From 891d26ded7966c35fdd7a99d7057c256a3bc982a Mon Sep 17 00:00:00 2001 From: George Berezhnoy Date: Mon, 24 Apr 2017 02:02:06 +0300 Subject: [PATCH 01/12] Now pasted text splits into paragraphs --- codex-editor.js | 4 +- codex-editor.js.map | 2 +- modules/callbacks.js | 90 ++++++++++++++++++++++---------------------- modules/caret.js | 25 ++++++++++++ modules/content.js | 15 ++++++-- 5 files changed, 84 insertions(+), 52 deletions(-) diff --git a/codex-editor.js b/codex-editor.js index a2dec0be..cb254ccc 100644 --- a/codex-editor.js +++ b/codex-editor.js @@ -1,3 +1,3 @@ -var codex=codex||{};codex.editor=function(e){function t(o){if(n[o])return n[o].exports;var r=n[o]={exports:{},id:o,loaded:!1};return e[o].call(r.exports,r,r.exports,t),r.loaded=!0,r.exports}var n={};return t.m=e,t.c=n,t.p="",t(0)}([function(e,t,n){"use strict";e.exports=function(e){e.version="1.6.3",e.scriptPrefix="cdx-script-";var t=function(){e.core=n(1),e.tools=n(2),e.ui=n(3),e.transport=n(4),e.renderer=n(5),e.saver=n(6),e.content=n(7),e.toolbar=n(8),e.callback=n(12),e.draw=n(13),e.caret=n(14),e.notifications=n(15),e.parser=n(16),e.sanitizer=n(17),e.listeners=n(19),e.destroyer=n(20),e.paste=n(21)};return e.settings={tools:["paragraph","header","picture","list","quote","code","twitter","instagram","smile"],holderId:"codex-editor",initialBlockPlugin:"paragraph"},e.nodes={holder:null,wrapper:null,toolbar:null,inlineToolbar:{wrapper:null,buttons:null,actions:null},toolbox:null,notifications:null,plusButton:null,showSettingsButton:null,showTrashButton:null,blockSettings:null,pluginSettings:null,defaultSettings:null,toolbarButtons:{},redactor:null},e.state={jsonOutput:[],blocks:[],inputs:[]},e.tools={},e.start=function(n){t(),e.core.prepare(n).then(e.ui.prepare).then(e.tools.prepare).then(e.sanitizer.prepare).then(e.paste.prepare).then(e.transport.prepare).then(e.renderer.makeBlocksFromData).then(e.ui.saveInputs).catch(function(t){e.core.log("Initialization failed with error: %o","warn",t)})},e}({})},function(e,t){"use strict";var n="function"==typeof Symbol&&"symbol"==typeof Symbol.iterator?function(e){return typeof e}:function(e){return e&&"function"==typeof Symbol&&e.constructor===Symbol&&e!==Symbol.prototype?"symbol":typeof e};e.exports=function(e){var t=codex.editor;e.prepare=function(e){return new Promise(function(o,r){e&&(t.settings.tools=e.tools||t.settings.tools),e.data&&(t.state.blocks=e.data),e.initialBlockPlugin&&(t.settings.initialBlockPlugin=e.initialBlockPlugin),e.sanitizer&&(t.settings.sanitizer=e.sanitizer),t.hideToolbar=e.hideToolbar,t.nodes.holder=document.getElementById(e.holderId||t.settings.holderId),void 0===n(t.nodes.holder)||null===t.nodes.holder?r(Error("Holder wasn't found by ID: #"+e.holderId)):o()})},e.log=function(e,t,n){t=t||"log",n?e="[codex-editor]: "+e:(n=e||"undefined",e="[codex-editor]: %o");try{"console"in window&&window.console[t]&&(n?window.console[t](e,n):window.console[t](e))}catch(e){}},e.insertAfter=function(e,t){e.parentNode.insertBefore(t,e.nextSibling)},e.nodeTypes={TAG:1,TEXT:3,COMMENT:8},e.keys={BACKSPACE:8,TAB:9,ENTER:13,SHIFT:16,CTRL:17,ALT:18,ESC:27,SPACE:32,LEFT:37,UP:38,DOWN:40,RIGHT:39,DELETE:46,META:91},e.isDomNode=function(e){return e&&"object"===("undefined"==typeof e?"undefined":n(e))&&e.nodeType&&e.nodeType==this.nodeTypes.TAG},e.isEmpty=function(e){return 0===Object.keys(e).length},e.ajax=function(e){if(e&&e.url){var t,n,r,i=window.XMLHttpRequest?new XMLHttpRequest:new ActiveXObject("Microsoft.XMLHTTP");if(e.async=!0,e.type=e.type||"GET",e.data=e.data||"",e["content-type"]=e["content-type"]||"application/json; charset=utf-8","GET"==e.type&&e.data)e.url=/\?/.test(e.url)?e.url+"&"+e.data:e.url+"?"+e.data;else{t="";for(r in e.data)t+=r+"="+encodeURIComponent(e.data[r])+"&"}e.withCredentials&&(i.withCredentials=!0);var a=void 0,s=void 0;if("function"!=typeof e.beforeSend||(s=e.beforeSend.call(),s!==!1))return i.open(e.type,e.url,e.async),n=o(e.data),n||("POST"!==e.type?i.setRequestHeader("Content-type",e["content-type"]):i.setRequestHeader("Content-type","application/x-www-form-urlencoded")),i.setRequestHeader("X-Requested-With","XMLHttpRequest"),a=s||i,"function"==typeof e.progress&&(i.upload.onprogress=e.progress.bind(a)),i.onreadystatechange=function(){4===i.readyState&&(200===i.status?"function"==typeof e.success&&e.success.call(a,i.responseText):"function"==typeof e.error&&e.error.call(a,i.responseText,i.status))},n?i.send(e.data):i.send(t),i}},e.importScript=function(e,n){return new Promise(function(o,r){var i=void 0;n?document.getElementById(t.scriptPrefix+n)&&o(e):r("Instance name is missed"),i=document.createElement("SCRIPT"),i.async=!0,i.defer=!0,i.id=t.scriptPrefix+n,i.onload=function(){o(e)},i.onerror=function(){r(e)},i.src=e,document.head.appendChild(i)})};var o=function(e){return e instanceof FormData};return e}({})},function(e,t){"use strict";e.exports=function(){function e(){return new Promise(function(e,o){Promise.resolve().then(function(){var t=[],o=n.tools;for(var r in o){var i=o[r];i.prepare&&"function"!=typeof i.prepare||!i.prepare||t.push(i)}return t.length||e(),t}).then(t).then(function(){n.core.log("Plugins loaded","info"),e()}).catch(function(e){o(e)})})}function t(e){return new Promise(function(t){e.reduce(function(r,i,a){return r.then(function(){return new Promise(function(r){o(i).then(r).then(function(){i.available=!0}).catch(function(e){n.core.log("Plugin «"+i.type+"» was not loaded. Preparation failed because %o","warn",e),i.available=!1,i.loadingMessage=e,r()}).then(function(){a==e.length-1&&t()})})})},Promise.resolve())})}var n=codex.editor,o=function(e){return e.prepare(e.config||{})};return{prepare:e}}()},function(e,t){"use strict";e.exports=function(e){var t=codex.editor;e.className={BLOCK_CLASSNAME:"ce-block",BLOCK_CONTENT:"ce-block__content",BLOCK_STRETCHED:"ce-block--stretched",BLOCK_HIGHLIGHTED:"ce-block--focused",SETTINGS_ITEM:"ce-settings__item"},e.prepare=function(){return new Promise(function(e){var n=t.draw.wrapper(),r=t.draw.redactor(),i=o();n.appendChild(i),n.appendChild(r),t.nodes.wrapper=n,t.nodes.redactor=r,t.nodes.holder.appendChild(n),e()}).then(s).then(n).then(l).then(a).then(c).catch(function(){t.core.log("Can't draw editor interface")})};var n=function(){var e=t.draw.inlineToolbar();t.nodes.inlineToolbar.wrapper=e,t.nodes.inlineToolbar.buttons=t.draw.inlineToolbarButtons(),t.nodes.inlineToolbar.actions=t.draw.inlineToolbarActions(),t.nodes.inlineToolbar.wrapper.appendChild(t.nodes.inlineToolbar.buttons),t.nodes.inlineToolbar.wrapper.appendChild(t.nodes.inlineToolbar.actions),t.nodes.wrapper.appendChild(t.nodes.inlineToolbar.wrapper)},o=function(){var e=t.draw.toolbar(),n=i(),o=r();return e.appendChild(n),e.appendChild(o),t.nodes.toolbar=e,e},r=function(){var e=t.draw.toolbarContent(),n=t.draw.toolbox(),o=t.draw.plusButton();return e.appendChild(o),e.appendChild(n),t.nodes.toolbox=n,t.nodes.plusButton=o,e},i=function(){var e=t.draw.blockSettings(),n=t.draw.blockButtons(),o=t.draw.defaultSettings(),r=t.draw.settingsButton(),i=t.toolbar.settings.makeRemoveBlockButton(),a=t.draw.pluginsSettings();return e.appendChild(a),e.appendChild(o),n.appendChild(r),n.appendChild(i),n.appendChild(e),t.nodes.blockSettings=e,t.nodes.pluginSettings=a,t.nodes.defaultSettings=o,t.nodes.showSettingsButton=r,t.nodes.showTrashButton=i,n},a=function(){t.nodes.notifications=t.notifications.createHolder()},s=function(){var e,n,o;for(n in t.settings.tools)e=t.settings.tools[n],t.tools[n]=e,e.iconClassname||!e.displayInToolbox?"function"==typeof e.render?e.displayInToolbox&&(o=t.draw.toolbarButton(n,e.iconClassname),t.nodes.toolbox.appendChild(o),t.nodes.toolbarButtons[n]=o):t.core.log("render method missed. Tool %o skipped","warn",n):t.core.log("Toolbar icon classname missed. Tool %o skipped","warn",n)},l=function(){var e,n,o={bold:{icon:"ce-icon-bold",command:"bold"},italic:{icon:"ce-icon-italic",command:"italic"},underline:{icon:"ce-icon-underline",command:"underline"},link:{icon:"ce-icon-link",command:"createLink"}};for(var r in o)n=o[r],e=t.draw.toolbarButtonInline(r,n.icon),t.nodes.inlineToolbar.buttons.appendChild(e),t.ui.setInlineToolbarButtonBehaviour(e,n.command)},c=function(){t.core.log("ui.bindEvents fired","info"),t.listeners.add(document,"keydown",t.callback.globalKeydown,!1),t.listeners.add(t.nodes.redactor,"keydown",t.callback.redactorKeyDown,!1),t.listeners.add(document,"keyup",t.callback.globalKeyup,!1),t.listeners.add(t.nodes.redactor,"click",t.callback.redactorClicked,!1),t.listeners.add(t.nodes.plusButton,"click",t.callback.plusButtonClicked,!1),t.listeners.add(t.nodes.showSettingsButton,"click",t.callback.showSettingsButtonClicked,!1);for(var e in t.nodes.toolbarButtons)t.listeners.add(t.nodes.toolbarButtons[e],"click",t.callback.toolbarButtonClicked,!1)};return e.addBlockHandlers=function(e){e&&(t.listeners.add(e,"keydown",t.callback.blockKeydown,!1),t.listeners.add(e,"paste",t.callback.blockPasteCallback,!1),t.listeners.add(e,"mouseup",t.toolbar.inline.show,!1))},e.saveInputs=function(){var e=t.nodes.redactor;t.state.inputs=[];var n=e.querySelectorAll("[contenteditable], input, textarea");Array.prototype.map.call(n,function(e){e.type&&"text"!=e.type&&"textarea"!=e.type||t.state.inputs.push(e)})},e.addInitialBlock=function(){var e,n=t.settings.initialBlockPlugin;return t.tools[n]?(e=t.tools[n].render(),e.setAttribute("data-placeholder","Расскажите свою историю..."),t.content.insertBlock({type:n,block:e}),void t.content.workingNodeChanged(e)):void t.core.log("Plugin %o was not implemented and can't be used as initial block","warn",n)},e.setInlineToolbarButtonBehaviour=function(e,n){t.listeners.add(e,"mousedown",function(e){t.toolbar.inline.toolClicked(e,n)},!1)},e}({})},function(e,t){"use strict";e.exports=function(e){var t=codex.editor,n=null;return e.input=null,e.arguments=null,e.prepare=function(){var e=t.draw.node("INPUT","",{type:"file"});t.listeners.add(e,"change",t.transport.fileSelected),t.transport.input=e},e.clearInput=function(){e.input=null,e.prepare()},e.fileSelected=function(){var o,r=this,i=r.files,a=new FormData;if(t.transport.arguments.multiple===!0)for(o=0;o'}),o=t.draw.node("DIV","ce-toolbar__remove-confirmation",{}),r=t.draw.node("DIV","ce-toolbar__remove-confirm",{textContent:"Удалить блок"}),i=t.draw.node("DIV","ce-toolbar__remove-cancel",{textContent:"Отмена"});return t.listeners.add(n,"click",t.toolbar.settings.removeButtonClicked,!1),t.listeners.add(r,"click",t.toolbar.settings.confirmRemovingRequest,!1),t.listeners.add(i,"click",t.toolbar.settings.cancelRemovingRequest,!1),o.appendChild(r),o.appendChild(i),e.appendChild(n),e.appendChild(o),t.toolbar.settings.setting=n,t.toolbar.settings.actions=o,e},e.removeButtonClicked=function(){var e=t.toolbar.settings.actions;e.classList.contains("opened")?t.toolbar.settings.hideRemoveActions():t.toolbar.settings.showRemoveActions(),t.toolbar.toolbox.close(),t.toolbar.settings.close()},e.cancelRemovingRequest=function(){t.toolbar.settings.actions.classList.remove("opened")},e.confirmRemovingRequest=function(){var e,n=t.content.currentNode;n.remove(),e=t.nodes.redactor.childNodes.length,0===e&&(t.content.currentNode=null,t.ui.addInitialBlock()),t.ui.saveInputs(),t.toolbar.close()},e.showRemoveActions=function(){t.toolbar.settings.actions.classList.add("opened")},e.hideRemoveActions=function(){t.toolbar.settings.actions.classList.remove("opened")},e}({})},function(e,t){"use strict";e.exports=function(e){var t=codex.editor;e.buttonsOpened=null,e.actionsOpened=null,e.wrappersOffset=null,e.storedSelection=null,e.show=function(){var n,o=t.content.currentNode,r=o.dataset.tool;if(n=t.tools[r],n.showInlineToolbar){var i=e.getSelectionText(),a=t.nodes.inlineToolbar.wrapper;i.length>0&&(t.toolbar.inline.move(),a.classList.add("opened"),t.toolbar.inline.showButtons())}},e.close=function(){var e=t.nodes.inlineToolbar.wrapper;e.classList.remove("opened")},e.move=function(){this.wrappersOffset||(this.wrappersOffset=this.getWrappersOffset());var e,n,o=this.getSelectionCoords(),r=0,i=t.nodes.inlineToolbar.wrapper;0===i.offsetHeight&&(r=40),e=o.x-this.wrappersOffset.left,n=o.y+window.scrollY-this.wrappersOffset.top-r-i.offsetHeight,i.style.transform="translate3D("+Math.floor(e)+"px, "+Math.floor(n)+"px, 0)",t.toolbar.inline.closeButtons(),t.toolbar.inline.closeAction()},e.toolClicked=function(e,n){switch(n){case"createLink":t.toolbar.inline.createLinkAction(e,n);break;default:t.toolbar.inline.defaultToolAction(n)}t.nodes.inlineToolbar.buttons.childNodes.forEach(t.toolbar.inline.hightlight)},e.getWrappersOffset=function(){var e=t.nodes.wrapper,n=this.getOffset(e);return this.wrappersOffset=n,n},e.getOffset=function(e){for(var t=0,n=0;e&&!isNaN(e.offsetLeft)&&!isNaN(e.offsetTop);)t+=e.offsetLeft+e.clientLeft,n+=e.offsetTop+e.clientTop,e=e.offsetParent;return{top:n,left:t}},e.getSelectionCoords=function(){var e,t=document.selection,n=0,o=0;if(t)"Control"!=t.type&&(e=t.createRange(),e.collapse(!0),n=e.boundingLeft,o=e.boundingTop);else if(window.getSelection&&(t=window.getSelection(),t.rangeCount&&(e=t.getRangeAt(0).cloneRange(),e.getClientRects))){e.collapse(!0);var r=e.getClientRects()[0];if(!r)return;n=r.left,o=r.top}return{x:n,y:o}},e.getSelectionText=function(){var e="";return window.getSelection&&(e=window.getSelection().toString()),e},e.showButtons=function(){var e=t.nodes.inlineToolbar.buttons;e.classList.add("opened"),t.toolbar.inline.buttonsOpened=!0,t.nodes.inlineToolbar.buttons.childNodes.forEach(t.toolbar.inline.hightlight)},e.closeButtons=function(){var e=t.nodes.inlineToolbar.buttons;e.classList.remove("opened"),t.toolbar.inline.buttonsOpened=!1},e.showActions=function(){var e=t.nodes.inlineToolbar.actions;e.classList.add("opened"),t.toolbar.inline.actionsOpened=!0},e.closeAction=function(){var e=t.nodes.inlineToolbar.actions;e.innerHTML="",e.classList.remove("opened"),t.toolbar.inline.actionsOpened=!1};var n=function(e){if(e.keyCode==t.core.keys.ENTER){var n=t.content.currentNode,o=t.toolbar.inline.storedSelection;t.toolbar.inline.restoreSelection(n,o),t.toolbar.inline.setAnchor(this.value),e.preventDefault(),e.stopImmediatePropagation(),t.toolbar.inline.clearRange()}};return e.createLinkAction=function(e){var o=this.isLinkActive(),r=t.content.currentNode,i=t.toolbar.inline.saveSelection(r);if(t.toolbar.inline.storedSelection=i,o)t.toolbar.inline.restoreSelection(r,i),t.toolbar.inline.defaultToolAction("unlink");else{var a=t.draw.inputForLink();t.nodes.inlineToolbar.actions.appendChild(a),t.toolbar.inline.closeButtons(),t.toolbar.inline.showActions(),a.focus(),e.preventDefault(),t.listeners.add(a,"keydown",n,!1)}},e.isLinkActive=function(){var e=!1;return t.nodes.inlineToolbar.buttons.childNodes.forEach(function(t){var n=t.dataset.type;"link"==n&&t.classList.contains("hightlighted")&&(e=!0)}),e},e.defaultToolAction=function(e){document.execCommand(e,!1,null)},e.setAnchor=function(e){document.execCommand("createLink",!1,e),t.toolbar.inline.closeAction()},e.saveSelection=function(e){var t,n=window.getSelection().getRangeAt(0),o=n.cloneRange();return o.selectNodeContents(e),o.setEnd(n.startContainer,n.startOffset),t=o.toString().length,{start:t,end:t+n.toString().length}},e.restoreSelection=function(e,t){var n=document.createRange(),o=0;n.setStart(e,0),n.collapse(!0);for(var r,i,a=[e],s=!1,l=!1;!l&&(r=a.pop());)if(3==r.nodeType)i=o+r.length,!s&&t.start>=o&&t.start<=i&&(n.setStart(r,t.start-o),s=!0),s&&t.end>=o&&t.end<=i&&(n.setEnd(r,t.end-o),l=!0),o=i;else for(var c=r.childNodes.length;c--;)a.push(r.childNodes[c]);var d=window.getSelection();d.removeAllRanges(),d.addRange(n)},e.clearRange=function(){var e=window.getSelection();e.removeAllRanges()},e.hightlight=function(e){var n=e.dataset.type;document.queryCommandState(n)?t.toolbar.inline.setButtonHighlighted(e):t.toolbar.inline.removeButtonsHighLight(e);var o=window.getSelection(),r=o.anchorNode.parentNode;"A"==r.tagName&&"link"==n&&t.toolbar.inline.setButtonHighlighted(e)},e.setButtonHighlighted=function(e){if(e.classList.add("hightlighted"),"link"==e.dataset.type){var t=e.childNodes[0];t.classList.remove("ce-icon-link"),t.classList.add("ce-icon-unlink")}},e.removeButtonsHighLight=function(e){if(e.classList.remove("hightlighted"),"link"==e.dataset.type){var t=e.childNodes[0];t.classList.remove("ce-icon-unlink"),t.classList.add("ce-icon-link")}},e}({})},function(e,t){"use strict";e.exports=function(e){var t=codex.editor;return e.opened=!1,e.open=function(){t.toolbar.settings.opened&&t.toolbar.settings.close(),t.nodes.toolbox.classList.add("opened"),t.nodes.plusButton.classList.add("clicked"),t.toolbar.toolbox.opened=!0},e.close=function(){t.nodes.toolbox.classList.remove("opened"),t.nodes.plusButton.classList.remove("clicked"),t.toolbar.toolbox.opened=!1,t.toolbar.current=null},e.leaf=function(){var e=t.toolbar.current,n=Object.keys(t.tools),o=t.nodes.toolbarButtons,r=0,i=void 0,a=void 0,s=void 0;if(e)for(r=(n.indexOf(e)+1)%n.length,a=n[r];!t.tools[a].displayInToolbox;)r=(r+1)%n.length,a=n[r];else for(s in t.tools){if(t.tools[s].displayInToolbox)break;r++}i=n[r];for(var l in o)o[l].classList.remove("selected");o[i].classList.add("selected"),t.toolbar.current=i},e.toolClicked=function(e){var n,o,r,i=["image","link","list","instagram","twitter","embed"],a=t.tools[t.toolbar.current],s=t.content.currentNode,l=t.caret.inputIndex;n=a.render(),r={block:n,type:a.type,stretched:!1},s&&i.indexOf(s.dataset.tool)===-1&&""===s.textContent.trim()?t.content.switchBlock(s,n,a.type):(t.content.insertBlock(r),l++),o=a.appendCallback,o&&"function"==typeof o&&o.call(e),window.setTimeout(function(){t.caret.setToBlock(l)},10),t.content.workingNodeChanged(),t.toolbar.move()},e}({})},function(e,t){"use strict";e.exports=function(e){var t=codex.editor;e.globalKeydown=function(e){switch(e.keyCode){case t.core.keys.ENTER:o(e)}},e.redactorKeyDown=function(e){switch(e.keyCode){case t.core.keys.TAB:n(e);break;case t.core.keys.ENTER:i(e);break;case t.core.keys.ESC:a(e);break;default:l(e)}},e.globalKeyup=function(e){switch(e.keyCode){case t.core.keys.UP:case t.core.keys.LEFT:case t.core.keys.RIGHT:case t.core.keys.DOWN:s(e)}};var n=function(e){e.preventDefault();var n=t.content.currentNode.querySelectorAll("textarea, input"),o=!0,r=!t.content.currentNode.textContent.trim();Array.prototype.map.call(n,function(e){"textarea"!=e.type&&"text"!=e.type||(o=o&&!e.value.trim())});var i=r&&o;i&&(t.toolbar.opened||t.toolbar.open(),t.toolbar.opened&&!t.toolbar.toolbox.opened?t.toolbar.toolbox.open():t.toolbar.toolbox.leaf())},o=function(){t.content.editorAreaHightlighted&&(t.caret.inputIndex=-1,r())},r=function(){var e=t.settings.initialBlockPlugin;t.content.insertBlock({type:e,block:t.tools[e].render()},!0),t.toolbar.move(),t.toolbar.open()},i=function(e){"true"==e.target.contentEditable&&t.caret.saveCurrentInputIndex();var n=t.caret.getCurrentInputIndex()||0,o=t.content.currentNode,r=o.dataset.tool,i=t.toolbar.opened&&t.toolbar.current&&e.target==t.state.inputs[n],a=t.tools[r].enableLineBreaks,s=t.settings.initialBlockPlugin;if(i)return e.preventDefault(),t.toolbar.toolbox.toolClicked(e),t.toolbar.close(),e.stopPropagation(),void e.stopImmediatePropagation();if(e.shiftKey||a)return e.stopPropagation(),void e.stopImmediatePropagation();var l=window.getSelection(),c=l.anchorNode,d=t.caret.position.atTheEnd(),u=!1;if(e.shiftKey&&!a)return t.callback.enterPressedOnBlock(t.content.currentBlock,e),void e.preventDefault();if(u=c&&"true"!=c.parentNode.contentEditable,c.nodeType!=t.core.nodeTypes.TEXT||u||d){var p=t.content.isLastNode(c);p&&d&&(e.preventDefault(),e.stopPropagation(),e.stopImmediatePropagation(),t.core.log("ENTER clicked in last textNode. Create new BLOCK"),t.content.insertBlock({type:s,block:t.tools[s].render()},!0),t.toolbar.move(),t.toolbar.open(),t.toolbar.showPlusButton())}else e.preventDefault(),t.core.log("Splitting Text node..."),t.content.splitBlock(n),t.state.inputs[n+1].textContent.trim()||t.toolbar.showPlusButton();t.ui.saveInputs()},a=function(e){t.toolbar.close(),t.toolbar.toolbox.close(),e.preventDefault()},s=function(e){t.content.workingNodeChanged(),t.toolbar.close(),t.toolbar.move()},l=function(){t.toolbar.close(),t.toolbar.inline.actionsOpened||(t.toolbar.inline.close(),t.content.clearMark())};e.redactorClicked=function(e){c(),t.content.workingNodeChanged(e.target),t.ui.saveInputs();var n,o=t.toolbar.inline.getSelectionText();if(0===o.length&&t.toolbar.inline.close(),"true"==e.target.contentEditable&&t.caret.saveCurrentInputIndex(),null===t.content.currentNode){var r=t.state.inputs.length>0?t.state.inputs.length-1:0;if(t.state.inputs.length&&(n=t.content.getFirstLevelBlock(t.state.inputs[r])),t.state.inputs.length&&""===t.state.inputs[r].textContent&&n.dataset.tool==t.settings.initialBlockPlugin)t.caret.setToBlock(r);else{var i=t.settings.initialBlockPlugin;t.content.insertBlock({type:i,block:t.tools[i].render()}),1===t.state.inputs.length?t.caret.setToBlock(r):t.caret.setToNextBlock(r)}}else t.toolbar.settings.close(),t.toolbar.toolbox.close();t.toolbar.move(),t.toolbar.open();var a=!t.content.currentNode.textContent.trim(),s=t.content.currentNode.dataset.tool,l=s==t.settings.initialBlockPlugin;t.toolbar.hidePlusButton(),a||t.content.markBlock(),l&&a&&t.toolbar.showPlusButton()};var c=function(){var e=window.getSelection(),n=e.anchorNode,o=!1;if(0===e.rangeCount)t.content.editorAreaHightlighted=!0;else{for(t.core.isDomNode(n)||(n=n.parentNode),"true"==n.contentEditable&&(o=!0);"true"!=n.contentEditable&&(n=n.parentNode,"true"==n.contentEditable&&(o=!0),n!=document.body););t.content.editorAreaHightlighted=!o}};e.toolbarButtonClicked=function(e){var n=this;t.toolbar.current=n.dataset.type,t.toolbar.toolbox.toolClicked(e),t.toolbar.close()},e.plusButtonClicked=function(){t.nodes.toolbox.classList.contains("opened")?t.toolbar.toolbox.close():t.toolbar.toolbox.open()},e.blockKeydown=function(e){var n=e.target;switch(e.keyCode){case t.core.keys.DOWN:case t.core.keys.RIGHT:d(e);break;case t.core.keys.BACKSPACE:p(n,e);break;case t.core.keys.UP:case t.core.keys.LEFT:u(e)}};var d=function(e){var n,o=window.getSelection(),r=t.state.inputs,i=o.anchorNode;if(!i)return!1;for(;"true"!=i.contentEditable;)n=i.parentNode,i=n;for(var a=0;i!=r[a];)a++;if(!i.textContent)return void t.caret.setToNextBlock(a);var s,l,c=!1,d=!1;return s=i.childNodes[i.childNodes.length-1],l=t.core.isDomNode(s)?t.content.getDeepestTextNodeFromPosition(s,s.childNodes.length):s,c=o.anchorNode==l,d=l.length==o.anchorOffset,c&&d?void t.caret.setToNextBlock(a):(t.core.log("arrow [down|right] : caret does not reached the end"),!1)},u=function(e){var n,o=window.getSelection(),r=t.state.inputs,i=o.anchorNode;if(!i)return!1;if(0!==o.anchorOffset)return!1;for(;"true"!=i.contentEditable;)n=i.parentNode,i=n;for(var a=0;i!=r[a];)a++;var s,l,c=!1,d=!1;return i.textContent?(s=i.childNodes[0],l=t.core.isDomNode(s)?t.content.getDeepestTextNodeFromPosition(s,0):s,c=o.anchorNode==l,d=0===o.anchorOffset,void(c&&d&&t.caret.setToPreviousBlock(a))):void t.caret.setToPreviousBlock(a)},p=function(e,n){var o,r,i,a=t.caret.getCurrentInputIndex();if(f(n.target)){if(""!=n.target.value.trim())return;e.remove()}if(e.textContent.trim()){if(o=t.content.getRange(),r=o.endOffset-o.startOffset,!t.caret.position.atStart()||r||!t.state.inputs[a-1])return;t.content.mergeBlocks(a)}r||e.remove(),i=t.nodes.redactor.childNodes.length,0===i?(t.content.currentNode=null,t.ui.addInitialBlock(),t.ui.saveInputs(),window.setTimeout(function(){t.caret.setToPreviousBlock(1)},10)):0!==t.caret.inputIndex?t.caret.setToPreviousBlock(t.caret.inputIndex):t.caret.setToNextBlock(t.caret.inputIndex),t.toolbar.move(), -t.toolbar.opened||t.toolbar.open(),t.ui.saveInputs(),n.preventDefault()};e.blockPasteCallback=function(e){if(!f(e.target)){e.preventDefault();var n=t.content.getEditableParent(e.target),o=t.content.currentNode;if(n){var r,i,a=e.clipboardData.getData("text/html"),s=e.clipboardData.getData("text/plain"),l=t.draw.node("DIV","",{});i=document.createDocumentFragment(),""!=a.trim()?(r=t.sanitizer.clean(a),l.innerHTML=r):l.innerText=s.toString();for(var c,d;c=l.firstChild;)d=i.appendChild(c);if(!t.tools[o.dataset.tool].allowRenderOnPaste||!t.paste.pasted(e)){var u,p;u=window.getSelection(),p=u.getRangeAt(0),p.deleteContents(),p.insertNode(i),d&&(p=p.cloneRange(),p.setStartAfter(d),p.collapse(!0),u.removeAllRanges(),u.addRange(p))}}}},e.showSettingsButtonClicked=function(e){var n=t.content.currentNode.dataset.tool;t.toolbar.settings.toggle(n),t.toolbar.toolbox.close(),t.toolbar.settings.hideRemoveActions()};var f=function(e){var t=["INPUT","TEXTAREA"];return t.indexOf(e.tagName)!=-1};return e}({})},function(e,t){"use strict";e.exports=function(e){return e.wrapper=function(){var e=document.createElement("div");return e.className+="codex-editor",e},e.redactor=function(){var e=document.createElement("div");return e.className+="ce-redactor",e},e.ceBlock=function(){var e=document.createElement("DIV");return e.className+="ce_block",e},e.toolbar=function(){var e=document.createElement("div");return e.className+="ce-toolbar",e},e.toolbarContent=function(){var e=document.createElement("DIV");return e.classList.add("ce-toolbar__content"),e},e.inlineToolbar=function(){var e=document.createElement("DIV");return e.className+="ce-toolbar-inline",e},e.inlineToolbarButtons=function(){var e=document.createElement("DIV");return e.className+="ce-toolbar-inline__buttons",e},e.inlineToolbarActions=function(){var e=document.createElement("DIV");return e.className+="ce-toolbar-inline__actions",e},e.inputForLink=function(){var e=document.createElement("INPUT");return e.type="input",e.className+="inputForLink",e.placeholder="Вставьте ссылку ...",e.setAttribute("form","defaultForm"),e.setAttribute("autofocus","autofocus"),e},e.blockButtons=function(){var e=document.createElement("div");return e.className+="ce-toolbar__actions",e},e.blockSettings=function(){var e=document.createElement("div");return e.className+="ce-settings",e},e.defaultSettings=function(){var e=document.createElement("div");return e.classList.add("ce-settings_default"),e},e.pluginsSettings=function(){var e=document.createElement("div");return e.classList.add("ce-settings_plugin"),e},e.plusButton=function(){var e=document.createElement("span");return e.className="ce-toolbar__plus",e},e.settingsButton=function(){var e=document.createElement("span");return e.className="ce-toolbar__settings-btn",e.innerHTML='',e},e.toolbox=function(){var e=document.createElement("div");return e.className="ce-toolbar__tools",e},e.toolbarButton=function(e,t){var n=document.createElement("li"),o=document.createElement("i"),r=document.createElement("span");return n.dataset.type=e,n.setAttribute("title",e),o.classList.add(t),r.classList.add("ce_toolbar_tools--title"),n.appendChild(o),n.appendChild(r),n},e.toolbarButtonInline=function(e,t){var n=document.createElement("BUTTON"),o=document.createElement("I");return n.type="button",n.dataset.type=e,o.classList.add(t),n.appendChild(o),n},e.block=function(e,t){var n=document.createElement(e);return n.innerHTML=t||"",n},e.node=function(e,t,n){var o=document.createElement(e);if(t&&(o.className=t),n)for(var r in n)o[r]=n[r];return o},e.unavailableBlock=function(){var e=document.createElement("DIV");return e.classList.add("cdx-unavailable-block"),e},e}({})},function(e,t){"use strict";e.exports=function(e){var t=codex.editor;return e.inputIndex=null,e.offset=null,e.focusedNodeIndex=null,e.set=function(n,o,r){r=r||e.offset||0,o=o||e.focusedNodeIndex||0;var i,a=n.childNodes;if(i=0===a.length?n:a[o],"true"!=n.contentEditable)return void n.focus();t.core.isDomNode(i)&&(i=t.content.getDeepestTextNodeFromPosition(i,i.childNodes.length));var s=document.createRange(),l=window.getSelection();window.setTimeout(function(){s.setStart(i,r),s.setEnd(i,r),l.removeAllRanges(),l.addRange(s),t.caret.saveCurrentInputIndex()},20)},e.saveCurrentInputIndex=function(){var n,o=window.getSelection(),r=t.state.inputs,i=o.anchorNode;if(i){for(;"true"!=i.contentEditable;)n=i.parentNode,i=n;for(var a=0;i!=r[a];)a++;e.inputIndex=a}},e.getCurrentInputIndex=function(){return e.inputIndex},e.setToNextBlock=function(e){var n=t.state.inputs,o=n[e+1];if(!o)return void t.core.log("We are reached the end");if(!o.childNodes.length){var r=document.createTextNode("");o.appendChild(r)}t.caret.inputIndex=e+1,t.caret.set(o,0,0),t.content.workingNodeChanged(o)},e.setToBlock=function(e){var n=t.state.inputs,o=n[e];if(o){if(!o.childNodes.length){var r=document.createTextNode("");o.appendChild(r)}t.caret.inputIndex=e,t.caret.set(o,0,0),t.content.workingNodeChanged(o)}},e.setToPreviousBlock=function(e){e=e||0;var n,o,r,i=t.state.inputs,a=i[e-1];return a?(n=t.content.getDeepestTextNodeFromPosition(a,a.childNodes.length),o=n.length,a.childNodes.length||(r=document.createTextNode(""),a.appendChild(r)),t.caret.inputIndex=e-1,t.caret.set(a,a.childNodes.length-1,o),void t.content.workingNodeChanged(i[e-1])):void t.core.log("We are reached first node")},e.position={atStart:function(){var e=window.getSelection(),n=e.anchorOffset,o=e.anchorNode,r=t.content.getFirstLevelBlock(o),i=r.childNodes[0];t.core.isDomNode(o)||(o=o.parentNode);var a=o===i.childNodes[0],s=0===n;return a&&s},atTheEnd:function(){var e=window.getSelection(),t=e.anchorOffset,n=e.anchorNode;return!n||!n.length||t===n.length}},e}({})},function(e,t){"use strict";e.exports=function(e){var t=codex.editor,n=[],o=function(e){n.push(e);for(var t=0;t5;)"confirm"!=n[t].type&&"prompt"!=n[t].type?(n[t].close(),n.splice(t,1)):t++};return e.createHolder=function(){var e=t.draw.node("DIV","cdx-notifications-block");return t.nodes.notifications=document.body.appendChild(e),e},e.errorThrown=function(e,n){t.notifications.notification({message:"This action is not available currently",type:n.type})},e.notification=function(e){function n(e){if(!e||!e.message)return void t.core.log("Can't create notification. Message is missed");e.type=e.type||"alert",e.time=1e3*e.time||1e4;var n=t.draw.node("DIV","cdx-notification"),o=t.draw.node("DIV","cdx-notification__message"),r=t.draw.node("INPUT","cdx-notification__input"),f=t.draw.node("SPAN","cdx-notification__ok-btn"),g=t.draw.node("SPAN","cdx-notification__cancel-btn");o.textContent=e.message,f.textContent=e.okMsg||"ОК",g.textContent=e.cancelMsg||"Отмена",t.listeners.add(f,"click",u),t.listeners.add(g,"click",p),n.appendChild(o),"prompt"==e.type&&n.appendChild(r),n.appendChild(f),"prompt"!=e.type&&"confirm"!=e.type||n.appendChild(g),n.classList.add("cdx-notification-"+e.type),n.dataset.type=e.type,a=n,l=e.type,c=e.confirm,s=e.cancel,d=r,"prompt"!=e.type&&"confirm"!=e.type&&window.setTimeout(i,e.time)}function r(){t.nodes.notifications.appendChild(a),d.focus(),t.nodes.notifications.classList.add("cdx-notification__notification-appending"),window.setTimeout(function(){t.nodes.notifications.classList.remove("cdx-notification__notification-appending")},100),o({type:l,close:i})}function i(){a.remove()}var a=null,s=null,l=null,c=null,d=null,u=function(){if(i(),"function"==typeof c)return"prompt"==l?void c(d.value):void c()},p=function(){i(),"function"==typeof s&&s()};return e&&(n(e),r()),{create:n,send:r,close:i}},e.clear=function(){t.nodes.notifications.innerHTML="",n=[]},e}({})},function(e,t){"use strict";e.exports=function(e){var t=codex.editor;return e.insertPastedContent=function(e,n){t.content.insertBlock({type:e.type,block:e.render({text:n.innerHTML})})},e.isFirstLevelBlock=function(e){return e.nodeType==t.core.nodeTypes.TAG&&e.classList.contains(t.ui.className.BLOCK_CLASSNAME)},e}({})},function(e,t,n){"use strict";e.exports=function(e){var t=n(18),o=codex.editor;e.prepare=function(){o.settings.sanitizer&&!o.core.isEmpty(o.settings.sanitizer)&&(r.CUSTOM=o.settings.sanitizer)};var r={CUSTOM:null,BASIC:{tags:{p:{},a:{href:!0,target:"_blank",rel:"nofollow"},i:{},b:{},strong:{},em:{},span:{}}}};e.Config=r;var i=function(e){var n=e||r.CUSTOM||r.BASIC;return new t(n)};return e.clean=function(e,t){var n=i(t);return n.clean(e)},e}({})},function(e,t,n){var o,r;!function(i,a){o=a,r="function"==typeof o?o.call(t,n,t,e):o,!(void 0!==r&&(e.exports=r))}(this,function(){function e(e){var t=e.tags,n=Object.keys(t),o=n.map(function(e){return typeof t[e]}).every(function(e){return"object"===e||"boolean"===e||"function"===e});if(!o)throw new Error("The configuration was invalid");this.config=e}function t(e){return s.indexOf(e.nodeName)!==-1}function n(e){return l.indexOf(e.nodeName)!==-1}function o(e){return document.createTreeWalker(e,NodeFilter.SHOW_TEXT|NodeFilter.SHOW_ELEMENT|NodeFilter.SHOW_COMMENT,null,!1)}function r(e,t,n){return"function"==typeof e.tags[t]?e.tags[t](n):e.tags[t]}function i(e,t){return"undefined"==typeof t||"boolean"==typeof t&&!t}function a(e,t,n){var o=e.name.toLowerCase();return t!==!0&&("function"==typeof t[o]?!t[o](e.value,n):"undefined"==typeof t[o]||(t[o]===!1||"string"==typeof t[o]&&t[o]!==e.value))}var s=["P","LI","TD","TH","DIV","H1","H2","H3","H4","H5","H6","PRE"],l=["A","B","STRONG","I","EM","SUB","SUP","U","STRIKE"];return e.prototype.clean=function(e){var t=document.createElement("div");return t.innerHTML=e,this._sanitize(t),t.innerHTML},e.prototype._sanitize=function(e){var s=o(e),l=s.firstChild();if(l)do if(!l._sanitized)if(l.nodeType!==Node.TEXT_NODE){if(l.nodeType===Node.COMMENT_NODE){e.removeChild(l),this._sanitize(e);break}var c,d=n(l);d&&(c=Array.prototype.some.call(l.childNodes,t));var u=!!e.parentNode,p=t(e)&&t(l)&&u,f=l.nodeName.toLowerCase(),g=r(this.config,f,l),h=d&&c;if(h||i(l,g)||!this.config.keepNestedBlockElements&&p){if("SCRIPT"!==l.nodeName&&"STYLE"!==l.nodeName)for(;l.childNodes.length>0;)e.insertBefore(l.childNodes[0],l);e.removeChild(l),this._sanitize(e);break}for(var v=0;v0&&t.splice(s,1)}},e.removeAll=function(){t.map(function(t){e.remove(t.element,t.type,t.handler)})},e.get=function(t,n,o){return e.search.all(t,n,o)},e}({})},function(e,t){"use strict";var n="function"==typeof Symbol&&"symbol"==typeof Symbol.iterator?function(e){return typeof e}:function(e){return e&&"function"==typeof Symbol&&e.constructor===Symbol&&e!==Symbol.prototype?"symbol":typeof e};e.exports=function(e){var t=codex.editor;return e.removeNodes=function(){t.nodes.wrapper.remove(),t.nodes.notifications.remove()},e.destroyPlugins=function(){for(var e in t.tools)"function"==typeof t.tools[e].destroy&&t.tools[e].destroy()},e.destroyScripts=function(){for(var e=document.getElementsByTagName("SCRIPT"),n=0;n"+t.split("\n\n").join("

")+"

";var n,o,r,i,a=document.createElement("DIV"),s=document.createElement("DIV"),c=["DIV","P"];for(a.innerHTML=e,o=document.createElement("P"),n=0;n'}),o=t.draw.node("DIV","ce-toolbar__remove-confirmation",{}),r=t.draw.node("DIV","ce-toolbar__remove-confirm",{textContent:"Удалить блок"}),i=t.draw.node("DIV","ce-toolbar__remove-cancel",{textContent:"Отмена"});return t.listeners.add(n,"click",t.toolbar.settings.removeButtonClicked,!1),t.listeners.add(r,"click",t.toolbar.settings.confirmRemovingRequest,!1),t.listeners.add(i,"click",t.toolbar.settings.cancelRemovingRequest,!1),o.appendChild(r),o.appendChild(i),e.appendChild(n),e.appendChild(o),t.toolbar.settings.setting=n,t.toolbar.settings.actions=o,e},e.removeButtonClicked=function(){var e=t.toolbar.settings.actions;e.classList.contains("opened")?t.toolbar.settings.hideRemoveActions():t.toolbar.settings.showRemoveActions(),t.toolbar.toolbox.close(),t.toolbar.settings.close()},e.cancelRemovingRequest=function(){t.toolbar.settings.actions.classList.remove("opened")},e.confirmRemovingRequest=function(){var e,n=t.content.currentNode;n.remove(),e=t.nodes.redactor.childNodes.length,0===e&&(t.content.currentNode=null,t.ui.addInitialBlock()),t.ui.saveInputs(),t.toolbar.close()},e.showRemoveActions=function(){t.toolbar.settings.actions.classList.add("opened")},e.hideRemoveActions=function(){t.toolbar.settings.actions.classList.remove("opened")},e}({})},function(e,t){"use strict";e.exports=function(e){var t=codex.editor;e.buttonsOpened=null,e.actionsOpened=null,e.wrappersOffset=null,e.storedSelection=null,e.show=function(){var n,o=t.content.currentNode,r=o.dataset.tool;if(n=t.tools[r],n.showInlineToolbar){var i=e.getSelectionText(),a=t.nodes.inlineToolbar.wrapper;i.length>0&&(t.toolbar.inline.move(),a.classList.add("opened"),t.toolbar.inline.showButtons())}},e.close=function(){var e=t.nodes.inlineToolbar.wrapper;e.classList.remove("opened")},e.move=function(){this.wrappersOffset||(this.wrappersOffset=this.getWrappersOffset());var e,n,o=this.getSelectionCoords(),r=0,i=t.nodes.inlineToolbar.wrapper;0===i.offsetHeight&&(r=40),e=o.x-this.wrappersOffset.left,n=o.y+window.scrollY-this.wrappersOffset.top-r-i.offsetHeight,i.style.transform="translate3D("+Math.floor(e)+"px, "+Math.floor(n)+"px, 0)",t.toolbar.inline.closeButtons(),t.toolbar.inline.closeAction()},e.toolClicked=function(e,n){switch(n){case"createLink":t.toolbar.inline.createLinkAction(e,n);break;default:t.toolbar.inline.defaultToolAction(n)}t.nodes.inlineToolbar.buttons.childNodes.forEach(t.toolbar.inline.hightlight)},e.getWrappersOffset=function(){var e=t.nodes.wrapper,n=this.getOffset(e);return this.wrappersOffset=n,n},e.getOffset=function(e){for(var t=0,n=0;e&&!isNaN(e.offsetLeft)&&!isNaN(e.offsetTop);)t+=e.offsetLeft+e.clientLeft,n+=e.offsetTop+e.clientTop,e=e.offsetParent;return{top:n,left:t}},e.getSelectionCoords=function(){var e,t=document.selection,n=0,o=0;if(t)"Control"!=t.type&&(e=t.createRange(),e.collapse(!0),n=e.boundingLeft,o=e.boundingTop);else if(window.getSelection&&(t=window.getSelection(),t.rangeCount&&(e=t.getRangeAt(0).cloneRange(),e.getClientRects))){e.collapse(!0);var r=e.getClientRects()[0];if(!r)return;n=r.left,o=r.top}return{x:n,y:o}},e.getSelectionText=function(){var e="";return window.getSelection&&(e=window.getSelection().toString()),e},e.showButtons=function(){var e=t.nodes.inlineToolbar.buttons;e.classList.add("opened"),t.toolbar.inline.buttonsOpened=!0,t.nodes.inlineToolbar.buttons.childNodes.forEach(t.toolbar.inline.hightlight)},e.closeButtons=function(){var e=t.nodes.inlineToolbar.buttons;e.classList.remove("opened"),t.toolbar.inline.buttonsOpened=!1},e.showActions=function(){var e=t.nodes.inlineToolbar.actions;e.classList.add("opened"),t.toolbar.inline.actionsOpened=!0},e.closeAction=function(){var e=t.nodes.inlineToolbar.actions;e.innerHTML="",e.classList.remove("opened"),t.toolbar.inline.actionsOpened=!1};var n=function(e){if(e.keyCode==t.core.keys.ENTER){var n=t.content.currentNode,o=t.toolbar.inline.storedSelection;t.toolbar.inline.restoreSelection(n,o),t.toolbar.inline.setAnchor(this.value),e.preventDefault(),e.stopImmediatePropagation(),t.toolbar.inline.clearRange()}};return e.createLinkAction=function(e){var o=this.isLinkActive(),r=t.content.currentNode,i=t.toolbar.inline.saveSelection(r);if(t.toolbar.inline.storedSelection=i,o)t.toolbar.inline.restoreSelection(r,i),t.toolbar.inline.defaultToolAction("unlink");else{var a=t.draw.inputForLink();t.nodes.inlineToolbar.actions.appendChild(a),t.toolbar.inline.closeButtons(),t.toolbar.inline.showActions(),a.focus(),e.preventDefault(),t.listeners.add(a,"keydown",n,!1)}},e.isLinkActive=function(){var e=!1;return t.nodes.inlineToolbar.buttons.childNodes.forEach(function(t){var n=t.dataset.type;"link"==n&&t.classList.contains("hightlighted")&&(e=!0)}),e},e.defaultToolAction=function(e){document.execCommand(e,!1,null)},e.setAnchor=function(e){document.execCommand("createLink",!1,e),t.toolbar.inline.closeAction()},e.saveSelection=function(e){var t,n=window.getSelection().getRangeAt(0),o=n.cloneRange();return o.selectNodeContents(e),o.setEnd(n.startContainer,n.startOffset),t=o.toString().length,{start:t,end:t+n.toString().length}},e.restoreSelection=function(e,t){var n=document.createRange(),o=0;n.setStart(e,0),n.collapse(!0);for(var r,i,a=[e],s=!1,c=!1;!c&&(r=a.pop());)if(3==r.nodeType)i=o+r.length,!s&&t.start>=o&&t.start<=i&&(n.setStart(r,t.start-o),s=!0),s&&t.end>=o&&t.end<=i&&(n.setEnd(r,t.end-o),c=!0),o=i;else for(var l=r.childNodes.length;l--;)a.push(r.childNodes[l]);var d=window.getSelection();d.removeAllRanges(),d.addRange(n)},e.clearRange=function(){var e=window.getSelection();e.removeAllRanges()},e.hightlight=function(e){var n=e.dataset.type;document.queryCommandState(n)?t.toolbar.inline.setButtonHighlighted(e):t.toolbar.inline.removeButtonsHighLight(e);var o=window.getSelection(),r=o.anchorNode.parentNode;"A"==r.tagName&&"link"==n&&t.toolbar.inline.setButtonHighlighted(e)},e.setButtonHighlighted=function(e){if(e.classList.add("hightlighted"),"link"==e.dataset.type){var t=e.childNodes[0];t.classList.remove("ce-icon-link"),t.classList.add("ce-icon-unlink")}},e.removeButtonsHighLight=function(e){if(e.classList.remove("hightlighted"),"link"==e.dataset.type){var t=e.childNodes[0];t.classList.remove("ce-icon-unlink"),t.classList.add("ce-icon-link")}},e}({})},function(e,t){"use strict";e.exports=function(e){var t=codex.editor;return e.opened=!1,e.open=function(){t.toolbar.settings.opened&&t.toolbar.settings.close(),t.nodes.toolbox.classList.add("opened"),t.nodes.plusButton.classList.add("clicked"),t.toolbar.toolbox.opened=!0},e.close=function(){t.nodes.toolbox.classList.remove("opened"),t.nodes.plusButton.classList.remove("clicked"),t.toolbar.toolbox.opened=!1,t.toolbar.current=null},e.leaf=function(){var e=t.toolbar.current,n=Object.keys(t.tools),o=t.nodes.toolbarButtons,r=0,i=void 0,a=void 0,s=void 0;if(e)for(r=(n.indexOf(e)+1)%n.length,a=n[r];!t.tools[a].displayInToolbox;)r=(r+1)%n.length,a=n[r];else for(s in t.tools){if(t.tools[s].displayInToolbox)break;r++}i=n[r];for(var c in o)o[c].classList.remove("selected");o[i].classList.add("selected"),t.toolbar.current=i},e.toolClicked=function(e){var n,o,r,i=["image","link","list","instagram","twitter","embed"],a=t.tools[t.toolbar.current],s=t.content.currentNode,c=t.caret.inputIndex;n=a.render(),r={block:n,type:a.type,stretched:!1},s&&i.indexOf(s.dataset.tool)===-1&&""===s.textContent.trim()?t.content.switchBlock(s,n,a.type):(t.content.insertBlock(r),c++),o=a.appendCallback,o&&"function"==typeof o&&o.call(e),window.setTimeout(function(){t.caret.setToBlock(c)},10),t.content.workingNodeChanged(),t.toolbar.move()},e}({})},function(e,t){"use strict";e.exports=function(e){var t=codex.editor;e.globalKeydown=function(e){switch(e.keyCode){case t.core.keys.ENTER:o(e)}},e.redactorKeyDown=function(e){switch(e.keyCode){case t.core.keys.TAB:n(e);break;case t.core.keys.ENTER:i(e);break;case t.core.keys.ESC:a(e);break;default:c(e)}},e.globalKeyup=function(e){switch(e.keyCode){case t.core.keys.UP:case t.core.keys.LEFT:case t.core.keys.RIGHT:case t.core.keys.DOWN:s(e)}};var n=function(e){e.preventDefault();var n=t.content.currentNode.querySelectorAll("textarea, input"),o=!0,r=!t.content.currentNode.textContent.trim();Array.prototype.map.call(n,function(e){"textarea"!=e.type&&"text"!=e.type||(o=o&&!e.value.trim())});var i=r&&o;i&&(t.toolbar.opened||t.toolbar.open(),t.toolbar.opened&&!t.toolbar.toolbox.opened?t.toolbar.toolbox.open():t.toolbar.toolbox.leaf())},o=function(){t.content.editorAreaHightlighted&&(t.caret.inputIndex=-1,r())},r=function(){var e=t.settings.initialBlockPlugin;t.content.insertBlock({type:e,block:t.tools[e].render()},!0),t.toolbar.move(),t.toolbar.open()},i=function(e){"true"==e.target.contentEditable&&t.caret.saveCurrentInputIndex();var n=t.caret.getCurrentInputIndex()||0,o=t.content.currentNode,r=o.dataset.tool,i=t.toolbar.opened&&t.toolbar.current&&e.target==t.state.inputs[n],a=t.tools[r].enableLineBreaks,s=t.settings.initialBlockPlugin;if(i)return e.preventDefault(),t.toolbar.toolbox.toolClicked(e),t.toolbar.close(),e.stopPropagation(),void e.stopImmediatePropagation();if(e.shiftKey||a)return e.stopPropagation(),void e.stopImmediatePropagation();var c=window.getSelection(),l=c.anchorNode,d=t.caret.position.atTheEnd(),u=!1;if(e.shiftKey&&!a)return t.callback.enterPressedOnBlock(t.content.currentBlock,e),void e.preventDefault();if(u=l&&"true"!=l.parentNode.contentEditable,l.nodeType!=t.core.nodeTypes.TEXT||u||d){var p=t.content.isLastNode(l);p&&d&&(e.preventDefault(),e.stopPropagation(),e.stopImmediatePropagation(),t.core.log("ENTER clicked in last textNode. Create new BLOCK"),t.content.insertBlock({type:s,block:t.tools[s].render()},!0),t.toolbar.move(),t.toolbar.open(),t.toolbar.showPlusButton())}else e.preventDefault(),t.core.log("Splitting Text node..."),t.content.splitBlock(n),t.state.inputs[n+1].textContent.trim()||t.toolbar.showPlusButton();t.ui.saveInputs()},a=function(e){t.toolbar.close(),t.toolbar.toolbox.close(),e.preventDefault()},s=function(e){t.content.workingNodeChanged(),t.toolbar.close(),t.toolbar.move()},c=function(){t.toolbar.close(),t.toolbar.inline.actionsOpened||(t.toolbar.inline.close(),t.content.clearMark())};e.redactorClicked=function(e){l(),t.content.workingNodeChanged(e.target),t.ui.saveInputs();var n,o=t.toolbar.inline.getSelectionText();if(0===o.length&&t.toolbar.inline.close(),"true"==e.target.contentEditable&&t.caret.saveCurrentInputIndex(),null===t.content.currentNode){var r=t.state.inputs.length>0?t.state.inputs.length-1:0;if(t.state.inputs.length&&(n=t.content.getFirstLevelBlock(t.state.inputs[r])),t.state.inputs.length&&""===t.state.inputs[r].textContent&&n.dataset.tool==t.settings.initialBlockPlugin)t.caret.setToBlock(r);else{var i=t.settings.initialBlockPlugin;t.content.insertBlock({type:i,block:t.tools[i].render()}),1===t.state.inputs.length?t.caret.setToBlock(r):t.caret.setToNextBlock(r)}}else t.toolbar.settings.close(),t.toolbar.toolbox.close();t.toolbar.move(),t.toolbar.open();var a=!t.content.currentNode.textContent.trim(),s=t.content.currentNode.dataset.tool,c=s==t.settings.initialBlockPlugin;t.toolbar.hidePlusButton(),a||t.content.markBlock(),c&&a&&t.toolbar.showPlusButton()};var l=function(){var e=window.getSelection(),n=e.anchorNode,o=!1;if(0===e.rangeCount)t.content.editorAreaHightlighted=!0;else{for(t.core.isDomNode(n)||(n=n.parentNode),"true"==n.contentEditable&&(o=!0);"true"!=n.contentEditable&&(n=n.parentNode,"true"==n.contentEditable&&(o=!0),n!=document.body););t.content.editorAreaHightlighted=!o}};e.toolbarButtonClicked=function(e){var n=this;t.toolbar.current=n.dataset.type,t.toolbar.toolbox.toolClicked(e),t.toolbar.close()},e.plusButtonClicked=function(){t.nodes.toolbox.classList.contains("opened")?t.toolbar.toolbox.close():t.toolbar.toolbox.open()},e.blockKeydown=function(e){var n=e.target;switch(e.keyCode){case t.core.keys.DOWN:case t.core.keys.RIGHT:d(e);break;case t.core.keys.BACKSPACE:p(n,e);break;case t.core.keys.UP:case t.core.keys.LEFT:u(e)}};var d=function(e){var n,o=window.getSelection(),r=t.state.inputs,i=o.anchorNode;if(!i)return!1;for(;"true"!=i.contentEditable;)n=i.parentNode,i=n;for(var a=0;i!=r[a];)a++;if(!i.textContent)return void t.caret.setToNextBlock(a);var s,c,l=!1,d=!1;return s=i.childNodes[i.childNodes.length-1],c=t.core.isDomNode(s)?t.content.getDeepestTextNodeFromPosition(s,s.childNodes.length):s,l=o.anchorNode==c,d=c.length==o.anchorOffset,l&&d?void t.caret.setToNextBlock(a):(t.core.log("arrow [down|right] : caret does not reached the end"),!1)},u=function(e){var n,o=window.getSelection(),r=t.state.inputs,i=o.anchorNode;if(!i)return!1;if(0!==o.anchorOffset)return!1;for(;"true"!=i.contentEditable;)n=i.parentNode,i=n;for(var a=0;i!=r[a];)a++;var s,c,l=!1,d=!1;return i.textContent?(s=i.childNodes[0],c=t.core.isDomNode(s)?t.content.getDeepestTextNodeFromPosition(s,0):s,l=o.anchorNode==c,d=0===o.anchorOffset,void(l&&d&&t.caret.setToPreviousBlock(a))):void t.caret.setToPreviousBlock(a)},p=function(e,n){var o,r,i,a=t.caret.getCurrentInputIndex();if(f(n.target)){if(""!=n.target.value.trim())return;e.remove()}if(e.textContent.trim()){if(o=t.content.getRange(),r=o.endOffset-o.startOffset,!t.caret.position.atStart()||r||!t.state.inputs[a-1])return;t.content.mergeBlocks(a)}r||e.remove(),i=t.nodes.redactor.childNodes.length,0===i?(t.content.currentNode=null,t.ui.addInitialBlock(),t.ui.saveInputs(),window.setTimeout(function(){t.caret.setToPreviousBlock(1)},10)):0!==t.caret.inputIndex?t.caret.setToPreviousBlock(t.caret.inputIndex):t.caret.setToNextBlock(t.caret.inputIndex), +t.toolbar.move(),t.toolbar.opened||t.toolbar.open(),t.ui.saveInputs(),n.preventDefault()};e.blockPasteCallback=function(e){if(!f(e.target)){e.preventDefault();var n=t.content.getEditableParent(e.target);if(n){var o,r,i=e.clipboardData.getData("text/html"),a=e.clipboardData.getData("text/plain"),s=t.draw.node("DIV","",{});o=t.sanitizer.clean(i),r=t.content.wrapTextWithParagraphs(o,a),s.innerHTML=r;var c=t.settings.initialBlockPlugin,l=t.content.currentNode.firstChild.firstChild;if(1==s.childNodes.length)return void t.caret.insertNode(document.createTextNode(s.firstChild.innerHTML));s.childNodes.forEach(function(e,n){return 0==n&&""===l.innerHTML.trim()?void t.content.switchBlock(t.content.currentNode,t.tools[c].render({text:e.innerHTML}),c):(t.content.insertBlock({type:c,block:t.tools[c].render({text:e.innerHTML})}),void t.caret.inputIndex++)}),t.caret.setToPreviousBlock(t.caret.getCurrentInputIndex()+1)}}},e.showSettingsButtonClicked=function(e){var n=t.content.currentNode.dataset.tool;t.toolbar.settings.toggle(n),t.toolbar.toolbox.close(),t.toolbar.settings.hideRemoveActions()};var f=function(e){var t=["INPUT","TEXTAREA"];return t.indexOf(e.tagName)!=-1};return e}({})},function(e,t){"use strict";e.exports=function(e){return e.wrapper=function(){var e=document.createElement("div");return e.className+="codex-editor",e},e.redactor=function(){var e=document.createElement("div");return e.className+="ce-redactor",e},e.ceBlock=function(){var e=document.createElement("DIV");return e.className+="ce_block",e},e.toolbar=function(){var e=document.createElement("div");return e.className+="ce-toolbar",e},e.toolbarContent=function(){var e=document.createElement("DIV");return e.classList.add("ce-toolbar__content"),e},e.inlineToolbar=function(){var e=document.createElement("DIV");return e.className+="ce-toolbar-inline",e},e.inlineToolbarButtons=function(){var e=document.createElement("DIV");return e.className+="ce-toolbar-inline__buttons",e},e.inlineToolbarActions=function(){var e=document.createElement("DIV");return e.className+="ce-toolbar-inline__actions",e},e.inputForLink=function(){var e=document.createElement("INPUT");return e.type="input",e.className+="inputForLink",e.placeholder="Вставьте ссылку ...",e.setAttribute("form","defaultForm"),e.setAttribute("autofocus","autofocus"),e},e.blockButtons=function(){var e=document.createElement("div");return e.className+="ce-toolbar__actions",e},e.blockSettings=function(){var e=document.createElement("div");return e.className+="ce-settings",e},e.defaultSettings=function(){var e=document.createElement("div");return e.classList.add("ce-settings_default"),e},e.pluginsSettings=function(){var e=document.createElement("div");return e.classList.add("ce-settings_plugin"),e},e.plusButton=function(){var e=document.createElement("span");return e.className="ce-toolbar__plus",e},e.settingsButton=function(){var e=document.createElement("span");return e.className="ce-toolbar__settings-btn",e.innerHTML='',e},e.toolbox=function(){var e=document.createElement("div");return e.className="ce-toolbar__tools",e},e.toolbarButton=function(e,t){var n=document.createElement("li"),o=document.createElement("i"),r=document.createElement("span");return n.dataset.type=e,n.setAttribute("title",e),o.classList.add(t),r.classList.add("ce_toolbar_tools--title"),n.appendChild(o),n.appendChild(r),n},e.toolbarButtonInline=function(e,t){var n=document.createElement("BUTTON"),o=document.createElement("I");return n.type="button",n.dataset.type=e,o.classList.add(t),n.appendChild(o),n},e.block=function(e,t){var n=document.createElement(e);return n.innerHTML=t||"",n},e.node=function(e,t,n){var o=document.createElement(e);if(t&&(o.className=t),n)for(var r in n)o[r]=n[r];return o},e.unavailableBlock=function(){var e=document.createElement("DIV");return e.classList.add("cdx-unavailable-block"),e},e}({})},function(e,t){"use strict";e.exports=function(e){var t=codex.editor;return e.inputIndex=null,e.offset=null,e.focusedNodeIndex=null,e.set=function(n,o,r){r=r||e.offset||0,o=o||e.focusedNodeIndex||0;var i,a=n.childNodes;if(i=0===a.length?n:a[o],"true"!=n.contentEditable)return void n.focus();t.core.isDomNode(i)&&(i=t.content.getDeepestTextNodeFromPosition(i,i.childNodes.length));var s=document.createRange(),c=window.getSelection();window.setTimeout(function(){s.setStart(i,r),s.setEnd(i,r),c.removeAllRanges(),c.addRange(s),t.caret.saveCurrentInputIndex()},20)},e.saveCurrentInputIndex=function(){var n,o=window.getSelection(),r=t.state.inputs,i=o.anchorNode;if(i){for(;"true"!=i.contentEditable;)n=i.parentNode,i=n;for(var a=0;i!=r[a];)a++;e.inputIndex=a}},e.getCurrentInputIndex=function(){return e.inputIndex},e.setToNextBlock=function(e){var n=t.state.inputs,o=n[e+1];if(!o)return void t.core.log("We are reached the end");if(!o.childNodes.length){var r=document.createTextNode("");o.appendChild(r)}t.caret.inputIndex=e+1,t.caret.set(o,0,0),t.content.workingNodeChanged(o)},e.setToBlock=function(e){var n=t.state.inputs,o=n[e];if(o){if(!o.childNodes.length){var r=document.createTextNode("");o.appendChild(r)}t.caret.inputIndex=e,t.caret.set(o,0,0),t.content.workingNodeChanged(o)}},e.setToPreviousBlock=function(e){e=e||0;var n,o,r,i=t.state.inputs,a=i[e-1];return a?(n=t.content.getDeepestTextNodeFromPosition(a,a.childNodes.length),o=n.length,a.childNodes.length||(r=document.createTextNode(""),a.appendChild(r)),t.caret.inputIndex=e-1,t.caret.set(a,a.childNodes.length-1,o),void t.content.workingNodeChanged(i[e-1])):void t.core.log("We are reached first node")},e.position={atStart:function(){var e=window.getSelection(),n=e.anchorOffset,o=e.anchorNode,r=t.content.getFirstLevelBlock(o),i=r.childNodes[0];t.core.isDomNode(o)||(o=o.parentNode);var a=o===i.childNodes[0],s=0===n;return a&&s},atTheEnd:function(){var e=window.getSelection(),t=e.anchorOffset,n=e.anchorNode;return!n||!n.length||t===n.length}},e.insertNode=function(e){var t,n;t=window.getSelection(),n=t.getRangeAt(0),n.deleteContents(),n.insertNode(e),n.setStartAfter(e),n.collapse(!0),t.removeAllRanges(),t.addRange(n)},e}({})},function(e,t){"use strict";e.exports=function(e){var t=codex.editor,n=[],o=function(e){n.push(e);for(var t=0;t5;)"confirm"!=n[t].type&&"prompt"!=n[t].type?(n[t].close(),n.splice(t,1)):t++};return e.createHolder=function(){var e=t.draw.node("DIV","cdx-notifications-block");return t.nodes.notifications=document.body.appendChild(e),e},e.errorThrown=function(e,n){t.notifications.notification({message:"This action is not available currently",type:n.type})},e.notification=function(e){function n(e){if(!e||!e.message)return void t.core.log("Can't create notification. Message is missed");e.type=e.type||"alert",e.time=1e3*e.time||1e4;var n=t.draw.node("DIV","cdx-notification"),o=t.draw.node("DIV","cdx-notification__message"),r=t.draw.node("INPUT","cdx-notification__input"),f=t.draw.node("SPAN","cdx-notification__ok-btn"),g=t.draw.node("SPAN","cdx-notification__cancel-btn");o.textContent=e.message,f.textContent=e.okMsg||"ОК",g.textContent=e.cancelMsg||"Отмена",t.listeners.add(f,"click",u),t.listeners.add(g,"click",p),n.appendChild(o),"prompt"==e.type&&n.appendChild(r),n.appendChild(f),"prompt"!=e.type&&"confirm"!=e.type||n.appendChild(g),n.classList.add("cdx-notification-"+e.type),n.dataset.type=e.type,a=n,c=e.type,l=e.confirm,s=e.cancel,d=r,"prompt"!=e.type&&"confirm"!=e.type&&window.setTimeout(i,e.time)}function r(){t.nodes.notifications.appendChild(a),d.focus(),t.nodes.notifications.classList.add("cdx-notification__notification-appending"),window.setTimeout(function(){t.nodes.notifications.classList.remove("cdx-notification__notification-appending")},100),o({type:c,close:i})}function i(){a.remove()}var a=null,s=null,c=null,l=null,d=null,u=function(){if(i(),"function"==typeof l)return"prompt"==c?void l(d.value):void l()},p=function(){i(),"function"==typeof s&&s()};return e&&(n(e),r()),{create:n,send:r,close:i}},e.clear=function(){t.nodes.notifications.innerHTML="",n=[]},e}({})},function(e,t){"use strict";e.exports=function(e){var t=codex.editor;return e.insertPastedContent=function(e,n){t.content.insertBlock({type:e.type,block:e.render({text:n.innerHTML})})},e.isFirstLevelBlock=function(e){return e.nodeType==t.core.nodeTypes.TAG&&e.classList.contains(t.ui.className.BLOCK_CLASSNAME)},e}({})},function(e,t,n){"use strict";e.exports=function(e){var t=n(18),o=codex.editor;e.prepare=function(){o.settings.sanitizer&&!o.core.isEmpty(o.settings.sanitizer)&&(r.CUSTOM=o.settings.sanitizer)};var r={CUSTOM:null,BASIC:{tags:{p:{},a:{href:!0,target:"_blank",rel:"nofollow"},i:{},b:{},strong:{},em:{},span:{}}}};e.Config=r;var i=function(e){var n=e||r.CUSTOM||r.BASIC;return new t(n)};return e.clean=function(e,t){var n=i(t);return n.clean(e)},e}({})},function(e,t,n){var o,r;!function(i,a){o=a,r="function"==typeof o?o.call(t,n,t,e):o,!(void 0!==r&&(e.exports=r))}(this,function(){function e(e){var t=e.tags,n=Object.keys(t),o=n.map(function(e){return typeof t[e]}).every(function(e){return"object"===e||"boolean"===e||"function"===e});if(!o)throw new Error("The configuration was invalid");this.config=e}function t(e){return s.indexOf(e.nodeName)!==-1}function n(e){return c.indexOf(e.nodeName)!==-1}function o(e){return document.createTreeWalker(e,NodeFilter.SHOW_TEXT|NodeFilter.SHOW_ELEMENT|NodeFilter.SHOW_COMMENT,null,!1)}function r(e,t,n){return"function"==typeof e.tags[t]?e.tags[t](n):e.tags[t]}function i(e,t){return"undefined"==typeof t||"boolean"==typeof t&&!t}function a(e,t,n){var o=e.name.toLowerCase();return t!==!0&&("function"==typeof t[o]?!t[o](e.value,n):"undefined"==typeof t[o]||(t[o]===!1||"string"==typeof t[o]&&t[o]!==e.value))}var s=["P","LI","TD","TH","DIV","H1","H2","H3","H4","H5","H6","PRE"],c=["A","B","STRONG","I","EM","SUB","SUP","U","STRIKE"];return e.prototype.clean=function(e){var t=document.createElement("div");return t.innerHTML=e,this._sanitize(t),t.innerHTML},e.prototype._sanitize=function(e){var s=o(e),c=s.firstChild();if(c)do if(!c._sanitized)if(c.nodeType!==Node.TEXT_NODE){if(c.nodeType===Node.COMMENT_NODE){e.removeChild(c),this._sanitize(e);break}var l,d=n(c);d&&(l=Array.prototype.some.call(c.childNodes,t));var u=!!e.parentNode,p=t(e)&&t(c)&&u,f=c.nodeName.toLowerCase(),g=r(this.config,f,c),h=d&&l;if(h||i(c,g)||!this.config.keepNestedBlockElements&&p){if("SCRIPT"!==c.nodeName&&"STYLE"!==c.nodeName)for(;c.childNodes.length>0;)e.insertBefore(c.childNodes[0],c);e.removeChild(c),this._sanitize(e);break}for(var v=0;v0&&t.splice(s,1)}},e.removeAll=function(){t.map(function(t){e.remove(t.element,t.type,t.handler)})},e.get=function(t,n,o){return e.search.all(t,n,o)},e}({})},function(e,t){"use strict";var n="function"==typeof Symbol&&"symbol"==typeof Symbol.iterator?function(e){return typeof e}:function(e){return e&&"function"==typeof Symbol&&e.constructor===Symbol&&e!==Symbol.prototype?"symbol":typeof e};e.exports=function(e){var t=codex.editor;return e.removeNodes=function(){t.nodes.wrapper.remove(),t.nodes.notifications.remove()},e.destroyPlugins=function(){for(var e in t.tools)"function"==typeof t.tools[e].destroy&&t.tools[e].destroy()},e.destroyScripts=function(){for(var e=document.getElementsByTagName("SCRIPT"),n=0;n nodeList\n\t */\n\t renderer.appendBlocks = function (data) {\n\t\n\t var blocks = data.items;\n\t\n\t /**\n\t * Sequence of one-by-one blocks appending\n\t * Uses to save blocks order after async-handler\n\t */\n\t var nodeSequence = Promise.resolve();\n\t\n\t for (var index = 0; index < blocks.length; index++) {\n\t\n\t /** Add node to sequence at specified index */\n\t editor.renderer.appendNodeAtIndex(nodeSequence, blocks, index);\n\t }\n\t };\n\t\n\t /**\n\t * Append node at specified index\n\t */\n\t renderer.appendNodeAtIndex = function (nodeSequence, blocks, index) {\n\t\n\t /** We need to append node to sequence */\n\t nodeSequence\n\t\n\t /** first, get node async-aware */\n\t .then(function () {\n\t\n\t return editor.renderer.getNodeAsync(blocks, index);\n\t })\n\t\n\t /**\n\t * second, compose editor-block from JSON object\n\t */\n\t .then(editor.renderer.createBlockFromData)\n\t\n\t /**\n\t * now insert block to redactor\n\t */\n\t .then(function (blockData) {\n\t\n\t /**\n\t * blockData has 'block', 'type' and 'stretched' information\n\t */\n\t editor.content.insertBlock(blockData);\n\t\n\t /** Pass created block to next step */\n\t return blockData.block;\n\t })\n\t\n\t /** Log if something wrong with node */\n\t .catch(function (error) {\n\t\n\t editor.core.log('Node skipped while parsing because %o', 'error', error);\n\t });\n\t };\n\t\n\t /**\n\t * Asynchronously returns block data from blocksList by index\n\t * @return Promise to node\n\t */\n\t renderer.getNodeAsync = function (blocksList, index) {\n\t\n\t return Promise.resolve().then(function () {\n\t\n\t return {\n\t tool: blocksList[index],\n\t position: index\n\t };\n\t });\n\t };\n\t\n\t /**\n\t * Creates editor block by JSON-data\n\t *\n\t * @uses render method of each plugin\n\t *\n\t * @param {Object} toolData.tool\n\t * { header : {\n\t * text: '',\n\t * type: 'H3', ...\n\t * }\n\t * }\n\t * @param {Number} toolData.position - index in input-blocks array\n\t * @return {Object} with type and Element\n\t */\n\t renderer.createBlockFromData = function (toolData) {\n\t\n\t /** New parser */\n\t var block,\n\t tool = toolData.tool,\n\t pluginName = tool.type;\n\t\n\t /** Get first key of object that stores plugin name */\n\t // for (var pluginName in blockData) break;\n\t\n\t /** Check for plugin existance */\n\t if (!editor.tools[pluginName]) {\n\t\n\t throw Error('Plugin \\xAB' + pluginName + '\\xBB not found');\n\t }\n\t\n\t /** Check for plugin having render method */\n\t if (typeof editor.tools[pluginName].render != 'function') {\n\t\n\t throw Error('Plugin \\xAB' + pluginName + '\\xBB must have \\xABrender\\xBB method');\n\t }\n\t\n\t if (editor.tools[pluginName].available === false) {\n\t\n\t block = editor.draw.unavailableBlock();\n\t\n\t block.innerHTML = editor.tools[pluginName].loadingMessage;\n\t\n\t /**\n\t * Saver will extract data from initial block data by position in array\n\t */\n\t block.dataset.inputPosition = toolData.position;\n\t } else {\n\t\n\t /** New Parser */\n\t block = editor.tools[pluginName].render(tool.data);\n\t }\n\t\n\t /** is first-level block stretched */\n\t var stretched = editor.tools[pluginName].isStretched || false;\n\t\n\t /** Retrun type and block */\n\t return {\n\t type: pluginName,\n\t block: block,\n\t stretched: stretched\n\t };\n\t };\n\t\n\t return renderer;\n\t}({});\n\n/***/ },\n/* 6 */\n/***/ function(module, exports) {\n\n\t'use strict';\n\t\n\t/**\n\t * Codex Editor Saver\n\t *\n\t * @author Codex Team\n\t * @version 1.0.2\n\t */\n\t\n\tmodule.exports = function (saver) {\n\t\n\t var editor = codex.editor;\n\t\n\t /**\n\t * Saves blocks\n\t * @private\n\t */\n\t saver.saveBlocks = function () {\n\t\n\t /** Save html content of redactor to memory */\n\t editor.state.html = editor.nodes.redactor.innerHTML;\n\t\n\t /** Empty jsonOutput state */\n\t editor.state.jsonOutput = [];\n\t\n\t Promise.resolve().then(function () {\n\t\n\t return editor.nodes.redactor.childNodes;\n\t })\n\t /** Making a sequence from separate blocks */\n\t .then(editor.saver.makeQueue).then(function () {\n\t // editor.nodes.textarea.innerHTML = editor.state.html;\n\t }).catch(function (error) {\n\t\n\t editor.core.log(error);\n\t });\n\t };\n\t\n\t saver.makeQueue = function (blocks) {\n\t\n\t var queue = Promise.resolve();\n\t\n\t for (var index = 0; index < blocks.length; index++) {\n\t\n\t /** Add node to sequence at specified index */\n\t editor.saver.getBlockData(queue, blocks, index);\n\t }\n\t };\n\t\n\t /** Gets every block and makes From Data */\n\t saver.getBlockData = function (queue, blocks, index) {\n\t\n\t queue.then(function () {\n\t\n\t return editor.saver.getNodeAsync(blocks, index);\n\t }).then(editor.saver.makeFormDataFromBlocks);\n\t };\n\t\n\t /**\n\t * Asynchronously returns block data from blocksList by index\n\t * @return Promise to node\n\t */\n\t saver.getNodeAsync = function (blocksList, index) {\n\t\n\t return Promise.resolve().then(function () {\n\t\n\t return blocksList[index];\n\t });\n\t };\n\t\n\t saver.makeFormDataFromBlocks = function (block) {\n\t\n\t var pluginName = block.dataset.tool;\n\t\n\t /** Check for plugin existance */\n\t if (!editor.tools[pluginName]) {\n\t\n\t throw Error('Plugin \\xAB' + pluginName + '\\xBB not found');\n\t }\n\t\n\t /** Check for plugin having render method */\n\t if (typeof editor.tools[pluginName].save != 'function') {\n\t\n\t throw Error('Plugin \\xAB' + pluginName + '\\xBB must have save method');\n\t }\n\t\n\t /** Result saver */\n\t var blockContent = block.childNodes[0],\n\t pluginsContent = blockContent.childNodes[0],\n\t savedData,\n\t position,\n\t output;\n\t\n\t /** If plugin wasn't available then return data from cache */\n\t if (editor.tools[pluginName].available === false) {\n\t\n\t position = pluginsContent.dataset.inputPosition;\n\t savedData = codex.editor.state.blocks.items[position].data;\n\t } else {\n\t\n\t savedData = editor.tools[pluginName].save(pluginsContent);\n\t\n\t if (editor.tools[pluginName].validate) {\n\t\n\t var result = editor.tools[pluginName].validate(savedData);\n\t\n\t /**\n\t * Do not allow invalid data\n\t */\n\t if (!result) return;\n\t }\n\t }\n\t\n\t output = {\n\t type: pluginName,\n\t data: savedData\n\t };\n\t\n\t editor.state.jsonOutput.push(output);\n\t };\n\t\n\t return saver;\n\t}({});\n\n/***/ },\n/* 7 */\n/***/ function(module, exports) {\n\n\t'use strict';\n\t\n\t/**\n\t * Codex Editor Content Module\n\t * Works with DOM\n\t *\n\t * @module Codex Editor content module\n\t *\n\t * @author Codex Team\n\t * @version 1.3.13\n\t *\n\t * @description Module works with Elements that have been appended to the main DOM\n\t */\n\t\n\tmodule.exports = function (content) {\n\t\n\t var editor = codex.editor;\n\t\n\t /**\n\t * Links to current active block\n\t * @type {null | Element}\n\t */\n\t content.currentNode = null;\n\t\n\t /**\n\t * clicked in redactor area\n\t * @type {null | Boolean}\n\t */\n\t content.editorAreaHightlighted = null;\n\t\n\t /**\n\t * @deprecated\n\t * Synchronizes redactor with original textarea\n\t */\n\t content.sync = function () {\n\t\n\t editor.core.log('syncing...');\n\t\n\t /**\n\t * Save redactor content to editor.state\n\t */\n\t editor.state.html = editor.nodes.redactor.innerHTML;\n\t };\n\t\n\t /**\n\t * Appends background to the block\n\t *\n\t * @description add CSS class to highlight visually first-level block area\n\t */\n\t content.markBlock = function () {\n\t\n\t editor.content.currentNode.classList.add(editor.ui.className.BLOCK_HIGHLIGHTED);\n\t };\n\t\n\t /**\n\t * Clear background\n\t *\n\t * @description clears styles that highlights block\n\t */\n\t content.clearMark = function () {\n\t\n\t if (editor.content.currentNode) {\n\t\n\t editor.content.currentNode.classList.remove(editor.ui.className.BLOCK_HIGHLIGHTED);\n\t }\n\t };\n\t\n\t /**\n\t * Finds first-level block\n\t *\n\t * @param {Element} node - selected or clicked in redactors area node\n\t * @protected\n\t *\n\t * @description looks for first-level block.\n\t * gets parent while node is not first-level\n\t */\n\t content.getFirstLevelBlock = function (node) {\n\t\n\t if (!editor.core.isDomNode(node)) {\n\t\n\t node = node.parentNode;\n\t }\n\t\n\t if (node === editor.nodes.redactor || node === document.body) {\n\t\n\t return null;\n\t } else {\n\t\n\t while (!node.classList.contains(editor.ui.className.BLOCK_CLASSNAME)) {\n\t\n\t node = node.parentNode;\n\t }\n\t\n\t return node;\n\t }\n\t };\n\t\n\t /**\n\t * Trigger this event when working node changed\n\t * @param {Element} targetNode - first-level of this node will be current\n\t * @protected\n\t *\n\t * @description If targetNode is first-level then we set it as current else we look for parents to find first-level\n\t */\n\t content.workingNodeChanged = function (targetNode) {\n\t\n\t /** Clear background from previous marked block before we change */\n\t editor.content.clearMark();\n\t\n\t if (!targetNode) {\n\t\n\t return;\n\t }\n\t\n\t this.currentNode = this.getFirstLevelBlock(targetNode);\n\t };\n\t\n\t /**\n\t * Replaces one redactor block with another\n\t * @protected\n\t * @param {Element} targetBlock - block to replace. Mostly currentNode.\n\t * @param {Element} newBlock\n\t * @param {string} newBlockType - type of new block; we need to store it to data-attribute\n\t *\n\t * [!] Function does not saves old block content.\n\t * You can get it manually and pass with newBlock.innerHTML\n\t */\n\t content.replaceBlock = function (targetBlock, newBlock) {\n\t\n\t if (!targetBlock || !newBlock) {\n\t\n\t editor.core.log('replaceBlock: missed params');\n\t return;\n\t }\n\t\n\t /** If target-block is not a frist-level block, then we iterate parents to find it */\n\t while (!targetBlock.classList.contains(editor.ui.className.BLOCK_CLASSNAME)) {\n\t\n\t targetBlock = targetBlock.parentNode;\n\t }\n\t\n\t /** Replacing */\n\t editor.nodes.redactor.replaceChild(newBlock, targetBlock);\n\t\n\t /**\n\t * Set new node as current\n\t */\n\t editor.content.workingNodeChanged(newBlock);\n\t\n\t /**\n\t * Add block handlers\n\t */\n\t editor.ui.addBlockHandlers(newBlock);\n\t\n\t /**\n\t * Save changes\n\t */\n\t editor.ui.saveInputs();\n\t };\n\t\n\t /**\n\t * @protected\n\t *\n\t * Inserts new block to redactor\n\t * Wrapps block into a DIV with BLOCK_CLASSNAME class\n\t *\n\t * @param blockData {object}\n\t * @param blockData.block {Element} element with block content\n\t * @param blockData.type {string} block plugin\n\t * @param needPlaceCaret {bool} pass true to set caret in new block\n\t *\n\t */\n\t content.insertBlock = function (blockData, needPlaceCaret) {\n\t\n\t var workingBlock = editor.content.currentNode,\n\t newBlockContent = blockData.block,\n\t blockType = blockData.type,\n\t isStretched = blockData.stretched;\n\t\n\t var newBlock = composeNewBlock_(newBlockContent, blockType, isStretched);\n\t\n\t if (workingBlock) {\n\t\n\t editor.core.insertAfter(workingBlock, newBlock);\n\t } else {\n\t\n\t /**\n\t * If redactor is empty, append as first child\n\t */\n\t editor.nodes.redactor.appendChild(newBlock);\n\t }\n\t\n\t /**\n\t * Block handler\n\t */\n\t editor.ui.addBlockHandlers(newBlock);\n\t\n\t /**\n\t * Set new node as current\n\t */\n\t editor.content.workingNodeChanged(newBlock);\n\t\n\t /**\n\t * Save changes\n\t */\n\t editor.ui.saveInputs();\n\t\n\t if (needPlaceCaret) {\n\t\n\t /**\n\t * If we don't know input index then we set default value -1\n\t */\n\t var currentInputIndex = editor.caret.getCurrentInputIndex() || -1;\n\t\n\t if (currentInputIndex == -1) {\n\t\n\t var editableElement = newBlock.querySelector('[contenteditable]'),\n\t emptyText = document.createTextNode('');\n\t\n\t editableElement.appendChild(emptyText);\n\t editor.caret.set(editableElement, 0, 0);\n\t\n\t editor.toolbar.move();\n\t editor.toolbar.showPlusButton();\n\t } else {\n\t\n\t if (currentInputIndex === editor.state.inputs.length - 1) return;\n\t\n\t /** Timeout for browsers execution */\n\t window.setTimeout(function () {\n\t\n\t /** Setting to the new input */\n\t editor.caret.setToNextBlock(currentInputIndex);\n\t editor.toolbar.move();\n\t editor.toolbar.open();\n\t }, 10);\n\t }\n\t }\n\t\n\t /**\n\t * Block is inserted, wait for new click that defined focusing on editors area\n\t * @type {boolean}\n\t */\n\t content.editorAreaHightlighted = false;\n\t };\n\t\n\t /**\n\t * Replaces blocks with saving content\n\t * @protected\n\t * @param {Element} noteToReplace\n\t * @param {Element} newNode\n\t * @param {Element} blockType\n\t */\n\t content.switchBlock = function (blockToReplace, newBlock, tool) {\n\t\n\t tool = tool || editor.content.currentNode.dataset.tool;\n\t var newBlockComposed = composeNewBlock_(newBlock, tool);\n\t\n\t /** Replacing */\n\t editor.content.replaceBlock(blockToReplace, newBlockComposed);\n\t\n\t /** Save new Inputs when block is changed */\n\t editor.ui.saveInputs();\n\t };\n\t\n\t /**\n\t * Iterates between child noted and looking for #text node on deepest level\n\t * @protected\n\t *\n\t * @param {Element} block - node where find\n\t * @param {int} postiton - starting postion\n\t * Example: childNodex.length to find from the end\n\t * or 0 to find from the start\n\t * @return {Text} block\n\t * @uses DFS\n\t */\n\t content.getDeepestTextNodeFromPosition = function (block, position) {\n\t\n\t /**\n\t * Clear Block from empty and useless spaces with trim.\n\t * Such nodes we should remove\n\t */\n\t var blockChilds = block.childNodes,\n\t index,\n\t node,\n\t text;\n\t\n\t for (index = 0; index < blockChilds.length; index++) {\n\t\n\t node = blockChilds[index];\n\t\n\t if (node.nodeType == editor.core.nodeTypes.TEXT) {\n\t\n\t text = node.textContent.trim();\n\t\n\t /** Text is empty. We should remove this child from node before we start DFS\n\t * decrease the quantity of childs.\n\t */\n\t if (text === '') {\n\t\n\t block.removeChild(node);\n\t position--;\n\t }\n\t }\n\t }\n\t\n\t if (block.childNodes.length === 0) {\n\t\n\t return document.createTextNode('');\n\t }\n\t\n\t /** Setting default position when we deleted all empty nodes */\n\t if (position < 0) position = 1;\n\t\n\t var lookingFromStart = false;\n\t\n\t /** For looking from START */\n\t if (position === 0) {\n\t\n\t lookingFromStart = true;\n\t position = 1;\n\t }\n\t\n\t while (position) {\n\t\n\t /** initial verticle of node. */\n\t if (lookingFromStart) {\n\t\n\t block = block.childNodes[0];\n\t } else {\n\t\n\t block = block.childNodes[position - 1];\n\t }\n\t\n\t if (block.nodeType == editor.core.nodeTypes.TAG) {\n\t\n\t position = block.childNodes.length;\n\t } else if (block.nodeType == editor.core.nodeTypes.TEXT) {\n\t\n\t position = 0;\n\t }\n\t }\n\t\n\t return block;\n\t };\n\t\n\t /**\n\t * @private\n\t * @param {Element} block - current plugins render\n\t * @param {String} tool - plugins name\n\t * @param {Boolean} isStretched - make stretched block or not\n\t *\n\t * @description adds necessary information to wrap new created block by first-level holder\n\t */\n\t var composeNewBlock_ = function composeNewBlock_(block, tool, isStretched) {\n\t\n\t var newBlock = editor.draw.node('DIV', editor.ui.className.BLOCK_CLASSNAME, {}),\n\t blockContent = editor.draw.node('DIV', editor.ui.className.BLOCK_CONTENT, {});\n\t\n\t blockContent.appendChild(block);\n\t newBlock.appendChild(blockContent);\n\t\n\t if (isStretched) {\n\t\n\t blockContent.classList.add(editor.ui.className.BLOCK_STRETCHED);\n\t }\n\t\n\t newBlock.dataset.tool = tool;\n\t return newBlock;\n\t };\n\t\n\t /**\n\t * Returns Range object of current selection\n\t * @protected\n\t */\n\t content.getRange = function () {\n\t\n\t var selection = window.getSelection().getRangeAt(0);\n\t\n\t return selection;\n\t };\n\t\n\t /**\n\t * Divides block in two blocks (after and before caret)\n\t *\n\t * @protected\n\t * @param {int} inputIndex - target input index\n\t *\n\t * @description splits current input content to the separate blocks\n\t * When enter is pressed among the words, that text will be splited.\n\t */\n\t content.splitBlock = function (inputIndex) {\n\t\n\t var selection = window.getSelection(),\n\t anchorNode = selection.anchorNode,\n\t anchorNodeText = anchorNode.textContent,\n\t caretOffset = selection.anchorOffset,\n\t textBeforeCaret,\n\t textNodeBeforeCaret,\n\t textAfterCaret,\n\t textNodeAfterCaret;\n\t\n\t var currentBlock = editor.content.currentNode.querySelector('[contentEditable]');\n\t\n\t textBeforeCaret = anchorNodeText.substring(0, caretOffset);\n\t textAfterCaret = anchorNodeText.substring(caretOffset);\n\t\n\t textNodeBeforeCaret = document.createTextNode(textBeforeCaret);\n\t\n\t if (textAfterCaret) {\n\t\n\t textNodeAfterCaret = document.createTextNode(textAfterCaret);\n\t }\n\t\n\t var previousChilds = [],\n\t nextChilds = [],\n\t reachedCurrent = false;\n\t\n\t if (textNodeAfterCaret) {\n\t\n\t nextChilds.push(textNodeAfterCaret);\n\t }\n\t\n\t for (var i = 0, child; !!(child = currentBlock.childNodes[i]); i++) {\n\t\n\t if (child != anchorNode) {\n\t\n\t if (!reachedCurrent) {\n\t\n\t previousChilds.push(child);\n\t } else {\n\t\n\t nextChilds.push(child);\n\t }\n\t } else {\n\t\n\t reachedCurrent = true;\n\t }\n\t }\n\t\n\t /** Clear current input */\n\t editor.state.inputs[inputIndex].innerHTML = '';\n\t\n\t /**\n\t * Append all childs founded before anchorNode\n\t */\n\t var previousChildsLength = previousChilds.length;\n\t\n\t for (i = 0; i < previousChildsLength; i++) {\n\t\n\t editor.state.inputs[inputIndex].appendChild(previousChilds[i]);\n\t }\n\t\n\t editor.state.inputs[inputIndex].appendChild(textNodeBeforeCaret);\n\t\n\t /**\n\t * Append text node which is after caret\n\t */\n\t var nextChildsLength = nextChilds.length,\n\t newNode = document.createElement('div');\n\t\n\t for (i = 0; i < nextChildsLength; i++) {\n\t\n\t newNode.appendChild(nextChilds[i]);\n\t }\n\t\n\t newNode = newNode.innerHTML;\n\t\n\t /** This type of block creates when enter is pressed */\n\t var NEW_BLOCK_TYPE = editor.settings.initialBlockPlugin;\n\t\n\t /**\n\t * Make new paragraph with text after caret\n\t */\n\t editor.content.insertBlock({\n\t type: NEW_BLOCK_TYPE,\n\t block: editor.tools[NEW_BLOCK_TYPE].render({\n\t text: newNode\n\t })\n\t }, true);\n\t };\n\t\n\t /**\n\t * Merges two blocks — current and target\n\t * If target index is not exist, then previous will be as target\n\t *\n\t * @protected\n\t * @param {int} currentInputIndex\n\t * @param {int} targetInputIndex\n\t *\n\t * @description gets two inputs indexes and merges into one\n\t */\n\t content.mergeBlocks = function (currentInputIndex, targetInputIndex) {\n\t\n\t /** If current input index is zero, then prevent method execution */\n\t if (currentInputIndex === 0) {\n\t\n\t return;\n\t }\n\t\n\t var targetInput,\n\t currentInputContent = editor.state.inputs[currentInputIndex].innerHTML;\n\t\n\t if (!targetInputIndex) {\n\t\n\t targetInput = editor.state.inputs[currentInputIndex - 1];\n\t } else {\n\t\n\t targetInput = editor.state.inputs[targetInputIndex];\n\t }\n\t\n\t targetInput.innerHTML += currentInputContent;\n\t };\n\t\n\t /**\n\t * Iterates all right siblings and parents, which has right siblings\n\t * while it does not reached the first-level block\n\t *\n\t * @param {Element} node\n\t * @return {boolean}\n\t */\n\t content.isLastNode = function (node) {\n\t\n\t // console.log('погнали перебор родителей');\n\t\n\t var allChecked = false;\n\t\n\t while (!allChecked) {\n\t\n\t // console.log('Смотрим на %o', node);\n\t // console.log('Проверим, пустые ли соседи справа');\n\t\n\t if (!allSiblingsEmpty_(node)) {\n\t\n\t // console.log('Есть непустые соседи. Узел не последний. Выходим.');\n\t return false;\n\t }\n\t\n\t node = node.parentNode;\n\t\n\t /**\n\t * Проверяем родителей до тех пор, пока не найдем блок первого уровня\n\t */\n\t if (node.classList.contains(editor.ui.className.BLOCK_CONTENT)) {\n\t\n\t allChecked = true;\n\t }\n\t }\n\t\n\t return true;\n\t };\n\t\n\t /**\n\t * Checks if all element right siblings is empty\n\t * @param node\n\t */\n\t var allSiblingsEmpty_ = function allSiblingsEmpty_(node) {\n\t\n\t /**\n\t * Нужно убедиться, что после пустого соседа ничего нет\n\t */\n\t var sibling = node.nextSibling;\n\t\n\t while (sibling) {\n\t\n\t if (sibling.textContent.length) {\n\t\n\t return false;\n\t }\n\t\n\t sibling = sibling.nextSibling;\n\t }\n\t\n\t return true;\n\t };\n\t\n\t /**\n\t * @public\n\t *\n\t * @param [String] htmlString - html content as string\n\t * @return {string} - html content as string\n\t */\n\t content.wrapTextWithParagraphs = function (htmlString) {\n\t\n\t var wrapper = document.createElement('DIV'),\n\t newWrapper = document.createElement('DIV'),\n\t i,\n\t paragraph,\n\t firstLevelBlocks = ['DIV', 'P'],\n\t blockTyped,\n\t node;\n\t\n\t /**\n\t * Make HTML Element to Wrap Text\n\t * It allows us to work with input data as HTML content\n\t */\n\t wrapper.innerHTML = htmlString;\n\t paragraph = document.createElement('P');\n\t\n\t for (i = 0; i < wrapper.childNodes.length; i++) {\n\t\n\t node = wrapper.childNodes[i];\n\t\n\t blockTyped = firstLevelBlocks.indexOf(node.tagName) != -1;\n\t\n\t /**\n\t * If node is first-levet\n\t * we add this node to our new wrapper\n\t */\n\t if (blockTyped) {\n\t\n\t /**\n\t * If we had splitted inline nodes to paragraph before\n\t */\n\t if (paragraph.childNodes.length) {\n\t\n\t newWrapper.appendChild(paragraph.cloneNode(true));\n\t\n\t /** empty paragraph */\n\t paragraph = null;\n\t paragraph = document.createElement('P');\n\t }\n\t\n\t newWrapper.appendChild(node.cloneNode(true));\n\t } else {\n\t\n\t /** Collect all inline nodes to one as paragraph */\n\t paragraph.appendChild(node.cloneNode(true));\n\t\n\t /** if node is last we should append this node to paragraph and paragraph to new wrapper */\n\t if (i == wrapper.childNodes.length - 1) {\n\t\n\t newWrapper.appendChild(paragraph.cloneNode(true));\n\t }\n\t }\n\t }\n\t\n\t return newWrapper.innerHTML;\n\t };\n\t\n\t /**\n\t * Finds closest Contenteditable parent from Element\n\t * @param {Element} node element looking from\n\t * @return {Element} node contenteditable\n\t */\n\t content.getEditableParent = function (node) {\n\t\n\t while (node && node.contentEditable != 'true') {\n\t\n\t node = node.parentNode;\n\t }\n\t\n\t return node;\n\t };\n\t\n\t return content;\n\t}({});\n\n/***/ },\n/* 8 */\n/***/ function(module, exports, __webpack_require__) {\n\n\t'use strict';\n\t\n\t/**\n\t * Codex Editor toolbar module\n\t *\n\t * Contains:\n\t * - Inline toolbox\n\t * - Toolbox within plus button\n\t * - Settings section\n\t *\n\t * @author Codex Team\n\t * @version 1.0\n\t */\n\t\n\tmodule.exports = function (toolbar) {\n\t\n\t var editor = codex.editor;\n\t\n\t toolbar.settings = __webpack_require__(9);\n\t toolbar.inline = __webpack_require__(10);\n\t toolbar.toolbox = __webpack_require__(11);\n\t\n\t /**\n\t * Margin between focused node and toolbar\n\t */\n\t toolbar.defaultToolbarHeight = 49;\n\t\n\t toolbar.defaultOffset = 34;\n\t\n\t toolbar.opened = false;\n\t\n\t toolbar.current = null;\n\t\n\t /**\n\t * @protected\n\t */\n\t toolbar.open = function () {\n\t\n\t if (editor.hideToolbar) {\n\t\n\t return;\n\t }\n\t\n\t editor.nodes.toolbar.classList.add('opened');\n\t this.opened = true;\n\t };\n\t\n\t /**\n\t * @protected\n\t */\n\t toolbar.close = function () {\n\t\n\t editor.nodes.toolbar.classList.remove('opened');\n\t\n\t toolbar.opened = false;\n\t toolbar.current = null;\n\t\n\t for (var button in editor.nodes.toolbarButtons) {\n\t\n\t editor.nodes.toolbarButtons[button].classList.remove('selected');\n\t }\n\t\n\t /** Close toolbox when toolbar is not displayed */\n\t editor.toolbar.toolbox.close();\n\t editor.toolbar.settings.close();\n\t };\n\t\n\t toolbar.toggle = function () {\n\t\n\t if (!this.opened) {\n\t\n\t this.open();\n\t } else {\n\t\n\t this.close();\n\t }\n\t };\n\t\n\t toolbar.hidePlusButton = function () {\n\t\n\t editor.nodes.plusButton.classList.add('hide');\n\t };\n\t\n\t toolbar.showPlusButton = function () {\n\t\n\t editor.nodes.plusButton.classList.remove('hide');\n\t };\n\t\n\t /**\n\t * Moving toolbar to the specified node\n\t */\n\t toolbar.move = function () {\n\t\n\t /** Close Toolbox when we move toolbar */\n\t editor.toolbar.toolbox.close();\n\t\n\t if (!editor.content.currentNode) {\n\t\n\t return;\n\t }\n\t\n\t var newYCoordinate = editor.content.currentNode.offsetTop - editor.toolbar.defaultToolbarHeight / 2 + editor.toolbar.defaultOffset;\n\t\n\t editor.nodes.toolbar.style.transform = 'translate3D(0, ' + Math.floor(newYCoordinate) + 'px, 0)';\n\t\n\t /** Close trash actions */\n\t editor.toolbar.settings.hideRemoveActions();\n\t };\n\t\n\t return toolbar;\n\t}({});\n\n/***/ },\n/* 9 */\n/***/ function(module, exports) {\n\n\t'use strict';\n\t\n\t/**\n\t * Toolbar settings\n\t *\n\t * @version 1.0.4\n\t */\n\t\n\tmodule.exports = function (settings) {\n\t\n\t var editor = codex.editor;\n\t\n\t settings.opened = false;\n\t\n\t settings.setting = null;\n\t settings.actions = null;\n\t\n\t /**\n\t * Append and open settings\n\t */\n\t settings.open = function (toolType) {\n\t\n\t /**\n\t * Append settings content\n\t * It's stored in tool.settings\n\t */\n\t if (!editor.tools[toolType] || !editor.tools[toolType].makeSettings) {\n\t\n\t editor.core.log('Plugin \\xAB' + toolType + '\\xBB has no settings', 'warn');\n\t editor.nodes.pluginSettings.innerHTML = '\\u041F\\u043B\\u0430\\u0433\\u0438\\u043D \\xAB' + toolType + '\\xBB \\u043D\\u0435 \\u0438\\u043C\\u0435\\u0435\\u0442 \\u043D\\u0430\\u0441\\u0442\\u0440\\u043E\\u0435\\u043A';\n\t } else {\n\t\n\t /**\n\t * Draw settings block\n\t */\n\t var settingsBlock = editor.tools[toolType].makeSettings();\n\t\n\t editor.nodes.pluginSettings.appendChild(settingsBlock);\n\t }\n\t\n\t /** Open settings block */\n\t editor.nodes.blockSettings.classList.add('opened');\n\t this.opened = true;\n\t };\n\t\n\t /**\n\t * Close and clear settings\n\t */\n\t settings.close = function () {\n\t\n\t editor.nodes.blockSettings.classList.remove('opened');\n\t editor.nodes.pluginSettings.innerHTML = '';\n\t\n\t this.opened = false;\n\t };\n\t\n\t /**\n\t * @param {string} toolType - plugin type\n\t */\n\t settings.toggle = function (toolType) {\n\t\n\t if (!this.opened) {\n\t\n\t this.open(toolType);\n\t } else {\n\t\n\t this.close();\n\t }\n\t };\n\t\n\t /**\n\t * Here we will draw buttons and add listeners to components\n\t */\n\t settings.makeRemoveBlockButton = function () {\n\t\n\t var removeBlockWrapper = editor.draw.node('SPAN', 'ce-toolbar__remove-btn', {}),\n\t settingButton = editor.draw.node('SPAN', 'ce-toolbar__remove-setting', { innerHTML: '' }),\n\t actionWrapper = editor.draw.node('DIV', 'ce-toolbar__remove-confirmation', {}),\n\t confirmAction = editor.draw.node('DIV', 'ce-toolbar__remove-confirm', { textContent: 'Удалить блок' }),\n\t cancelAction = editor.draw.node('DIV', 'ce-toolbar__remove-cancel', { textContent: 'Отмена' });\n\t\n\t editor.listeners.add(settingButton, 'click', editor.toolbar.settings.removeButtonClicked, false);\n\t\n\t editor.listeners.add(confirmAction, 'click', editor.toolbar.settings.confirmRemovingRequest, false);\n\t\n\t editor.listeners.add(cancelAction, 'click', editor.toolbar.settings.cancelRemovingRequest, false);\n\t\n\t actionWrapper.appendChild(confirmAction);\n\t actionWrapper.appendChild(cancelAction);\n\t\n\t removeBlockWrapper.appendChild(settingButton);\n\t removeBlockWrapper.appendChild(actionWrapper);\n\t\n\t /** Save setting */\n\t editor.toolbar.settings.setting = settingButton;\n\t editor.toolbar.settings.actions = actionWrapper;\n\t\n\t return removeBlockWrapper;\n\t };\n\t\n\t settings.removeButtonClicked = function () {\n\t\n\t var action = editor.toolbar.settings.actions;\n\t\n\t if (action.classList.contains('opened')) {\n\t\n\t editor.toolbar.settings.hideRemoveActions();\n\t } else {\n\t\n\t editor.toolbar.settings.showRemoveActions();\n\t }\n\t\n\t editor.toolbar.toolbox.close();\n\t editor.toolbar.settings.close();\n\t };\n\t\n\t settings.cancelRemovingRequest = function () {\n\t\n\t editor.toolbar.settings.actions.classList.remove('opened');\n\t };\n\t\n\t settings.confirmRemovingRequest = function () {\n\t\n\t var currentBlock = editor.content.currentNode,\n\t firstLevelBlocksCount;\n\t\n\t currentBlock.remove();\n\t\n\t firstLevelBlocksCount = editor.nodes.redactor.childNodes.length;\n\t\n\t /**\n\t * If all blocks are removed\n\t */\n\t if (firstLevelBlocksCount === 0) {\n\t\n\t /** update currentNode variable */\n\t editor.content.currentNode = null;\n\t\n\t /** Inserting new empty initial block */\n\t editor.ui.addInitialBlock();\n\t }\n\t\n\t editor.ui.saveInputs();\n\t\n\t editor.toolbar.close();\n\t };\n\t\n\t settings.showRemoveActions = function () {\n\t\n\t editor.toolbar.settings.actions.classList.add('opened');\n\t };\n\t\n\t settings.hideRemoveActions = function () {\n\t\n\t editor.toolbar.settings.actions.classList.remove('opened');\n\t };\n\t\n\t return settings;\n\t}({});\n\n/***/ },\n/* 10 */\n/***/ function(module, exports) {\n\n\t'use strict';\n\t\n\t/**\n\t * Inline toolbar\n\t *\n\t * Contains from tools:\n\t * Bold, Italic, Underline and Anchor\n\t *\n\t * @author Codex Team\n\t * @version 1.0\n\t */\n\t\n\tmodule.exports = function (inline) {\n\t\n\t var editor = codex.editor;\n\t\n\t inline.buttonsOpened = null;\n\t inline.actionsOpened = null;\n\t inline.wrappersOffset = null;\n\t\n\t /**\n\t * saving selection that need for execCommand for styling\n\t *\n\t */\n\t inline.storedSelection = null;\n\t\n\t /**\n\t * @protected\n\t *\n\t * Open inline toobar\n\t */\n\t inline.show = function () {\n\t\n\t var currentNode = editor.content.currentNode,\n\t tool = currentNode.dataset.tool,\n\t plugin;\n\t\n\t /**\n\t * tool allowed to open inline toolbar\n\t */\n\t plugin = editor.tools[tool];\n\t\n\t if (!plugin.showInlineToolbar) return;\n\t\n\t var selectedText = inline.getSelectionText(),\n\t toolbar = editor.nodes.inlineToolbar.wrapper;\n\t\n\t if (selectedText.length > 0) {\n\t\n\t /** Move toolbar and open */\n\t editor.toolbar.inline.move();\n\t\n\t /** Open inline toolbar */\n\t toolbar.classList.add('opened');\n\t\n\t /** show buttons of inline toolbar */\n\t editor.toolbar.inline.showButtons();\n\t }\n\t };\n\t\n\t /**\n\t * @protected\n\t *\n\t * Closes inline toolbar\n\t */\n\t inline.close = function () {\n\t\n\t var toolbar = editor.nodes.inlineToolbar.wrapper;\n\t\n\t toolbar.classList.remove('opened');\n\t };\n\t\n\t /**\n\t * @private\n\t *\n\t * Moving toolbar\n\t */\n\t inline.move = function () {\n\t\n\t if (!this.wrappersOffset) {\n\t\n\t this.wrappersOffset = this.getWrappersOffset();\n\t }\n\t\n\t var coords = this.getSelectionCoords(),\n\t defaultOffset = 0,\n\t toolbar = editor.nodes.inlineToolbar.wrapper,\n\t newCoordinateX,\n\t newCoordinateY;\n\t\n\t if (toolbar.offsetHeight === 0) {\n\t\n\t defaultOffset = 40;\n\t }\n\t\n\t newCoordinateX = coords.x - this.wrappersOffset.left;\n\t newCoordinateY = coords.y + window.scrollY - this.wrappersOffset.top - defaultOffset - toolbar.offsetHeight;\n\t\n\t toolbar.style.transform = 'translate3D(' + Math.floor(newCoordinateX) + 'px, ' + Math.floor(newCoordinateY) + 'px, 0)';\n\t\n\t /** Close everything */\n\t editor.toolbar.inline.closeButtons();\n\t editor.toolbar.inline.closeAction();\n\t };\n\t\n\t /**\n\t * @private\n\t *\n\t * Tool Clicked\n\t */\n\t\n\t inline.toolClicked = function (event, type) {\n\t\n\t /**\n\t * For simple tools we use default browser function\n\t * For more complicated tools, we should write our own behavior\n\t */\n\t switch (type) {\n\t case 'createLink':\n\t editor.toolbar.inline.createLinkAction(event, type);break;\n\t default:\n\t editor.toolbar.inline.defaultToolAction(type);break;\n\t }\n\t\n\t /**\n\t * highlight buttons\n\t * after making some action\n\t */\n\t editor.nodes.inlineToolbar.buttons.childNodes.forEach(editor.toolbar.inline.hightlight);\n\t };\n\t\n\t /**\n\t * @private\n\t *\n\t * Saving wrappers offset in DOM\n\t */\n\t inline.getWrappersOffset = function () {\n\t\n\t var wrapper = editor.nodes.wrapper,\n\t offset = this.getOffset(wrapper);\n\t\n\t this.wrappersOffset = offset;\n\t return offset;\n\t };\n\t\n\t /**\n\t * @private\n\t *\n\t * Calculates offset of DOM element\n\t *\n\t * @param el\n\t * @returns {{top: number, left: number}}\n\t */\n\t inline.getOffset = function (el) {\n\t\n\t var _x = 0;\n\t var _y = 0;\n\t\n\t while (el && !isNaN(el.offsetLeft) && !isNaN(el.offsetTop)) {\n\t\n\t _x += el.offsetLeft + el.clientLeft;\n\t _y += el.offsetTop + el.clientTop;\n\t el = el.offsetParent;\n\t }\n\t return { top: _y, left: _x };\n\t };\n\t\n\t /**\n\t * @private\n\t *\n\t * Calculates position of selected text\n\t * @returns {{x: number, y: number}}\n\t */\n\t inline.getSelectionCoords = function () {\n\t\n\t var sel = document.selection,\n\t range;\n\t var x = 0,\n\t y = 0;\n\t\n\t if (sel) {\n\t\n\t if (sel.type != 'Control') {\n\t\n\t range = sel.createRange();\n\t range.collapse(true);\n\t x = range.boundingLeft;\n\t y = range.boundingTop;\n\t }\n\t } else if (window.getSelection) {\n\t\n\t sel = window.getSelection();\n\t\n\t if (sel.rangeCount) {\n\t\n\t range = sel.getRangeAt(0).cloneRange();\n\t if (range.getClientRects) {\n\t\n\t range.collapse(true);\n\t var rect = range.getClientRects()[0];\n\t\n\t if (!rect) {\n\t\n\t return;\n\t }\n\t\n\t x = rect.left;\n\t y = rect.top;\n\t }\n\t }\n\t }\n\t return { x: x, y: y };\n\t };\n\t\n\t /**\n\t * @private\n\t *\n\t * Returns selected text as String\n\t * @returns {string}\n\t */\n\t inline.getSelectionText = function () {\n\t\n\t var selectedText = '';\n\t\n\t // all modern browsers and IE9+\n\t if (window.getSelection) {\n\t\n\t selectedText = window.getSelection().toString();\n\t }\n\t\n\t return selectedText;\n\t };\n\t\n\t /** Opens buttons block */\n\t inline.showButtons = function () {\n\t\n\t var buttons = editor.nodes.inlineToolbar.buttons;\n\t\n\t buttons.classList.add('opened');\n\t\n\t editor.toolbar.inline.buttonsOpened = true;\n\t\n\t /** highlight buttons */\n\t editor.nodes.inlineToolbar.buttons.childNodes.forEach(editor.toolbar.inline.hightlight);\n\t };\n\t\n\t /** Makes buttons disappear */\n\t inline.closeButtons = function () {\n\t\n\t var buttons = editor.nodes.inlineToolbar.buttons;\n\t\n\t buttons.classList.remove('opened');\n\t\n\t editor.toolbar.inline.buttonsOpened = false;\n\t };\n\t\n\t /** Open buttons defined action if exist */\n\t inline.showActions = function () {\n\t\n\t var action = editor.nodes.inlineToolbar.actions;\n\t\n\t action.classList.add('opened');\n\t\n\t editor.toolbar.inline.actionsOpened = true;\n\t };\n\t\n\t /** Close actions block */\n\t inline.closeAction = function () {\n\t\n\t var action = editor.nodes.inlineToolbar.actions;\n\t\n\t action.innerHTML = '';\n\t action.classList.remove('opened');\n\t editor.toolbar.inline.actionsOpened = false;\n\t };\n\t\n\t /**\n\t * Callback for keydowns in inline toolbar \"Insert link...\" input\n\t */\n\t var inlineToolbarAnchorInputKeydown_ = function inlineToolbarAnchorInputKeydown_(event) {\n\t\n\t if (event.keyCode != editor.core.keys.ENTER) {\n\t\n\t return;\n\t }\n\t\n\t var editable = editor.content.currentNode,\n\t storedSelection = editor.toolbar.inline.storedSelection;\n\t\n\t editor.toolbar.inline.restoreSelection(editable, storedSelection);\n\t editor.toolbar.inline.setAnchor(this.value);\n\t\n\t /**\n\t * Preventing events that will be able to happen\n\t */\n\t event.preventDefault();\n\t event.stopImmediatePropagation();\n\t\n\t editor.toolbar.inline.clearRange();\n\t };\n\t\n\t /** Action for link creation or for setting anchor */\n\t inline.createLinkAction = function (event) {\n\t\n\t var isActive = this.isLinkActive();\n\t\n\t var editable = editor.content.currentNode,\n\t storedSelection = editor.toolbar.inline.saveSelection(editable);\n\t\n\t /** Save globally selection */\n\t editor.toolbar.inline.storedSelection = storedSelection;\n\t\n\t if (isActive) {\n\t\n\t /**\n\t * Changing stored selection. if we want to remove anchor from word\n\t * we should remove anchor from whole word, not only selected part.\n\t * The solution is than we get the length of current link\n\t * Change start position to - end of selection minus length of anchor\n\t */\n\t editor.toolbar.inline.restoreSelection(editable, storedSelection);\n\t\n\t editor.toolbar.inline.defaultToolAction('unlink');\n\t } else {\n\t\n\t /** Create input and close buttons */\n\t var action = editor.draw.inputForLink();\n\t\n\t editor.nodes.inlineToolbar.actions.appendChild(action);\n\t\n\t editor.toolbar.inline.closeButtons();\n\t editor.toolbar.inline.showActions();\n\t\n\t /**\n\t * focus to input\n\t * Solution: https://developer.mozilla.org/ru/docs/Web/API/HTMLElement/focus\n\t * Prevents event after showing input and when we need to focus an input which is in unexisted form\n\t */\n\t action.focus();\n\t event.preventDefault();\n\t\n\t /** Callback to link action */\n\t editor.listeners.add(action, 'keydown', inlineToolbarAnchorInputKeydown_, false);\n\t }\n\t };\n\t\n\t inline.isLinkActive = function () {\n\t\n\t var isActive = false;\n\t\n\t editor.nodes.inlineToolbar.buttons.childNodes.forEach(function (tool) {\n\t\n\t var dataType = tool.dataset.type;\n\t\n\t if (dataType == 'link' && tool.classList.contains('hightlighted')) {\n\t\n\t isActive = true;\n\t }\n\t });\n\t\n\t return isActive;\n\t };\n\t\n\t /** default action behavior of tool */\n\t inline.defaultToolAction = function (type) {\n\t\n\t document.execCommand(type, false, null);\n\t };\n\t\n\t /**\n\t * @private\n\t *\n\t * Sets URL\n\t *\n\t * @param {String} url - URL\n\t */\n\t inline.setAnchor = function (url) {\n\t\n\t document.execCommand('createLink', false, url);\n\t\n\t /** Close after URL inserting */\n\t editor.toolbar.inline.closeAction();\n\t };\n\t\n\t /**\n\t * @private\n\t *\n\t * Saves selection\n\t */\n\t inline.saveSelection = function (containerEl) {\n\t\n\t var range = window.getSelection().getRangeAt(0),\n\t preSelectionRange = range.cloneRange(),\n\t start;\n\t\n\t preSelectionRange.selectNodeContents(containerEl);\n\t preSelectionRange.setEnd(range.startContainer, range.startOffset);\n\t\n\t start = preSelectionRange.toString().length;\n\t\n\t return {\n\t start: start,\n\t end: start + range.toString().length\n\t };\n\t };\n\t\n\t /**\n\t * @private\n\t *\n\t * Sets to previous selection (Range)\n\t *\n\t * @param {Element} containerEl - editable element where we restore range\n\t * @param {Object} savedSel - range basic information to restore\n\t */\n\t inline.restoreSelection = function (containerEl, savedSel) {\n\t\n\t var range = document.createRange(),\n\t charIndex = 0;\n\t\n\t range.setStart(containerEl, 0);\n\t range.collapse(true);\n\t\n\t var nodeStack = [containerEl],\n\t node,\n\t foundStart = false,\n\t stop = false,\n\t nextCharIndex;\n\t\n\t while (!stop && (node = nodeStack.pop())) {\n\t\n\t if (node.nodeType == 3) {\n\t\n\t nextCharIndex = charIndex + node.length;\n\t\n\t if (!foundStart && savedSel.start >= charIndex && savedSel.start <= nextCharIndex) {\n\t\n\t range.setStart(node, savedSel.start - charIndex);\n\t foundStart = true;\n\t }\n\t if (foundStart && savedSel.end >= charIndex && savedSel.end <= nextCharIndex) {\n\t\n\t range.setEnd(node, savedSel.end - charIndex);\n\t stop = true;\n\t }\n\t charIndex = nextCharIndex;\n\t } else {\n\t\n\t var i = node.childNodes.length;\n\t\n\t while (i--) {\n\t\n\t nodeStack.push(node.childNodes[i]);\n\t }\n\t }\n\t }\n\t\n\t var sel = window.getSelection();\n\t\n\t sel.removeAllRanges();\n\t sel.addRange(range);\n\t };\n\t\n\t /**\n\t * @private\n\t *\n\t * Removes all ranges from window selection\n\t */\n\t inline.clearRange = function () {\n\t\n\t var selection = window.getSelection();\n\t\n\t selection.removeAllRanges();\n\t };\n\t\n\t /**\n\t * @private\n\t *\n\t * sets or removes hightlight\n\t */\n\t inline.hightlight = function (tool) {\n\t\n\t var dataType = tool.dataset.type;\n\t\n\t if (document.queryCommandState(dataType)) {\n\t\n\t editor.toolbar.inline.setButtonHighlighted(tool);\n\t } else {\n\t\n\t editor.toolbar.inline.removeButtonsHighLight(tool);\n\t }\n\t\n\t /**\n\t *\n\t * hightlight for anchors\n\t */\n\t var selection = window.getSelection(),\n\t tag = selection.anchorNode.parentNode;\n\t\n\t if (tag.tagName == 'A' && dataType == 'link') {\n\t\n\t editor.toolbar.inline.setButtonHighlighted(tool);\n\t }\n\t };\n\t\n\t /**\n\t * @private\n\t *\n\t * Mark button if text is already executed\n\t */\n\t inline.setButtonHighlighted = function (button) {\n\t\n\t button.classList.add('hightlighted');\n\t\n\t /** At link tool we also change icon */\n\t if (button.dataset.type == 'link') {\n\t\n\t var icon = button.childNodes[0];\n\t\n\t icon.classList.remove('ce-icon-link');\n\t icon.classList.add('ce-icon-unlink');\n\t }\n\t };\n\t\n\t /**\n\t * @private\n\t *\n\t * Removes hightlight\n\t */\n\t inline.removeButtonsHighLight = function (button) {\n\t\n\t button.classList.remove('hightlighted');\n\t\n\t /** At link tool we also change icon */\n\t if (button.dataset.type == 'link') {\n\t\n\t var icon = button.childNodes[0];\n\t\n\t icon.classList.remove('ce-icon-unlink');\n\t icon.classList.add('ce-icon-link');\n\t }\n\t };\n\t\n\t return inline;\n\t}({});\n\n/***/ },\n/* 11 */\n/***/ function(module, exports) {\n\n\t'use strict';\n\t\n\t/**\n\t * Codex Editor toolbox\n\t *\n\t * All tools be able to appended here\n\t *\n\t * @author Codex Team\n\t * @version 1.0\n\t */\n\t\n\tmodule.exports = function (toolbox) {\n\t\n\t var editor = codex.editor;\n\t\n\t toolbox.opened = false;\n\t\n\t /** Shows toolbox */\n\t toolbox.open = function () {\n\t\n\t /** Close setting if toolbox is opened */\n\t if (editor.toolbar.settings.opened) {\n\t\n\t editor.toolbar.settings.close();\n\t }\n\t\n\t /** display toolbox */\n\t editor.nodes.toolbox.classList.add('opened');\n\t\n\t /** Animate plus button */\n\t editor.nodes.plusButton.classList.add('clicked');\n\t\n\t /** toolbox state */\n\t editor.toolbar.toolbox.opened = true;\n\t };\n\t\n\t /** Closes toolbox */\n\t toolbox.close = function () {\n\t\n\t /** Makes toolbox disapear */\n\t editor.nodes.toolbox.classList.remove('opened');\n\t\n\t /** Rotate plus button */\n\t editor.nodes.plusButton.classList.remove('clicked');\n\t\n\t /** toolbox state */\n\t editor.toolbar.toolbox.opened = false;\n\t\n\t editor.toolbar.current = null;\n\t };\n\t\n\t toolbox.leaf = function () {\n\t\n\t var currentTool = editor.toolbar.current,\n\t tools = Object.keys(editor.tools),\n\t barButtons = editor.nodes.toolbarButtons,\n\t nextToolIndex = 0,\n\t toolToSelect = void 0,\n\t visibleTool = void 0,\n\t tool = void 0;\n\t\n\t if (!currentTool) {\n\t\n\t /** Get first tool from object*/\n\t for (tool in editor.tools) {\n\t\n\t if (editor.tools[tool].displayInToolbox) {\n\t\n\t break;\n\t }\n\t\n\t nextToolIndex++;\n\t }\n\t } else {\n\t\n\t nextToolIndex = (tools.indexOf(currentTool) + 1) % tools.length;\n\t visibleTool = tools[nextToolIndex];\n\t\n\t while (!editor.tools[visibleTool].displayInToolbox) {\n\t\n\t nextToolIndex = (nextToolIndex + 1) % tools.length;\n\t visibleTool = tools[nextToolIndex];\n\t }\n\t }\n\t\n\t toolToSelect = tools[nextToolIndex];\n\t\n\t for (var button in barButtons) {\n\t\n\t barButtons[button].classList.remove('selected');\n\t }\n\t\n\t barButtons[toolToSelect].classList.add('selected');\n\t editor.toolbar.current = toolToSelect;\n\t };\n\t\n\t /**\n\t * Transforming selected node type into selected toolbar element type\n\t * @param {event} event\n\t */\n\t toolbox.toolClicked = function (event) {\n\t\n\t /**\n\t * UNREPLACEBLE_TOOLS this types of tools are forbidden to replace even they are empty\n\t */\n\t var UNREPLACEBLE_TOOLS = ['image', 'link', 'list', 'instagram', 'twitter', 'embed'],\n\t tool = editor.tools[editor.toolbar.current],\n\t workingNode = editor.content.currentNode,\n\t currentInputIndex = editor.caret.inputIndex,\n\t newBlockContent,\n\t appendCallback,\n\t blockData;\n\t\n\t /** Make block from plugin */\n\t newBlockContent = tool.render();\n\t\n\t /** information about block */\n\t blockData = {\n\t block: newBlockContent,\n\t type: tool.type,\n\t stretched: false\n\t };\n\t\n\t if (workingNode && UNREPLACEBLE_TOOLS.indexOf(workingNode.dataset.tool) === -1 && workingNode.textContent.trim() === '') {\n\t\n\t /** Replace current block */\n\t editor.content.switchBlock(workingNode, newBlockContent, tool.type);\n\t } else {\n\t\n\t /** Insert new Block from plugin */\n\t editor.content.insertBlock(blockData);\n\t\n\t /** increase input index */\n\t currentInputIndex++;\n\t }\n\t\n\t /** Fire tool append callback */\n\t appendCallback = tool.appendCallback;\n\t\n\t if (appendCallback && typeof appendCallback == 'function') {\n\t\n\t appendCallback.call(event);\n\t }\n\t\n\t window.setTimeout(function () {\n\t\n\t /** Set caret to current block */\n\t editor.caret.setToBlock(currentInputIndex);\n\t }, 10);\n\t\n\t /**\n\t * Changing current Node\n\t */\n\t editor.content.workingNodeChanged();\n\t\n\t /**\n\t * Move toolbar when node is changed\n\t */\n\t editor.toolbar.move();\n\t };\n\t\n\t return toolbox;\n\t}({});\n\n/***/ },\n/* 12 */\n/***/ function(module, exports) {\n\n\t'use strict';\n\t\n\t/**\n\t * @module Codex Editor Callbacks module\n\t * @description Module works with editor added Elements\n\t *\n\t * @author Codex Team\n\t * @version 1.4.0\n\t */\n\t\n\tmodule.exports = function (callbacks) {\n\t\n\t var editor = codex.editor;\n\t\n\t /**\n\t * used by UI module\n\t * @description Routes all keydowns on document\n\t * @param {Object} event\n\t */\n\t callbacks.globalKeydown = function (event) {\n\t\n\t switch (event.keyCode) {\n\t case editor.core.keys.ENTER:\n\t enterKeyPressed_(event);break;\n\t }\n\t };\n\t\n\t /**\n\t * used by UI module\n\t * @description Routes all keydowns on redactors area\n\t * @param {Object} event\n\t */\n\t callbacks.redactorKeyDown = function (event) {\n\t\n\t switch (event.keyCode) {\n\t case editor.core.keys.TAB:\n\t tabKeyPressedOnRedactorsZone_(event);break;\n\t case editor.core.keys.ENTER:\n\t enterKeyPressedOnRedactorsZone_(event);break;\n\t case editor.core.keys.ESC:\n\t escapeKeyPressedOnRedactorsZone_(event);break;\n\t default:\n\t defaultKeyPressedOnRedactorsZone_(event);break;\n\t }\n\t };\n\t\n\t /**\n\t * used by UI module\n\t * @description Routes all keyup events\n\t * @param {Object} event\n\t */\n\t callbacks.globalKeyup = function (event) {\n\t\n\t switch (event.keyCode) {\n\t case editor.core.keys.UP:\n\t case editor.core.keys.LEFT:\n\t case editor.core.keys.RIGHT:\n\t case editor.core.keys.DOWN:\n\t arrowKeyPressed_(event);break;\n\t }\n\t };\n\t\n\t /**\n\t * @param {Object} event\n\t * @private\n\t *\n\t * Handles behaviour when tab pressed\n\t * @description if Content is empty show toolbox (if it is closed) or leaf tools\n\t * uses Toolbars toolbox module to handle the situation\n\t */\n\t var tabKeyPressedOnRedactorsZone_ = function tabKeyPressedOnRedactorsZone_(event) {\n\t\n\t /**\n\t * Wait for solution. Would like to know the behaviour\n\t * @todo Add spaces\n\t */\n\t event.preventDefault();\n\t\n\t var nativeInputs = editor.content.currentNode.querySelectorAll('textarea, input'),\n\t nativeInputsAreEmpty = true,\n\t textContentIsEmpty = !editor.content.currentNode.textContent.trim();\n\t\n\t Array.prototype.map.call(nativeInputs, function (input) {\n\t\n\t if (input.type == 'textarea' || input.type == 'text') {\n\t\n\t nativeInputsAreEmpty = nativeInputsAreEmpty && !input.value.trim();\n\t }\n\t });\n\t\n\t var blockIsEmpty = textContentIsEmpty && nativeInputsAreEmpty;\n\t\n\t if (!blockIsEmpty) {\n\t\n\t return;\n\t }\n\t\n\t if (!editor.toolbar.opened) {\n\t\n\t editor.toolbar.open();\n\t }\n\t\n\t if (editor.toolbar.opened && !editor.toolbar.toolbox.opened) {\n\t\n\t editor.toolbar.toolbox.open();\n\t } else {\n\t\n\t editor.toolbar.toolbox.leaf();\n\t }\n\t };\n\t\n\t /**\n\t * Handles global EnterKey Press\n\t * @see enterPressedOnBlock_\n\t * @param {Object} event\n\t */\n\t var enterKeyPressed_ = function enterKeyPressed_() {\n\t\n\t if (editor.content.editorAreaHightlighted) {\n\t\n\t /**\n\t * it means that we lose input index, saved index before is not correct\n\t * therefore we need to set caret when we insert new block\n\t */\n\t editor.caret.inputIndex = -1;\n\t\n\t enterPressedOnBlock_();\n\t }\n\t };\n\t\n\t /**\n\t * Callback for enter key pressing in first-level block area\n\t *\n\t * @param {Event} event\n\t * @private\n\t *\n\t * @description Inserts new block with initial type from settings\n\t */\n\t var enterPressedOnBlock_ = function enterPressedOnBlock_() {\n\t\n\t var NEW_BLOCK_TYPE = editor.settings.initialBlockPlugin;\n\t\n\t editor.content.insertBlock({\n\t type: NEW_BLOCK_TYPE,\n\t block: editor.tools[NEW_BLOCK_TYPE].render()\n\t }, true);\n\t\n\t editor.toolbar.move();\n\t editor.toolbar.open();\n\t };\n\t\n\t /**\n\t * ENTER key handler\n\t *\n\t * @param {Object} event\n\t * @private\n\t *\n\t * @description Makes new block with initial type from settings\n\t */\n\t var enterKeyPressedOnRedactorsZone_ = function enterKeyPressedOnRedactorsZone_(event) {\n\t\n\t if (event.target.contentEditable == 'true') {\n\t\n\t /** Update input index */\n\t editor.caret.saveCurrentInputIndex();\n\t }\n\t\n\t var currentInputIndex = editor.caret.getCurrentInputIndex() || 0,\n\t workingNode = editor.content.currentNode,\n\t tool = workingNode.dataset.tool,\n\t isEnterPressedOnToolbar = editor.toolbar.opened && editor.toolbar.current && event.target == editor.state.inputs[currentInputIndex];\n\t\n\t /** The list of tools which needs the default browser behaviour */\n\t var enableLineBreaks = editor.tools[tool].enableLineBreaks;\n\t\n\t /** This type of block creates when enter is pressed */\n\t var NEW_BLOCK_TYPE = editor.settings.initialBlockPlugin;\n\t\n\t /**\n\t * When toolbar is opened, select tool instead of making new paragraph\n\t */\n\t if (isEnterPressedOnToolbar) {\n\t\n\t event.preventDefault();\n\t\n\t editor.toolbar.toolbox.toolClicked(event);\n\t\n\t editor.toolbar.close();\n\t\n\t /**\n\t * Stop other listeners callback executions\n\t */\n\t event.stopPropagation();\n\t event.stopImmediatePropagation();\n\t\n\t return;\n\t }\n\t\n\t /**\n\t * Allow paragraph lineBreaks with shift enter\n\t * Or if shiftkey pressed and enter and enabledLineBreaks, the let new block creation\n\t */\n\t if (event.shiftKey || enableLineBreaks) {\n\t\n\t event.stopPropagation();\n\t event.stopImmediatePropagation();\n\t return;\n\t }\n\t\n\t var currentSelection = window.getSelection(),\n\t currentSelectedNode = currentSelection.anchorNode,\n\t caretAtTheEndOfText = editor.caret.position.atTheEnd(),\n\t isTextNodeHasParentBetweenContenteditable = false;\n\t\n\t /**\n\t * Allow making new

in same block by SHIFT+ENTER and forbids to prevent default browser behaviour\n\t */\n\t if (event.shiftKey && !enableLineBreaks) {\n\t\n\t editor.callback.enterPressedOnBlock(editor.content.currentBlock, event);\n\t event.preventDefault();\n\t return;\n\t }\n\t\n\t /**\n\t * Workaround situation when caret at the Text node that has some wrapper Elements\n\t * Split block cant handle this.\n\t * We need to save default behavior\n\t */\n\t isTextNodeHasParentBetweenContenteditable = currentSelectedNode && currentSelectedNode.parentNode.contentEditable != 'true';\n\t\n\t /**\n\t * Split blocks when input has several nodes and caret placed in textNode\n\t */\n\t if (currentSelectedNode.nodeType == editor.core.nodeTypes.TEXT && !isTextNodeHasParentBetweenContenteditable && !caretAtTheEndOfText) {\n\t\n\t event.preventDefault();\n\t\n\t editor.core.log('Splitting Text node...');\n\t\n\t editor.content.splitBlock(currentInputIndex);\n\t\n\t /** Show plus button when next input after split is empty*/\n\t if (!editor.state.inputs[currentInputIndex + 1].textContent.trim()) {\n\t\n\t editor.toolbar.showPlusButton();\n\t }\n\t } else {\n\t\n\t var islastNode = editor.content.isLastNode(currentSelectedNode);\n\t\n\t if (islastNode && caretAtTheEndOfText) {\n\t\n\t event.preventDefault();\n\t event.stopPropagation();\n\t event.stopImmediatePropagation();\n\t\n\t editor.core.log('ENTER clicked in last textNode. Create new BLOCK');\n\t\n\t editor.content.insertBlock({\n\t type: NEW_BLOCK_TYPE,\n\t block: editor.tools[NEW_BLOCK_TYPE].render()\n\t }, true);\n\t\n\t editor.toolbar.move();\n\t editor.toolbar.open();\n\t\n\t /** Show plus button with empty block */\n\t editor.toolbar.showPlusButton();\n\t }\n\t }\n\t\n\t /** get all inputs after new appending block */\n\t editor.ui.saveInputs();\n\t };\n\t\n\t /**\n\t * Escape behaviour\n\t * @param event\n\t * @private\n\t *\n\t * @description Closes toolbox and toolbar. Prevents default behaviour\n\t */\n\t var escapeKeyPressedOnRedactorsZone_ = function escapeKeyPressedOnRedactorsZone_(event) {\n\t\n\t /** Close all toolbar */\n\t editor.toolbar.close();\n\t\n\t /** Close toolbox */\n\t editor.toolbar.toolbox.close();\n\t\n\t event.preventDefault();\n\t };\n\t\n\t /**\n\t * @param {Event} event\n\t * @private\n\t *\n\t * closes and moves toolbar\n\t */\n\t var arrowKeyPressed_ = function arrowKeyPressed_(event) {\n\t\n\t editor.content.workingNodeChanged();\n\t\n\t /* Closing toolbar */\n\t editor.toolbar.close();\n\t editor.toolbar.move();\n\t };\n\t\n\t /**\n\t * @private\n\t * @param {Event} event\n\t *\n\t * @description Closes all opened bars from toolbar.\n\t * If block is mark, clears highlightning\n\t */\n\t var defaultKeyPressedOnRedactorsZone_ = function defaultKeyPressedOnRedactorsZone_() {\n\t\n\t editor.toolbar.close();\n\t\n\t if (!editor.toolbar.inline.actionsOpened) {\n\t\n\t editor.toolbar.inline.close();\n\t editor.content.clearMark();\n\t }\n\t };\n\t\n\t /**\n\t * Handler when clicked on redactors area\n\t *\n\t * @protected\n\t * @param event\n\t *\n\t * @description Detects clicked area. If it is first-level block area, marks as detected and\n\t * on next enter press will be inserted new block\n\t * Otherwise, save carets position (input index) and put caret to the editable zone.\n\t *\n\t * @see detectWhenClickedOnFirstLevelBlockArea_\n\t *\n\t */\n\t callbacks.redactorClicked = function (event) {\n\t\n\t detectWhenClickedOnFirstLevelBlockArea_();\n\t\n\t editor.content.workingNodeChanged(event.target);\n\t editor.ui.saveInputs();\n\t\n\t var selectedText = editor.toolbar.inline.getSelectionText(),\n\t firstLevelBlock;\n\t\n\t /** If selection range took off, then we hide inline toolbar */\n\t if (selectedText.length === 0) {\n\t\n\t editor.toolbar.inline.close();\n\t }\n\t\n\t /** Update current input index in memory when caret focused into existed input */\n\t if (event.target.contentEditable == 'true') {\n\t\n\t editor.caret.saveCurrentInputIndex();\n\t }\n\t\n\t if (editor.content.currentNode === null) {\n\t\n\t /**\n\t * If inputs in redactor does not exits, then we put input index 0 not -1\n\t */\n\t var indexOfLastInput = editor.state.inputs.length > 0 ? editor.state.inputs.length - 1 : 0;\n\t\n\t /** If we have any inputs */\n\t if (editor.state.inputs.length) {\n\t\n\t /** getting firstlevel parent of input */\n\t firstLevelBlock = editor.content.getFirstLevelBlock(editor.state.inputs[indexOfLastInput]);\n\t }\n\t\n\t /** If input is empty, then we set caret to the last input */\n\t if (editor.state.inputs.length && editor.state.inputs[indexOfLastInput].textContent === '' && firstLevelBlock.dataset.tool == editor.settings.initialBlockPlugin) {\n\t\n\t editor.caret.setToBlock(indexOfLastInput);\n\t } else {\n\t\n\t /** Create new input when caret clicked in redactors area */\n\t var NEW_BLOCK_TYPE = editor.settings.initialBlockPlugin;\n\t\n\t editor.content.insertBlock({\n\t type: NEW_BLOCK_TYPE,\n\t block: editor.tools[NEW_BLOCK_TYPE].render()\n\t });\n\t\n\t /** If there is no inputs except inserted */\n\t if (editor.state.inputs.length === 1) {\n\t\n\t editor.caret.setToBlock(indexOfLastInput);\n\t } else {\n\t\n\t /** Set caret to this appended input */\n\t editor.caret.setToNextBlock(indexOfLastInput);\n\t }\n\t }\n\t } else {\n\t\n\t /** Close all panels */\n\t editor.toolbar.settings.close();\n\t editor.toolbar.toolbox.close();\n\t }\n\t\n\t /**\n\t * Move toolbar and open\n\t */\n\t editor.toolbar.move();\n\t editor.toolbar.open();\n\t\n\t var inputIsEmpty = !editor.content.currentNode.textContent.trim(),\n\t currentNodeType = editor.content.currentNode.dataset.tool,\n\t isInitialType = currentNodeType == editor.settings.initialBlockPlugin;\n\t\n\t /** Hide plus buttons */\n\t editor.toolbar.hidePlusButton();\n\t\n\t if (!inputIsEmpty) {\n\t\n\t /** Mark current block */\n\t editor.content.markBlock();\n\t }\n\t\n\t if (isInitialType && inputIsEmpty) {\n\t\n\t /** Show plus button */\n\t editor.toolbar.showPlusButton();\n\t }\n\t };\n\t\n\t /**\n\t * This method allows to define, is caret in contenteditable element or not.\n\t *\n\t * @private\n\t *\n\t * @description Otherwise, if we get TEXT node from range container, that will means we have input index.\n\t * In this case we use default browsers behaviour (if plugin allows that) or overwritten action.\n\t * Therefore, to be sure that we've clicked first-level block area, we should have currentNode, which always\n\t * specifies to the first-level block. Other cases we just ignore.\n\t */\n\t var detectWhenClickedOnFirstLevelBlockArea_ = function detectWhenClickedOnFirstLevelBlockArea_() {\n\t\n\t var selection = window.getSelection(),\n\t anchorNode = selection.anchorNode,\n\t flag = false;\n\t\n\t if (selection.rangeCount === 0) {\n\t\n\t editor.content.editorAreaHightlighted = true;\n\t } else {\n\t\n\t if (!editor.core.isDomNode(anchorNode)) {\n\t\n\t anchorNode = anchorNode.parentNode;\n\t }\n\t\n\t /** Already founded, without loop */\n\t if (anchorNode.contentEditable == 'true') {\n\t\n\t flag = true;\n\t }\n\t\n\t while (anchorNode.contentEditable != 'true') {\n\t\n\t anchorNode = anchorNode.parentNode;\n\t\n\t if (anchorNode.contentEditable == 'true') {\n\t\n\t flag = true;\n\t }\n\t\n\t if (anchorNode == document.body) {\n\t\n\t break;\n\t }\n\t }\n\t\n\t /** If editable element founded, flag is \"TRUE\", Therefore we return \"FALSE\" */\n\t editor.content.editorAreaHightlighted = !flag;\n\t }\n\t };\n\t\n\t /**\n\t * Toolbar button click handler\n\t *\n\t * @param {Object} event - cursor to the button\n\t * @protected\n\t *\n\t * @description gets current tool and calls render method\n\t */\n\t callbacks.toolbarButtonClicked = function (event) {\n\t\n\t var button = this;\n\t\n\t editor.toolbar.current = button.dataset.type;\n\t\n\t editor.toolbar.toolbox.toolClicked(event);\n\t editor.toolbar.close();\n\t };\n\t\n\t /**\n\t * Show or Hide toolbox when plus button is clicked\n\t */\n\t callbacks.plusButtonClicked = function () {\n\t\n\t if (!editor.nodes.toolbox.classList.contains('opened')) {\n\t\n\t editor.toolbar.toolbox.open();\n\t } else {\n\t\n\t editor.toolbar.toolbox.close();\n\t }\n\t };\n\t\n\t /**\n\t * Block handlers for KeyDown events\n\t *\n\t * @protected\n\t * @param {Object} event\n\t *\n\t * Handles keydowns on block\n\t * @see blockRightOrDownArrowPressed_\n\t * @see backspacePressed_\n\t * @see blockLeftOrUpArrowPressed_\n\t */\n\t callbacks.blockKeydown = function (event) {\n\t\n\t var block = event.target; // event.target is input\n\t\n\t switch (event.keyCode) {\n\t\n\t case editor.core.keys.DOWN:\n\t case editor.core.keys.RIGHT:\n\t blockRightOrDownArrowPressed_(event);\n\t break;\n\t\n\t case editor.core.keys.BACKSPACE:\n\t backspacePressed_(block, event);\n\t break;\n\t\n\t case editor.core.keys.UP:\n\t case editor.core.keys.LEFT:\n\t blockLeftOrUpArrowPressed_(event);\n\t break;\n\t\n\t }\n\t };\n\t\n\t /**\n\t * RIGHT or DOWN keydowns on block\n\t *\n\t * @param {Object} event\n\t * @private\n\t *\n\t * @description watches the selection and gets closest editable element.\n\t * Uses method getDeepestTextNodeFromPosition to get the last node of next block\n\t * Sets caret if it is contenteditable\n\t */\n\t var blockRightOrDownArrowPressed_ = function blockRightOrDownArrowPressed_(event) {\n\t\n\t var selection = window.getSelection(),\n\t inputs = editor.state.inputs,\n\t focusedNode = selection.anchorNode,\n\t focusedNodeHolder;\n\t\n\t /** Check for caret existance */\n\t if (!focusedNode) {\n\t\n\t return false;\n\t }\n\t\n\t /** Looking for closest (parent) contentEditable element of focused node */\n\t while (focusedNode.contentEditable != 'true') {\n\t\n\t focusedNodeHolder = focusedNode.parentNode;\n\t focusedNode = focusedNodeHolder;\n\t }\n\t\n\t /** Input index in DOM level */\n\t var editableElementIndex = 0;\n\t\n\t while (focusedNode != inputs[editableElementIndex]) {\n\t\n\t editableElementIndex++;\n\t }\n\t\n\t /**\n\t * Founded contentEditable element doesn't have childs\n\t * Or maybe New created block\n\t */\n\t if (!focusedNode.textContent) {\n\t\n\t editor.caret.setToNextBlock(editableElementIndex);\n\t return;\n\t }\n\t\n\t /**\n\t * Do nothing when caret doesn not reaches the end of last child\n\t */\n\t var caretInLastChild = false,\n\t caretAtTheEndOfText = false;\n\t\n\t var lastChild, deepestTextnode;\n\t\n\t lastChild = focusedNode.childNodes[focusedNode.childNodes.length - 1];\n\t\n\t if (editor.core.isDomNode(lastChild)) {\n\t\n\t deepestTextnode = editor.content.getDeepestTextNodeFromPosition(lastChild, lastChild.childNodes.length);\n\t } else {\n\t\n\t deepestTextnode = lastChild;\n\t }\n\t\n\t caretInLastChild = selection.anchorNode == deepestTextnode;\n\t caretAtTheEndOfText = deepestTextnode.length == selection.anchorOffset;\n\t\n\t if (!caretInLastChild || !caretAtTheEndOfText) {\n\t\n\t editor.core.log('arrow [down|right] : caret does not reached the end');\n\t return false;\n\t }\n\t\n\t editor.caret.setToNextBlock(editableElementIndex);\n\t };\n\t\n\t /**\n\t * LEFT or UP keydowns on block\n\t *\n\t * @param {Object} event\n\t * @private\n\t *\n\t * watches the selection and gets closest editable element.\n\t * Uses method getDeepestTextNodeFromPosition to get the last node of previous block\n\t * Sets caret if it is contenteditable\n\t *\n\t */\n\t var blockLeftOrUpArrowPressed_ = function blockLeftOrUpArrowPressed_(event) {\n\t\n\t var selection = window.getSelection(),\n\t inputs = editor.state.inputs,\n\t focusedNode = selection.anchorNode,\n\t focusedNodeHolder;\n\t\n\t /** Check for caret existance */\n\t if (!focusedNode) {\n\t\n\t return false;\n\t }\n\t\n\t /**\n\t * LEFT or UP not at the beginning\n\t */\n\t if (selection.anchorOffset !== 0) {\n\t\n\t return false;\n\t }\n\t\n\t /** Looking for parent contentEditable block */\n\t while (focusedNode.contentEditable != 'true') {\n\t\n\t focusedNodeHolder = focusedNode.parentNode;\n\t focusedNode = focusedNodeHolder;\n\t }\n\t\n\t /** Input index in DOM level */\n\t var editableElementIndex = 0;\n\t\n\t while (focusedNode != inputs[editableElementIndex]) {\n\t\n\t editableElementIndex++;\n\t }\n\t\n\t /**\n\t * Do nothing if caret is not at the beginning of first child\n\t */\n\t var caretInFirstChild = false,\n\t caretAtTheBeginning = false;\n\t\n\t var firstChild, deepestTextnode;\n\t\n\t /**\n\t * Founded contentEditable element doesn't have childs\n\t * Or maybe New created block\n\t */\n\t if (!focusedNode.textContent) {\n\t\n\t editor.caret.setToPreviousBlock(editableElementIndex);\n\t return;\n\t }\n\t\n\t firstChild = focusedNode.childNodes[0];\n\t\n\t if (editor.core.isDomNode(firstChild)) {\n\t\n\t deepestTextnode = editor.content.getDeepestTextNodeFromPosition(firstChild, 0);\n\t } else {\n\t\n\t deepestTextnode = firstChild;\n\t }\n\t\n\t caretInFirstChild = selection.anchorNode == deepestTextnode;\n\t caretAtTheBeginning = selection.anchorOffset === 0;\n\t\n\t if (caretInFirstChild && caretAtTheBeginning) {\n\t\n\t editor.caret.setToPreviousBlock(editableElementIndex);\n\t }\n\t };\n\t\n\t /**\n\t * Handles backspace keydown\n\t *\n\t * @param {Element} block\n\t * @param {Object} event\n\t * @private\n\t *\n\t * @description if block is empty, delete the block and set caret to the previous block\n\t * If block is not empty, try to merge two blocks - current and previous\n\t * But it we try'n to remove first block, then we should set caret to the next block, not previous.\n\t * If we removed the last block, create new one\n\t */\n\t var backspacePressed_ = function backspacePressed_(block, event) {\n\t\n\t var currentInputIndex = editor.caret.getCurrentInputIndex(),\n\t range,\n\t selectionLength,\n\t firstLevelBlocksCount;\n\t\n\t if (isNativeInput_(event.target)) {\n\t\n\t /** If input value is empty - remove block */\n\t if (event.target.value.trim() == '') {\n\t\n\t block.remove();\n\t } else {\n\t\n\t return;\n\t }\n\t }\n\t\n\t if (block.textContent.trim()) {\n\t\n\t range = editor.content.getRange();\n\t selectionLength = range.endOffset - range.startOffset;\n\t\n\t if (editor.caret.position.atStart() && !selectionLength && editor.state.inputs[currentInputIndex - 1]) {\n\t\n\t editor.content.mergeBlocks(currentInputIndex);\n\t } else {\n\t\n\t return;\n\t }\n\t }\n\t\n\t if (!selectionLength) {\n\t\n\t block.remove();\n\t }\n\t\n\t firstLevelBlocksCount = editor.nodes.redactor.childNodes.length;\n\t\n\t /**\n\t * If all blocks are removed\n\t */\n\t if (firstLevelBlocksCount === 0) {\n\t\n\t /** update currentNode variable */\n\t editor.content.currentNode = null;\n\t\n\t /** Inserting new empty initial block */\n\t editor.ui.addInitialBlock();\n\t\n\t /** Updating inputs state after deleting last block */\n\t editor.ui.saveInputs();\n\t\n\t /** Set to current appended block */\n\t window.setTimeout(function () {\n\t\n\t editor.caret.setToPreviousBlock(1);\n\t }, 10);\n\t } else {\n\t\n\t if (editor.caret.inputIndex !== 0) {\n\t\n\t /** Target block is not first */\n\t editor.caret.setToPreviousBlock(editor.caret.inputIndex);\n\t } else {\n\t\n\t /** If we try to delete first block */\n\t editor.caret.setToNextBlock(editor.caret.inputIndex);\n\t }\n\t }\n\t\n\t editor.toolbar.move();\n\t\n\t if (!editor.toolbar.opened) {\n\t\n\t editor.toolbar.open();\n\t }\n\t\n\t /** Updating inputs state */\n\t editor.ui.saveInputs();\n\t\n\t /** Prevent default browser behaviour */\n\t event.preventDefault();\n\t };\n\t\n\t /**\n\t * This method prevents default behaviour.\n\t *\n\t * @param {Object} event\n\t * @protected\n\t *\n\t * @description We get from clipboard pasted data, sanitize, make a fragment that contains of this sanitized nodes.\n\t * Firstly, we need to memorize the caret position. We can do that by getting the range of selection.\n\t * After all, we insert clear fragment into caret placed position. Then, we should move the caret to the last node\n\t */\n\t callbacks.blockPasteCallback = function (event) {\n\t\n\t /** If area is input or textarea then allow default behaviour */\n\t if (isNativeInput_(event.target)) {\n\t\n\t return;\n\t }\n\t\n\t /** Prevent default behaviour */\n\t event.preventDefault();\n\t\n\t var editableParent = editor.content.getEditableParent(event.target),\n\t currentNode = editor.content.currentNode;\n\t\n\t /** Allow paste when event target placed in Editable element */\n\t if (!editableParent) {\n\t\n\t return;\n\t }\n\t\n\t /** get html pasted data - dirty data */\n\t var htmlData = event.clipboardData.getData('text/html'),\n\t plainData = event.clipboardData.getData('text/plain');\n\t\n\t /** Temporary DIV that is used to work with childs as arrays item */\n\t var div = editor.draw.node('DIV', '', {}),\n\t cleanData,\n\t fragment;\n\t\n\t /** Create fragment, that we paste to range after proccesing */\n\t fragment = document.createDocumentFragment();\n\t\n\t if (htmlData.trim() != '') {\n\t\n\t cleanData = editor.sanitizer.clean(htmlData);\n\t div.innerHTML = cleanData;\n\t } else {\n\t\n\t div.innerText = plainData.toString();\n\t }\n\t\n\t var node, lastNode;\n\t\n\t /**\n\t * and fill in fragment\n\t */\n\t while (node = div.firstChild) {\n\t\n\t lastNode = fragment.appendChild(node);\n\t }\n\t\n\t if (editor.tools[currentNode.dataset.tool].allowRenderOnPaste) {\n\t\n\t if (editor.paste.pasted(event)) return;\n\t }\n\t\n\t /**\n\t * work with selection and range\n\t */\n\t var selection, range;\n\t\n\t selection = window.getSelection();\n\t\n\t range = selection.getRangeAt(0);\n\t range.deleteContents();\n\t\n\t range.insertNode(fragment);\n\t\n\t /** Preserve the selection */\n\t if (lastNode) {\n\t\n\t range = range.cloneRange();\n\t range.setStartAfter(lastNode);\n\t range.collapse(true);\n\t selection.removeAllRanges();\n\t selection.addRange(range);\n\t }\n\t };\n\t\n\t /**\n\t * used by UI module\n\t * Clicks on block settings button\n\t *\n\t * @param {Object} event\n\t * @protected\n\t * @description Opens toolbar settings\n\t */\n\t callbacks.showSettingsButtonClicked = function (event) {\n\t\n\t /**\n\t * Get type of current block\n\t * It uses to append settings from tool.settings property.\n\t * ...\n\t * Type is stored in data-type attribute on block\n\t */\n\t var currentToolType = editor.content.currentNode.dataset.tool;\n\t\n\t editor.toolbar.settings.toggle(currentToolType);\n\t\n\t /** Close toolbox when settings button is active */\n\t editor.toolbar.toolbox.close();\n\t editor.toolbar.settings.hideRemoveActions();\n\t };\n\t\n\t /**\n\t * Check block\n\t * @param target\n\t * @private\n\t *\n\t * @description Checks target is it native input\n\t */\n\t var isNativeInput_ = function isNativeInput_(target) {\n\t\n\t var nativeInputAreas = ['INPUT', 'TEXTAREA'];\n\t\n\t return nativeInputAreas.indexOf(target.tagName) != -1;\n\t };\n\t\n\t return callbacks;\n\t}({});\n\n/***/ },\n/* 13 */\n/***/ function(module, exports) {\n\n\t'use strict';\n\t\n\t/**\n\t * Codex Editor Draw module\n\t *\n\t * @author Codex Team\n\t * @version 1.0.\n\t */\n\t\n\tmodule.exports = function (draw) {\n\t\n\t /**\n\t * Base editor wrapper\n\t */\n\t draw.wrapper = function () {\n\t\n\t var wrapper = document.createElement('div');\n\t\n\t wrapper.className += 'codex-editor';\n\t\n\t return wrapper;\n\t };\n\t\n\t /**\n\t * Content-editable holder\n\t */\n\t draw.redactor = function () {\n\t\n\t var redactor = document.createElement('div');\n\t\n\t redactor.className += 'ce-redactor';\n\t\n\t return redactor;\n\t };\n\t\n\t draw.ceBlock = function () {\n\t\n\t var block = document.createElement('DIV');\n\t\n\t block.className += 'ce_block';\n\t\n\t return block;\n\t };\n\t\n\t /**\n\t * Empty toolbar with toggler\n\t */\n\t draw.toolbar = function () {\n\t\n\t var bar = document.createElement('div');\n\t\n\t bar.className += 'ce-toolbar';\n\t\n\t return bar;\n\t };\n\t\n\t draw.toolbarContent = function () {\n\t\n\t var wrapper = document.createElement('DIV');\n\t\n\t wrapper.classList.add('ce-toolbar__content');\n\t\n\t return wrapper;\n\t };\n\t\n\t /**\n\t * Inline toolbar\n\t */\n\t draw.inlineToolbar = function () {\n\t\n\t var bar = document.createElement('DIV');\n\t\n\t bar.className += 'ce-toolbar-inline';\n\t\n\t return bar;\n\t };\n\t\n\t /**\n\t * Wrapper for inline toobar buttons\n\t */\n\t draw.inlineToolbarButtons = function () {\n\t\n\t var wrapper = document.createElement('DIV');\n\t\n\t wrapper.className += 'ce-toolbar-inline__buttons';\n\t\n\t return wrapper;\n\t };\n\t\n\t /**\n\t * For some actions\n\t */\n\t draw.inlineToolbarActions = function () {\n\t\n\t var wrapper = document.createElement('DIV');\n\t\n\t wrapper.className += 'ce-toolbar-inline__actions';\n\t\n\t return wrapper;\n\t };\n\t\n\t draw.inputForLink = function () {\n\t\n\t var input = document.createElement('INPUT');\n\t\n\t input.type = 'input';\n\t input.className += 'inputForLink';\n\t input.placeholder = 'Вставьте ссылку ...';\n\t input.setAttribute('form', 'defaultForm');\n\t\n\t input.setAttribute('autofocus', 'autofocus');\n\t\n\t return input;\n\t };\n\t\n\t /**\n\t * @todo Desc\n\t */\n\t draw.blockButtons = function () {\n\t\n\t var block = document.createElement('div');\n\t\n\t block.className += 'ce-toolbar__actions';\n\t\n\t return block;\n\t };\n\t\n\t /**\n\t * Block settings panel\n\t */\n\t draw.blockSettings = function () {\n\t\n\t var settings = document.createElement('div');\n\t\n\t settings.className += 'ce-settings';\n\t\n\t return settings;\n\t };\n\t\n\t draw.defaultSettings = function () {\n\t\n\t var div = document.createElement('div');\n\t\n\t div.classList.add('ce-settings_default');\n\t\n\t return div;\n\t };\n\t\n\t draw.pluginsSettings = function () {\n\t\n\t var div = document.createElement('div');\n\t\n\t div.classList.add('ce-settings_plugin');\n\t\n\t return div;\n\t };\n\t\n\t draw.plusButton = function () {\n\t\n\t var button = document.createElement('span');\n\t\n\t button.className = 'ce-toolbar__plus';\n\t // button.innerHTML = '';\n\t\n\t return button;\n\t };\n\t\n\t /**\n\t * Settings button in toolbar\n\t */\n\t draw.settingsButton = function () {\n\t\n\t var toggler = document.createElement('span');\n\t\n\t toggler.className = 'ce-toolbar__settings-btn';\n\t\n\t /** Toggler button*/\n\t toggler.innerHTML = '';\n\t\n\t return toggler;\n\t };\n\t\n\t /**\n\t * Redactor tools wrapper\n\t */\n\t\n\t draw.toolbox = function () {\n\t\n\t var wrapper = document.createElement('div');\n\t\n\t wrapper.className = 'ce-toolbar__tools';\n\t\n\t return wrapper;\n\t };\n\t\n\t /**\n\t * @protected\n\t *\n\t * Draws tool buttons for toolbox\n\t *\n\t * @param {String} type\n\t * @param {String} classname\n\t * @returns {Element}\n\t */\n\t draw.toolbarButton = function (type, classname) {\n\t\n\t var button = document.createElement('li'),\n\t toolIcon = document.createElement('i'),\n\t toolTitle = document.createElement('span');\n\t\n\t button.dataset.type = type;\n\t button.setAttribute('title', type);\n\t\n\t toolIcon.classList.add(classname);\n\t toolTitle.classList.add('ce_toolbar_tools--title');\n\t\n\t button.appendChild(toolIcon);\n\t button.appendChild(toolTitle);\n\t\n\t return button;\n\t };\n\t\n\t /**\n\t * @protected\n\t *\n\t * Draws tools for inline toolbar\n\t *\n\t * @param {String} type\n\t * @param {String} classname\n\t */\n\t draw.toolbarButtonInline = function (type, classname) {\n\t\n\t var button = document.createElement('BUTTON'),\n\t toolIcon = document.createElement('I');\n\t\n\t button.type = 'button';\n\t button.dataset.type = type;\n\t toolIcon.classList.add(classname);\n\t\n\t button.appendChild(toolIcon);\n\t\n\t return button;\n\t };\n\t\n\t /**\n\t * Redactor block\n\t */\n\t draw.block = function (tagName, content) {\n\t\n\t var node = document.createElement(tagName);\n\t\n\t node.innerHTML = content || '';\n\t\n\t return node;\n\t };\n\t\n\t /**\n\t * Creates Node with passed tagName and className\n\t * @param {string} tagName\n\t * @param {string} className\n\t * @param {object} properties - allow to assign properties\n\t */\n\t draw.node = function (tagName, className, properties) {\n\t\n\t var el = document.createElement(tagName);\n\t\n\t if (className) el.className = className;\n\t\n\t if (properties) {\n\t\n\t for (var name in properties) {\n\t\n\t el[name] = properties[name];\n\t }\n\t }\n\t\n\t return el;\n\t };\n\t\n\t /**\n\t * Unavailable plugin block\n\t */\n\t draw.unavailableBlock = function () {\n\t\n\t var wrapper = document.createElement('DIV');\n\t\n\t wrapper.classList.add('cdx-unavailable-block');\n\t\n\t return wrapper;\n\t };\n\t\n\t return draw;\n\t}({});\n\n/***/ },\n/* 14 */\n/***/ function(module, exports) {\n\n\t'use strict';\n\t\n\t/**\n\t * Codex Editor Caret Module\n\t *\n\t * @author Codex Team\n\t * @version 1.0\n\t */\n\t\n\tmodule.exports = function (caret) {\n\t\n\t var editor = codex.editor;\n\t\n\t /**\n\t * @var {int} InputIndex - editable element in DOM\n\t */\n\t caret.inputIndex = null;\n\t\n\t /**\n\t * @var {int} offset - caret position in a text node.\n\t */\n\t caret.offset = null;\n\t\n\t /**\n\t * @var {int} focusedNodeIndex - we get index of child node from first-level block\n\t */\n\t caret.focusedNodeIndex = null;\n\t\n\t /**\n\t * Creates Document Range and sets caret to the element.\n\t * @protected\n\t * @uses caret.save — if you need to save caret position\n\t * @param {Element} el - Changed Node.\n\t */\n\t caret.set = function (el, index, offset) {\n\t\n\t offset = offset || caret.offset || 0;\n\t index = index || caret.focusedNodeIndex || 0;\n\t\n\t var childs = el.childNodes,\n\t nodeToSet;\n\t\n\t if (childs.length === 0) {\n\t\n\t nodeToSet = el;\n\t } else {\n\t\n\t nodeToSet = childs[index];\n\t }\n\t\n\t /** If Element is INPUT */\n\t if (el.contentEditable != 'true') {\n\t\n\t el.focus();\n\t return;\n\t }\n\t\n\t if (editor.core.isDomNode(nodeToSet)) {\n\t\n\t nodeToSet = editor.content.getDeepestTextNodeFromPosition(nodeToSet, nodeToSet.childNodes.length);\n\t }\n\t\n\t var range = document.createRange(),\n\t selection = window.getSelection();\n\t\n\t window.setTimeout(function () {\n\t\n\t range.setStart(nodeToSet, offset);\n\t range.setEnd(nodeToSet, offset);\n\t\n\t selection.removeAllRanges();\n\t selection.addRange(range);\n\t\n\t editor.caret.saveCurrentInputIndex();\n\t }, 20);\n\t };\n\t\n\t /**\n\t * @protected\n\t * Updates index of input and saves it in caret object\n\t */\n\t caret.saveCurrentInputIndex = function () {\n\t\n\t /** Index of Input that we paste sanitized content */\n\t var selection = window.getSelection(),\n\t inputs = editor.state.inputs,\n\t focusedNode = selection.anchorNode,\n\t focusedNodeHolder;\n\t\n\t if (!focusedNode) {\n\t\n\t return;\n\t }\n\t\n\t /** Looking for parent contentEditable block */\n\t while (focusedNode.contentEditable != 'true') {\n\t\n\t focusedNodeHolder = focusedNode.parentNode;\n\t focusedNode = focusedNodeHolder;\n\t }\n\t\n\t /** Input index in DOM level */\n\t var editableElementIndex = 0;\n\t\n\t while (focusedNode != inputs[editableElementIndex]) {\n\t\n\t editableElementIndex++;\n\t }\n\t\n\t caret.inputIndex = editableElementIndex;\n\t };\n\t\n\t /**\n\t * Returns current input index (caret object)\n\t */\n\t caret.getCurrentInputIndex = function () {\n\t\n\t return caret.inputIndex;\n\t };\n\t\n\t /**\n\t * @param {int} index - index of first-level block after that we set caret into next input\n\t */\n\t caret.setToNextBlock = function (index) {\n\t\n\t var inputs = editor.state.inputs,\n\t nextInput = inputs[index + 1];\n\t\n\t if (!nextInput) {\n\t\n\t editor.core.log('We are reached the end');\n\t return;\n\t }\n\t\n\t /**\n\t * When new Block created or deleted content of input\n\t * We should add some text node to set caret\n\t */\n\t if (!nextInput.childNodes.length) {\n\t\n\t var emptyTextElement = document.createTextNode('');\n\t\n\t nextInput.appendChild(emptyTextElement);\n\t }\n\t\n\t editor.caret.inputIndex = index + 1;\n\t editor.caret.set(nextInput, 0, 0);\n\t editor.content.workingNodeChanged(nextInput);\n\t };\n\t\n\t /**\n\t * @param {int} index - index of target input.\n\t * Sets caret to input with this index\n\t */\n\t caret.setToBlock = function (index) {\n\t\n\t var inputs = editor.state.inputs,\n\t targetInput = inputs[index];\n\t\n\t if (!targetInput) {\n\t\n\t return;\n\t }\n\t\n\t /**\n\t * When new Block created or deleted content of input\n\t * We should add some text node to set caret\n\t */\n\t if (!targetInput.childNodes.length) {\n\t\n\t var emptyTextElement = document.createTextNode('');\n\t\n\t targetInput.appendChild(emptyTextElement);\n\t }\n\t\n\t editor.caret.inputIndex = index;\n\t editor.caret.set(targetInput, 0, 0);\n\t editor.content.workingNodeChanged(targetInput);\n\t };\n\t\n\t /**\n\t * @param {int} index - index of input\n\t */\n\t caret.setToPreviousBlock = function (index) {\n\t\n\t index = index || 0;\n\t\n\t var inputs = editor.state.inputs,\n\t previousInput = inputs[index - 1],\n\t lastChildNode,\n\t lengthOfLastChildNode,\n\t emptyTextElement;\n\t\n\t if (!previousInput) {\n\t\n\t editor.core.log('We are reached first node');\n\t return;\n\t }\n\t\n\t lastChildNode = editor.content.getDeepestTextNodeFromPosition(previousInput, previousInput.childNodes.length);\n\t lengthOfLastChildNode = lastChildNode.length;\n\t\n\t /**\n\t * When new Block created or deleted content of input\n\t * We should add some text node to set caret\n\t */\n\t if (!previousInput.childNodes.length) {\n\t\n\t emptyTextElement = document.createTextNode('');\n\t previousInput.appendChild(emptyTextElement);\n\t }\n\t editor.caret.inputIndex = index - 1;\n\t editor.caret.set(previousInput, previousInput.childNodes.length - 1, lengthOfLastChildNode);\n\t editor.content.workingNodeChanged(inputs[index - 1]);\n\t };\n\t\n\t caret.position = {\n\t\n\t atStart: function atStart() {\n\t\n\t var selection = window.getSelection(),\n\t anchorOffset = selection.anchorOffset,\n\t anchorNode = selection.anchorNode,\n\t firstLevelBlock = editor.content.getFirstLevelBlock(anchorNode),\n\t pluginsRender = firstLevelBlock.childNodes[0];\n\t\n\t if (!editor.core.isDomNode(anchorNode)) {\n\t\n\t anchorNode = anchorNode.parentNode;\n\t }\n\t\n\t var isFirstNode = anchorNode === pluginsRender.childNodes[0],\n\t isOffsetZero = anchorOffset === 0;\n\t\n\t return isFirstNode && isOffsetZero;\n\t },\n\t\n\t atTheEnd: function atTheEnd() {\n\t\n\t var selection = window.getSelection(),\n\t anchorOffset = selection.anchorOffset,\n\t anchorNode = selection.anchorNode;\n\t\n\t /** Caret is at the end of input */\n\t return !anchorNode || !anchorNode.length || anchorOffset === anchorNode.length;\n\t }\n\t };\n\t\n\t return caret;\n\t}({});\n\n/***/ },\n/* 15 */\n/***/ function(module, exports) {\n\n\t'use strict';\n\t\n\t/**\n\t * Codex Editor Notification Module\n\t *\n\t * @author Codex Team\n\t * @version 1.0\n\t */\n\t\n\tmodule.exports = function (notifications) {\n\t\n\t var editor = codex.editor;\n\t\n\t var queue = [];\n\t\n\t var addToQueue = function addToQueue(settings) {\n\t\n\t queue.push(settings);\n\t\n\t var index = 0;\n\t\n\t while (index < queue.length && queue.length > 5) {\n\t\n\t if (queue[index].type == 'confirm' || queue[index].type == 'prompt') {\n\t\n\t index++;\n\t continue;\n\t }\n\t\n\t queue[index].close();\n\t queue.splice(index, 1);\n\t }\n\t };\n\t\n\t notifications.createHolder = function () {\n\t\n\t var holder = editor.draw.node('DIV', 'cdx-notifications-block');\n\t\n\t editor.nodes.notifications = document.body.appendChild(holder);\n\t\n\t return holder;\n\t };\n\t\n\t /**\n\t * Error notificator. Shows block with message\n\t * @protected\n\t */\n\t notifications.errorThrown = function (errorMsg, event) {\n\t\n\t editor.notifications.notification({ message: 'This action is not available currently', type: event.type });\n\t };\n\t\n\t /**\n\t *\n\t * Appends notification\n\t *\n\t * settings = {\n\t * type - notification type (reserved types: alert, confirm, prompt). Just add class 'cdx-notification-'+type\n\t * message - notification message\n\t * okMsg - confirm button text (default - 'Ok')\n\t * cancelBtn - cancel button text (default - 'Cancel'). Only for confirm and prompt types\n\t * confirm - function-handler for ok button click\n\t * cancel - function-handler for cancel button click. Only for confirm and prompt types\n\t * time - time (in seconds) after which notification will close (default - 10s)\n\t * }\n\t *\n\t * @param settings\n\t */\n\t notifications.notification = function (constructorSettings) {\n\t\n\t /** Private vars and methods */\n\t var notification = null,\n\t cancel = null,\n\t type = null,\n\t confirm = null,\n\t inputField = null;\n\t\n\t var confirmHandler = function confirmHandler() {\n\t\n\t close();\n\t\n\t if (typeof confirm !== 'function') {\n\t\n\t return;\n\t }\n\t\n\t if (type == 'prompt') {\n\t\n\t confirm(inputField.value);\n\t return;\n\t }\n\t\n\t confirm();\n\t };\n\t\n\t var cancelHandler = function cancelHandler() {\n\t\n\t close();\n\t\n\t if (typeof cancel !== 'function') {\n\t\n\t return;\n\t }\n\t\n\t cancel();\n\t };\n\t\n\t /** Public methods */\n\t function create(settings) {\n\t\n\t if (!(settings && settings.message)) {\n\t\n\t editor.core.log('Can\\'t create notification. Message is missed');\n\t return;\n\t }\n\t\n\t settings.type = settings.type || 'alert';\n\t settings.time = settings.time * 1000 || 10000;\n\t\n\t var wrapper = editor.draw.node('DIV', 'cdx-notification'),\n\t message = editor.draw.node('DIV', 'cdx-notification__message'),\n\t input = editor.draw.node('INPUT', 'cdx-notification__input'),\n\t okBtn = editor.draw.node('SPAN', 'cdx-notification__ok-btn'),\n\t cancelBtn = editor.draw.node('SPAN', 'cdx-notification__cancel-btn');\n\t\n\t message.textContent = settings.message;\n\t okBtn.textContent = settings.okMsg || 'ОК';\n\t cancelBtn.textContent = settings.cancelMsg || 'Отмена';\n\t\n\t editor.listeners.add(okBtn, 'click', confirmHandler);\n\t editor.listeners.add(cancelBtn, 'click', cancelHandler);\n\t\n\t wrapper.appendChild(message);\n\t\n\t if (settings.type == 'prompt') {\n\t\n\t wrapper.appendChild(input);\n\t }\n\t\n\t wrapper.appendChild(okBtn);\n\t\n\t if (settings.type == 'prompt' || settings.type == 'confirm') {\n\t\n\t wrapper.appendChild(cancelBtn);\n\t }\n\t\n\t wrapper.classList.add('cdx-notification-' + settings.type);\n\t wrapper.dataset.type = settings.type;\n\t\n\t notification = wrapper;\n\t type = settings.type;\n\t confirm = settings.confirm;\n\t cancel = settings.cancel;\n\t inputField = input;\n\t\n\t if (settings.type != 'prompt' && settings.type != 'confirm') {\n\t\n\t window.setTimeout(close, settings.time);\n\t }\n\t };\n\t\n\t function send() {\n\t\n\t editor.nodes.notifications.appendChild(notification);\n\t inputField.focus();\n\t\n\t editor.nodes.notifications.classList.add('cdx-notification__notification-appending');\n\t\n\t window.setTimeout(function () {\n\t\n\t editor.nodes.notifications.classList.remove('cdx-notification__notification-appending');\n\t }, 100);\n\t\n\t addToQueue({ type: type, close: close });\n\t };\n\t\n\t function close() {\n\t\n\t notification.remove();\n\t };\n\t\n\t if (constructorSettings) {\n\t\n\t create(constructorSettings);\n\t send();\n\t }\n\t\n\t return {\n\t create: create,\n\t send: send,\n\t close: close\n\t };\n\t };\n\t\n\t notifications.clear = function () {\n\t\n\t editor.nodes.notifications.innerHTML = '';\n\t queue = [];\n\t };\n\t\n\t return notifications;\n\t}({});\n\n/***/ },\n/* 16 */\n/***/ function(module, exports) {\n\n\t\"use strict\";\n\t\n\t/**\n\t * Codex Editor Parser Module\n\t *\n\t * @author Codex Team\n\t * @version 1.1\n\t */\n\t\n\tmodule.exports = function (parser) {\n\t\n\t var editor = codex.editor;\n\t\n\t /** inserting text */\n\t parser.insertPastedContent = function (blockType, tag) {\n\t\n\t editor.content.insertBlock({\n\t type: blockType.type,\n\t block: blockType.render({\n\t text: tag.innerHTML\n\t })\n\t });\n\t };\n\t\n\t /**\n\t * Check DOM node for display style: separated block or child-view\n\t */\n\t parser.isFirstLevelBlock = function (node) {\n\t\n\t return node.nodeType == editor.core.nodeTypes.TAG && node.classList.contains(editor.ui.className.BLOCK_CLASSNAME);\n\t };\n\t\n\t return parser;\n\t}({});\n\n/***/ },\n/* 17 */\n/***/ function(module, exports, __webpack_require__) {\n\n\t'use strict';\n\t\n\t/**\n\t * Codex Sanitizer\n\t */\n\t\n\tmodule.exports = function (sanitizer) {\n\t\n\t /** HTML Janitor library */\n\t var janitor = __webpack_require__(18);\n\t\n\t /** Codex Editor */\n\t var editor = codex.editor;\n\t\n\t sanitizer.prepare = function () {\n\t\n\t if (editor.settings.sanitizer && !editor.core.isEmpty(editor.settings.sanitizer)) {\n\t\n\t Config.CUSTOM = editor.settings.sanitizer;\n\t }\n\t };\n\t\n\t /**\n\t * Basic config\n\t */\n\t var Config = {\n\t\n\t /** User configuration */\n\t CUSTOM: null,\n\t\n\t BASIC: {\n\t\n\t tags: {\n\t p: {},\n\t a: {\n\t href: true,\n\t target: '_blank',\n\t rel: 'nofollow'\n\t },\n\t i: {},\n\t b: {},\n\t strong: {},\n\t em: {},\n\t span: {}\n\t }\n\t }\n\t };\n\t\n\t sanitizer.Config = Config;\n\t\n\t /**\n\t *\n\t * @param userCustomConfig\n\t * @returns {*}\n\t * @private\n\t *\n\t * @description If developer uses editor's API, then he can customize sane restrictions.\n\t * Or, sane config can be defined globally in editors initialization. That config will be used everywhere\n\t * At least, if there is no config overrides, that API uses BASIC Default configation\n\t */\n\t var init_ = function init_(userCustomConfig) {\n\t\n\t var configuration = userCustomConfig || Config.CUSTOM || Config.BASIC;\n\t\n\t return new janitor(configuration);\n\t };\n\t\n\t /**\n\t * Cleans string from unwanted tags\n\t * @protected\n\t * @param {String} dirtyString - taint string\n\t * @param {Object} customConfig - allowed tags\n\t */\n\t sanitizer.clean = function (dirtyString, customConfig) {\n\t\n\t var janitorInstance = init_(customConfig);\n\t\n\t return janitorInstance.clean(dirtyString);\n\t };\n\t\n\t return sanitizer;\n\t}({});\n\n/***/ },\n/* 18 */\n/***/ function(module, exports, __webpack_require__) {\n\n\tvar __WEBPACK_AMD_DEFINE_FACTORY__, __WEBPACK_AMD_DEFINE_RESULT__;(function (root, factory) {\n\t if (true) {\n\t !(__WEBPACK_AMD_DEFINE_FACTORY__ = (factory), __WEBPACK_AMD_DEFINE_RESULT__ = (typeof __WEBPACK_AMD_DEFINE_FACTORY__ === 'function' ? (__WEBPACK_AMD_DEFINE_FACTORY__.call(exports, __webpack_require__, exports, module)) : __WEBPACK_AMD_DEFINE_FACTORY__), __WEBPACK_AMD_DEFINE_RESULT__ !== undefined && (module.exports = __WEBPACK_AMD_DEFINE_RESULT__));\n\t } else if (typeof exports === 'object') {\n\t module.exports = factory();\n\t } else {\n\t root.HTMLJanitor = factory();\n\t }\n\t}(this, function () {\n\t\n\t /**\n\t * @param {Object} config.tags Dictionary of allowed tags.\n\t * @param {boolean} config.keepNestedBlockElements Default false.\n\t */\n\t function HTMLJanitor(config) {\n\t\n\t var tagDefinitions = config['tags'];\n\t var tags = Object.keys(tagDefinitions);\n\t\n\t var validConfigValues = tags\n\t .map(function(k) { return typeof tagDefinitions[k]; })\n\t .every(function(type) { return type === 'object' || type === 'boolean' || type === 'function'; });\n\t\n\t if(!validConfigValues) {\n\t throw new Error(\"The configuration was invalid\");\n\t }\n\t\n\t this.config = config;\n\t }\n\t\n\t // TODO: not exhaustive?\n\t var blockElementNames = ['P', 'LI', 'TD', 'TH', 'DIV', 'H1', 'H2', 'H3', 'H4', 'H5', 'H6', 'PRE'];\n\t function isBlockElement(node) {\n\t return blockElementNames.indexOf(node.nodeName) !== -1;\n\t }\n\t\n\t var inlineElementNames = ['A', 'B', 'STRONG', 'I', 'EM', 'SUB', 'SUP', 'U', 'STRIKE'];\n\t function isInlineElement(node) {\n\t return inlineElementNames.indexOf(node.nodeName) !== -1;\n\t }\n\t\n\t HTMLJanitor.prototype.clean = function (html) {\n\t var sandbox = document.createElement('div');\n\t sandbox.innerHTML = html;\n\t\n\t this._sanitize(sandbox);\n\t\n\t return sandbox.innerHTML;\n\t };\n\t\n\t HTMLJanitor.prototype._sanitize = function (parentNode) {\n\t var treeWalker = createTreeWalker(parentNode);\n\t var node = treeWalker.firstChild();\n\t if (!node) { return; }\n\t\n\t do {\n\t // Ignore nodes that have already been sanitized\n\t if (node._sanitized) {\n\t continue;\n\t }\n\t\n\t if (node.nodeType === Node.TEXT_NODE) {\n\t // If this text node is just whitespace and the previous or next element\n\t // sibling is a block element, remove it\n\t // N.B.: This heuristic could change. Very specific to a bug with\n\t // `contenteditable` in Firefox: http://jsbin.com/EyuKase/1/edit?js,output\n\t // FIXME: make this an option?\n\t if (node.data.trim() === ''\n\t && ((node.previousElementSibling && isBlockElement(node.previousElementSibling))\n\t || (node.nextElementSibling && isBlockElement(node.nextElementSibling)))) {\n\t parentNode.removeChild(node);\n\t this._sanitize(parentNode);\n\t break;\n\t } else {\n\t continue;\n\t }\n\t }\n\t\n\t // Remove all comments\n\t if (node.nodeType === Node.COMMENT_NODE) {\n\t parentNode.removeChild(node);\n\t this._sanitize(parentNode);\n\t break;\n\t }\n\t\n\t var isInline = isInlineElement(node);\n\t var containsBlockElement;\n\t if (isInline) {\n\t containsBlockElement = Array.prototype.some.call(node.childNodes, isBlockElement);\n\t }\n\t\n\t // Block elements should not be nested (e.g.

  • ...); if\n\t // they are, we want to unwrap the inner block element.\n\t var isNotTopContainer = !! parentNode.parentNode;\n\t var isNestedBlockElement =\n\t isBlockElement(parentNode) &&\n\t isBlockElement(node) &&\n\t isNotTopContainer;\n\t\n\t var nodeName = node.nodeName.toLowerCase();\n\t\n\t var allowedAttrs = getAllowedAttrs(this.config, nodeName, node);\n\t\n\t var isInvalid = isInline && containsBlockElement;\n\t\n\t // Drop tag entirely according to the whitelist *and* if the markup\n\t // is invalid.\n\t if (isInvalid || shouldRejectNode(node, allowedAttrs)\n\t || (!this.config.keepNestedBlockElements && isNestedBlockElement)) {\n\t // Do not keep the inner text of SCRIPT/STYLE elements.\n\t if (! (node.nodeName === 'SCRIPT' || node.nodeName === 'STYLE')) {\n\t while (node.childNodes.length > 0) {\n\t parentNode.insertBefore(node.childNodes[0], node);\n\t }\n\t }\n\t parentNode.removeChild(node);\n\t\n\t this._sanitize(parentNode);\n\t break;\n\t }\n\t\n\t // Sanitize attributes\n\t for (var a = 0; a < node.attributes.length; a += 1) {\n\t var attr = node.attributes[a];\n\t\n\t if (shouldRejectAttr(attr, allowedAttrs, node)) {\n\t node.removeAttribute(attr.name);\n\t // Shift the array to continue looping.\n\t a = a - 1;\n\t }\n\t }\n\t\n\t // Sanitize children\n\t this._sanitize(node);\n\t\n\t // Mark node as sanitized so it's ignored in future runs\n\t node._sanitized = true;\n\t } while ((node = treeWalker.nextSibling()));\n\t };\n\t\n\t function createTreeWalker(node) {\n\t return document.createTreeWalker(node,\n\t NodeFilter.SHOW_TEXT | NodeFilter.SHOW_ELEMENT | NodeFilter.SHOW_COMMENT,\n\t null, false);\n\t }\n\t\n\t function getAllowedAttrs(config, nodeName, node){\n\t if (typeof config.tags[nodeName] === 'function') {\n\t return config.tags[nodeName](node);\n\t } else {\n\t return config.tags[nodeName];\n\t }\n\t }\n\t\n\t function shouldRejectNode(node, allowedAttrs){\n\t if (typeof allowedAttrs === 'undefined') {\n\t return true;\n\t } else if (typeof allowedAttrs === 'boolean') {\n\t return !allowedAttrs;\n\t }\n\t\n\t return false;\n\t }\n\t\n\t function shouldRejectAttr(attr, allowedAttrs, node){\n\t var attrName = attr.name.toLowerCase();\n\t\n\t if (allowedAttrs === true){\n\t return false;\n\t } else if (typeof allowedAttrs[attrName] === 'function'){\n\t return !allowedAttrs[attrName](attr.value, node);\n\t } else if (typeof allowedAttrs[attrName] === 'undefined'){\n\t return true;\n\t } else if (allowedAttrs[attrName] === false) {\n\t return true;\n\t } else if (typeof allowedAttrs[attrName] === 'string') {\n\t return (allowedAttrs[attrName] !== attr.value);\n\t }\n\t\n\t return false;\n\t }\n\t\n\t return HTMLJanitor;\n\t\n\t}));\n\n\n/***/ },\n/* 19 */\n/***/ function(module, exports) {\n\n\t\"use strict\";\n\t\n\t/**\n\t * Codex Editor Listeners module\n\t *\n\t * @author Codex Team\n\t * @version 1.0\n\t */\n\t\n\t/**\n\t * Module-decorator for event listeners assignment\n\t */\n\tmodule.exports = function (listeners) {\n\t\n\t var allListeners = [];\n\t\n\t /**\n\t * Search methods\n\t *\n\t * byElement, byType and byHandler returns array of suitable listeners\n\t * one and all takes element, eventType, and handler and returns first (all) suitable listener\n\t *\n\t */\n\t listeners.search = function () {\n\t\n\t var byElement = function byElement(element, context) {\n\t\n\t var listenersOnElement = [];\n\t\n\t context = context || allListeners;\n\t\n\t for (var i = 0; i < context.length; i++) {\n\t\n\t var listener = context[i];\n\t\n\t if (listener.element === element) {\n\t\n\t listenersOnElement.push(listener);\n\t }\n\t }\n\t\n\t return listenersOnElement;\n\t };\n\t\n\t var byType = function byType(eventType, context) {\n\t\n\t var listenersWithType = [];\n\t\n\t context = context || allListeners;\n\t\n\t for (var i = 0; i < context.length; i++) {\n\t\n\t var listener = context[i];\n\t\n\t if (listener.type === eventType) {\n\t\n\t listenersWithType.push(listener);\n\t }\n\t }\n\t\n\t return listenersWithType;\n\t };\n\t\n\t var byHandler = function byHandler(handler, context) {\n\t\n\t var listenersWithHandler = [];\n\t\n\t context = context || allListeners;\n\t\n\t for (var i = 0; i < context.length; i++) {\n\t\n\t var listener = context[i];\n\t\n\t if (listener.handler === handler) {\n\t\n\t listenersWithHandler.push(listener);\n\t }\n\t }\n\t\n\t return listenersWithHandler;\n\t };\n\t\n\t var one = function one(element, eventType, handler) {\n\t\n\t var result = allListeners;\n\t\n\t if (element) result = byElement(element, result);\n\t\n\t if (eventType) result = byType(eventType, result);\n\t\n\t if (handler) result = byHandler(handler, result);\n\t\n\t return result[0];\n\t };\n\t\n\t var all = function all(element, eventType, handler) {\n\t\n\t var result = allListeners;\n\t\n\t if (element) result = byElement(element, result);\n\t\n\t if (eventType) result = byType(eventType, result);\n\t\n\t if (handler) result = byHandler(handler, result);\n\t\n\t return result;\n\t };\n\t\n\t return {\n\t byElement: byElement,\n\t byType: byType,\n\t byHandler: byHandler,\n\t one: one,\n\t all: all\n\t };\n\t }();\n\t\n\t listeners.add = function (element, eventType, handler, isCapture) {\n\t\n\t element.addEventListener(eventType, handler, isCapture);\n\t\n\t var data = {\n\t element: element,\n\t type: eventType,\n\t handler: handler\n\t };\n\t\n\t var alreadyAddedListener = listeners.search.one(element, eventType, handler);\n\t\n\t if (!alreadyAddedListener) {\n\t\n\t allListeners.push(data);\n\t }\n\t };\n\t\n\t listeners.remove = function (element, eventType, handler) {\n\t\n\t element.removeEventListener(eventType, handler);\n\t\n\t var existingListeners = listeners.search.all(element, eventType, handler);\n\t\n\t for (var i = 0; i < existingListeners.length; i++) {\n\t\n\t var index = allListeners.indexOf(existingListeners[i]);\n\t\n\t if (index > 0) {\n\t\n\t allListeners.splice(index, 1);\n\t }\n\t }\n\t };\n\t\n\t listeners.removeAll = function () {\n\t\n\t allListeners.map(function (current) {\n\t\n\t listeners.remove(current.element, current.type, current.handler);\n\t });\n\t };\n\t\n\t listeners.get = function (element, eventType, handler) {\n\t\n\t return listeners.search.all(element, eventType, handler);\n\t };\n\t\n\t return listeners;\n\t}({});\n\n/***/ },\n/* 20 */\n/***/ function(module, exports) {\n\n\t'use strict';\n\t\n\tvar _typeof = typeof Symbol === \"function\" && typeof Symbol.iterator === \"symbol\" ? function (obj) { return typeof obj; } : function (obj) { return obj && typeof Symbol === \"function\" && obj.constructor === Symbol && obj !== Symbol.prototype ? \"symbol\" : typeof obj; };\n\t\n\t/**\n\t * Codex Editor Destroyer module\n\t *\n\t * @auhor Codex Team\n\t * @version 1.0\n\t */\n\t\n\tmodule.exports = function (destroyer) {\n\t\n\t var editor = codex.editor;\n\t\n\t destroyer.removeNodes = function () {\n\t\n\t editor.nodes.wrapper.remove();\n\t editor.nodes.notifications.remove();\n\t };\n\t\n\t destroyer.destroyPlugins = function () {\n\t\n\t for (var tool in editor.tools) {\n\t\n\t if (typeof editor.tools[tool].destroy === 'function') {\n\t\n\t editor.tools[tool].destroy();\n\t }\n\t }\n\t };\n\t\n\t destroyer.destroyScripts = function () {\n\t\n\t var scripts = document.getElementsByTagName('SCRIPT');\n\t\n\t for (var i = 0; i < scripts.length; i++) {\n\t\n\t if (scripts[i].id.indexOf(editor.scriptPrefix) + 1) {\n\t\n\t scripts[i].remove();\n\t i--;\n\t }\n\t }\n\t };\n\t\n\t /**\n\t * Delete editor data from webpage.\n\t * You should send settings argument with boolean flags:\n\t * @param settings.ui- remove redactor event listeners and DOM nodes\n\t * @param settings.scripts - remove redactor scripts from DOM\n\t * @param settings.plugins - remove plugin's objects\n\t * @param settings.core - remove editor core. You can remove core only if UI and scripts flags is true\n\t * }\n\t *\n\t */\n\t destroyer.destroy = function (settings) {\n\t\n\t if (!settings || (typeof settings === 'undefined' ? 'undefined' : _typeof(settings)) !== 'object') {\n\t\n\t return;\n\t }\n\t\n\t if (settings.ui) {\n\t\n\t destroyer.removeNodes();\n\t editor.listeners.removeAll();\n\t }\n\t\n\t if (settings.scripts) {\n\t\n\t destroyer.destroyScripts();\n\t }\n\t\n\t if (settings.plugins) {\n\t\n\t destroyer.destroyPlugins();\n\t }\n\t\n\t if (settings.ui && settings.scripts && settings.core) {\n\t\n\t delete codex.editor;\n\t }\n\t };\n\t\n\t return destroyer;\n\t}({});\n\n/***/ },\n/* 21 */\n/***/ function(module, exports) {\n\n\t'use strict';\n\t\n\t/**\n\t * Codex Editor Paste module\n\t *\n\t * @author Codex Team\n\t * @version 1.1.1\n\t */\n\t\n\tmodule.exports = function (paste) {\n\t\n\t var editor = codex.editor;\n\t\n\t var patterns = [];\n\t\n\t paste.prepare = function () {\n\t\n\t var tools = editor.tools;\n\t\n\t for (var tool in tools) {\n\t\n\t if (!tools[tool].renderOnPastePatterns || !Array.isArray(tools[tool].renderOnPastePatterns)) {\n\t\n\t continue;\n\t }\n\t\n\t tools[tool].renderOnPastePatterns.map(function (pattern) {\n\t\n\t patterns.push(pattern);\n\t });\n\t }\n\t\n\t return Promise.resolve();\n\t };\n\t\n\t /**\n\t * Saves data\n\t * @param event\n\t */\n\t paste.pasted = function (event) {\n\t\n\t var clipBoardData = event.clipboardData || window.clipboardData,\n\t content = clipBoardData.getData('Text');\n\t\n\t var result = analize(content);\n\t\n\t if (result) {\n\t\n\t event.preventDefault();\n\t event.stopImmediatePropagation();\n\t }\n\t\n\t return result;\n\t };\n\t\n\t /**\n\t * Analizes pated string and calls necessary method\n\t */\n\t\n\t var analize = function analize(string) {\n\t\n\t var result = false,\n\t content = editor.content.currentNode,\n\t plugin = content.dataset.tool;\n\t\n\t patterns.map(function (pattern) {\n\t\n\t var execArray = pattern.regex.exec(string),\n\t match = execArray && execArray[0];\n\t\n\t if (match && match === string.trim()) {\n\t\n\t /** current block is not empty */\n\t if (content.textContent.trim() && plugin == editor.settings.initialBlockPlugin) {\n\t\n\t pasteToNewBlock_();\n\t }\n\t\n\t pattern.callback(string, pattern);\n\t result = true;\n\t }\n\t });\n\t\n\t return result;\n\t };\n\t\n\t var pasteToNewBlock_ = function pasteToNewBlock_() {\n\t\n\t /** Create new initial block */\n\t editor.content.insertBlock({\n\t\n\t type: editor.settings.initialBlockPlugin,\n\t block: editor.tools[editor.settings.initialBlockPlugin].render({\n\t text: ''\n\t })\n\t\n\t }, false);\n\t };\n\t\n\t return paste;\n\t}({});\n\n/***/ }\n/******/ ]);\n\n\n// WEBPACK FOOTER //\n// codex-editor.js"," \t// The module cache\n \tvar installedModules = {};\n\n \t// The require function\n \tfunction __webpack_require__(moduleId) {\n\n \t\t// Check if module is in cache\n \t\tif(installedModules[moduleId])\n \t\t\treturn installedModules[moduleId].exports;\n\n \t\t// Create a new module (and put it into the cache)\n \t\tvar module = installedModules[moduleId] = {\n \t\t\texports: {},\n \t\t\tid: moduleId,\n \t\t\tloaded: false\n \t\t};\n\n \t\t// Execute the module function\n \t\tmodules[moduleId].call(module.exports, module, module.exports, __webpack_require__);\n\n \t\t// Flag the module as loaded\n \t\tmodule.loaded = true;\n\n \t\t// Return the exports of the module\n \t\treturn module.exports;\n \t}\n\n\n \t// expose the modules object (__webpack_modules__)\n \t__webpack_require__.m = modules;\n\n \t// expose the module cache\n \t__webpack_require__.c = installedModules;\n\n \t// __webpack_public_path__\n \t__webpack_require__.p = \"\";\n\n \t// Load entry module and return exports\n \treturn __webpack_require__(0);\n\n\n\n// WEBPACK FOOTER //\n// webpack/bootstrap acc8d353ac48698ee4f1","/**\n *\n * Codex Editor\n *\n * @author Codex Team\n */\n\nmodule.exports = (function (editor) {\n\n 'use strict';\n\n editor.version = VERSION;\n editor.scriptPrefix = 'cdx-script-';\n\n var init = function () {\n\n editor.core = require('./modules/core');\n editor.tools = require('./modules/tools');\n editor.ui = require('./modules/ui');\n editor.transport = require('./modules/transport');\n editor.renderer = require('./modules/renderer');\n editor.saver = require('./modules/saver');\n editor.content = require('./modules/content');\n editor.toolbar = require('./modules/toolbar/toolbar');\n editor.callback = require('./modules/callbacks');\n editor.draw = require('./modules/draw');\n editor.caret = require('./modules/caret');\n editor.notifications = require('./modules/notifications');\n editor.parser = require('./modules/parser');\n editor.sanitizer = require('./modules/sanitizer');\n editor.listeners = require('./modules/listeners');\n editor.destroyer = require('./modules/destroyer');\n editor.paste = require('./modules/paste');\n\n };\n\n /**\n * @public\n * holds initial settings\n */\n editor.settings = {\n tools : ['paragraph', 'header', 'picture', 'list', 'quote', 'code', 'twitter', 'instagram', 'smile'],\n holderId : 'codex-editor',\n\n // Type of block showing on empty editor\n initialBlockPlugin: 'paragraph'\n };\n\n /**\n * public\n *\n * Static nodes\n */\n editor.nodes = {\n holder : null,\n wrapper : null,\n toolbar : null,\n inlineToolbar : {\n wrapper : null,\n buttons : null,\n actions : null\n },\n toolbox : null,\n notifications : null,\n plusButton : null,\n showSettingsButton: null,\n showTrashButton : null,\n blockSettings : null,\n pluginSettings : null,\n defaultSettings : null,\n toolbarButtons : {}, // { type : DomEl, ... }\n redactor : null\n };\n\n /**\n * @public\n *\n * Output state\n */\n editor.state = {\n jsonOutput : [],\n blocks : [],\n inputs : []\n };\n\n /**\n * @public\n * Editor plugins\n */\n editor.tools = {};\n\n /**\n * Initialization\n * @uses Promise cEditor.core.prepare\n * @param {Object} userSettings\n * @param {Array} userSettings.tools list of plugins\n * @param {String} userSettings.holderId Element's id to append editor\n *\n * Load user defined tools\n * Tools must contain this important objects :\n * @param {String} type - this is a type of plugin. It can be used as plugin name\n * @param {String} iconClassname - this a icon in toolbar\n * @param {Object} make - what should plugin do, when it is clicked\n * @param {Object} appendCallback - callback after clicking\n * @param {Element} settings - what settings does it have\n * @param {Object} render - plugin get JSON, and should return HTML\n * @param {Object} save - plugin gets HTML content, returns JSON\n * @param {Boolean} displayInToolbox - will be displayed in toolbox. Default value is TRUE\n * @param {Boolean} enableLineBreaks - inserts new block or break lines. Default value is FALSE\n *\n * @example\n * - type : 'header',\n * - iconClassname : 'ce-icon-header',\n * - make : headerTool.make,\n * - appendCallback : headerTool.appendCallback,\n * - settings : headerTool.makeSettings(),\n * - render : headerTool.render,\n * - save : headerTool.save,\n * - displayInToolbox : true,\n * - enableLineBreaks : false\n */\n editor.start = function (userSettings) {\n\n init();\n\n editor.core.prepare(userSettings)\n\n // If all ok, make UI, bind events and parse initial-content\n .then(editor.ui.prepare)\n .then(editor.tools.prepare)\n .then(editor.sanitizer.prepare)\n .then(editor.paste.prepare)\n .then(editor.transport.prepare)\n .then(editor.renderer.makeBlocksFromData)\n .then(editor.ui.saveInputs)\n .catch(function (error) {\n\n editor.core.log('Initialization failed with error: %o', 'warn', error);\n\n });\n\n };\n\n return editor;\n\n})({});\n\n\n\n// WEBPACK FOOTER //\n// ./codex.js","/**\n * Codex Editor Core\n *\n * @author Codex Team\n * @version 1.1.3\n */\n\nmodule.exports = (function (core) {\n\n let editor = codex.editor;\n\n /**\n * @public\n *\n * Editor preparing method\n * @return Promise\n */\n core.prepare = function (userSettings) {\n\n return new Promise(function (resolve, reject) {\n\n if ( userSettings ) {\n\n editor.settings.tools = userSettings.tools || editor.settings.tools;\n\n }\n\n if (userSettings.data) {\n\n editor.state.blocks = userSettings.data;\n\n }\n\n if (userSettings.initialBlockPlugin) {\n\n editor.settings.initialBlockPlugin = userSettings.initialBlockPlugin;\n\n }\n\n if (userSettings.sanitizer) {\n\n editor.settings.sanitizer = userSettings.sanitizer;\n\n }\n\n editor.hideToolbar = userSettings.hideToolbar;\n\n editor.nodes.holder = document.getElementById(userSettings.holderId || editor.settings.holderId);\n\n if (typeof editor.nodes.holder === undefined || editor.nodes.holder === null) {\n\n reject(Error(\"Holder wasn't found by ID: #\" + userSettings.holderId));\n\n } else {\n\n resolve();\n\n }\n\n });\n\n };\n\n /**\n * Logging method\n * @param type = ['log', 'info', 'warn']\n */\n core.log = function (msg, type, arg) {\n\n type = type || 'log';\n\n if (!arg) {\n\n arg = msg || 'undefined';\n msg = '[codex-editor]: %o';\n\n } else {\n\n msg = '[codex-editor]: ' + msg;\n\n }\n\n try{\n\n if ( 'console' in window && window.console[ type ] ) {\n\n if ( arg ) window.console[ type ]( msg, arg );\n else window.console[ type ]( msg );\n\n }\n\n }catch(e) {}\n\n };\n\n /**\n * @protected\n *\n * Helper for insert one element after another\n */\n core.insertAfter = function (target, element) {\n\n target.parentNode.insertBefore(element, target.nextSibling);\n\n };\n\n /**\n * @const\n *\n * Readable DOM-node types map\n */\n core.nodeTypes = {\n TAG : 1,\n TEXT : 3,\n COMMENT : 8\n };\n\n /**\n * @const\n * Readable keys map\n */\n core.keys = { BACKSPACE: 8, TAB: 9, ENTER: 13, SHIFT: 16, CTRL: 17, ALT: 18, ESC: 27, SPACE: 32, LEFT: 37, UP: 38, DOWN: 40, RIGHT: 39, DELETE: 46, META: 91 };\n\n /**\n * @protected\n *\n * Check object for DOM node\n */\n core.isDomNode = function (el) {\n\n return el && typeof el === 'object' && el.nodeType && el.nodeType == this.nodeTypes.TAG;\n\n };\n\n /**\n * Checks passed object for emptiness\n * @require ES5 - Object.keys\n * @param {object}\n */\n core.isEmpty = function ( obj ) {\n\n return Object.keys(obj).length === 0;\n\n };\n\n /**\n * Native Ajax\n * @param {String} settings.url - request URL\n * @param {function} settings.beforeSend - returned value will be passed as context to the Success, Error and Progress callbacks\n * @param {function} settings.success\n * @param {function} settings.progress\n */\n core.ajax = function (settings) {\n\n if (!settings || !settings.url) {\n\n return;\n\n }\n\n var XMLHTTP = window.XMLHttpRequest ? new XMLHttpRequest() : new ActiveXObject('Microsoft.XMLHTTP'),\n encodedString,\n isFormData,\n prop;\n\n\n settings.async = true;\n settings.type = settings.type || 'GET';\n settings.data = settings.data || '';\n settings['content-type'] = settings['content-type'] || 'application/json; charset=utf-8';\n\n if (settings.type == 'GET' && settings.data) {\n\n settings.url = /\\?/.test(settings.url) ? settings.url + '&' + settings.data : settings.url + '?' + settings.data;\n\n } else {\n\n encodedString = '';\n for(prop in settings.data) {\n\n encodedString += (prop + '=' + encodeURIComponent(settings.data[prop]) + '&');\n\n }\n\n }\n\n if (settings.withCredentials) {\n\n XMLHTTP.withCredentials = true;\n\n }\n\n /**\n * Value returned in beforeSend funtion will be passed as context to the other response callbacks\n * If beforeSend returns false, AJAX will be blocked\n */\n let responseContext,\n beforeSendResult;\n\n if (typeof settings.beforeSend === 'function') {\n\n beforeSendResult = settings.beforeSend.call();\n\n if (beforeSendResult === false) {\n\n return;\n\n }\n\n }\n\n XMLHTTP.open( settings.type, settings.url, settings.async );\n\n /**\n * If we send FormData, we need no content-type header\n */\n isFormData = isFormData_(settings.data);\n\n if (!isFormData) {\n\n if (settings.type !== 'POST') {\n\n XMLHTTP.setRequestHeader('Content-type', settings['content-type']);\n\n } else {\n\n XMLHTTP.setRequestHeader('Content-type', 'application/x-www-form-urlencoded');\n\n }\n\n }\n\n XMLHTTP.setRequestHeader('X-Requested-With', 'XMLHttpRequest');\n\n responseContext = beforeSendResult || XMLHTTP;\n\n if (typeof settings.progress === 'function') {\n\n XMLHTTP.upload.onprogress = settings.progress.bind(responseContext);\n\n }\n\n XMLHTTP.onreadystatechange = function () {\n\n if (XMLHTTP.readyState === 4) {\n\n if (XMLHTTP.status === 200) {\n\n if (typeof settings.success === 'function') {\n\n settings.success.call(responseContext, XMLHTTP.responseText);\n\n }\n\n } else {\n\n if (typeof settings.error === 'function') {\n\n settings.error.call(responseContext, XMLHTTP.responseText, XMLHTTP.status);\n\n }\n\n }\n\n }\n\n };\n\n if (isFormData) {\n\n // Sending FormData\n XMLHTTP.send(settings.data);\n\n } else {\n\n // POST requests\n XMLHTTP.send(encodedString);\n\n }\n\n return XMLHTTP;\n\n };\n\n /**\n * Appends script to head of document\n * @return Promise\n */\n core.importScript = function (scriptPath, instanceName) {\n\n return new Promise(function (resolve, reject) {\n\n let script;\n\n /** Script is already loaded */\n if ( !instanceName ) {\n\n reject('Instance name is missed');\n\n } else if ( document.getElementById(editor.scriptPrefix + instanceName) ) {\n\n resolve(scriptPath);\n\n }\n\n script = document.createElement('SCRIPT');\n script.async = true;\n script.defer = true;\n script.id = editor.scriptPrefix + instanceName;\n\n script.onload = function () {\n\n resolve(scriptPath);\n\n };\n\n script.onerror = function () {\n\n reject(scriptPath);\n\n };\n\n script.src = scriptPath;\n document.head.appendChild(script);\n\n });\n\n };\n\n /**\n * Function for checking is it FormData object to send.\n * @param {Object} object to check\n * @return boolean\n */\n var isFormData_ = function (object) {\n\n return object instanceof FormData;\n\n };\n\n return core;\n\n})({});\n\n\n\n// WEBPACK FOOTER //\n// ./modules/core.js","/**\n* Module working with plugins\n*/\nmodule.exports = (function () {\n\n let editor = codex.editor;\n\n /**\n * Initialize plugins before using\n * Ex. Load scripts or call some internal methods\n * @return Promise\n */\n function prepare() {\n\n return new Promise(function (resolve_, reject_) {\n\n Promise.resolve()\n\n /**\n * Compose a sequence of plugins that requires preparation\n */\n .then(function () {\n\n let pluginsRequiresPreparation = [],\n allPlugins = editor.tools;\n\n for ( let pluginName in allPlugins ) {\n\n let plugin = allPlugins[pluginName];\n\n if (plugin.prepare && typeof plugin.prepare != 'function' || !plugin.prepare) {\n\n continue;\n\n }\n\n pluginsRequiresPreparation.push(plugin);\n\n }\n\n /**\n * If no one passed plugins requires preparation, finish prepare() and go ahead\n */\n if (!pluginsRequiresPreparation.length) {\n\n resolve_();\n\n }\n\n return pluginsRequiresPreparation;\n\n })\n\n /** Wait plugins while they prepares */\n .then(waitAllPluginsPreparation_)\n\n .then(function () {\n\n editor.core.log('Plugins loaded', 'info');\n resolve_();\n\n }).catch(function (error) {\n\n reject_(error);\n\n });\n\n });\n\n }\n\n /**\n * @param {array} plugins - list of tools that requires preparation\n * @return {Promise} resolved while all plugins will be ready or failed\n */\n function waitAllPluginsPreparation_(plugins) {\n\n /**\n * @calls allPluginsProcessed__ when all plugins prepared or failed\n */\n return new Promise (function (allPluginsProcessed__) {\n\n /**\n * pluck each element from queue\n * First, send resolved Promise as previous value\n * Each plugins \"prepare\" method returns a Promise, that's why\n * reduce current element will not be able to continue while can't get\n * a resolved Promise\n *\n * If last plugin is \"prepared\" then go to the next stage of initialization\n */\n plugins.reduce(function (previousValue, plugin, iteration) {\n\n return previousValue.then(function () {\n\n /**\n * Wait till plugins prepared\n * @calls pluginIsReady__ when plugin is ready or failed\n */\n return new Promise ( function (pluginIsReady__) {\n\n callPluginsPrepareMethod_( plugin )\n\n .then( pluginIsReady__ )\n .then( function () {\n\n plugin.available = true;\n\n })\n\n .catch(function (error) {\n\n editor.core.log(`Plugin «${plugin.type}» was not loaded. Preparation failed because %o`, 'warn', error);\n plugin.available = false;\n plugin.loadingMessage = error;\n\n /** Go ahead even some plugin has problems */\n pluginIsReady__();\n\n })\n\n .then(function () {\n\n /** If last plugin has problems then just ignore and continue */\n if (iteration == plugins.length - 1) {\n\n allPluginsProcessed__();\n\n }\n\n });\n\n });\n\n });\n\n }, Promise.resolve() );\n\n });\n\n }\n\n var callPluginsPrepareMethod_ = function (plugin) {\n\n return plugin.prepare( plugin.config || {} );\n\n };\n\n return {\n prepare: prepare\n };\n\n}());\n\n\n// WEBPACK FOOTER //\n// ./modules/tools.js","/**\n * Codex Editor UI module\n *\n * @author Codex Team\n * @version 1.2.0\n */\n\nmodule.exports = (function (ui) {\n\n let editor = codex.editor;\n\n /**\n * Basic editor classnames\n */\n ui.className = {\n\n /**\n * @const {string} BLOCK_CLASSNAME - redactor blocks name\n */\n BLOCK_CLASSNAME : 'ce-block',\n\n /**\n * @const {String} wrapper for plugins content\n */\n BLOCK_CONTENT : 'ce-block__content',\n\n /**\n * @const {String} BLOCK_STRETCHED - makes block stretched\n */\n BLOCK_STRETCHED : 'ce-block--stretched',\n\n /**\n * @const {String} BLOCK_HIGHLIGHTED - adds background\n */\n BLOCK_HIGHLIGHTED : 'ce-block--focused',\n\n /**\n * @const {String} - for all default settings\n */\n SETTINGS_ITEM : 'ce-settings__item'\n\n };\n\n /**\n * @protected\n *\n * Making main interface\n */\n ui.prepare = function () {\n\n return new Promise(function (resolve) {\n\n let wrapper = editor.draw.wrapper(),\n redactor = editor.draw.redactor(),\n toolbar = makeToolBar_();\n\n wrapper.appendChild(toolbar);\n wrapper.appendChild(redactor);\n\n /** Save created ui-elements to static nodes state */\n editor.nodes.wrapper = wrapper;\n editor.nodes.redactor = redactor;\n\n /** Append editor wrapper with redactor zone into holder */\n editor.nodes.holder.appendChild(wrapper);\n\n resolve();\n\n })\n\n /** Add toolbox tools */\n .then(addTools_)\n\n /** Make container for inline toolbar */\n .then(makeInlineToolbar_)\n\n /** Add inline toolbar tools */\n .then(addInlineToolbarTools_)\n\n /** Draw wrapper for notifications */\n .then(makeNotificationHolder_)\n\n /** Add eventlisteners to redactor elements */\n .then(bindEvents_)\n\n .catch( function () {\n\n editor.core.log(\"Can't draw editor interface\");\n\n });\n\n };\n\n /**\n * @private\n * Draws inline toolbar zone\n */\n var makeInlineToolbar_ = function () {\n\n var container = editor.draw.inlineToolbar();\n\n /** Append to redactor new inline block */\n editor.nodes.inlineToolbar.wrapper = container;\n\n /** Draw toolbar buttons */\n editor.nodes.inlineToolbar.buttons = editor.draw.inlineToolbarButtons();\n\n /** Buttons action or settings */\n editor.nodes.inlineToolbar.actions = editor.draw.inlineToolbarActions();\n\n /** Append to inline toolbar buttons as part of it */\n editor.nodes.inlineToolbar.wrapper.appendChild(editor.nodes.inlineToolbar.buttons);\n editor.nodes.inlineToolbar.wrapper.appendChild(editor.nodes.inlineToolbar.actions);\n\n editor.nodes.wrapper.appendChild(editor.nodes.inlineToolbar.wrapper);\n\n };\n\n var makeToolBar_ = function () {\n\n let toolbar = editor.draw.toolbar(),\n blockButtons = makeToolbarSettings_(),\n toolbarContent = makeToolbarContent_();\n\n /** Appending first-level block buttons */\n toolbar.appendChild(blockButtons);\n\n /** Append toolbarContent to toolbar */\n toolbar.appendChild(toolbarContent);\n\n /** Make toolbar global */\n editor.nodes.toolbar = toolbar;\n\n return toolbar;\n\n };\n\n var makeToolbarContent_ = function () {\n\n let toolbarContent = editor.draw.toolbarContent(),\n toolbox = editor.draw.toolbox(),\n plusButton = editor.draw.plusButton();\n\n /** Append plus button */\n toolbarContent.appendChild(plusButton);\n\n /** Appending toolbar tools */\n toolbarContent.appendChild(toolbox);\n\n /** Make Toolbox and plusButton global */\n editor.nodes.toolbox = toolbox;\n editor.nodes.plusButton = plusButton;\n\n return toolbarContent;\n\n };\n\n var makeToolbarSettings_ = function () {\n\n let blockSettings = editor.draw.blockSettings(),\n blockButtons = editor.draw.blockButtons(),\n defaultSettings = editor.draw.defaultSettings(),\n showSettingsButton = editor.draw.settingsButton(),\n showTrashButton = editor.toolbar.settings.makeRemoveBlockButton(),\n pluginSettings = editor.draw.pluginsSettings();\n\n /** Add default and plugins settings */\n blockSettings.appendChild(pluginSettings);\n blockSettings.appendChild(defaultSettings);\n\n /**\n * Make blocks buttons\n * This block contains settings button and remove block button\n */\n blockButtons.appendChild(showSettingsButton);\n blockButtons.appendChild(showTrashButton);\n blockButtons.appendChild(blockSettings);\n\n /** Make BlockSettings, PluginSettings, DefaultSettings global */\n editor.nodes.blockSettings = blockSettings;\n editor.nodes.pluginSettings = pluginSettings;\n editor.nodes.defaultSettings = defaultSettings;\n editor.nodes.showSettingsButton = showSettingsButton;\n editor.nodes.showTrashButton = showTrashButton;\n\n return blockButtons;\n\n };\n\n /** Draw notifications holder */\n var makeNotificationHolder_ = function () {\n\n /** Append block with notifications to the document */\n editor.nodes.notifications = editor.notifications.createHolder();\n\n };\n\n /**\n * @private\n * Append tools passed in editor.tools\n */\n var addTools_ = function () {\n\n var tool,\n toolName,\n toolButton;\n\n for ( toolName in editor.settings.tools ) {\n\n tool = editor.settings.tools[toolName];\n\n editor.tools[toolName] = tool;\n\n if (!tool.iconClassname && tool.displayInToolbox) {\n\n editor.core.log('Toolbar icon classname missed. Tool %o skipped', 'warn', toolName);\n continue;\n\n }\n\n if (typeof tool.render != 'function') {\n\n editor.core.log('render method missed. Tool %o skipped', 'warn', toolName);\n continue;\n\n }\n\n if (!tool.displayInToolbox) {\n\n continue;\n\n } else {\n\n /** if tools is for toolbox */\n toolButton = editor.draw.toolbarButton(toolName, tool.iconClassname);\n\n editor.nodes.toolbox.appendChild(toolButton);\n\n editor.nodes.toolbarButtons[toolName] = toolButton;\n\n }\n\n }\n\n };\n\n var addInlineToolbarTools_ = function () {\n\n var tools = {\n\n bold: {\n icon : 'ce-icon-bold',\n command : 'bold'\n },\n\n italic: {\n icon : 'ce-icon-italic',\n command : 'italic'\n },\n\n underline: {\n icon : 'ce-icon-underline',\n command : 'underline'\n },\n\n link: {\n icon : 'ce-icon-link',\n command : 'createLink'\n }\n };\n\n var toolButton,\n tool;\n\n for(var name in tools) {\n\n tool = tools[name];\n\n toolButton = editor.draw.toolbarButtonInline(name, tool.icon);\n\n editor.nodes.inlineToolbar.buttons.appendChild(toolButton);\n /**\n * Add callbacks to this buttons\n */\n editor.ui.setInlineToolbarButtonBehaviour(toolButton, tool.command);\n\n }\n\n };\n\n /**\n * @private\n * Bind editor UI events\n */\n var bindEvents_ = function () {\n\n editor.core.log('ui.bindEvents fired', 'info');\n\n // window.addEventListener('error', function (errorMsg, url, lineNumber) {\n // editor.notifications.errorThrown(errorMsg, event);\n // }, false );\n\n /** All keydowns on Document */\n editor.listeners.add(document, 'keydown', editor.callback.globalKeydown, false);\n\n /** All keydowns on Redactor zone */\n editor.listeners.add(editor.nodes.redactor, 'keydown', editor.callback.redactorKeyDown, false);\n\n /** All keydowns on Document */\n editor.listeners.add(document, 'keyup', editor.callback.globalKeyup, false );\n\n /**\n * Mouse click to radactor\n */\n editor.listeners.add(editor.nodes.redactor, 'click', editor.callback.redactorClicked, false );\n\n /**\n * Clicks to the Plus button\n */\n editor.listeners.add(editor.nodes.plusButton, 'click', editor.callback.plusButtonClicked, false);\n\n /**\n * Clicks to SETTINGS button in toolbar\n */\n editor.listeners.add(editor.nodes.showSettingsButton, 'click', editor.callback.showSettingsButtonClicked, false );\n\n /** Bind click listeners on toolbar buttons */\n for (var button in editor.nodes.toolbarButtons) {\n\n editor.listeners.add(editor.nodes.toolbarButtons[button], 'click', editor.callback.toolbarButtonClicked, false);\n\n }\n\n };\n\n ui.addBlockHandlers = function (block) {\n\n if (!block) return;\n\n /**\n * Block keydowns\n */\n editor.listeners.add(block, 'keydown', editor.callback.blockKeydown, false);\n\n /**\n * Pasting content from another source\n * We have two type of sanitization\n * First - uses deep-first search algorithm to get sub nodes,\n * sanitizes whole Block_content and replaces cleared nodes\n * This method is deprecated\n * Method is used in editor.callback.blockPaste(event)\n *\n * Secont - uses Mutation observer.\n * Observer \"observe\" DOM changes and send changings to callback.\n * Callback gets changed node, not whole Block_content.\n * Inserted or changed node, which we've gotten have been cleared and replaced with diry node\n *\n * Method is used in editor.callback.blockPasteViaSanitize(event)\n *\n * @uses html-janitor\n * @example editor.callback.blockPasteViaSanitize(event), the second method.\n *\n */\n editor.listeners.add(block, 'paste', editor.callback.blockPasteCallback, false);\n\n editor.listeners.add(block, 'mouseup', editor.toolbar.inline.show, false);\n\n };\n\n /** getting all contenteditable elements */\n ui.saveInputs = function () {\n\n var redactor = editor.nodes.redactor;\n\n editor.state.inputs = [];\n\n /** Save all inputs in global variable state */\n var inputs = redactor.querySelectorAll('[contenteditable], input, textarea');\n\n Array.prototype.map.call(inputs, function (current) {\n\n if (!current.type || current.type == 'text' || current.type == 'textarea') {\n\n editor.state.inputs.push(current);\n\n }\n\n });\n\n };\n\n /**\n * Adds first initial block on empty redactor\n */\n ui.addInitialBlock = function () {\n\n var initialBlockType = editor.settings.initialBlockPlugin,\n initialBlock;\n\n if ( !editor.tools[initialBlockType] ) {\n\n editor.core.log('Plugin %o was not implemented and can\\'t be used as initial block', 'warn', initialBlockType);\n return;\n\n }\n\n initialBlock = editor.tools[initialBlockType].render();\n\n initialBlock.setAttribute('data-placeholder', 'Расскажите свою историю...');\n\n editor.content.insertBlock({\n type : initialBlockType,\n block : initialBlock\n });\n\n editor.content.workingNodeChanged(initialBlock);\n\n };\n\n ui.setInlineToolbarButtonBehaviour = function (button, type) {\n\n editor.listeners.add(button, 'mousedown', function (event) {\n\n editor.toolbar.inline.toolClicked(event, type);\n\n }, false);\n\n };\n\n return ui;\n\n})({});\n\n\n\n// WEBPACK FOOTER //\n// ./modules/ui.js","/**\n *\n * Codex.Editor Transport Module\n *\n * @copyright 2017 Codex-Team\n * @version 1.2.0\n */\n\nmodule.exports = (function (transport) {\n\n let editor = codex.editor;\n\n\n /**\n * @private {Object} current XmlHttpRequest instance\n */\n var currentRequest = null;\n\n\n /**\n * @type {null} | {DOMElement} input - keeps input element in memory\n */\n transport.input = null;\n\n /**\n * @property {Object} arguments - keep plugin settings and defined callbacks\n */\n transport.arguments = null;\n\n /**\n * Prepares input element where will be files\n */\n transport.prepare = function () {\n\n let input = editor.draw.node( 'INPUT', '', { type : 'file' } );\n\n editor.listeners.add(input, 'change', editor.transport.fileSelected);\n editor.transport.input = input;\n\n };\n\n /** Clear input when files is uploaded */\n transport.clearInput = function () {\n\n /** Remove old input */\n transport.input = null;\n\n /** Prepare new one */\n transport.prepare();\n\n };\n\n /**\n * Callback for file selection\n * @param {Event} event\n */\n transport.fileSelected = function () {\n\n var input = this,\n i,\n files = input.files,\n formData = new FormData();\n\n if (editor.transport.arguments.multiple === true) {\n\n for ( i = 0; i < files.length; i++) {\n\n formData.append('files[]', files[i], files[i].name);\n\n }\n\n } else {\n\n formData.append('files', files[0], files[0].name);\n\n }\n\n currentRequest = editor.core.ajax({\n type : 'POST',\n data : formData,\n url : editor.transport.arguments.url,\n beforeSend : editor.transport.arguments.beforeSend,\n success : editor.transport.arguments.success,\n error : editor.transport.arguments.error,\n progress : editor.transport.arguments.progress\n });\n\n /** Clear input */\n transport.clearInput();\n\n };\n\n /**\n * Use plugin callbacks\n * @protected\n *\n * @param {Object} args - can have :\n * @param {String} args.url - fetch URL\n * @param {Function} args.beforeSend - function calls before sending ajax\n * @param {Function} args.success - success callback\n * @param {Function} args.error - on error handler\n * @param {Function} args.progress - xhr onprogress handler\n * @param {Boolean} args.multiple - allow select several files\n * @param {String} args.accept - adds accept attribute\n */\n transport.selectAndUpload = function (args) {\n\n transport.arguments = args;\n\n if ( args.multiple === true) {\n\n transport.input.setAttribute('multiple', 'multiple');\n\n }\n\n if ( args.accept ) {\n\n transport.input.setAttribute('accept', args.accept);\n\n }\n\n transport.input.click();\n\n };\n\n transport.abort = function () {\n\n currentRequest.abort();\n\n currentRequest = null;\n\n };\n\n return transport;\n\n})({});\n\n\n// WEBPACK FOOTER //\n// ./modules/transport.js","/**\n * Codex Editor Renderer Module\n *\n * @author Codex Team\n * @version 1.0\n */\n\nmodule.exports = (function (renderer) {\n\n let editor = codex.editor;\n\n /**\n * Asyncronously parses input JSON to redactor blocks\n */\n renderer.makeBlocksFromData = function () {\n\n /**\n * If redactor is empty, add first paragraph to start writing\n */\n if (editor.core.isEmpty(editor.state.blocks) || !editor.state.blocks.items.length) {\n\n editor.ui.addInitialBlock();\n return;\n\n }\n\n Promise.resolve()\n\n /** First, get JSON from state */\n .then(function () {\n\n return editor.state.blocks;\n\n })\n\n /** Then, start to iterate they */\n .then(editor.renderer.appendBlocks)\n\n /** Write log if something goes wrong */\n .catch(function (error) {\n\n editor.core.log('Error while parsing JSON: %o', 'error', error);\n\n });\n\n };\n\n /**\n * Parses JSON to blocks\n * @param {object} data\n * @return Primise -> nodeList\n */\n renderer.appendBlocks = function (data) {\n\n var blocks = data.items;\n\n /**\n * Sequence of one-by-one blocks appending\n * Uses to save blocks order after async-handler\n */\n var nodeSequence = Promise.resolve();\n\n for (var index = 0; index < blocks.length ; index++ ) {\n\n /** Add node to sequence at specified index */\n editor.renderer.appendNodeAtIndex(nodeSequence, blocks, index);\n\n }\n\n };\n\n /**\n * Append node at specified index\n */\n renderer.appendNodeAtIndex = function (nodeSequence, blocks, index) {\n\n /** We need to append node to sequence */\n nodeSequence\n\n /** first, get node async-aware */\n .then(function () {\n\n return editor.renderer.getNodeAsync(blocks, index);\n\n })\n\n /**\n * second, compose editor-block from JSON object\n */\n .then(editor.renderer.createBlockFromData)\n\n /**\n * now insert block to redactor\n */\n .then(function (blockData) {\n\n /**\n * blockData has 'block', 'type' and 'stretched' information\n */\n editor.content.insertBlock(blockData);\n\n /** Pass created block to next step */\n return blockData.block;\n\n })\n\n /** Log if something wrong with node */\n .catch(function (error) {\n\n editor.core.log('Node skipped while parsing because %o', 'error', error);\n\n });\n\n };\n\n /**\n * Asynchronously returns block data from blocksList by index\n * @return Promise to node\n */\n renderer.getNodeAsync = function (blocksList, index) {\n\n return Promise.resolve().then(function () {\n\n return {\n tool : blocksList[index],\n position : index\n };\n\n });\n\n };\n\n /**\n * Creates editor block by JSON-data\n *\n * @uses render method of each plugin\n *\n * @param {Object} toolData.tool\n * { header : {\n * text: '',\n * type: 'H3', ...\n * }\n * }\n * @param {Number} toolData.position - index in input-blocks array\n * @return {Object} with type and Element\n */\n renderer.createBlockFromData = function ( toolData ) {\n\n /** New parser */\n var block,\n tool = toolData.tool,\n pluginName = tool.type;\n\n /** Get first key of object that stores plugin name */\n // for (var pluginName in blockData) break;\n\n /** Check for plugin existance */\n if (!editor.tools[pluginName]) {\n\n throw Error(`Plugin «${pluginName}» not found`);\n\n }\n\n /** Check for plugin having render method */\n if (typeof editor.tools[pluginName].render != 'function') {\n\n throw Error(`Plugin «${pluginName}» must have «render» method`);\n\n }\n\n if ( editor.tools[pluginName].available === false ) {\n\n block = editor.draw.unavailableBlock();\n\n block.innerHTML = editor.tools[pluginName].loadingMessage;\n\n /**\n * Saver will extract data from initial block data by position in array\n */\n block.dataset.inputPosition = toolData.position;\n\n } else {\n\n /** New Parser */\n block = editor.tools[pluginName].render(tool.data);\n\n }\n\n /** is first-level block stretched */\n var stretched = editor.tools[pluginName].isStretched || false;\n\n /** Retrun type and block */\n return {\n type : pluginName,\n block : block,\n stretched : stretched\n };\n\n };\n\n return renderer;\n\n})({});\n\n\n// WEBPACK FOOTER //\n// ./modules/renderer.js","/**\n * Codex Editor Saver\n *\n * @author Codex Team\n * @version 1.0.2\n */\n\nmodule.exports = (function (saver) {\n\n let editor = codex.editor;\n\n /**\n * Saves blocks\n * @private\n */\n saver.saveBlocks = function () {\n\n /** Save html content of redactor to memory */\n editor.state.html = editor.nodes.redactor.innerHTML;\n\n /** Empty jsonOutput state */\n editor.state.jsonOutput = [];\n\n Promise.resolve()\n\n .then(function () {\n\n return editor.nodes.redactor.childNodes;\n\n })\n /** Making a sequence from separate blocks */\n .then(editor.saver.makeQueue)\n\n .then(function () {\n // editor.nodes.textarea.innerHTML = editor.state.html;\n })\n\n .catch( function (error) {\n\n editor.core.log(error);\n\n });\n\n };\n\n saver.makeQueue = function (blocks) {\n\n var queue = Promise.resolve();\n\n for(var index = 0; index < blocks.length; index++) {\n\n /** Add node to sequence at specified index */\n editor.saver.getBlockData(queue, blocks, index);\n\n }\n\n };\n\n /** Gets every block and makes From Data */\n saver.getBlockData = function (queue, blocks, index) {\n\n queue.then(function () {\n\n return editor.saver.getNodeAsync(blocks, index);\n\n })\n\n .then(editor.saver.makeFormDataFromBlocks);\n\n };\n\n\n /**\n * Asynchronously returns block data from blocksList by index\n * @return Promise to node\n */\n saver.getNodeAsync = function (blocksList, index) {\n\n return Promise.resolve().then(function () {\n\n return blocksList[index];\n\n });\n\n };\n\n saver.makeFormDataFromBlocks = function (block) {\n\n var pluginName = block.dataset.tool;\n\n /** Check for plugin existance */\n if (!editor.tools[pluginName]) {\n\n throw Error(`Plugin «${pluginName}» not found`);\n\n }\n\n /** Check for plugin having render method */\n if (typeof editor.tools[pluginName].save != 'function') {\n\n throw Error(`Plugin «${pluginName}» must have save method`);\n\n }\n\n /** Result saver */\n var blockContent = block.childNodes[0],\n pluginsContent = blockContent.childNodes[0],\n savedData,\n position,\n output;\n\n /** If plugin wasn't available then return data from cache */\n if ( editor.tools[pluginName].available === false ) {\n\n position = pluginsContent.dataset.inputPosition;\n savedData = codex.editor.state.blocks.items[position].data;\n\n } else {\n\n savedData = editor.tools[pluginName].save(pluginsContent);\n\n if (editor.tools[pluginName].validate) {\n\n var result = editor.tools[pluginName].validate(savedData);\n\n /**\n * Do not allow invalid data\n */\n if (!result)\n return;\n\n }\n\n }\n\n output = {\n type : pluginName,\n data : savedData\n };\n\n editor.state.jsonOutput.push(output);\n\n };\n\n return saver;\n\n})({});\n\n\n// WEBPACK FOOTER //\n// ./modules/saver.js","/**\n * Codex Editor Content Module\n * Works with DOM\n *\n * @module Codex Editor content module\n *\n * @author Codex Team\n * @version 1.3.13\n *\n * @description Module works with Elements that have been appended to the main DOM\n */\n\nmodule.exports = (function (content) {\n\n let editor = codex.editor;\n\n /**\n * Links to current active block\n * @type {null | Element}\n */\n content.currentNode = null;\n\n /**\n * clicked in redactor area\n * @type {null | Boolean}\n */\n content.editorAreaHightlighted = null;\n\n /**\n * @deprecated\n * Synchronizes redactor with original textarea\n */\n content.sync = function () {\n\n editor.core.log('syncing...');\n\n /**\n * Save redactor content to editor.state\n */\n editor.state.html = editor.nodes.redactor.innerHTML;\n\n };\n\n /**\n * Appends background to the block\n *\n * @description add CSS class to highlight visually first-level block area\n */\n content.markBlock = function () {\n\n editor.content.currentNode.classList.add(editor.ui.className.BLOCK_HIGHLIGHTED);\n\n };\n\n /**\n * Clear background\n *\n * @description clears styles that highlights block\n */\n content.clearMark = function () {\n\n if (editor.content.currentNode) {\n\n editor.content.currentNode.classList.remove(editor.ui.className.BLOCK_HIGHLIGHTED);\n\n }\n\n };\n\n /**\n * Finds first-level block\n *\n * @param {Element} node - selected or clicked in redactors area node\n * @protected\n *\n * @description looks for first-level block.\n * gets parent while node is not first-level\n */\n content.getFirstLevelBlock = function (node) {\n\n if (!editor.core.isDomNode(node)) {\n\n node = node.parentNode;\n\n }\n\n if (node === editor.nodes.redactor || node === document.body) {\n\n return null;\n\n } else {\n\n while(!node.classList.contains(editor.ui.className.BLOCK_CLASSNAME)) {\n\n node = node.parentNode;\n\n }\n\n return node;\n\n }\n\n };\n\n /**\n * Trigger this event when working node changed\n * @param {Element} targetNode - first-level of this node will be current\n * @protected\n *\n * @description If targetNode is first-level then we set it as current else we look for parents to find first-level\n */\n content.workingNodeChanged = function (targetNode) {\n\n /** Clear background from previous marked block before we change */\n editor.content.clearMark();\n\n if (!targetNode) {\n\n return;\n\n }\n\n this.currentNode = this.getFirstLevelBlock(targetNode);\n\n };\n\n /**\n * Replaces one redactor block with another\n * @protected\n * @param {Element} targetBlock - block to replace. Mostly currentNode.\n * @param {Element} newBlock\n * @param {string} newBlockType - type of new block; we need to store it to data-attribute\n *\n * [!] Function does not saves old block content.\n * You can get it manually and pass with newBlock.innerHTML\n */\n content.replaceBlock = function (targetBlock, newBlock) {\n\n if (!targetBlock || !newBlock) {\n\n editor.core.log('replaceBlock: missed params');\n return;\n\n }\n\n /** If target-block is not a frist-level block, then we iterate parents to find it */\n while(!targetBlock.classList.contains(editor.ui.className.BLOCK_CLASSNAME)) {\n\n targetBlock = targetBlock.parentNode;\n\n }\n\n /** Replacing */\n editor.nodes.redactor.replaceChild(newBlock, targetBlock);\n\n /**\n * Set new node as current\n */\n editor.content.workingNodeChanged(newBlock);\n\n /**\n * Add block handlers\n */\n editor.ui.addBlockHandlers(newBlock);\n\n /**\n * Save changes\n */\n editor.ui.saveInputs();\n\n };\n\n /**\n * @protected\n *\n * Inserts new block to redactor\n * Wrapps block into a DIV with BLOCK_CLASSNAME class\n *\n * @param blockData {object}\n * @param blockData.block {Element} element with block content\n * @param blockData.type {string} block plugin\n * @param needPlaceCaret {bool} pass true to set caret in new block\n *\n */\n content.insertBlock = function ( blockData, needPlaceCaret ) {\n\n var workingBlock = editor.content.currentNode,\n newBlockContent = blockData.block,\n blockType = blockData.type,\n isStretched = blockData.stretched;\n\n var newBlock = composeNewBlock_(newBlockContent, blockType, isStretched);\n\n if (workingBlock) {\n\n editor.core.insertAfter(workingBlock, newBlock);\n\n } else {\n\n /**\n * If redactor is empty, append as first child\n */\n editor.nodes.redactor.appendChild(newBlock);\n\n }\n\n /**\n * Block handler\n */\n editor.ui.addBlockHandlers(newBlock);\n\n /**\n * Set new node as current\n */\n editor.content.workingNodeChanged(newBlock);\n\n /**\n * Save changes\n */\n editor.ui.saveInputs();\n\n\n if ( needPlaceCaret ) {\n\n /**\n * If we don't know input index then we set default value -1\n */\n var currentInputIndex = editor.caret.getCurrentInputIndex() || -1;\n\n\n if (currentInputIndex == -1) {\n\n\n var editableElement = newBlock.querySelector('[contenteditable]'),\n emptyText = document.createTextNode('');\n\n editableElement.appendChild(emptyText);\n editor.caret.set(editableElement, 0, 0);\n\n editor.toolbar.move();\n editor.toolbar.showPlusButton();\n\n\n } else {\n\n if (currentInputIndex === editor.state.inputs.length - 1)\n return;\n\n /** Timeout for browsers execution */\n window.setTimeout(function () {\n\n /** Setting to the new input */\n editor.caret.setToNextBlock(currentInputIndex);\n editor.toolbar.move();\n editor.toolbar.open();\n\n }, 10);\n\n }\n\n }\n\n /**\n * Block is inserted, wait for new click that defined focusing on editors area\n * @type {boolean}\n */\n content.editorAreaHightlighted = false;\n\n };\n\n /**\n * Replaces blocks with saving content\n * @protected\n * @param {Element} noteToReplace\n * @param {Element} newNode\n * @param {Element} blockType\n */\n content.switchBlock = function (blockToReplace, newBlock, tool) {\n\n tool = tool || editor.content.currentNode.dataset.tool;\n var newBlockComposed = composeNewBlock_(newBlock, tool);\n\n /** Replacing */\n editor.content.replaceBlock(blockToReplace, newBlockComposed);\n\n /** Save new Inputs when block is changed */\n editor.ui.saveInputs();\n\n };\n\n /**\n * Iterates between child noted and looking for #text node on deepest level\n * @protected\n *\n * @param {Element} block - node where find\n * @param {int} postiton - starting postion\n * Example: childNodex.length to find from the end\n * or 0 to find from the start\n * @return {Text} block\n * @uses DFS\n */\n content.getDeepestTextNodeFromPosition = function (block, position) {\n\n /**\n * Clear Block from empty and useless spaces with trim.\n * Such nodes we should remove\n */\n var blockChilds = block.childNodes,\n index,\n node,\n text;\n\n for(index = 0; index < blockChilds.length; index++) {\n\n node = blockChilds[index];\n\n if (node.nodeType == editor.core.nodeTypes.TEXT) {\n\n text = node.textContent.trim();\n\n /** Text is empty. We should remove this child from node before we start DFS\n * decrease the quantity of childs.\n */\n if (text === '') {\n\n block.removeChild(node);\n position--;\n\n }\n\n }\n\n }\n\n if (block.childNodes.length === 0) {\n\n return document.createTextNode('');\n\n }\n\n /** Setting default position when we deleted all empty nodes */\n if ( position < 0 )\n position = 1;\n\n var lookingFromStart = false;\n\n /** For looking from START */\n if (position === 0) {\n\n lookingFromStart = true;\n position = 1;\n\n }\n\n while ( position ) {\n\n /** initial verticle of node. */\n if ( lookingFromStart ) {\n\n block = block.childNodes[0];\n\n } else {\n\n block = block.childNodes[position - 1];\n\n }\n\n if ( block.nodeType == editor.core.nodeTypes.TAG ) {\n\n position = block.childNodes.length;\n\n } else if (block.nodeType == editor.core.nodeTypes.TEXT ) {\n\n position = 0;\n\n }\n\n }\n\n return block;\n\n };\n\n /**\n * @private\n * @param {Element} block - current plugins render\n * @param {String} tool - plugins name\n * @param {Boolean} isStretched - make stretched block or not\n *\n * @description adds necessary information to wrap new created block by first-level holder\n */\n var composeNewBlock_ = function (block, tool, isStretched) {\n\n var newBlock = editor.draw.node('DIV', editor.ui.className.BLOCK_CLASSNAME, {}),\n blockContent = editor.draw.node('DIV', editor.ui.className.BLOCK_CONTENT, {});\n\n blockContent.appendChild(block);\n newBlock.appendChild(blockContent);\n\n if (isStretched) {\n\n blockContent.classList.add(editor.ui.className.BLOCK_STRETCHED);\n\n }\n\n newBlock.dataset.tool = tool;\n return newBlock;\n\n };\n\n /**\n * Returns Range object of current selection\n * @protected\n */\n content.getRange = function () {\n\n var selection = window.getSelection().getRangeAt(0);\n\n return selection;\n\n };\n\n /**\n * Divides block in two blocks (after and before caret)\n *\n * @protected\n * @param {int} inputIndex - target input index\n *\n * @description splits current input content to the separate blocks\n * When enter is pressed among the words, that text will be splited.\n */\n content.splitBlock = function (inputIndex) {\n\n var selection = window.getSelection(),\n anchorNode = selection.anchorNode,\n anchorNodeText = anchorNode.textContent,\n caretOffset = selection.anchorOffset,\n textBeforeCaret,\n textNodeBeforeCaret,\n textAfterCaret,\n textNodeAfterCaret;\n\n var currentBlock = editor.content.currentNode.querySelector('[contentEditable]');\n\n\n textBeforeCaret = anchorNodeText.substring(0, caretOffset);\n textAfterCaret = anchorNodeText.substring(caretOffset);\n\n textNodeBeforeCaret = document.createTextNode(textBeforeCaret);\n\n if (textAfterCaret) {\n\n textNodeAfterCaret = document.createTextNode(textAfterCaret);\n\n }\n\n var previousChilds = [],\n nextChilds = [],\n reachedCurrent = false;\n\n if (textNodeAfterCaret) {\n\n nextChilds.push(textNodeAfterCaret);\n\n }\n\n for ( var i = 0, child; !!(child = currentBlock.childNodes[i]); i++) {\n\n if ( child != anchorNode ) {\n\n if ( !reachedCurrent ) {\n\n previousChilds.push(child);\n\n } else {\n\n nextChilds.push(child);\n\n }\n\n } else {\n\n reachedCurrent = true;\n\n }\n\n }\n\n /** Clear current input */\n editor.state.inputs[inputIndex].innerHTML = '';\n\n /**\n * Append all childs founded before anchorNode\n */\n var previousChildsLength = previousChilds.length;\n\n for(i = 0; i < previousChildsLength; i++) {\n\n editor.state.inputs[inputIndex].appendChild(previousChilds[i]);\n\n }\n\n editor.state.inputs[inputIndex].appendChild(textNodeBeforeCaret);\n\n /**\n * Append text node which is after caret\n */\n var nextChildsLength = nextChilds.length,\n newNode = document.createElement('div');\n\n for(i = 0; i < nextChildsLength; i++) {\n\n newNode.appendChild(nextChilds[i]);\n\n }\n\n newNode = newNode.innerHTML;\n\n /** This type of block creates when enter is pressed */\n var NEW_BLOCK_TYPE = editor.settings.initialBlockPlugin;\n\n /**\n * Make new paragraph with text after caret\n */\n editor.content.insertBlock({\n type : NEW_BLOCK_TYPE,\n block : editor.tools[NEW_BLOCK_TYPE].render({\n text : newNode\n })\n }, true );\n\n };\n\n /**\n * Merges two blocks — current and target\n * If target index is not exist, then previous will be as target\n *\n * @protected\n * @param {int} currentInputIndex\n * @param {int} targetInputIndex\n *\n * @description gets two inputs indexes and merges into one\n */\n content.mergeBlocks = function (currentInputIndex, targetInputIndex) {\n\n /** If current input index is zero, then prevent method execution */\n if (currentInputIndex === 0) {\n\n return;\n\n }\n\n var targetInput,\n currentInputContent = editor.state.inputs[currentInputIndex].innerHTML;\n\n if (!targetInputIndex) {\n\n targetInput = editor.state.inputs[currentInputIndex - 1];\n\n } else {\n\n targetInput = editor.state.inputs[targetInputIndex];\n\n }\n\n targetInput.innerHTML += currentInputContent;\n\n };\n\n /**\n * Iterates all right siblings and parents, which has right siblings\n * while it does not reached the first-level block\n *\n * @param {Element} node\n * @return {boolean}\n */\n content.isLastNode = function (node) {\n\n // console.log('погнали перебор родителей');\n\n var allChecked = false;\n\n while ( !allChecked ) {\n\n // console.log('Смотрим на %o', node);\n // console.log('Проверим, пустые ли соседи справа');\n\n if ( !allSiblingsEmpty_(node) ) {\n\n // console.log('Есть непустые соседи. Узел не последний. Выходим.');\n return false;\n\n }\n\n node = node.parentNode;\n\n /**\n * Проверяем родителей до тех пор, пока не найдем блок первого уровня\n */\n if ( node.classList.contains(editor.ui.className.BLOCK_CONTENT) ) {\n\n allChecked = true;\n\n }\n\n }\n\n return true;\n\n };\n\n /**\n * Checks if all element right siblings is empty\n * @param node\n */\n var allSiblingsEmpty_ = function (node) {\n\n /**\n * Нужно убедиться, что после пустого соседа ничего нет\n */\n var sibling = node.nextSibling;\n\n while ( sibling ) {\n\n if (sibling.textContent.length) {\n\n return false;\n\n }\n\n sibling = sibling.nextSibling;\n\n }\n\n return true;\n\n };\n\n /**\n * @public\n *\n * @param [String] htmlString - html content as string\n * @return {string} - html content as string\n */\n content.wrapTextWithParagraphs = function (htmlString) {\n\n var wrapper = document.createElement('DIV'),\n newWrapper = document.createElement('DIV'),\n i,\n paragraph,\n firstLevelBlocks = ['DIV', 'P'],\n blockTyped,\n node;\n\n /**\n * Make HTML Element to Wrap Text\n * It allows us to work with input data as HTML content\n */\n wrapper.innerHTML = htmlString;\n paragraph = document.createElement('P');\n\n for (i = 0; i < wrapper.childNodes.length; i++) {\n\n node = wrapper.childNodes[i];\n\n blockTyped = firstLevelBlocks.indexOf(node.tagName) != -1;\n\n /**\n * If node is first-levet\n * we add this node to our new wrapper\n */\n if ( blockTyped ) {\n\n /**\n * If we had splitted inline nodes to paragraph before\n */\n if ( paragraph.childNodes.length ) {\n\n newWrapper.appendChild(paragraph.cloneNode(true));\n\n /** empty paragraph */\n paragraph = null;\n paragraph = document.createElement('P');\n\n }\n\n newWrapper.appendChild(node.cloneNode(true));\n\n } else {\n\n /** Collect all inline nodes to one as paragraph */\n paragraph.appendChild(node.cloneNode(true));\n\n /** if node is last we should append this node to paragraph and paragraph to new wrapper */\n if ( i == wrapper.childNodes.length - 1 ) {\n\n newWrapper.appendChild(paragraph.cloneNode(true));\n\n }\n\n }\n\n }\n\n return newWrapper.innerHTML;\n\n };\n\n /**\n * Finds closest Contenteditable parent from Element\n * @param {Element} node element looking from\n * @return {Element} node contenteditable\n */\n content.getEditableParent = function (node) {\n\n while (node && node.contentEditable != 'true') {\n\n node = node.parentNode;\n\n }\n\n return node;\n\n };\n\n return content;\n\n})({});\n\n\n// WEBPACK FOOTER //\n// ./modules/content.js","/**\n * Codex Editor toolbar module\n *\n * Contains:\n * - Inline toolbox\n * - Toolbox within plus button\n * - Settings section\n *\n * @author Codex Team\n * @version 1.0\n */\n\nmodule.exports = (function (toolbar) {\n\n let editor = codex.editor;\n\n toolbar.settings = require('./settings');\n toolbar.inline = require('./inline');\n toolbar.toolbox = require('./toolbox');\n\n /**\n * Margin between focused node and toolbar\n */\n toolbar.defaultToolbarHeight = 49;\n\n toolbar.defaultOffset = 34;\n\n toolbar.opened = false;\n\n toolbar.current = null;\n\n /**\n * @protected\n */\n toolbar.open = function () {\n\n if (editor.hideToolbar) {\n\n return;\n\n }\n\n editor.nodes.toolbar.classList.add('opened');\n this.opened = true;\n\n };\n\n /**\n * @protected\n */\n toolbar.close = function () {\n\n editor.nodes.toolbar.classList.remove('opened');\n\n toolbar.opened = false;\n toolbar.current = null;\n\n for (var button in editor.nodes.toolbarButtons) {\n\n editor.nodes.toolbarButtons[button].classList.remove('selected');\n\n }\n\n /** Close toolbox when toolbar is not displayed */\n editor.toolbar.toolbox.close();\n editor.toolbar.settings.close();\n\n };\n\n toolbar.toggle = function () {\n\n if ( !this.opened ) {\n\n this.open();\n\n } else {\n\n this.close();\n\n }\n\n };\n\n toolbar.hidePlusButton = function () {\n\n editor.nodes.plusButton.classList.add('hide');\n\n };\n\n toolbar.showPlusButton = function () {\n\n editor.nodes.plusButton.classList.remove('hide');\n\n };\n\n /**\n * Moving toolbar to the specified node\n */\n toolbar.move = function () {\n\n /** Close Toolbox when we move toolbar */\n editor.toolbar.toolbox.close();\n\n if (!editor.content.currentNode) {\n\n return;\n\n }\n\n var newYCoordinate = editor.content.currentNode.offsetTop - (editor.toolbar.defaultToolbarHeight / 2) + editor.toolbar.defaultOffset;\n\n editor.nodes.toolbar.style.transform = `translate3D(0, ${Math.floor(newYCoordinate)}px, 0)`;\n\n /** Close trash actions */\n editor.toolbar.settings.hideRemoveActions();\n\n };\n\n return toolbar;\n\n})({});\n\n\n// WEBPACK FOOTER //\n// ./modules/toolbar/toolbar.js","/**\n * Toolbar settings\n *\n * @version 1.0.4\n */\n\nmodule.exports = (function (settings) {\n\n let editor = codex.editor;\n\n settings.opened = false;\n\n settings.setting = null;\n settings.actions = null;\n\n /**\n * Append and open settings\n */\n settings.open = function (toolType) {\n\n /**\n * Append settings content\n * It's stored in tool.settings\n */\n if (!editor.tools[toolType] || !editor.tools[toolType].makeSettings ) {\n\n editor.core.log(`Plugin «${toolType}» has no settings`, 'warn');\n editor.nodes.pluginSettings.innerHTML = `Плагин «${toolType}» не имеет настроек`;\n\n } else {\n\n /**\n * Draw settings block\n */\n var settingsBlock = editor.tools[toolType].makeSettings();\n\n editor.nodes.pluginSettings.appendChild(settingsBlock);\n\n }\n\n /** Open settings block */\n editor.nodes.blockSettings.classList.add('opened');\n this.opened = true;\n\n };\n\n /**\n * Close and clear settings\n */\n settings.close = function () {\n\n editor.nodes.blockSettings.classList.remove('opened');\n editor.nodes.pluginSettings.innerHTML = '';\n\n this.opened = false;\n\n };\n\n /**\n * @param {string} toolType - plugin type\n */\n settings.toggle = function ( toolType ) {\n\n if ( !this.opened ) {\n\n this.open(toolType);\n\n } else {\n\n this.close();\n\n }\n\n };\n\n /**\n * Here we will draw buttons and add listeners to components\n */\n settings.makeRemoveBlockButton = function () {\n\n var removeBlockWrapper = editor.draw.node('SPAN', 'ce-toolbar__remove-btn', {}),\n settingButton = editor.draw.node('SPAN', 'ce-toolbar__remove-setting', { innerHTML : '' }),\n actionWrapper = editor.draw.node('DIV', 'ce-toolbar__remove-confirmation', {}),\n confirmAction = editor.draw.node('DIV', 'ce-toolbar__remove-confirm', { textContent : 'Удалить блок' }),\n cancelAction = editor.draw.node('DIV', 'ce-toolbar__remove-cancel', { textContent : 'Отмена' });\n\n editor.listeners.add(settingButton, 'click', editor.toolbar.settings.removeButtonClicked, false);\n\n editor.listeners.add(confirmAction, 'click', editor.toolbar.settings.confirmRemovingRequest, false);\n\n editor.listeners.add(cancelAction, 'click', editor.toolbar.settings.cancelRemovingRequest, false);\n\n actionWrapper.appendChild(confirmAction);\n actionWrapper.appendChild(cancelAction);\n\n removeBlockWrapper.appendChild(settingButton);\n removeBlockWrapper.appendChild(actionWrapper);\n\n /** Save setting */\n editor.toolbar.settings.setting = settingButton;\n editor.toolbar.settings.actions = actionWrapper;\n\n return removeBlockWrapper;\n\n };\n\n settings.removeButtonClicked = function () {\n\n var action = editor.toolbar.settings.actions;\n\n if (action.classList.contains('opened')) {\n\n editor.toolbar.settings.hideRemoveActions();\n\n } else {\n\n editor.toolbar.settings.showRemoveActions();\n\n }\n\n editor.toolbar.toolbox.close();\n editor.toolbar.settings.close();\n\n };\n\n settings.cancelRemovingRequest = function () {\n\n editor.toolbar.settings.actions.classList.remove('opened');\n\n };\n\n settings.confirmRemovingRequest = function () {\n\n var currentBlock = editor.content.currentNode,\n firstLevelBlocksCount;\n\n currentBlock.remove();\n\n firstLevelBlocksCount = editor.nodes.redactor.childNodes.length;\n\n /**\n * If all blocks are removed\n */\n if (firstLevelBlocksCount === 0) {\n\n /** update currentNode variable */\n editor.content.currentNode = null;\n\n /** Inserting new empty initial block */\n editor.ui.addInitialBlock();\n\n }\n\n editor.ui.saveInputs();\n\n editor.toolbar.close();\n\n };\n\n settings.showRemoveActions = function () {\n\n editor.toolbar.settings.actions.classList.add('opened');\n\n };\n\n settings.hideRemoveActions = function () {\n\n editor.toolbar.settings.actions.classList.remove('opened');\n\n };\n\n return settings;\n\n})({});\n\n\n// WEBPACK FOOTER //\n// ./modules/toolbar/settings.js","/**\n * Inline toolbar\n *\n * Contains from tools:\n * Bold, Italic, Underline and Anchor\n *\n * @author Codex Team\n * @version 1.0\n */\n\nmodule.exports = (function (inline) {\n\n let editor = codex.editor;\n\n inline.buttonsOpened = null;\n inline.actionsOpened = null;\n inline.wrappersOffset = null;\n\n /**\n * saving selection that need for execCommand for styling\n *\n */\n inline.storedSelection = null;\n\n /**\n * @protected\n *\n * Open inline toobar\n */\n inline.show = function () {\n\n var currentNode = editor.content.currentNode,\n tool = currentNode.dataset.tool,\n plugin;\n\n /**\n * tool allowed to open inline toolbar\n */\n plugin = editor.tools[tool];\n\n if (!plugin.showInlineToolbar)\n return;\n\n var selectedText = inline.getSelectionText(),\n toolbar = editor.nodes.inlineToolbar.wrapper;\n\n if (selectedText.length > 0) {\n\n /** Move toolbar and open */\n editor.toolbar.inline.move();\n\n /** Open inline toolbar */\n toolbar.classList.add('opened');\n\n /** show buttons of inline toolbar */\n editor.toolbar.inline.showButtons();\n\n }\n\n };\n\n /**\n * @protected\n *\n * Closes inline toolbar\n */\n inline.close = function () {\n\n var toolbar = editor.nodes.inlineToolbar.wrapper;\n\n toolbar.classList.remove('opened');\n\n };\n\n /**\n * @private\n *\n * Moving toolbar\n */\n inline.move = function () {\n\n if (!this.wrappersOffset) {\n\n this.wrappersOffset = this.getWrappersOffset();\n\n }\n\n var coords = this.getSelectionCoords(),\n defaultOffset = 0,\n toolbar = editor.nodes.inlineToolbar.wrapper,\n newCoordinateX,\n newCoordinateY;\n\n if (toolbar.offsetHeight === 0) {\n\n defaultOffset = 40;\n\n }\n\n newCoordinateX = coords.x - this.wrappersOffset.left;\n newCoordinateY = coords.y + window.scrollY - this.wrappersOffset.top - defaultOffset - toolbar.offsetHeight;\n\n toolbar.style.transform = `translate3D(${Math.floor(newCoordinateX)}px, ${Math.floor(newCoordinateY)}px, 0)`;\n\n /** Close everything */\n editor.toolbar.inline.closeButtons();\n editor.toolbar.inline.closeAction();\n\n };\n\n /**\n * @private\n *\n * Tool Clicked\n */\n\n inline.toolClicked = function (event, type) {\n\n /**\n * For simple tools we use default browser function\n * For more complicated tools, we should write our own behavior\n */\n switch (type) {\n case 'createLink' : editor.toolbar.inline.createLinkAction(event, type); break;\n default : editor.toolbar.inline.defaultToolAction(type); break;\n }\n\n /**\n * highlight buttons\n * after making some action\n */\n editor.nodes.inlineToolbar.buttons.childNodes.forEach(editor.toolbar.inline.hightlight);\n\n };\n\n /**\n * @private\n *\n * Saving wrappers offset in DOM\n */\n inline.getWrappersOffset = function () {\n\n var wrapper = editor.nodes.wrapper,\n offset = this.getOffset(wrapper);\n\n this.wrappersOffset = offset;\n return offset;\n\n };\n\n /**\n * @private\n *\n * Calculates offset of DOM element\n *\n * @param el\n * @returns {{top: number, left: number}}\n */\n inline.getOffset = function ( el ) {\n\n var _x = 0;\n var _y = 0;\n\n while( el && !isNaN( el.offsetLeft ) && !isNaN( el.offsetTop ) ) {\n\n _x += (el.offsetLeft + el.clientLeft);\n _y += (el.offsetTop + el.clientTop);\n el = el.offsetParent;\n\n }\n return { top: _y, left: _x };\n\n };\n\n /**\n * @private\n *\n * Calculates position of selected text\n * @returns {{x: number, y: number}}\n */\n inline.getSelectionCoords = function () {\n\n var sel = document.selection, range;\n var x = 0, y = 0;\n\n if (sel) {\n\n if (sel.type != 'Control') {\n\n range = sel.createRange();\n range.collapse(true);\n x = range.boundingLeft;\n y = range.boundingTop;\n\n }\n\n } else if (window.getSelection) {\n\n sel = window.getSelection();\n\n if (sel.rangeCount) {\n\n range = sel.getRangeAt(0).cloneRange();\n if (range.getClientRects) {\n\n range.collapse(true);\n var rect = range.getClientRects()[0];\n\n if (!rect) {\n\n return;\n\n }\n\n x = rect.left;\n y = rect.top;\n\n }\n\n }\n\n }\n return { x: x, y: y };\n\n };\n\n /**\n * @private\n *\n * Returns selected text as String\n * @returns {string}\n */\n inline.getSelectionText = function () {\n\n var selectedText = '';\n\n // all modern browsers and IE9+\n if (window.getSelection) {\n\n selectedText = window.getSelection().toString();\n\n }\n\n return selectedText;\n\n };\n\n /** Opens buttons block */\n inline.showButtons = function () {\n\n var buttons = editor.nodes.inlineToolbar.buttons;\n\n buttons.classList.add('opened');\n\n editor.toolbar.inline.buttonsOpened = true;\n\n /** highlight buttons */\n editor.nodes.inlineToolbar.buttons.childNodes.forEach(editor.toolbar.inline.hightlight);\n\n };\n\n /** Makes buttons disappear */\n inline.closeButtons = function () {\n\n var buttons = editor.nodes.inlineToolbar.buttons;\n\n buttons.classList.remove('opened');\n\n editor.toolbar.inline.buttonsOpened = false;\n\n };\n\n /** Open buttons defined action if exist */\n inline.showActions = function () {\n\n var action = editor.nodes.inlineToolbar.actions;\n\n action.classList.add('opened');\n\n editor.toolbar.inline.actionsOpened = true;\n\n };\n\n /** Close actions block */\n inline.closeAction = function () {\n\n var action = editor.nodes.inlineToolbar.actions;\n\n action.innerHTML = '';\n action.classList.remove('opened');\n editor.toolbar.inline.actionsOpened = false;\n\n };\n\n\n /**\n * Callback for keydowns in inline toolbar \"Insert link...\" input\n */\n let inlineToolbarAnchorInputKeydown_ = function (event) {\n\n if (event.keyCode != editor.core.keys.ENTER) {\n\n return;\n\n }\n\n let editable = editor.content.currentNode,\n storedSelection = editor.toolbar.inline.storedSelection;\n\n editor.toolbar.inline.restoreSelection(editable, storedSelection);\n editor.toolbar.inline.setAnchor(this.value);\n\n /**\n * Preventing events that will be able to happen\n */\n event.preventDefault();\n event.stopImmediatePropagation();\n\n editor.toolbar.inline.clearRange();\n\n };\n\n /** Action for link creation or for setting anchor */\n inline.createLinkAction = function (event) {\n\n var isActive = this.isLinkActive();\n\n var editable = editor.content.currentNode,\n storedSelection = editor.toolbar.inline.saveSelection(editable);\n\n /** Save globally selection */\n editor.toolbar.inline.storedSelection = storedSelection;\n\n if (isActive) {\n\n\n /**\n * Changing stored selection. if we want to remove anchor from word\n * we should remove anchor from whole word, not only selected part.\n * The solution is than we get the length of current link\n * Change start position to - end of selection minus length of anchor\n */\n editor.toolbar.inline.restoreSelection(editable, storedSelection);\n\n editor.toolbar.inline.defaultToolAction('unlink');\n\n } else {\n\n /** Create input and close buttons */\n var action = editor.draw.inputForLink();\n\n editor.nodes.inlineToolbar.actions.appendChild(action);\n\n editor.toolbar.inline.closeButtons();\n editor.toolbar.inline.showActions();\n\n /**\n * focus to input\n * Solution: https://developer.mozilla.org/ru/docs/Web/API/HTMLElement/focus\n * Prevents event after showing input and when we need to focus an input which is in unexisted form\n */\n action.focus();\n event.preventDefault();\n\n /** Callback to link action */\n editor.listeners.add(action, 'keydown', inlineToolbarAnchorInputKeydown_, false);\n\n }\n\n };\n\n inline.isLinkActive = function () {\n\n var isActive = false;\n\n editor.nodes.inlineToolbar.buttons.childNodes.forEach(function (tool) {\n\n var dataType = tool.dataset.type;\n\n if (dataType == 'link' && tool.classList.contains('hightlighted')) {\n\n isActive = true;\n\n }\n\n });\n\n return isActive;\n\n };\n\n /** default action behavior of tool */\n inline.defaultToolAction = function (type) {\n\n document.execCommand(type, false, null);\n\n };\n\n /**\n * @private\n *\n * Sets URL\n *\n * @param {String} url - URL\n */\n inline.setAnchor = function (url) {\n\n document.execCommand('createLink', false, url);\n\n /** Close after URL inserting */\n editor.toolbar.inline.closeAction();\n\n };\n\n /**\n * @private\n *\n * Saves selection\n */\n inline.saveSelection = function (containerEl) {\n\n var range = window.getSelection().getRangeAt(0),\n preSelectionRange = range.cloneRange(),\n start;\n\n preSelectionRange.selectNodeContents(containerEl);\n preSelectionRange.setEnd(range.startContainer, range.startOffset);\n\n start = preSelectionRange.toString().length;\n\n return {\n start: start,\n end: start + range.toString().length\n };\n\n };\n\n /**\n * @private\n *\n * Sets to previous selection (Range)\n *\n * @param {Element} containerEl - editable element where we restore range\n * @param {Object} savedSel - range basic information to restore\n */\n inline.restoreSelection = function (containerEl, savedSel) {\n\n var range = document.createRange(),\n charIndex = 0;\n\n range.setStart(containerEl, 0);\n range.collapse(true);\n\n var nodeStack = [ containerEl ],\n node,\n foundStart = false,\n stop = false,\n nextCharIndex;\n\n while (!stop && (node = nodeStack.pop())) {\n\n if (node.nodeType == 3) {\n\n nextCharIndex = charIndex + node.length;\n\n if (!foundStart && savedSel.start >= charIndex && savedSel.start <= nextCharIndex) {\n\n range.setStart(node, savedSel.start - charIndex);\n foundStart = true;\n\n }\n if (foundStart && savedSel.end >= charIndex && savedSel.end <= nextCharIndex) {\n\n range.setEnd(node, savedSel.end - charIndex);\n stop = true;\n\n }\n charIndex = nextCharIndex;\n\n } else {\n\n var i = node.childNodes.length;\n\n while (i--) {\n\n nodeStack.push(node.childNodes[i]);\n\n }\n\n }\n\n }\n\n var sel = window.getSelection();\n\n sel.removeAllRanges();\n sel.addRange(range);\n\n };\n\n /**\n * @private\n *\n * Removes all ranges from window selection\n */\n inline.clearRange = function () {\n\n var selection = window.getSelection();\n\n selection.removeAllRanges();\n\n };\n\n /**\n * @private\n *\n * sets or removes hightlight\n */\n inline.hightlight = function (tool) {\n\n var dataType = tool.dataset.type;\n\n if (document.queryCommandState(dataType)) {\n\n editor.toolbar.inline.setButtonHighlighted(tool);\n\n } else {\n\n editor.toolbar.inline.removeButtonsHighLight(tool);\n\n }\n\n /**\n *\n * hightlight for anchors\n */\n var selection = window.getSelection(),\n tag = selection.anchorNode.parentNode;\n\n if (tag.tagName == 'A' && dataType == 'link') {\n\n editor.toolbar.inline.setButtonHighlighted(tool);\n\n }\n\n };\n\n /**\n * @private\n *\n * Mark button if text is already executed\n */\n inline.setButtonHighlighted = function (button) {\n\n button.classList.add('hightlighted');\n\n /** At link tool we also change icon */\n if (button.dataset.type == 'link') {\n\n var icon = button.childNodes[0];\n\n icon.classList.remove('ce-icon-link');\n icon.classList.add('ce-icon-unlink');\n\n }\n\n };\n\n /**\n * @private\n *\n * Removes hightlight\n */\n inline.removeButtonsHighLight = function (button) {\n\n button.classList.remove('hightlighted');\n\n /** At link tool we also change icon */\n if (button.dataset.type == 'link') {\n\n var icon = button.childNodes[0];\n\n icon.classList.remove('ce-icon-unlink');\n icon.classList.add('ce-icon-link');\n\n }\n\n };\n\n\n return inline;\n\n})({});\n\n\n// WEBPACK FOOTER //\n// ./modules/toolbar/inline.js","/**\n * Codex Editor toolbox\n *\n * All tools be able to appended here\n *\n * @author Codex Team\n * @version 1.0\n */\n\nmodule.exports = (function (toolbox) {\n\n let editor = codex.editor;\n\n toolbox.opened = false;\n\n /** Shows toolbox */\n toolbox.open = function () {\n\n /** Close setting if toolbox is opened */\n if (editor.toolbar.settings.opened) {\n\n editor.toolbar.settings.close();\n\n }\n\n /** display toolbox */\n editor.nodes.toolbox.classList.add('opened');\n\n /** Animate plus button */\n editor.nodes.plusButton.classList.add('clicked');\n\n /** toolbox state */\n editor.toolbar.toolbox.opened = true;\n\n };\n\n /** Closes toolbox */\n toolbox.close = function () {\n\n /** Makes toolbox disapear */\n editor.nodes.toolbox.classList.remove('opened');\n\n /** Rotate plus button */\n editor.nodes.plusButton.classList.remove('clicked');\n\n /** toolbox state */\n editor.toolbar.toolbox.opened = false;\n\n editor.toolbar.current = null;\n\n };\n\n toolbox.leaf = function () {\n\n let currentTool = editor.toolbar.current,\n tools = Object.keys(editor.tools),\n barButtons = editor.nodes.toolbarButtons,\n nextToolIndex = 0,\n toolToSelect,\n visibleTool,\n tool;\n\n if ( !currentTool ) {\n\n /** Get first tool from object*/\n for(tool in editor.tools) {\n\n if (editor.tools[tool].displayInToolbox) {\n\n break;\n\n }\n\n nextToolIndex ++;\n\n }\n\n } else {\n\n nextToolIndex = (tools.indexOf(currentTool) + 1) % tools.length;\n visibleTool = tools[nextToolIndex];\n\n while (!editor.tools[visibleTool].displayInToolbox) {\n\n nextToolIndex = (nextToolIndex + 1) % tools.length;\n visibleTool = tools[nextToolIndex];\n\n }\n\n }\n\n toolToSelect = tools[nextToolIndex];\n\n for ( var button in barButtons ) {\n\n barButtons[button].classList.remove('selected');\n\n }\n\n barButtons[toolToSelect].classList.add('selected');\n editor.toolbar.current = toolToSelect;\n\n };\n\n /**\n * Transforming selected node type into selected toolbar element type\n * @param {event} event\n */\n toolbox.toolClicked = function (event) {\n\n /**\n * UNREPLACEBLE_TOOLS this types of tools are forbidden to replace even they are empty\n */\n var UNREPLACEBLE_TOOLS = ['image', 'link', 'list', 'instagram', 'twitter', 'embed'],\n tool = editor.tools[editor.toolbar.current],\n workingNode = editor.content.currentNode,\n currentInputIndex = editor.caret.inputIndex,\n newBlockContent,\n appendCallback,\n blockData;\n\n /** Make block from plugin */\n newBlockContent = tool.render();\n\n /** information about block */\n blockData = {\n block : newBlockContent,\n type : tool.type,\n stretched : false\n };\n\n if (\n workingNode &&\n UNREPLACEBLE_TOOLS.indexOf(workingNode.dataset.tool) === -1 &&\n workingNode.textContent.trim() === ''\n ) {\n\n /** Replace current block */\n editor.content.switchBlock(workingNode, newBlockContent, tool.type);\n\n } else {\n\n /** Insert new Block from plugin */\n editor.content.insertBlock(blockData);\n\n /** increase input index */\n currentInputIndex++;\n\n }\n\n /** Fire tool append callback */\n appendCallback = tool.appendCallback;\n\n if (appendCallback && typeof appendCallback == 'function') {\n\n appendCallback.call(event);\n\n }\n\n window.setTimeout(function () {\n\n /** Set caret to current block */\n editor.caret.setToBlock(currentInputIndex);\n\n }, 10);\n\n\n /**\n * Changing current Node\n */\n editor.content.workingNodeChanged();\n\n /**\n * Move toolbar when node is changed\n */\n editor.toolbar.move();\n\n };\n\n return toolbox;\n\n})({});\n\n\n// WEBPACK FOOTER //\n// ./modules/toolbar/toolbox.js","/**\n * @module Codex Editor Callbacks module\n * @description Module works with editor added Elements\n *\n * @author Codex Team\n * @version 1.4.0\n */\n\nmodule.exports = (function (callbacks) {\n\n let editor = codex.editor;\n\n /**\n * used by UI module\n * @description Routes all keydowns on document\n * @param {Object} event\n */\n callbacks.globalKeydown = function (event) {\n\n switch (event.keyCode) {\n case editor.core.keys.ENTER : enterKeyPressed_(event); break;\n }\n\n };\n\n /**\n * used by UI module\n * @description Routes all keydowns on redactors area\n * @param {Object} event\n */\n callbacks.redactorKeyDown = function (event) {\n\n switch (event.keyCode) {\n case editor.core.keys.TAB : tabKeyPressedOnRedactorsZone_(event); break;\n case editor.core.keys.ENTER : enterKeyPressedOnRedactorsZone_(event); break;\n case editor.core.keys.ESC : escapeKeyPressedOnRedactorsZone_(event); break;\n default : defaultKeyPressedOnRedactorsZone_(event); break;\n }\n\n };\n\n /**\n * used by UI module\n * @description Routes all keyup events\n * @param {Object} event\n */\n callbacks.globalKeyup = function (event) {\n\n switch (event.keyCode) {\n case editor.core.keys.UP :\n case editor.core.keys.LEFT :\n case editor.core.keys.RIGHT :\n case editor.core.keys.DOWN : arrowKeyPressed_(event); break;\n }\n\n };\n\n /**\n * @param {Object} event\n * @private\n *\n * Handles behaviour when tab pressed\n * @description if Content is empty show toolbox (if it is closed) or leaf tools\n * uses Toolbars toolbox module to handle the situation\n */\n var tabKeyPressedOnRedactorsZone_ = function (event) {\n\n /**\n * Wait for solution. Would like to know the behaviour\n * @todo Add spaces\n */\n event.preventDefault();\n\n var nativeInputs = editor.content.currentNode.querySelectorAll('textarea, input'),\n nativeInputsAreEmpty = true,\n textContentIsEmpty = !editor.content.currentNode.textContent.trim();\n\n Array.prototype.map.call(nativeInputs, function (input) {\n\n if (input.type == 'textarea' || input.type == 'text') {\n\n nativeInputsAreEmpty = nativeInputsAreEmpty && !input.value.trim();\n\n }\n\n });\n\n var blockIsEmpty = textContentIsEmpty && nativeInputsAreEmpty;\n\n if (!blockIsEmpty) {\n\n return;\n\n }\n\n if ( !editor.toolbar.opened ) {\n\n editor.toolbar.open();\n\n }\n\n if (editor.toolbar.opened && !editor.toolbar.toolbox.opened) {\n\n editor.toolbar.toolbox.open();\n\n } else {\n\n editor.toolbar.toolbox.leaf();\n\n }\n\n };\n\n /**\n * Handles global EnterKey Press\n * @see enterPressedOnBlock_\n * @param {Object} event\n */\n var enterKeyPressed_ = function () {\n\n if (editor.content.editorAreaHightlighted) {\n\n /**\n * it means that we lose input index, saved index before is not correct\n * therefore we need to set caret when we insert new block\n */\n editor.caret.inputIndex = -1;\n\n enterPressedOnBlock_();\n\n }\n\n };\n\n /**\n * Callback for enter key pressing in first-level block area\n *\n * @param {Event} event\n * @private\n *\n * @description Inserts new block with initial type from settings\n */\n var enterPressedOnBlock_ = function () {\n\n var NEW_BLOCK_TYPE = editor.settings.initialBlockPlugin;\n\n editor.content.insertBlock({\n type : NEW_BLOCK_TYPE,\n block : editor.tools[NEW_BLOCK_TYPE].render()\n }, true );\n\n editor.toolbar.move();\n editor.toolbar.open();\n\n };\n\n\n /**\n * ENTER key handler\n *\n * @param {Object} event\n * @private\n *\n * @description Makes new block with initial type from settings\n */\n var enterKeyPressedOnRedactorsZone_ = function (event) {\n\n if (event.target.contentEditable == 'true') {\n\n /** Update input index */\n editor.caret.saveCurrentInputIndex();\n\n }\n\n var currentInputIndex = editor.caret.getCurrentInputIndex() || 0,\n workingNode = editor.content.currentNode,\n tool = workingNode.dataset.tool,\n isEnterPressedOnToolbar = editor.toolbar.opened &&\n editor.toolbar.current &&\n event.target == editor.state.inputs[currentInputIndex];\n\n /** The list of tools which needs the default browser behaviour */\n var enableLineBreaks = editor.tools[tool].enableLineBreaks;\n\n /** This type of block creates when enter is pressed */\n var NEW_BLOCK_TYPE = editor.settings.initialBlockPlugin;\n\n /**\n * When toolbar is opened, select tool instead of making new paragraph\n */\n if ( isEnterPressedOnToolbar ) {\n\n event.preventDefault();\n\n editor.toolbar.toolbox.toolClicked(event);\n\n editor.toolbar.close();\n\n /**\n * Stop other listeners callback executions\n */\n event.stopPropagation();\n event.stopImmediatePropagation();\n\n return;\n\n }\n\n /**\n * Allow paragraph lineBreaks with shift enter\n * Or if shiftkey pressed and enter and enabledLineBreaks, the let new block creation\n */\n if ( event.shiftKey || enableLineBreaks ) {\n\n event.stopPropagation();\n event.stopImmediatePropagation();\n return;\n\n }\n\n var currentSelection = window.getSelection(),\n currentSelectedNode = currentSelection.anchorNode,\n caretAtTheEndOfText = editor.caret.position.atTheEnd(),\n isTextNodeHasParentBetweenContenteditable = false;\n\n /**\n * Allow making new

    in same block by SHIFT+ENTER and forbids to prevent default browser behaviour\n */\n if ( event.shiftKey && !enableLineBreaks ) {\n\n editor.callback.enterPressedOnBlock(editor.content.currentBlock, event);\n event.preventDefault();\n return;\n\n }\n\n /**\n * Workaround situation when caret at the Text node that has some wrapper Elements\n * Split block cant handle this.\n * We need to save default behavior\n */\n isTextNodeHasParentBetweenContenteditable = currentSelectedNode && currentSelectedNode.parentNode.contentEditable != 'true';\n\n /**\n * Split blocks when input has several nodes and caret placed in textNode\n */\n if (\n currentSelectedNode.nodeType == editor.core.nodeTypes.TEXT &&\n !isTextNodeHasParentBetweenContenteditable &&\n !caretAtTheEndOfText\n ) {\n\n event.preventDefault();\n\n editor.core.log('Splitting Text node...');\n\n editor.content.splitBlock(currentInputIndex);\n\n /** Show plus button when next input after split is empty*/\n if (!editor.state.inputs[currentInputIndex + 1].textContent.trim()) {\n\n editor.toolbar.showPlusButton();\n\n }\n\n } else {\n\n var islastNode = editor.content.isLastNode(currentSelectedNode);\n\n if ( islastNode && caretAtTheEndOfText ) {\n\n event.preventDefault();\n event.stopPropagation();\n event.stopImmediatePropagation();\n\n editor.core.log('ENTER clicked in last textNode. Create new BLOCK');\n\n editor.content.insertBlock({\n type: NEW_BLOCK_TYPE,\n block: editor.tools[NEW_BLOCK_TYPE].render()\n }, true);\n\n editor.toolbar.move();\n editor.toolbar.open();\n\n /** Show plus button with empty block */\n editor.toolbar.showPlusButton();\n\n }\n\n }\n\n /** get all inputs after new appending block */\n editor.ui.saveInputs();\n\n };\n\n /**\n * Escape behaviour\n * @param event\n * @private\n *\n * @description Closes toolbox and toolbar. Prevents default behaviour\n */\n var escapeKeyPressedOnRedactorsZone_ = function (event) {\n\n /** Close all toolbar */\n editor.toolbar.close();\n\n /** Close toolbox */\n editor.toolbar.toolbox.close();\n\n event.preventDefault();\n\n };\n\n /**\n * @param {Event} event\n * @private\n *\n * closes and moves toolbar\n */\n var arrowKeyPressed_ = function (event) {\n\n editor.content.workingNodeChanged();\n\n /* Closing toolbar */\n editor.toolbar.close();\n editor.toolbar.move();\n\n };\n\n /**\n * @private\n * @param {Event} event\n *\n * @description Closes all opened bars from toolbar.\n * If block is mark, clears highlightning\n */\n var defaultKeyPressedOnRedactorsZone_ = function () {\n\n editor.toolbar.close();\n\n if (!editor.toolbar.inline.actionsOpened) {\n\n editor.toolbar.inline.close();\n editor.content.clearMark();\n\n }\n\n };\n\n /**\n * Handler when clicked on redactors area\n *\n * @protected\n * @param event\n *\n * @description Detects clicked area. If it is first-level block area, marks as detected and\n * on next enter press will be inserted new block\n * Otherwise, save carets position (input index) and put caret to the editable zone.\n *\n * @see detectWhenClickedOnFirstLevelBlockArea_\n *\n */\n callbacks.redactorClicked = function (event) {\n\n detectWhenClickedOnFirstLevelBlockArea_();\n\n editor.content.workingNodeChanged(event.target);\n editor.ui.saveInputs();\n\n var selectedText = editor.toolbar.inline.getSelectionText(),\n firstLevelBlock;\n\n /** If selection range took off, then we hide inline toolbar */\n if (selectedText.length === 0) {\n\n editor.toolbar.inline.close();\n\n }\n\n /** Update current input index in memory when caret focused into existed input */\n if (event.target.contentEditable == 'true') {\n\n editor.caret.saveCurrentInputIndex();\n\n }\n\n if (editor.content.currentNode === null) {\n\n /**\n * If inputs in redactor does not exits, then we put input index 0 not -1\n */\n var indexOfLastInput = editor.state.inputs.length > 0 ? editor.state.inputs.length - 1 : 0;\n\n /** If we have any inputs */\n if (editor.state.inputs.length) {\n\n /** getting firstlevel parent of input */\n firstLevelBlock = editor.content.getFirstLevelBlock(editor.state.inputs[indexOfLastInput]);\n\n }\n\n /** If input is empty, then we set caret to the last input */\n if (editor.state.inputs.length && editor.state.inputs[indexOfLastInput].textContent === '' && firstLevelBlock.dataset.tool == editor.settings.initialBlockPlugin) {\n\n editor.caret.setToBlock(indexOfLastInput);\n\n } else {\n\n /** Create new input when caret clicked in redactors area */\n var NEW_BLOCK_TYPE = editor.settings.initialBlockPlugin;\n\n editor.content.insertBlock({\n type : NEW_BLOCK_TYPE,\n block : editor.tools[NEW_BLOCK_TYPE].render()\n });\n\n /** If there is no inputs except inserted */\n if (editor.state.inputs.length === 1) {\n\n editor.caret.setToBlock(indexOfLastInput);\n\n } else {\n\n /** Set caret to this appended input */\n editor.caret.setToNextBlock(indexOfLastInput);\n\n }\n\n }\n\n } else {\n\n /** Close all panels */\n editor.toolbar.settings.close();\n editor.toolbar.toolbox.close();\n\n }\n\n /**\n * Move toolbar and open\n */\n editor.toolbar.move();\n editor.toolbar.open();\n\n var inputIsEmpty = !editor.content.currentNode.textContent.trim(),\n currentNodeType = editor.content.currentNode.dataset.tool,\n isInitialType = currentNodeType == editor.settings.initialBlockPlugin;\n\n\n /** Hide plus buttons */\n editor.toolbar.hidePlusButton();\n\n if (!inputIsEmpty) {\n\n /** Mark current block */\n editor.content.markBlock();\n\n }\n\n if ( isInitialType && inputIsEmpty ) {\n\n /** Show plus button */\n editor.toolbar.showPlusButton();\n\n }\n\n\n };\n\n /**\n * This method allows to define, is caret in contenteditable element or not.\n *\n * @private\n *\n * @description Otherwise, if we get TEXT node from range container, that will means we have input index.\n * In this case we use default browsers behaviour (if plugin allows that) or overwritten action.\n * Therefore, to be sure that we've clicked first-level block area, we should have currentNode, which always\n * specifies to the first-level block. Other cases we just ignore.\n */\n var detectWhenClickedOnFirstLevelBlockArea_ = function () {\n\n var selection = window.getSelection(),\n anchorNode = selection.anchorNode,\n flag = false;\n\n if (selection.rangeCount === 0) {\n\n editor.content.editorAreaHightlighted = true;\n\n } else {\n\n if (!editor.core.isDomNode(anchorNode)) {\n\n anchorNode = anchorNode.parentNode;\n\n }\n\n /** Already founded, without loop */\n if (anchorNode.contentEditable == 'true') {\n\n flag = true;\n\n }\n\n while (anchorNode.contentEditable != 'true') {\n\n anchorNode = anchorNode.parentNode;\n\n if (anchorNode.contentEditable == 'true') {\n\n flag = true;\n\n }\n\n if (anchorNode == document.body) {\n\n break;\n\n }\n\n }\n\n /** If editable element founded, flag is \"TRUE\", Therefore we return \"FALSE\" */\n editor.content.editorAreaHightlighted = !flag;\n\n }\n\n };\n\n /**\n * Toolbar button click handler\n *\n * @param {Object} event - cursor to the button\n * @protected\n *\n * @description gets current tool and calls render method\n */\n callbacks.toolbarButtonClicked = function (event) {\n\n var button = this;\n\n editor.toolbar.current = button.dataset.type;\n\n editor.toolbar.toolbox.toolClicked(event);\n editor.toolbar.close();\n\n };\n\n /**\n * Show or Hide toolbox when plus button is clicked\n */\n callbacks.plusButtonClicked = function () {\n\n if (!editor.nodes.toolbox.classList.contains('opened')) {\n\n editor.toolbar.toolbox.open();\n\n } else {\n\n editor.toolbar.toolbox.close();\n\n }\n\n };\n\n /**\n * Block handlers for KeyDown events\n *\n * @protected\n * @param {Object} event\n *\n * Handles keydowns on block\n * @see blockRightOrDownArrowPressed_\n * @see backspacePressed_\n * @see blockLeftOrUpArrowPressed_\n */\n callbacks.blockKeydown = function (event) {\n\n let block = event.target; // event.target is input\n\n switch (event.keyCode) {\n\n case editor.core.keys.DOWN:\n case editor.core.keys.RIGHT:\n blockRightOrDownArrowPressed_(event);\n break;\n\n case editor.core.keys.BACKSPACE:\n backspacePressed_(block, event);\n break;\n\n case editor.core.keys.UP:\n case editor.core.keys.LEFT:\n blockLeftOrUpArrowPressed_(event);\n break;\n\n }\n\n };\n\n /**\n * RIGHT or DOWN keydowns on block\n *\n * @param {Object} event\n * @private\n *\n * @description watches the selection and gets closest editable element.\n * Uses method getDeepestTextNodeFromPosition to get the last node of next block\n * Sets caret if it is contenteditable\n */\n var blockRightOrDownArrowPressed_ = function (event) {\n\n var selection = window.getSelection(),\n inputs = editor.state.inputs,\n focusedNode = selection.anchorNode,\n focusedNodeHolder;\n\n /** Check for caret existance */\n if (!focusedNode) {\n\n return false;\n\n }\n\n /** Looking for closest (parent) contentEditable element of focused node */\n while (focusedNode.contentEditable != 'true') {\n\n focusedNodeHolder = focusedNode.parentNode;\n focusedNode = focusedNodeHolder;\n\n }\n\n /** Input index in DOM level */\n var editableElementIndex = 0;\n\n while (focusedNode != inputs[editableElementIndex]) {\n\n editableElementIndex ++;\n\n }\n\n /**\n * Founded contentEditable element doesn't have childs\n * Or maybe New created block\n */\n if (!focusedNode.textContent) {\n\n editor.caret.setToNextBlock(editableElementIndex);\n return;\n\n }\n\n /**\n * Do nothing when caret doesn not reaches the end of last child\n */\n var caretInLastChild = false,\n caretAtTheEndOfText = false;\n\n var lastChild,\n deepestTextnode;\n\n lastChild = focusedNode.childNodes[focusedNode.childNodes.length - 1 ];\n\n if (editor.core.isDomNode(lastChild)) {\n\n deepestTextnode = editor.content.getDeepestTextNodeFromPosition(lastChild, lastChild.childNodes.length);\n\n } else {\n\n deepestTextnode = lastChild;\n\n }\n\n caretInLastChild = selection.anchorNode == deepestTextnode;\n caretAtTheEndOfText = deepestTextnode.length == selection.anchorOffset;\n\n if ( !caretInLastChild || !caretAtTheEndOfText ) {\n\n editor.core.log('arrow [down|right] : caret does not reached the end');\n return false;\n\n }\n\n editor.caret.setToNextBlock(editableElementIndex);\n\n };\n\n /**\n * LEFT or UP keydowns on block\n *\n * @param {Object} event\n * @private\n *\n * watches the selection and gets closest editable element.\n * Uses method getDeepestTextNodeFromPosition to get the last node of previous block\n * Sets caret if it is contenteditable\n *\n */\n var blockLeftOrUpArrowPressed_ = function (event) {\n\n var selection = window.getSelection(),\n inputs = editor.state.inputs,\n focusedNode = selection.anchorNode,\n focusedNodeHolder;\n\n /** Check for caret existance */\n if (!focusedNode) {\n\n return false;\n\n }\n\n /**\n * LEFT or UP not at the beginning\n */\n if ( selection.anchorOffset !== 0) {\n\n return false;\n\n }\n\n /** Looking for parent contentEditable block */\n while (focusedNode.contentEditable != 'true') {\n\n focusedNodeHolder = focusedNode.parentNode;\n focusedNode = focusedNodeHolder;\n\n }\n\n /** Input index in DOM level */\n var editableElementIndex = 0;\n\n while (focusedNode != inputs[editableElementIndex]) {\n\n editableElementIndex ++;\n\n }\n\n /**\n * Do nothing if caret is not at the beginning of first child\n */\n var caretInFirstChild = false,\n caretAtTheBeginning = false;\n\n var firstChild,\n deepestTextnode;\n\n /**\n * Founded contentEditable element doesn't have childs\n * Or maybe New created block\n */\n if (!focusedNode.textContent) {\n\n editor.caret.setToPreviousBlock(editableElementIndex);\n return;\n\n }\n\n firstChild = focusedNode.childNodes[0];\n\n if (editor.core.isDomNode(firstChild)) {\n\n deepestTextnode = editor.content.getDeepestTextNodeFromPosition(firstChild, 0);\n\n } else {\n\n deepestTextnode = firstChild;\n\n }\n\n caretInFirstChild = selection.anchorNode == deepestTextnode;\n caretAtTheBeginning = selection.anchorOffset === 0;\n\n if ( caretInFirstChild && caretAtTheBeginning ) {\n\n editor.caret.setToPreviousBlock(editableElementIndex);\n\n }\n\n };\n\n /**\n * Handles backspace keydown\n *\n * @param {Element} block\n * @param {Object} event\n * @private\n *\n * @description if block is empty, delete the block and set caret to the previous block\n * If block is not empty, try to merge two blocks - current and previous\n * But it we try'n to remove first block, then we should set caret to the next block, not previous.\n * If we removed the last block, create new one\n */\n var backspacePressed_ = function (block, event) {\n\n var currentInputIndex = editor.caret.getCurrentInputIndex(),\n range,\n selectionLength,\n firstLevelBlocksCount;\n\n if (isNativeInput_(event.target)) {\n\n /** If input value is empty - remove block */\n if (event.target.value.trim() == '') {\n\n block.remove();\n\n } else {\n\n return;\n\n }\n\n }\n\n if (block.textContent.trim()) {\n\n range = editor.content.getRange();\n selectionLength = range.endOffset - range.startOffset;\n\n if (editor.caret.position.atStart() && !selectionLength && editor.state.inputs[currentInputIndex - 1]) {\n\n editor.content.mergeBlocks(currentInputIndex);\n\n } else {\n\n return;\n\n }\n\n }\n\n if (!selectionLength) {\n\n block.remove();\n\n }\n\n\n firstLevelBlocksCount = editor.nodes.redactor.childNodes.length;\n\n /**\n * If all blocks are removed\n */\n if (firstLevelBlocksCount === 0) {\n\n /** update currentNode variable */\n editor.content.currentNode = null;\n\n /** Inserting new empty initial block */\n editor.ui.addInitialBlock();\n\n /** Updating inputs state after deleting last block */\n editor.ui.saveInputs();\n\n /** Set to current appended block */\n window.setTimeout(function () {\n\n editor.caret.setToPreviousBlock(1);\n\n }, 10);\n\n } else {\n\n if (editor.caret.inputIndex !== 0) {\n\n /** Target block is not first */\n editor.caret.setToPreviousBlock(editor.caret.inputIndex);\n\n } else {\n\n /** If we try to delete first block */\n editor.caret.setToNextBlock(editor.caret.inputIndex);\n\n }\n\n }\n\n editor.toolbar.move();\n\n if (!editor.toolbar.opened) {\n\n editor.toolbar.open();\n\n }\n\n /** Updating inputs state */\n editor.ui.saveInputs();\n\n /** Prevent default browser behaviour */\n event.preventDefault();\n\n };\n\n /**\n * This method prevents default behaviour.\n *\n * @param {Object} event\n * @protected\n *\n * @description We get from clipboard pasted data, sanitize, make a fragment that contains of this sanitized nodes.\n * Firstly, we need to memorize the caret position. We can do that by getting the range of selection.\n * After all, we insert clear fragment into caret placed position. Then, we should move the caret to the last node\n */\n callbacks.blockPasteCallback = function (event) {\n\n /** If area is input or textarea then allow default behaviour */\n if ( isNativeInput_(event.target) ) {\n\n return;\n\n }\n\n /** Prevent default behaviour */\n event.preventDefault();\n\n var editableParent = editor.content.getEditableParent(event.target),\n currentNode = editor.content.currentNode;\n\n /** Allow paste when event target placed in Editable element */\n if (!editableParent) {\n\n return;\n\n }\n\n /** get html pasted data - dirty data */\n var htmlData = event.clipboardData.getData('text/html'),\n plainData = event.clipboardData.getData('text/plain');\n\n /** Temporary DIV that is used to work with childs as arrays item */\n var div = editor.draw.node('DIV', '', {}),\n cleanData,\n fragment;\n\n /** Create fragment, that we paste to range after proccesing */\n fragment = document.createDocumentFragment();\n\n if ( htmlData.trim() != '' ) {\n\n cleanData = editor.sanitizer.clean(htmlData);\n div.innerHTML = cleanData;\n\n } else {\n\n div.innerText = plainData.toString();\n\n }\n\n var node, lastNode;\n\n /**\n * and fill in fragment\n */\n while (( node = div.firstChild) ) {\n\n lastNode = fragment.appendChild(node);\n\n }\n\n\n if (editor.tools[currentNode.dataset.tool].allowRenderOnPaste) {\n\n if (editor.paste.pasted(event)) return;\n\n }\n\n /**\n * work with selection and range\n */\n var selection, range;\n\n selection = window.getSelection();\n\n range = selection.getRangeAt(0);\n range.deleteContents();\n\n range.insertNode(fragment);\n\n /** Preserve the selection */\n if (lastNode) {\n\n range = range.cloneRange();\n range.setStartAfter(lastNode);\n range.collapse(true);\n selection.removeAllRanges();\n selection.addRange(range);\n\n }\n\n };\n\n /**\n * used by UI module\n * Clicks on block settings button\n *\n * @param {Object} event\n * @protected\n * @description Opens toolbar settings\n */\n callbacks.showSettingsButtonClicked = function (event) {\n\n /**\n * Get type of current block\n * It uses to append settings from tool.settings property.\n * ...\n * Type is stored in data-type attribute on block\n */\n var currentToolType = editor.content.currentNode.dataset.tool;\n\n editor.toolbar.settings.toggle(currentToolType);\n\n /** Close toolbox when settings button is active */\n editor.toolbar.toolbox.close();\n editor.toolbar.settings.hideRemoveActions();\n\n };\n\n /**\n * Check block\n * @param target\n * @private\n *\n * @description Checks target is it native input\n */\n var isNativeInput_ = function (target) {\n\n var nativeInputAreas = ['INPUT', 'TEXTAREA'];\n\n return (nativeInputAreas.indexOf(target.tagName) != -1);\n\n };\n\n return callbacks;\n\n})({});\n\n\n// WEBPACK FOOTER //\n// ./modules/callbacks.js","/**\n * Codex Editor Draw module\n *\n * @author Codex Team\n * @version 1.0.\n */\n\nmodule.exports = (function (draw) {\n\n /**\n * Base editor wrapper\n */\n draw.wrapper = function () {\n\n var wrapper = document.createElement('div');\n\n wrapper.className += 'codex-editor';\n\n return wrapper;\n\n };\n\n /**\n * Content-editable holder\n */\n draw.redactor = function () {\n\n var redactor = document.createElement('div');\n\n redactor.className += 'ce-redactor';\n\n return redactor;\n\n };\n\n draw.ceBlock = function () {\n\n var block = document.createElement('DIV');\n\n block.className += 'ce_block';\n\n return block;\n\n };\n\n /**\n * Empty toolbar with toggler\n */\n draw.toolbar = function () {\n\n var bar = document.createElement('div');\n\n bar.className += 'ce-toolbar';\n\n return bar;\n\n };\n\n draw.toolbarContent = function () {\n\n var wrapper = document.createElement('DIV');\n\n wrapper.classList.add('ce-toolbar__content');\n\n return wrapper;\n\n };\n\n /**\n * Inline toolbar\n */\n draw.inlineToolbar = function () {\n\n var bar = document.createElement('DIV');\n\n bar.className += 'ce-toolbar-inline';\n\n return bar;\n\n };\n\n /**\n * Wrapper for inline toobar buttons\n */\n draw.inlineToolbarButtons = function () {\n\n var wrapper = document.createElement('DIV');\n\n wrapper.className += 'ce-toolbar-inline__buttons';\n\n return wrapper;\n\n };\n\n /**\n * For some actions\n */\n draw.inlineToolbarActions = function () {\n\n var wrapper = document.createElement('DIV');\n\n wrapper.className += 'ce-toolbar-inline__actions';\n\n return wrapper;\n\n };\n\n draw.inputForLink = function () {\n\n var input = document.createElement('INPUT');\n\n input.type = 'input';\n input.className += 'inputForLink';\n input.placeholder = 'Вставьте ссылку ...';\n input.setAttribute('form', 'defaultForm');\n\n input.setAttribute('autofocus', 'autofocus');\n\n return input;\n\n };\n\n /**\n * @todo Desc\n */\n draw.blockButtons = function () {\n\n var block = document.createElement('div');\n\n block.className += 'ce-toolbar__actions';\n\n return block;\n\n };\n\n /**\n * Block settings panel\n */\n draw.blockSettings = function () {\n\n var settings = document.createElement('div');\n\n settings.className += 'ce-settings';\n\n return settings;\n\n };\n\n draw.defaultSettings = function () {\n\n var div = document.createElement('div');\n\n div.classList.add('ce-settings_default');\n\n return div;\n\n };\n\n draw.pluginsSettings = function () {\n\n var div = document.createElement('div');\n\n div.classList.add('ce-settings_plugin');\n\n return div;\n\n };\n\n draw.plusButton = function () {\n\n var button = document.createElement('span');\n\n button.className = 'ce-toolbar__plus';\n // button.innerHTML = '';\n\n return button;\n\n };\n\n /**\n * Settings button in toolbar\n */\n draw.settingsButton = function () {\n\n var toggler = document.createElement('span');\n\n toggler.className = 'ce-toolbar__settings-btn';\n\n /** Toggler button*/\n toggler.innerHTML = '';\n\n return toggler;\n\n };\n\n /**\n * Redactor tools wrapper\n */\n\n draw.toolbox = function () {\n\n var wrapper = document.createElement('div');\n\n wrapper.className = 'ce-toolbar__tools';\n\n return wrapper;\n\n };\n\n /**\n * @protected\n *\n * Draws tool buttons for toolbox\n *\n * @param {String} type\n * @param {String} classname\n * @returns {Element}\n */\n draw.toolbarButton = function (type, classname) {\n\n var button = document.createElement('li'),\n toolIcon = document.createElement('i'),\n toolTitle = document.createElement('span');\n\n button.dataset.type = type;\n button.setAttribute('title', type);\n\n toolIcon.classList.add(classname);\n toolTitle.classList.add('ce_toolbar_tools--title');\n\n\n button.appendChild(toolIcon);\n button.appendChild(toolTitle);\n\n return button;\n\n };\n\n /**\n * @protected\n *\n * Draws tools for inline toolbar\n *\n * @param {String} type\n * @param {String} classname\n */\n draw.toolbarButtonInline = function (type, classname) {\n\n var button = document.createElement('BUTTON'),\n toolIcon = document.createElement('I');\n\n button.type = 'button';\n button.dataset.type = type;\n toolIcon.classList.add(classname);\n\n button.appendChild(toolIcon);\n\n return button;\n\n };\n\n /**\n * Redactor block\n */\n draw.block = function (tagName, content) {\n\n var node = document.createElement(tagName);\n\n node.innerHTML = content || '';\n\n return node;\n\n };\n\n /**\n * Creates Node with passed tagName and className\n * @param {string} tagName\n * @param {string} className\n * @param {object} properties - allow to assign properties\n */\n draw.node = function ( tagName, className, properties ) {\n\n var el = document.createElement( tagName );\n\n if ( className ) el.className = className;\n\n if ( properties ) {\n\n for (var name in properties) {\n\n el[name] = properties[name];\n\n }\n\n }\n\n return el;\n\n };\n\n /**\n * Unavailable plugin block\n */\n draw.unavailableBlock = function () {\n\n var wrapper = document.createElement('DIV');\n\n wrapper.classList.add('cdx-unavailable-block');\n\n return wrapper;\n\n };\n\n return draw;\n\n})({});\n\n\n// WEBPACK FOOTER //\n// ./modules/draw.js","/**\n * Codex Editor Caret Module\n *\n * @author Codex Team\n * @version 1.0\n */\n\nmodule.exports = (function (caret) {\n\n let editor = codex.editor;\n\n /**\n * @var {int} InputIndex - editable element in DOM\n */\n caret.inputIndex = null;\n\n /**\n * @var {int} offset - caret position in a text node.\n */\n caret.offset = null;\n\n /**\n * @var {int} focusedNodeIndex - we get index of child node from first-level block\n */\n caret.focusedNodeIndex = null;\n\n /**\n * Creates Document Range and sets caret to the element.\n * @protected\n * @uses caret.save — if you need to save caret position\n * @param {Element} el - Changed Node.\n */\n caret.set = function ( el, index, offset) {\n\n offset = offset || caret.offset || 0;\n index = index || caret.focusedNodeIndex || 0;\n\n var childs = el.childNodes,\n nodeToSet;\n\n if ( childs.length === 0 ) {\n\n nodeToSet = el;\n\n } else {\n\n nodeToSet = childs[index];\n\n }\n\n /** If Element is INPUT */\n if (el.contentEditable != 'true') {\n\n el.focus();\n return;\n\n }\n\n if (editor.core.isDomNode(nodeToSet)) {\n\n nodeToSet = editor.content.getDeepestTextNodeFromPosition(nodeToSet, nodeToSet.childNodes.length);\n\n }\n\n var range = document.createRange(),\n selection = window.getSelection();\n\n window.setTimeout(function () {\n\n range.setStart(nodeToSet, offset);\n range.setEnd(nodeToSet, offset);\n\n selection.removeAllRanges();\n selection.addRange(range);\n\n editor.caret.saveCurrentInputIndex();\n\n }, 20);\n\n };\n\n /**\n * @protected\n * Updates index of input and saves it in caret object\n */\n caret.saveCurrentInputIndex = function () {\n\n /** Index of Input that we paste sanitized content */\n var selection = window.getSelection(),\n inputs = editor.state.inputs,\n focusedNode = selection.anchorNode,\n focusedNodeHolder;\n\n if (!focusedNode) {\n\n return;\n\n }\n\n /** Looking for parent contentEditable block */\n while (focusedNode.contentEditable != 'true') {\n\n focusedNodeHolder = focusedNode.parentNode;\n focusedNode = focusedNodeHolder;\n\n }\n\n /** Input index in DOM level */\n var editableElementIndex = 0;\n\n while (focusedNode != inputs[editableElementIndex]) {\n\n editableElementIndex ++;\n\n }\n\n caret.inputIndex = editableElementIndex;\n\n };\n\n /**\n * Returns current input index (caret object)\n */\n caret.getCurrentInputIndex = function () {\n\n return caret.inputIndex;\n\n };\n\n /**\n * @param {int} index - index of first-level block after that we set caret into next input\n */\n caret.setToNextBlock = function (index) {\n\n var inputs = editor.state.inputs,\n nextInput = inputs[index + 1];\n\n if (!nextInput) {\n\n editor.core.log('We are reached the end');\n return;\n\n }\n\n /**\n * When new Block created or deleted content of input\n * We should add some text node to set caret\n */\n if (!nextInput.childNodes.length) {\n\n var emptyTextElement = document.createTextNode('');\n\n nextInput.appendChild(emptyTextElement);\n\n }\n\n editor.caret.inputIndex = index + 1;\n editor.caret.set(nextInput, 0, 0);\n editor.content.workingNodeChanged(nextInput);\n\n };\n\n /**\n * @param {int} index - index of target input.\n * Sets caret to input with this index\n */\n caret.setToBlock = function (index) {\n\n var inputs = editor.state.inputs,\n targetInput = inputs[index];\n\n if ( !targetInput ) {\n\n return;\n\n }\n\n /**\n * When new Block created or deleted content of input\n * We should add some text node to set caret\n */\n if (!targetInput.childNodes.length) {\n\n var emptyTextElement = document.createTextNode('');\n\n targetInput.appendChild(emptyTextElement);\n\n }\n\n editor.caret.inputIndex = index;\n editor.caret.set(targetInput, 0, 0);\n editor.content.workingNodeChanged(targetInput);\n\n };\n\n /**\n * @param {int} index - index of input\n */\n caret.setToPreviousBlock = function (index) {\n\n index = index || 0;\n\n var inputs = editor.state.inputs,\n previousInput = inputs[index - 1],\n lastChildNode,\n lengthOfLastChildNode,\n emptyTextElement;\n\n\n if (!previousInput) {\n\n editor.core.log('We are reached first node');\n return;\n\n }\n\n lastChildNode = editor.content.getDeepestTextNodeFromPosition(previousInput, previousInput.childNodes.length);\n lengthOfLastChildNode = lastChildNode.length;\n\n /**\n * When new Block created or deleted content of input\n * We should add some text node to set caret\n */\n if (!previousInput.childNodes.length) {\n\n emptyTextElement = document.createTextNode('');\n previousInput.appendChild(emptyTextElement);\n\n }\n editor.caret.inputIndex = index - 1;\n editor.caret.set(previousInput, previousInput.childNodes.length - 1, lengthOfLastChildNode);\n editor.content.workingNodeChanged(inputs[index - 1]);\n\n };\n\n caret.position = {\n\n atStart : function () {\n\n var selection = window.getSelection(),\n anchorOffset = selection.anchorOffset,\n anchorNode = selection.anchorNode,\n firstLevelBlock = editor.content.getFirstLevelBlock(anchorNode),\n pluginsRender = firstLevelBlock.childNodes[0];\n\n if (!editor.core.isDomNode(anchorNode)) {\n\n anchorNode = anchorNode.parentNode;\n\n }\n\n var isFirstNode = anchorNode === pluginsRender.childNodes[0],\n isOffsetZero = anchorOffset === 0;\n\n return isFirstNode && isOffsetZero;\n\n },\n\n atTheEnd : function () {\n\n var selection = window.getSelection(),\n anchorOffset = selection.anchorOffset,\n anchorNode = selection.anchorNode;\n\n /** Caret is at the end of input */\n return !anchorNode || !anchorNode.length || anchorOffset === anchorNode.length;\n\n }\n };\n\n return caret;\n\n})({});\n\n\n// WEBPACK FOOTER //\n// ./modules/caret.js","/**\n * Codex Editor Notification Module\n *\n * @author Codex Team\n * @version 1.0\n */\n\nmodule.exports = (function (notifications) {\n\n let editor = codex.editor;\n\n var queue = [];\n\n var addToQueue = function (settings) {\n\n queue.push(settings);\n\n var index = 0;\n\n while ( index < queue.length && queue.length > 5) {\n\n if (queue[index].type == 'confirm' || queue[index].type == 'prompt') {\n\n index++;\n continue;\n\n }\n\n queue[index].close();\n queue.splice(index, 1);\n\n }\n\n };\n\n notifications.createHolder = function () {\n\n var holder = editor.draw.node('DIV', 'cdx-notifications-block');\n\n editor.nodes.notifications = document.body.appendChild(holder);\n\n return holder;\n\n };\n\n\n /**\n * Error notificator. Shows block with message\n * @protected\n */\n notifications.errorThrown = function (errorMsg, event) {\n\n editor.notifications.notification({message: 'This action is not available currently', type: event.type});\n\n };\n\n /**\n *\n * Appends notification\n *\n * settings = {\n * type - notification type (reserved types: alert, confirm, prompt). Just add class 'cdx-notification-'+type\n * message - notification message\n * okMsg - confirm button text (default - 'Ok')\n * cancelBtn - cancel button text (default - 'Cancel'). Only for confirm and prompt types\n * confirm - function-handler for ok button click\n * cancel - function-handler for cancel button click. Only for confirm and prompt types\n * time - time (in seconds) after which notification will close (default - 10s)\n * }\n *\n * @param settings\n */\n notifications.notification = function (constructorSettings) {\n\n /** Private vars and methods */\n var notification = null,\n cancel = null,\n type = null,\n confirm = null,\n inputField = null;\n\n var confirmHandler = function () {\n\n close();\n\n if (typeof confirm !== 'function' ) {\n\n return;\n\n }\n\n if (type == 'prompt') {\n\n confirm(inputField.value);\n return;\n\n }\n\n confirm();\n\n };\n\n var cancelHandler = function () {\n\n close();\n\n if (typeof cancel !== 'function' ) {\n\n return;\n\n }\n\n cancel();\n\n };\n\n\n /** Public methods */\n function create(settings) {\n\n if (!(settings && settings.message)) {\n\n editor.core.log('Can\\'t create notification. Message is missed');\n return;\n\n }\n\n settings.type = settings.type || 'alert';\n settings.time = settings.time*1000 || 10000;\n\n var wrapper = editor.draw.node('DIV', 'cdx-notification'),\n message = editor.draw.node('DIV', 'cdx-notification__message'),\n input = editor.draw.node('INPUT', 'cdx-notification__input'),\n okBtn = editor.draw.node('SPAN', 'cdx-notification__ok-btn'),\n cancelBtn = editor.draw.node('SPAN', 'cdx-notification__cancel-btn');\n\n message.textContent = settings.message;\n okBtn.textContent = settings.okMsg || 'ОК';\n cancelBtn.textContent = settings.cancelMsg || 'Отмена';\n\n editor.listeners.add(okBtn, 'click', confirmHandler);\n editor.listeners.add(cancelBtn, 'click', cancelHandler);\n\n wrapper.appendChild(message);\n\n if (settings.type == 'prompt') {\n\n wrapper.appendChild(input);\n\n }\n\n wrapper.appendChild(okBtn);\n\n if (settings.type == 'prompt' || settings.type == 'confirm') {\n\n wrapper.appendChild(cancelBtn);\n\n }\n\n wrapper.classList.add('cdx-notification-' + settings.type);\n wrapper.dataset.type = settings.type;\n\n notification = wrapper;\n type = settings.type;\n confirm = settings.confirm;\n cancel = settings.cancel;\n inputField = input;\n\n if (settings.type != 'prompt' && settings.type != 'confirm') {\n\n window.setTimeout(close, settings.time);\n\n }\n\n };\n\n function send() {\n\n editor.nodes.notifications.appendChild(notification);\n inputField.focus();\n\n editor.nodes.notifications.classList.add('cdx-notification__notification-appending');\n\n window.setTimeout(function () {\n\n editor.nodes.notifications.classList.remove('cdx-notification__notification-appending');\n\n }, 100);\n\n addToQueue({type: type, close: close});\n\n };\n\n function close() {\n\n notification.remove();\n\n };\n\n\n if (constructorSettings) {\n\n create(constructorSettings);\n send();\n\n }\n\n return {\n create: create,\n send: send,\n close: close\n };\n\n };\n\n notifications.clear = function () {\n\n editor.nodes.notifications.innerHTML = '';\n queue = [];\n\n };\n\n return notifications;\n\n})({});\n\n\n// WEBPACK FOOTER //\n// ./modules/notifications.js","/**\n * Codex Editor Parser Module\n *\n * @author Codex Team\n * @version 1.1\n */\n\nmodule.exports = (function (parser) {\n\n let editor = codex.editor;\n\n /** inserting text */\n parser.insertPastedContent = function (blockType, tag) {\n\n editor.content.insertBlock({\n type : blockType.type,\n block : blockType.render({\n text : tag.innerHTML\n })\n });\n\n };\n\n /**\n * Check DOM node for display style: separated block or child-view\n */\n parser.isFirstLevelBlock = function (node) {\n\n return node.nodeType == editor.core.nodeTypes.TAG &&\n node.classList.contains(editor.ui.className.BLOCK_CLASSNAME);\n\n };\n\n return parser;\n\n})({});\n\n\n\n// WEBPACK FOOTER //\n// ./modules/parser.js","/**\n * Codex Sanitizer\n */\n\nmodule.exports = (function (sanitizer) {\n\n /** HTML Janitor library */\n let janitor = require('html-janitor');\n\n /** Codex Editor */\n let editor = codex.editor;\n\n sanitizer.prepare = function () {\n\n if (editor.settings.sanitizer && !editor.core.isEmpty(editor.settings.sanitizer)) {\n\n Config.CUSTOM = editor.settings.sanitizer;\n\n }\n\n };\n\n /**\n * Basic config\n */\n var Config = {\n\n /** User configuration */\n CUSTOM : null,\n\n BASIC : {\n\n tags: {\n p: {},\n a: {\n href: true,\n target: '_blank',\n rel: 'nofollow'\n },\n i: {},\n b: {},\n strong: {},\n em: {},\n span: {}\n }\n }\n };\n\n sanitizer.Config = Config;\n\n /**\n *\n * @param userCustomConfig\n * @returns {*}\n * @private\n *\n * @description If developer uses editor's API, then he can customize sane restrictions.\n * Or, sane config can be defined globally in editors initialization. That config will be used everywhere\n * At least, if there is no config overrides, that API uses BASIC Default configation\n */\n let init_ = function (userCustomConfig) {\n\n let configuration = userCustomConfig || Config.CUSTOM || Config.BASIC;\n\n return new janitor(configuration);\n\n };\n\n /**\n * Cleans string from unwanted tags\n * @protected\n * @param {String} dirtyString - taint string\n * @param {Object} customConfig - allowed tags\n */\n sanitizer.clean = function(dirtyString, customConfig) {\n\n let janitorInstance = init_(customConfig);\n\n return janitorInstance.clean(dirtyString);\n\n };\n\n return sanitizer;\n\n})({});\n\n\n// WEBPACK FOOTER //\n// ./modules/sanitizer.js","(function (root, factory) {\n if (typeof define === 'function' && define.amd) {\n define('html-janitor', factory);\n } else if (typeof exports === 'object') {\n module.exports = factory();\n } else {\n root.HTMLJanitor = factory();\n }\n}(this, function () {\n\n /**\n * @param {Object} config.tags Dictionary of allowed tags.\n * @param {boolean} config.keepNestedBlockElements Default false.\n */\n function HTMLJanitor(config) {\n\n var tagDefinitions = config['tags'];\n var tags = Object.keys(tagDefinitions);\n\n var validConfigValues = tags\n .map(function(k) { return typeof tagDefinitions[k]; })\n .every(function(type) { return type === 'object' || type === 'boolean' || type === 'function'; });\n\n if(!validConfigValues) {\n throw new Error(\"The configuration was invalid\");\n }\n\n this.config = config;\n }\n\n // TODO: not exhaustive?\n var blockElementNames = ['P', 'LI', 'TD', 'TH', 'DIV', 'H1', 'H2', 'H3', 'H4', 'H5', 'H6', 'PRE'];\n function isBlockElement(node) {\n return blockElementNames.indexOf(node.nodeName) !== -1;\n }\n\n var inlineElementNames = ['A', 'B', 'STRONG', 'I', 'EM', 'SUB', 'SUP', 'U', 'STRIKE'];\n function isInlineElement(node) {\n return inlineElementNames.indexOf(node.nodeName) !== -1;\n }\n\n HTMLJanitor.prototype.clean = function (html) {\n var sandbox = document.createElement('div');\n sandbox.innerHTML = html;\n\n this._sanitize(sandbox);\n\n return sandbox.innerHTML;\n };\n\n HTMLJanitor.prototype._sanitize = function (parentNode) {\n var treeWalker = createTreeWalker(parentNode);\n var node = treeWalker.firstChild();\n if (!node) { return; }\n\n do {\n // Ignore nodes that have already been sanitized\n if (node._sanitized) {\n continue;\n }\n\n if (node.nodeType === Node.TEXT_NODE) {\n // If this text node is just whitespace and the previous or next element\n // sibling is a block element, remove it\n // N.B.: This heuristic could change. Very specific to a bug with\n // `contenteditable` in Firefox: http://jsbin.com/EyuKase/1/edit?js,output\n // FIXME: make this an option?\n if (node.data.trim() === ''\n && ((node.previousElementSibling && isBlockElement(node.previousElementSibling))\n || (node.nextElementSibling && isBlockElement(node.nextElementSibling)))) {\n parentNode.removeChild(node);\n this._sanitize(parentNode);\n break;\n } else {\n continue;\n }\n }\n\n // Remove all comments\n if (node.nodeType === Node.COMMENT_NODE) {\n parentNode.removeChild(node);\n this._sanitize(parentNode);\n break;\n }\n\n var isInline = isInlineElement(node);\n var containsBlockElement;\n if (isInline) {\n containsBlockElement = Array.prototype.some.call(node.childNodes, isBlockElement);\n }\n\n // Block elements should not be nested (e.g.

  • ...); if\n // they are, we want to unwrap the inner block element.\n var isNotTopContainer = !! parentNode.parentNode;\n var isNestedBlockElement =\n isBlockElement(parentNode) &&\n isBlockElement(node) &&\n isNotTopContainer;\n\n var nodeName = node.nodeName.toLowerCase();\n\n var allowedAttrs = getAllowedAttrs(this.config, nodeName, node);\n\n var isInvalid = isInline && containsBlockElement;\n\n // Drop tag entirely according to the whitelist *and* if the markup\n // is invalid.\n if (isInvalid || shouldRejectNode(node, allowedAttrs)\n || (!this.config.keepNestedBlockElements && isNestedBlockElement)) {\n // Do not keep the inner text of SCRIPT/STYLE elements.\n if (! (node.nodeName === 'SCRIPT' || node.nodeName === 'STYLE')) {\n while (node.childNodes.length > 0) {\n parentNode.insertBefore(node.childNodes[0], node);\n }\n }\n parentNode.removeChild(node);\n\n this._sanitize(parentNode);\n break;\n }\n\n // Sanitize attributes\n for (var a = 0; a < node.attributes.length; a += 1) {\n var attr = node.attributes[a];\n\n if (shouldRejectAttr(attr, allowedAttrs, node)) {\n node.removeAttribute(attr.name);\n // Shift the array to continue looping.\n a = a - 1;\n }\n }\n\n // Sanitize children\n this._sanitize(node);\n\n // Mark node as sanitized so it's ignored in future runs\n node._sanitized = true;\n } while ((node = treeWalker.nextSibling()));\n };\n\n function createTreeWalker(node) {\n return document.createTreeWalker(node,\n NodeFilter.SHOW_TEXT | NodeFilter.SHOW_ELEMENT | NodeFilter.SHOW_COMMENT,\n null, false);\n }\n\n function getAllowedAttrs(config, nodeName, node){\n if (typeof config.tags[nodeName] === 'function') {\n return config.tags[nodeName](node);\n } else {\n return config.tags[nodeName];\n }\n }\n\n function shouldRejectNode(node, allowedAttrs){\n if (typeof allowedAttrs === 'undefined') {\n return true;\n } else if (typeof allowedAttrs === 'boolean') {\n return !allowedAttrs;\n }\n\n return false;\n }\n\n function shouldRejectAttr(attr, allowedAttrs, node){\n var attrName = attr.name.toLowerCase();\n\n if (allowedAttrs === true){\n return false;\n } else if (typeof allowedAttrs[attrName] === 'function'){\n return !allowedAttrs[attrName](attr.value, node);\n } else if (typeof allowedAttrs[attrName] === 'undefined'){\n return true;\n } else if (allowedAttrs[attrName] === false) {\n return true;\n } else if (typeof allowedAttrs[attrName] === 'string') {\n return (allowedAttrs[attrName] !== attr.value);\n }\n\n return false;\n }\n\n return HTMLJanitor;\n\n}));\n\n\n\n//////////////////\n// WEBPACK FOOTER\n// ./~/html-janitor/src/html-janitor.js\n// module id = 18\n// module chunks = 0","/**\n * Codex Editor Listeners module\n *\n * @author Codex Team\n * @version 1.0\n */\n\n/**\n * Module-decorator for event listeners assignment\n */\nmodule.exports = function (listeners) {\n\n var allListeners = [];\n\n /**\n * Search methods\n *\n * byElement, byType and byHandler returns array of suitable listeners\n * one and all takes element, eventType, and handler and returns first (all) suitable listener\n *\n */\n listeners.search = function () {\n\n var byElement = function (element, context) {\n\n var listenersOnElement = [];\n\n context = context || allListeners;\n\n for (var i = 0; i < context.length; i++) {\n\n var listener = context[i];\n\n if (listener.element === element) {\n\n listenersOnElement.push(listener);\n\n }\n\n }\n\n return listenersOnElement;\n\n };\n\n var byType = function (eventType, context) {\n\n var listenersWithType = [];\n\n context = context || allListeners;\n\n for (var i = 0; i < context.length; i++) {\n\n var listener = context[i];\n\n if (listener.type === eventType) {\n\n listenersWithType.push(listener);\n\n }\n\n }\n\n return listenersWithType;\n\n };\n\n var byHandler = function (handler, context) {\n\n var listenersWithHandler = [];\n\n context = context || allListeners;\n\n for (var i = 0; i < context.length; i++) {\n\n var listener = context[i];\n\n if (listener.handler === handler) {\n\n listenersWithHandler.push(listener);\n\n }\n\n }\n\n return listenersWithHandler;\n\n };\n\n var one = function (element, eventType, handler) {\n\n var result = allListeners;\n\n if (element)\n result = byElement(element, result);\n\n if (eventType)\n result = byType(eventType, result);\n\n if (handler)\n result = byHandler(handler, result);\n\n return result[0];\n\n };\n\n var all = function (element, eventType, handler) {\n\n var result = allListeners;\n\n if (element)\n result = byElement(element, result);\n\n if (eventType)\n result = byType(eventType, result);\n\n if (handler)\n result = byHandler(handler, result);\n\n return result;\n\n };\n\n return {\n byElement : byElement,\n byType : byType,\n byHandler : byHandler,\n one : one,\n all : all\n };\n\n }();\n\n listeners.add = function (element, eventType, handler, isCapture) {\n\n element.addEventListener(eventType, handler, isCapture);\n\n var data = {\n element: element,\n type: eventType,\n handler: handler\n };\n\n var alreadyAddedListener = listeners.search.one(element, eventType, handler);\n\n if (!alreadyAddedListener) {\n\n allListeners.push(data);\n\n }\n\n };\n\n listeners.remove = function (element, eventType, handler) {\n\n element.removeEventListener(eventType, handler);\n\n var existingListeners = listeners.search.all(element, eventType, handler);\n\n for (var i = 0; i < existingListeners.length; i++) {\n\n var index = allListeners.indexOf(existingListeners[i]);\n\n if (index > 0) {\n\n allListeners.splice(index, 1);\n\n }\n\n }\n\n };\n\n listeners.removeAll = function () {\n\n allListeners.map(function (current) {\n\n listeners.remove(current.element, current.type, current.handler);\n\n });\n\n };\n\n listeners.get = function (element, eventType, handler) {\n\n return listeners.search.all(element, eventType, handler);\n\n };\n\n return listeners;\n\n}({});\n\n\n// WEBPACK FOOTER //\n// ./modules/listeners.js","/**\n * Codex Editor Destroyer module\n *\n * @auhor Codex Team\n * @version 1.0\n */\n\nmodule.exports = function (destroyer) {\n\n let editor = codex.editor;\n\n destroyer.removeNodes = function () {\n\n editor.nodes.wrapper.remove();\n editor.nodes.notifications.remove();\n\n };\n\n destroyer.destroyPlugins = function () {\n\n for (var tool in editor.tools) {\n\n if (typeof editor.tools[tool].destroy === 'function') {\n\n editor.tools[tool].destroy();\n\n }\n\n }\n\n };\n\n destroyer.destroyScripts = function () {\n\n var scripts = document.getElementsByTagName('SCRIPT');\n\n for (var i = 0; i < scripts.length; i++) {\n\n if (scripts[i].id.indexOf(editor.scriptPrefix) + 1) {\n\n scripts[i].remove();\n i--;\n\n }\n\n }\n\n };\n\n\n /**\n * Delete editor data from webpage.\n * You should send settings argument with boolean flags:\n * @param settings.ui- remove redactor event listeners and DOM nodes\n * @param settings.scripts - remove redactor scripts from DOM\n * @param settings.plugins - remove plugin's objects\n * @param settings.core - remove editor core. You can remove core only if UI and scripts flags is true\n * }\n *\n */\n destroyer.destroy = function (settings) {\n\n if (!settings || typeof settings !== 'object') {\n\n return;\n\n }\n\n if (settings.ui) {\n\n destroyer.removeNodes();\n editor.listeners.removeAll();\n\n }\n\n if (settings.scripts) {\n\n destroyer.destroyScripts();\n\n }\n\n if (settings.plugins) {\n\n destroyer.destroyPlugins();\n\n }\n\n if (settings.ui && settings.scripts && settings.core) {\n\n delete codex.editor;\n\n }\n\n };\n\n return destroyer;\n\n}({});\n\n\n// WEBPACK FOOTER //\n// ./modules/destroyer.js","/**\n * Codex Editor Paste module\n *\n * @author Codex Team\n * @version 1.1.1\n */\n\nmodule.exports = function (paste) {\n\n let editor = codex.editor;\n\n var patterns = [];\n\n paste.prepare = function () {\n\n var tools = editor.tools;\n\n for (var tool in tools) {\n\n if (!tools[tool].renderOnPastePatterns || !Array.isArray(tools[tool].renderOnPastePatterns)) {\n\n continue;\n\n }\n\n tools[tool].renderOnPastePatterns.map(function (pattern) {\n\n\n patterns.push(pattern);\n\n });\n\n }\n\n return Promise.resolve();\n\n };\n\n /**\n * Saves data\n * @param event\n */\n paste.pasted = function (event) {\n\n var clipBoardData = event.clipboardData || window.clipboardData,\n content = clipBoardData.getData('Text');\n\n var result = analize(content);\n\n if (result) {\n\n event.preventDefault();\n event.stopImmediatePropagation();\n\n }\n\n return result;\n\n };\n\n /**\n * Analizes pated string and calls necessary method\n */\n\n var analize = function (string) {\n\n var result = false,\n content = editor.content.currentNode,\n plugin = content.dataset.tool;\n\n patterns.map( function (pattern) {\n\n var execArray = pattern.regex.exec(string),\n match = execArray && execArray[0];\n\n if ( match && match === string.trim()) {\n\n /** current block is not empty */\n if ( content.textContent.trim() && plugin == editor.settings.initialBlockPlugin ) {\n\n pasteToNewBlock_();\n\n }\n\n pattern.callback(string, pattern);\n result = true;\n\n }\n\n });\n\n return result;\n\n };\n\n var pasteToNewBlock_ = function () {\n\n /** Create new initial block */\n editor.content.insertBlock({\n\n type : editor.settings.initialBlockPlugin,\n block : editor.tools[editor.settings.initialBlockPlugin].render({\n text : ''\n })\n\n }, false);\n\n };\n\n return paste;\n\n}({});\n\n\n// WEBPACK FOOTER //\n// ./modules/paste.js"],"sourceRoot":""} \ No newline at end of file +{"version":3,"sources":["webpack:///codex-editor.js","webpack:///webpack/bootstrap b4a1ef56ed45877046f1","webpack:///./codex.js","webpack:///./modules/core.js","webpack:///./modules/tools.js","webpack:///./modules/ui.js","webpack:///./modules/transport.js","webpack:///./modules/renderer.js","webpack:///./modules/saver.js","webpack:///./modules/content.js","webpack:///./modules/toolbar/toolbar.js","webpack:///./modules/toolbar/settings.js","webpack:///./modules/toolbar/inline.js","webpack:///./modules/toolbar/toolbox.js","webpack:///./modules/callbacks.js","webpack:///./modules/draw.js","webpack:///./modules/caret.js","webpack:///./modules/notifications.js","webpack:///./modules/parser.js","webpack:///./modules/sanitizer.js","webpack:///./~/html-janitor/src/html-janitor.js","webpack:///./modules/listeners.js","webpack:///./modules/destroyer.js","webpack:///./modules/paste.js"],"names":["codex","modules","__webpack_require__","moduleId","installedModules","exports","module","id","loaded","call","m","c","p","editor","version","scriptPrefix","init","core","tools","ui","transport","renderer","saver","content","toolbar","callback","draw","caret","notifications","parser","sanitizer","listeners","destroyer","paste","settings","holderId","initialBlockPlugin","nodes","holder","wrapper","inlineToolbar","buttons","actions","toolbox","plusButton","showSettingsButton","showTrashButton","blockSettings","pluginSettings","defaultSettings","toolbarButtons","redactor","state","jsonOutput","blocks","inputs","start","userSettings","prepare","then","makeBlocksFromData","saveInputs","catch","error","log","_typeof","Symbol","iterator","obj","constructor","prototype","Promise","resolve","reject","data","hideToolbar","document","getElementById","undefined","Error","msg","type","arg","window","console","e","insertAfter","target","element","parentNode","insertBefore","nextSibling","nodeTypes","TAG","TEXT","COMMENT","keys","BACKSPACE","TAB","ENTER","SHIFT","CTRL","ALT","ESC","SPACE","LEFT","UP","DOWN","RIGHT","DELETE","META","isDomNode","el","nodeType","this","isEmpty","Object","length","ajax","url","encodedString","isFormData","prop","XMLHTTP","XMLHttpRequest","ActiveXObject","async","test","encodeURIComponent","withCredentials","responseContext","beforeSendResult","beforeSend","open","isFormData_","setRequestHeader","progress","upload","onprogress","bind","onreadystatechange","readyState","status","success","responseText","send","importScript","scriptPath","instanceName","script","createElement","defer","onload","onerror","src","head","appendChild","object","FormData","resolve_","reject_","pluginsRequiresPreparation","allPlugins","pluginName","plugin","push","waitAllPluginsPreparation_","plugins","allPluginsProcessed__","reduce","previousValue","iteration","pluginIsReady__","callPluginsPrepareMethod_","available","loadingMessage","config","className","BLOCK_CLASSNAME","BLOCK_CONTENT","BLOCK_STRETCHED","BLOCK_HIGHLIGHTED","SETTINGS_ITEM","makeToolBar_","addTools_","makeInlineToolbar_","addInlineToolbarTools_","makeNotificationHolder_","bindEvents_","container","inlineToolbarButtons","inlineToolbarActions","blockButtons","makeToolbarSettings_","toolbarContent","makeToolbarContent_","settingsButton","makeRemoveBlockButton","pluginsSettings","createHolder","tool","toolName","toolButton","iconClassname","displayInToolbox","render","toolbarButton","bold","icon","command","italic","underline","link","name","toolbarButtonInline","setInlineToolbarButtonBehaviour","add","globalKeydown","redactorKeyDown","globalKeyup","redactorClicked","plusButtonClicked","showSettingsButtonClicked","button","toolbarButtonClicked","addBlockHandlers","block","blockKeydown","blockPasteCallback","inline","show","querySelectorAll","Array","map","current","addInitialBlock","initialBlock","initialBlockType","setAttribute","insertBlock","workingNodeChanged","event","toolClicked","currentRequest","input","arguments","node","fileSelected","clearInput","i","files","formData","multiple","append","selectAndUpload","args","accept","click","abort","items","appendBlocks","nodeSequence","index","appendNodeAtIndex","getNodeAsync","createBlockFromData","blockData","blocksList","position","toolData","unavailableBlock","innerHTML","dataset","inputPosition","stretched","isStretched","saveBlocks","html","childNodes","makeQueue","queue","getBlockData","makeFormDataFromBlocks","save","savedData","output","blockContent","pluginsContent","validate","result","currentNode","editorAreaHightlighted","sync","markBlock","classList","clearMark","remove","getFirstLevelBlock","body","contains","targetNode","replaceBlock","targetBlock","newBlock","replaceChild","needPlaceCaret","workingBlock","newBlockContent","blockType","composeNewBlock_","currentInputIndex","getCurrentInputIndex","editableElement","querySelector","emptyText","createTextNode","set","move","showPlusButton","setTimeout","setToNextBlock","switchBlock","blockToReplace","newBlockComposed","getDeepestTextNodeFromPosition","text","blockChilds","textContent","trim","removeChild","lookingFromStart","getRange","selection","getSelection","getRangeAt","splitBlock","inputIndex","textBeforeCaret","textNodeBeforeCaret","textAfterCaret","textNodeAfterCaret","anchorNode","anchorNodeText","caretOffset","anchorOffset","currentBlock","substring","previousChilds","nextChilds","reachedCurrent","child","previousChildsLength","nextChildsLength","newNode","NEW_BLOCK_TYPE","mergeBlocks","targetInputIndex","targetInput","currentInputContent","isLastNode","allChecked","allSiblingsEmpty_","sibling","wrapTextWithParagraphs","htmlData","plainData","split","join","paragraph","blockTyped","newWrapper","firstLevelBlocks","indexOf","tagName","cloneNode","getEditableParent","contentEditable","defaultToolbarHeight","defaultOffset","opened","close","toggle","hidePlusButton","newYCoordinate","offsetTop","style","transform","Math","floor","hideRemoveActions","setting","toolType","makeSettings","settingsBlock","removeBlockWrapper","settingButton","actionWrapper","confirmAction","cancelAction","removeButtonClicked","confirmRemovingRequest","cancelRemovingRequest","action","showRemoveActions","firstLevelBlocksCount","buttonsOpened","actionsOpened","wrappersOffset","storedSelection","showInlineToolbar","selectedText","getSelectionText","showButtons","getWrappersOffset","newCoordinateX","newCoordinateY","coords","getSelectionCoords","offsetHeight","x","left","y","scrollY","top","closeButtons","closeAction","createLinkAction","defaultToolAction","forEach","hightlight","offset","getOffset","_x","_y","isNaN","offsetLeft","clientLeft","clientTop","offsetParent","range","sel","createRange","collapse","boundingLeft","boundingTop","rangeCount","cloneRange","getClientRects","rect","toString","showActions","inlineToolbarAnchorInputKeydown_","keyCode","editable","restoreSelection","setAnchor","value","preventDefault","stopImmediatePropagation","clearRange","isActive","isLinkActive","saveSelection","inputForLink","focus","dataType","execCommand","containerEl","preSelectionRange","selectNodeContents","setEnd","startContainer","startOffset","end","savedSel","charIndex","setStart","nextCharIndex","nodeStack","foundStart","stop","pop","removeAllRanges","addRange","queryCommandState","setButtonHighlighted","removeButtonsHighLight","tag","leaf","currentTool","barButtons","nextToolIndex","toolToSelect","visibleTool","appendCallback","UNREPLACEBLE_TOOLS","workingNode","setToBlock","callbacks","enterKeyPressed_","tabKeyPressedOnRedactorsZone_","enterKeyPressedOnRedactorsZone_","escapeKeyPressedOnRedactorsZone_","defaultKeyPressedOnRedactorsZone_","arrowKeyPressed_","nativeInputs","nativeInputsAreEmpty","textContentIsEmpty","blockIsEmpty","enterPressedOnBlock_","saveCurrentInputIndex","isEnterPressedOnToolbar","enableLineBreaks","stopPropagation","shiftKey","currentSelection","currentSelectedNode","caretAtTheEndOfText","atTheEnd","isTextNodeHasParentBetweenContenteditable","enterPressedOnBlock","islastNode","detectWhenClickedOnFirstLevelBlockArea_","firstLevelBlock","indexOfLastInput","inputIsEmpty","currentNodeType","isInitialType","flag","blockRightOrDownArrowPressed_","backspacePressed_","blockLeftOrUpArrowPressed_","focusedNodeHolder","focusedNode","editableElementIndex","lastChild","deepestTextnode","caretInLastChild","firstChild","caretInFirstChild","caretAtTheBeginning","setToPreviousBlock","selectionLength","isNativeInput_","endOffset","atStart","editableParent","cleanData","wrappedData","clipboardData","getData","div","clean","currentBlockContent","insertNode","currentToolType","nativeInputAreas","ceBlock","bar","placeholder","toggler","classname","toolIcon","toolTitle","properties","focusedNodeIndex","nodeToSet","childs","nextInput","emptyTextElement","lastChildNode","lengthOfLastChildNode","previousInput","pluginsRender","isFirstNode","isOffsetZero","deleteContents","setStartAfter","addToQueue","splice","errorThrown","errorMsg","notification","message","constructorSettings","create","time","okBtn","cancelBtn","okMsg","cancelMsg","confirmHandler","cancelHandler","confirm","cancel","inputField","clear","insertPastedContent","isFirstLevelBlock","janitor","Config","CUSTOM","BASIC","tags","a","href","rel","b","strong","em","span","init_","userCustomConfig","configuration","dirtyString","customConfig","janitorInstance","__WEBPACK_AMD_DEFINE_FACTORY__","__WEBPACK_AMD_DEFINE_RESULT__","root","factory","HTMLJanitor","tagDefinitions","validConfigValues","k","every","isBlockElement","blockElementNames","nodeName","isInlineElement","inlineElementNames","createTreeWalker","NodeFilter","SHOW_TEXT","SHOW_ELEMENT","SHOW_COMMENT","getAllowedAttrs","shouldRejectNode","allowedAttrs","shouldRejectAttr","attr","attrName","toLowerCase","sandbox","_sanitize","treeWalker","_sanitized","Node","TEXT_NODE","COMMENT_NODE","containsBlockElement","isInline","some","isNotTopContainer","isNestedBlockElement","isInvalid","keepNestedBlockElements","attributes","removeAttribute","previousElementSibling","nextElementSibling","allListeners","search","byElement","context","listenersOnElement","listener","byType","eventType","listenersWithType","byHandler","handler","listenersWithHandler","one","all","isCapture","addEventListener","alreadyAddedListener","removeEventListener","existingListeners","removeAll","get","removeNodes","destroyPlugins","destroy","destroyScripts","scripts","getElementsByTagName","patterns","renderOnPastePatterns","isArray","pattern","pasted","clipBoardData","analize","string","execArray","regex","exec","match","pasteToNewBlock_"],"mappings":"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","file":"codex-editor.js","sourcesContent":["var codex = codex || {}; codex[\"editor\"] =\n/******/ (function(modules) { // webpackBootstrap\n/******/ \t// The module cache\n/******/ \tvar installedModules = {};\n/******/\n/******/ \t// The require function\n/******/ \tfunction __webpack_require__(moduleId) {\n/******/\n/******/ \t\t// Check if module is in cache\n/******/ \t\tif(installedModules[moduleId])\n/******/ \t\t\treturn installedModules[moduleId].exports;\n/******/\n/******/ \t\t// Create a new module (and put it into the cache)\n/******/ \t\tvar module = installedModules[moduleId] = {\n/******/ \t\t\texports: {},\n/******/ \t\t\tid: moduleId,\n/******/ \t\t\tloaded: false\n/******/ \t\t};\n/******/\n/******/ \t\t// Execute the module function\n/******/ \t\tmodules[moduleId].call(module.exports, module, module.exports, __webpack_require__);\n/******/\n/******/ \t\t// Flag the module as loaded\n/******/ \t\tmodule.loaded = true;\n/******/\n/******/ \t\t// Return the exports of the module\n/******/ \t\treturn module.exports;\n/******/ \t}\n/******/\n/******/\n/******/ \t// expose the modules object (__webpack_modules__)\n/******/ \t__webpack_require__.m = modules;\n/******/\n/******/ \t// expose the module cache\n/******/ \t__webpack_require__.c = installedModules;\n/******/\n/******/ \t// __webpack_public_path__\n/******/ \t__webpack_require__.p = \"\";\n/******/\n/******/ \t// Load entry module and return exports\n/******/ \treturn __webpack_require__(0);\n/******/ })\n/************************************************************************/\n/******/ ([\n/* 0 */\n/***/ function(module, exports, __webpack_require__) {\n\n\t'use strict';\n\t\n\t/**\r\n\t *\r\n\t * Codex Editor\r\n\t *\r\n\t * @author Codex Team\r\n\t */\n\t\n\tmodule.exports = function (editor) {\n\t\n\t 'use strict';\n\t\n\t editor.version = (\"1.6.3\");\n\t editor.scriptPrefix = 'cdx-script-';\n\t\n\t var init = function init() {\n\t\n\t editor.core = __webpack_require__(1);\n\t editor.tools = __webpack_require__(2);\n\t editor.ui = __webpack_require__(3);\n\t editor.transport = __webpack_require__(4);\n\t editor.renderer = __webpack_require__(5);\n\t editor.saver = __webpack_require__(6);\n\t editor.content = __webpack_require__(7);\n\t editor.toolbar = __webpack_require__(8);\n\t editor.callback = __webpack_require__(12);\n\t editor.draw = __webpack_require__(13);\n\t editor.caret = __webpack_require__(14);\n\t editor.notifications = __webpack_require__(15);\n\t editor.parser = __webpack_require__(16);\n\t editor.sanitizer = __webpack_require__(17);\n\t editor.listeners = __webpack_require__(19);\n\t editor.destroyer = __webpack_require__(20);\n\t editor.paste = __webpack_require__(21);\n\t };\n\t\n\t /**\r\n\t * @public\r\n\t * holds initial settings\r\n\t */\n\t editor.settings = {\n\t tools: ['paragraph', 'header', 'picture', 'list', 'quote', 'code', 'twitter', 'instagram', 'smile'],\n\t holderId: 'codex-editor',\n\t\n\t // Type of block showing on empty editor\n\t initialBlockPlugin: 'paragraph'\n\t };\n\t\n\t /**\r\n\t * public\r\n\t *\r\n\t * Static nodes\r\n\t */\n\t editor.nodes = {\n\t holder: null,\n\t wrapper: null,\n\t toolbar: null,\n\t inlineToolbar: {\n\t wrapper: null,\n\t buttons: null,\n\t actions: null\n\t },\n\t toolbox: null,\n\t notifications: null,\n\t plusButton: null,\n\t showSettingsButton: null,\n\t showTrashButton: null,\n\t blockSettings: null,\n\t pluginSettings: null,\n\t defaultSettings: null,\n\t toolbarButtons: {}, // { type : DomEl, ... }\n\t redactor: null\n\t };\n\t\n\t /**\r\n\t * @public\r\n\t *\r\n\t * Output state\r\n\t */\n\t editor.state = {\n\t jsonOutput: [],\n\t blocks: [],\n\t inputs: []\n\t };\n\t\n\t /**\r\n\t * @public\r\n\t * Editor plugins\r\n\t */\n\t editor.tools = {};\n\t\n\t /**\r\n\t * Initialization\r\n\t * @uses Promise cEditor.core.prepare\r\n\t * @param {Object} userSettings\r\n\t * @param {Array} userSettings.tools list of plugins\r\n\t * @param {String} userSettings.holderId Element's id to append editor\r\n\t *\r\n\t * Load user defined tools\r\n\t * Tools must contain this important objects :\r\n\t * @param {String} type - this is a type of plugin. It can be used as plugin name\r\n\t * @param {String} iconClassname - this a icon in toolbar\r\n\t * @param {Object} make - what should plugin do, when it is clicked\r\n\t * @param {Object} appendCallback - callback after clicking\r\n\t * @param {Element} settings - what settings does it have\r\n\t * @param {Object} render - plugin get JSON, and should return HTML\r\n\t * @param {Object} save - plugin gets HTML content, returns JSON\r\n\t * @param {Boolean} displayInToolbox - will be displayed in toolbox. Default value is TRUE\r\n\t * @param {Boolean} enableLineBreaks - inserts new block or break lines. Default value is FALSE\r\n\t *\r\n\t * @example\r\n\t * - type : 'header',\r\n\t * - iconClassname : 'ce-icon-header',\r\n\t * - make : headerTool.make,\r\n\t * - appendCallback : headerTool.appendCallback,\r\n\t * - settings : headerTool.makeSettings(),\r\n\t * - render : headerTool.render,\r\n\t * - save : headerTool.save,\r\n\t * - displayInToolbox : true,\r\n\t * - enableLineBreaks : false\r\n\t */\n\t editor.start = function (userSettings) {\n\t\n\t init();\n\t\n\t editor.core.prepare(userSettings)\n\t\n\t // If all ok, make UI, bind events and parse initial-content\n\t .then(editor.ui.prepare).then(editor.tools.prepare).then(editor.sanitizer.prepare).then(editor.paste.prepare).then(editor.transport.prepare).then(editor.renderer.makeBlocksFromData).then(editor.ui.saveInputs).catch(function (error) {\n\t\n\t editor.core.log('Initialization failed with error: %o', 'warn', error);\n\t });\n\t };\n\t\n\t return editor;\n\t}({});\n\n/***/ },\n/* 1 */\n/***/ function(module, exports) {\n\n\t'use strict';\n\t\n\tvar _typeof = typeof Symbol === \"function\" && typeof Symbol.iterator === \"symbol\" ? function (obj) { return typeof obj; } : function (obj) { return obj && typeof Symbol === \"function\" && obj.constructor === Symbol && obj !== Symbol.prototype ? \"symbol\" : typeof obj; };\n\t\n\t/**\r\n\t * Codex Editor Core\r\n\t *\r\n\t * @author Codex Team\r\n\t * @version 1.1.3\r\n\t */\n\t\n\tmodule.exports = function (core) {\n\t\n\t var editor = codex.editor;\n\t\n\t /**\r\n\t * @public\r\n\t *\r\n\t * Editor preparing method\r\n\t * @return Promise\r\n\t */\n\t core.prepare = function (userSettings) {\n\t\n\t return new Promise(function (resolve, reject) {\n\t\n\t if (userSettings) {\n\t\n\t editor.settings.tools = userSettings.tools || editor.settings.tools;\n\t }\n\t\n\t if (userSettings.data) {\n\t\n\t editor.state.blocks = userSettings.data;\n\t }\n\t\n\t if (userSettings.initialBlockPlugin) {\n\t\n\t editor.settings.initialBlockPlugin = userSettings.initialBlockPlugin;\n\t }\n\t\n\t if (userSettings.sanitizer) {\n\t\n\t editor.settings.sanitizer = userSettings.sanitizer;\n\t }\n\t\n\t editor.hideToolbar = userSettings.hideToolbar;\n\t\n\t editor.nodes.holder = document.getElementById(userSettings.holderId || editor.settings.holderId);\n\t\n\t if (_typeof(editor.nodes.holder) === undefined || editor.nodes.holder === null) {\n\t\n\t reject(Error(\"Holder wasn't found by ID: #\" + userSettings.holderId));\n\t } else {\n\t\n\t resolve();\n\t }\n\t });\n\t };\n\t\n\t /**\r\n\t * Logging method\r\n\t * @param type = ['log', 'info', 'warn']\r\n\t */\n\t core.log = function (msg, type, arg) {\n\t\n\t type = type || 'log';\n\t\n\t if (!arg) {\n\t\n\t arg = msg || 'undefined';\n\t msg = '[codex-editor]: %o';\n\t } else {\n\t\n\t msg = '[codex-editor]: ' + msg;\n\t }\n\t\n\t try {\n\t\n\t if ('console' in window && window.console[type]) {\n\t\n\t if (arg) window.console[type](msg, arg);else window.console[type](msg);\n\t }\n\t } catch (e) {}\n\t };\n\t\n\t /**\r\n\t * @protected\r\n\t *\r\n\t * Helper for insert one element after another\r\n\t */\n\t core.insertAfter = function (target, element) {\n\t\n\t target.parentNode.insertBefore(element, target.nextSibling);\n\t };\n\t\n\t /**\r\n\t * @const\r\n\t *\r\n\t * Readable DOM-node types map\r\n\t */\n\t core.nodeTypes = {\n\t TAG: 1,\n\t TEXT: 3,\n\t COMMENT: 8\n\t };\n\t\n\t /**\r\n\t * @const\r\n\t * Readable keys map\r\n\t */\n\t core.keys = { BACKSPACE: 8, TAB: 9, ENTER: 13, SHIFT: 16, CTRL: 17, ALT: 18, ESC: 27, SPACE: 32, LEFT: 37, UP: 38, DOWN: 40, RIGHT: 39, DELETE: 46, META: 91 };\n\t\n\t /**\r\n\t * @protected\r\n\t *\r\n\t * Check object for DOM node\r\n\t */\n\t core.isDomNode = function (el) {\n\t\n\t return el && (typeof el === 'undefined' ? 'undefined' : _typeof(el)) === 'object' && el.nodeType && el.nodeType == this.nodeTypes.TAG;\n\t };\n\t\n\t /**\r\n\t * Checks passed object for emptiness\r\n\t * @require ES5 - Object.keys\r\n\t * @param {object}\r\n\t */\n\t core.isEmpty = function (obj) {\n\t\n\t return Object.keys(obj).length === 0;\n\t };\n\t\n\t /**\r\n\t * Native Ajax\r\n\t * @param {String} settings.url - request URL\r\n\t * @param {function} settings.beforeSend - returned value will be passed as context to the Success, Error and Progress callbacks\r\n\t * @param {function} settings.success\r\n\t * @param {function} settings.progress\r\n\t */\n\t core.ajax = function (settings) {\n\t\n\t if (!settings || !settings.url) {\n\t\n\t return;\n\t }\n\t\n\t var XMLHTTP = window.XMLHttpRequest ? new XMLHttpRequest() : new ActiveXObject('Microsoft.XMLHTTP'),\n\t encodedString,\n\t isFormData,\n\t prop;\n\t\n\t settings.async = true;\n\t settings.type = settings.type || 'GET';\n\t settings.data = settings.data || '';\n\t settings['content-type'] = settings['content-type'] || 'application/json; charset=utf-8';\n\t\n\t if (settings.type == 'GET' && settings.data) {\n\t\n\t settings.url = /\\?/.test(settings.url) ? settings.url + '&' + settings.data : settings.url + '?' + settings.data;\n\t } else {\n\t\n\t encodedString = '';\n\t for (prop in settings.data) {\n\t\n\t encodedString += prop + '=' + encodeURIComponent(settings.data[prop]) + '&';\n\t }\n\t }\n\t\n\t if (settings.withCredentials) {\n\t\n\t XMLHTTP.withCredentials = true;\n\t }\n\t\n\t /**\r\n\t * Value returned in beforeSend funtion will be passed as context to the other response callbacks\r\n\t * If beforeSend returns false, AJAX will be blocked\r\n\t */\n\t var responseContext = void 0,\n\t beforeSendResult = void 0;\n\t\n\t if (typeof settings.beforeSend === 'function') {\n\t\n\t beforeSendResult = settings.beforeSend.call();\n\t\n\t if (beforeSendResult === false) {\n\t\n\t return;\n\t }\n\t }\n\t\n\t XMLHTTP.open(settings.type, settings.url, settings.async);\n\t\n\t /**\r\n\t * If we send FormData, we need no content-type header\r\n\t */\n\t isFormData = isFormData_(settings.data);\n\t\n\t if (!isFormData) {\n\t\n\t if (settings.type !== 'POST') {\n\t\n\t XMLHTTP.setRequestHeader('Content-type', settings['content-type']);\n\t } else {\n\t\n\t XMLHTTP.setRequestHeader('Content-type', 'application/x-www-form-urlencoded');\n\t }\n\t }\n\t\n\t XMLHTTP.setRequestHeader('X-Requested-With', 'XMLHttpRequest');\n\t\n\t responseContext = beforeSendResult || XMLHTTP;\n\t\n\t if (typeof settings.progress === 'function') {\n\t\n\t XMLHTTP.upload.onprogress = settings.progress.bind(responseContext);\n\t }\n\t\n\t XMLHTTP.onreadystatechange = function () {\n\t\n\t if (XMLHTTP.readyState === 4) {\n\t\n\t if (XMLHTTP.status === 200) {\n\t\n\t if (typeof settings.success === 'function') {\n\t\n\t settings.success.call(responseContext, XMLHTTP.responseText);\n\t }\n\t } else {\n\t\n\t if (typeof settings.error === 'function') {\n\t\n\t settings.error.call(responseContext, XMLHTTP.responseText, XMLHTTP.status);\n\t }\n\t }\n\t }\n\t };\n\t\n\t if (isFormData) {\n\t\n\t // Sending FormData\n\t XMLHTTP.send(settings.data);\n\t } else {\n\t\n\t // POST requests\n\t XMLHTTP.send(encodedString);\n\t }\n\t\n\t return XMLHTTP;\n\t };\n\t\n\t /**\r\n\t * Appends script to head of document\r\n\t * @return Promise\r\n\t */\n\t core.importScript = function (scriptPath, instanceName) {\n\t\n\t return new Promise(function (resolve, reject) {\n\t\n\t var script = void 0;\n\t\n\t /** Script is already loaded */\n\t if (!instanceName) {\n\t\n\t reject('Instance name is missed');\n\t } else if (document.getElementById(editor.scriptPrefix + instanceName)) {\n\t\n\t resolve(scriptPath);\n\t }\n\t\n\t script = document.createElement('SCRIPT');\n\t script.async = true;\n\t script.defer = true;\n\t script.id = editor.scriptPrefix + instanceName;\n\t\n\t script.onload = function () {\n\t\n\t resolve(scriptPath);\n\t };\n\t\n\t script.onerror = function () {\n\t\n\t reject(scriptPath);\n\t };\n\t\n\t script.src = scriptPath;\n\t document.head.appendChild(script);\n\t });\n\t };\n\t\n\t /**\r\n\t * Function for checking is it FormData object to send.\r\n\t * @param {Object} object to check\r\n\t * @return boolean\r\n\t */\n\t var isFormData_ = function isFormData_(object) {\n\t\n\t return object instanceof FormData;\n\t };\n\t\n\t return core;\n\t}({});\n\n/***/ },\n/* 2 */\n/***/ function(module, exports) {\n\n\t'use strict';\n\t\n\t/**\r\n\t* Module working with plugins\r\n\t*/\n\tmodule.exports = function () {\n\t\n\t var editor = codex.editor;\n\t\n\t /**\r\n\t * Initialize plugins before using\r\n\t * Ex. Load scripts or call some internal methods\r\n\t * @return Promise\r\n\t */\n\t function prepare() {\n\t\n\t return new Promise(function (resolve_, reject_) {\n\t\n\t Promise.resolve()\n\t\n\t /**\r\n\t * Compose a sequence of plugins that requires preparation\r\n\t */\n\t .then(function () {\n\t\n\t var pluginsRequiresPreparation = [],\n\t allPlugins = editor.tools;\n\t\n\t for (var pluginName in allPlugins) {\n\t\n\t var plugin = allPlugins[pluginName];\n\t\n\t if (plugin.prepare && typeof plugin.prepare != 'function' || !plugin.prepare) {\n\t\n\t continue;\n\t }\n\t\n\t pluginsRequiresPreparation.push(plugin);\n\t }\n\t\n\t /**\r\n\t * If no one passed plugins requires preparation, finish prepare() and go ahead\r\n\t */\n\t if (!pluginsRequiresPreparation.length) {\n\t\n\t resolve_();\n\t }\n\t\n\t return pluginsRequiresPreparation;\n\t })\n\t\n\t /** Wait plugins while they prepares */\n\t .then(waitAllPluginsPreparation_).then(function () {\n\t\n\t editor.core.log('Plugins loaded', 'info');\n\t resolve_();\n\t }).catch(function (error) {\n\t\n\t reject_(error);\n\t });\n\t });\n\t }\n\t\n\t /**\r\n\t * @param {array} plugins - list of tools that requires preparation\r\n\t * @return {Promise} resolved while all plugins will be ready or failed\r\n\t */\n\t function waitAllPluginsPreparation_(plugins) {\n\t\n\t /**\r\n\t * @calls allPluginsProcessed__ when all plugins prepared or failed\r\n\t */\n\t return new Promise(function (allPluginsProcessed__) {\n\t\n\t /**\r\n\t * pluck each element from queue\r\n\t * First, send resolved Promise as previous value\r\n\t * Each plugins \"prepare\" method returns a Promise, that's why\r\n\t * reduce current element will not be able to continue while can't get\r\n\t * a resolved Promise\r\n\t *\r\n\t * If last plugin is \"prepared\" then go to the next stage of initialization\r\n\t */\n\t plugins.reduce(function (previousValue, plugin, iteration) {\n\t\n\t return previousValue.then(function () {\n\t\n\t /**\r\n\t * Wait till plugins prepared\r\n\t * @calls pluginIsReady__ when plugin is ready or failed\r\n\t */\n\t return new Promise(function (pluginIsReady__) {\n\t\n\t callPluginsPrepareMethod_(plugin).then(pluginIsReady__).then(function () {\n\t\n\t plugin.available = true;\n\t }).catch(function (error) {\n\t\n\t editor.core.log('Plugin \\xAB' + plugin.type + '\\xBB was not loaded. Preparation failed because %o', 'warn', error);\n\t plugin.available = false;\n\t plugin.loadingMessage = error;\n\t\n\t /** Go ahead even some plugin has problems */\n\t pluginIsReady__();\n\t }).then(function () {\n\t\n\t /** If last plugin has problems then just ignore and continue */\n\t if (iteration == plugins.length - 1) {\n\t\n\t allPluginsProcessed__();\n\t }\n\t });\n\t });\n\t });\n\t }, Promise.resolve());\n\t });\n\t }\n\t\n\t var callPluginsPrepareMethod_ = function callPluginsPrepareMethod_(plugin) {\n\t\n\t return plugin.prepare(plugin.config || {});\n\t };\n\t\n\t return {\n\t prepare: prepare\n\t };\n\t}();\n\n/***/ },\n/* 3 */\n/***/ function(module, exports) {\n\n\t'use strict';\n\t\n\t/**\r\n\t * Codex Editor UI module\r\n\t *\r\n\t * @author Codex Team\r\n\t * @version 1.2.0\r\n\t */\n\t\n\tmodule.exports = function (ui) {\n\t\n\t var editor = codex.editor;\n\t\n\t /**\r\n\t * Basic editor classnames\r\n\t */\n\t ui.className = {\n\t\n\t /**\r\n\t * @const {string} BLOCK_CLASSNAME - redactor blocks name\r\n\t */\n\t BLOCK_CLASSNAME: 'ce-block',\n\t\n\t /**\r\n\t * @const {String} wrapper for plugins content\r\n\t */\n\t BLOCK_CONTENT: 'ce-block__content',\n\t\n\t /**\r\n\t * @const {String} BLOCK_STRETCHED - makes block stretched\r\n\t */\n\t BLOCK_STRETCHED: 'ce-block--stretched',\n\t\n\t /**\r\n\t * @const {String} BLOCK_HIGHLIGHTED - adds background\r\n\t */\n\t BLOCK_HIGHLIGHTED: 'ce-block--focused',\n\t\n\t /**\r\n\t * @const {String} - for all default settings\r\n\t */\n\t SETTINGS_ITEM: 'ce-settings__item'\n\t\n\t };\n\t\n\t /**\r\n\t * @protected\r\n\t *\r\n\t * Making main interface\r\n\t */\n\t ui.prepare = function () {\n\t\n\t return new Promise(function (resolve) {\n\t\n\t var wrapper = editor.draw.wrapper(),\n\t redactor = editor.draw.redactor(),\n\t toolbar = makeToolBar_();\n\t\n\t wrapper.appendChild(toolbar);\n\t wrapper.appendChild(redactor);\n\t\n\t /** Save created ui-elements to static nodes state */\n\t editor.nodes.wrapper = wrapper;\n\t editor.nodes.redactor = redactor;\n\t\n\t /** Append editor wrapper with redactor zone into holder */\n\t editor.nodes.holder.appendChild(wrapper);\n\t\n\t resolve();\n\t })\n\t\n\t /** Add toolbox tools */\n\t .then(addTools_)\n\t\n\t /** Make container for inline toolbar */\n\t .then(makeInlineToolbar_)\n\t\n\t /** Add inline toolbar tools */\n\t .then(addInlineToolbarTools_)\n\t\n\t /** Draw wrapper for notifications */\n\t .then(makeNotificationHolder_)\n\t\n\t /** Add eventlisteners to redactor elements */\n\t .then(bindEvents_).catch(function () {\n\t\n\t editor.core.log(\"Can't draw editor interface\");\n\t });\n\t };\n\t\n\t /**\r\n\t * @private\r\n\t * Draws inline toolbar zone\r\n\t */\n\t var makeInlineToolbar_ = function makeInlineToolbar_() {\n\t\n\t var container = editor.draw.inlineToolbar();\n\t\n\t /** Append to redactor new inline block */\n\t editor.nodes.inlineToolbar.wrapper = container;\n\t\n\t /** Draw toolbar buttons */\n\t editor.nodes.inlineToolbar.buttons = editor.draw.inlineToolbarButtons();\n\t\n\t /** Buttons action or settings */\n\t editor.nodes.inlineToolbar.actions = editor.draw.inlineToolbarActions();\n\t\n\t /** Append to inline toolbar buttons as part of it */\n\t editor.nodes.inlineToolbar.wrapper.appendChild(editor.nodes.inlineToolbar.buttons);\n\t editor.nodes.inlineToolbar.wrapper.appendChild(editor.nodes.inlineToolbar.actions);\n\t\n\t editor.nodes.wrapper.appendChild(editor.nodes.inlineToolbar.wrapper);\n\t };\n\t\n\t var makeToolBar_ = function makeToolBar_() {\n\t\n\t var toolbar = editor.draw.toolbar(),\n\t blockButtons = makeToolbarSettings_(),\n\t toolbarContent = makeToolbarContent_();\n\t\n\t /** Appending first-level block buttons */\n\t toolbar.appendChild(blockButtons);\n\t\n\t /** Append toolbarContent to toolbar */\n\t toolbar.appendChild(toolbarContent);\n\t\n\t /** Make toolbar global */\n\t editor.nodes.toolbar = toolbar;\n\t\n\t return toolbar;\n\t };\n\t\n\t var makeToolbarContent_ = function makeToolbarContent_() {\n\t\n\t var toolbarContent = editor.draw.toolbarContent(),\n\t toolbox = editor.draw.toolbox(),\n\t plusButton = editor.draw.plusButton();\n\t\n\t /** Append plus button */\n\t toolbarContent.appendChild(plusButton);\n\t\n\t /** Appending toolbar tools */\n\t toolbarContent.appendChild(toolbox);\n\t\n\t /** Make Toolbox and plusButton global */\n\t editor.nodes.toolbox = toolbox;\n\t editor.nodes.plusButton = plusButton;\n\t\n\t return toolbarContent;\n\t };\n\t\n\t var makeToolbarSettings_ = function makeToolbarSettings_() {\n\t\n\t var blockSettings = editor.draw.blockSettings(),\n\t blockButtons = editor.draw.blockButtons(),\n\t defaultSettings = editor.draw.defaultSettings(),\n\t showSettingsButton = editor.draw.settingsButton(),\n\t showTrashButton = editor.toolbar.settings.makeRemoveBlockButton(),\n\t pluginSettings = editor.draw.pluginsSettings();\n\t\n\t /** Add default and plugins settings */\n\t blockSettings.appendChild(pluginSettings);\n\t blockSettings.appendChild(defaultSettings);\n\t\n\t /**\r\n\t * Make blocks buttons\r\n\t * This block contains settings button and remove block button\r\n\t */\n\t blockButtons.appendChild(showSettingsButton);\n\t blockButtons.appendChild(showTrashButton);\n\t blockButtons.appendChild(blockSettings);\n\t\n\t /** Make BlockSettings, PluginSettings, DefaultSettings global */\n\t editor.nodes.blockSettings = blockSettings;\n\t editor.nodes.pluginSettings = pluginSettings;\n\t editor.nodes.defaultSettings = defaultSettings;\n\t editor.nodes.showSettingsButton = showSettingsButton;\n\t editor.nodes.showTrashButton = showTrashButton;\n\t\n\t return blockButtons;\n\t };\n\t\n\t /** Draw notifications holder */\n\t var makeNotificationHolder_ = function makeNotificationHolder_() {\n\t\n\t /** Append block with notifications to the document */\n\t editor.nodes.notifications = editor.notifications.createHolder();\n\t };\n\t\n\t /**\r\n\t * @private\r\n\t * Append tools passed in editor.tools\r\n\t */\n\t var addTools_ = function addTools_() {\n\t\n\t var tool, toolName, toolButton;\n\t\n\t for (toolName in editor.settings.tools) {\n\t\n\t tool = editor.settings.tools[toolName];\n\t\n\t editor.tools[toolName] = tool;\n\t\n\t if (!tool.iconClassname && tool.displayInToolbox) {\n\t\n\t editor.core.log('Toolbar icon classname missed. Tool %o skipped', 'warn', toolName);\n\t continue;\n\t }\n\t\n\t if (typeof tool.render != 'function') {\n\t\n\t editor.core.log('render method missed. Tool %o skipped', 'warn', toolName);\n\t continue;\n\t }\n\t\n\t if (!tool.displayInToolbox) {\n\t\n\t continue;\n\t } else {\n\t\n\t /** if tools is for toolbox */\n\t toolButton = editor.draw.toolbarButton(toolName, tool.iconClassname);\n\t\n\t editor.nodes.toolbox.appendChild(toolButton);\n\t\n\t editor.nodes.toolbarButtons[toolName] = toolButton;\n\t }\n\t }\n\t };\n\t\n\t var addInlineToolbarTools_ = function addInlineToolbarTools_() {\n\t\n\t var tools = {\n\t\n\t bold: {\n\t icon: 'ce-icon-bold',\n\t command: 'bold'\n\t },\n\t\n\t italic: {\n\t icon: 'ce-icon-italic',\n\t command: 'italic'\n\t },\n\t\n\t underline: {\n\t icon: 'ce-icon-underline',\n\t command: 'underline'\n\t },\n\t\n\t link: {\n\t icon: 'ce-icon-link',\n\t command: 'createLink'\n\t }\n\t };\n\t\n\t var toolButton, tool;\n\t\n\t for (var name in tools) {\n\t\n\t tool = tools[name];\n\t\n\t toolButton = editor.draw.toolbarButtonInline(name, tool.icon);\n\t\n\t editor.nodes.inlineToolbar.buttons.appendChild(toolButton);\n\t /**\r\n\t * Add callbacks to this buttons\r\n\t */\n\t editor.ui.setInlineToolbarButtonBehaviour(toolButton, tool.command);\n\t }\n\t };\n\t\n\t /**\r\n\t * @private\r\n\t * Bind editor UI events\r\n\t */\n\t var bindEvents_ = function bindEvents_() {\n\t\n\t editor.core.log('ui.bindEvents fired', 'info');\n\t\n\t // window.addEventListener('error', function (errorMsg, url, lineNumber) {\n\t // editor.notifications.errorThrown(errorMsg, event);\n\t // }, false );\n\t\n\t /** All keydowns on Document */\n\t editor.listeners.add(document, 'keydown', editor.callback.globalKeydown, false);\n\t\n\t /** All keydowns on Redactor zone */\n\t editor.listeners.add(editor.nodes.redactor, 'keydown', editor.callback.redactorKeyDown, false);\n\t\n\t /** All keydowns on Document */\n\t editor.listeners.add(document, 'keyup', editor.callback.globalKeyup, false);\n\t\n\t /**\r\n\t * Mouse click to radactor\r\n\t */\n\t editor.listeners.add(editor.nodes.redactor, 'click', editor.callback.redactorClicked, false);\n\t\n\t /**\r\n\t * Clicks to the Plus button\r\n\t */\n\t editor.listeners.add(editor.nodes.plusButton, 'click', editor.callback.plusButtonClicked, false);\n\t\n\t /**\r\n\t * Clicks to SETTINGS button in toolbar\r\n\t */\n\t editor.listeners.add(editor.nodes.showSettingsButton, 'click', editor.callback.showSettingsButtonClicked, false);\n\t\n\t /** Bind click listeners on toolbar buttons */\n\t for (var button in editor.nodes.toolbarButtons) {\n\t\n\t editor.listeners.add(editor.nodes.toolbarButtons[button], 'click', editor.callback.toolbarButtonClicked, false);\n\t }\n\t };\n\t\n\t ui.addBlockHandlers = function (block) {\n\t\n\t if (!block) return;\n\t\n\t /**\r\n\t * Block keydowns\r\n\t */\n\t editor.listeners.add(block, 'keydown', editor.callback.blockKeydown, false);\n\t\n\t /**\r\n\t * Pasting content from another source\r\n\t * We have two type of sanitization\r\n\t * First - uses deep-first search algorithm to get sub nodes,\r\n\t * sanitizes whole Block_content and replaces cleared nodes\r\n\t * This method is deprecated\r\n\t * Method is used in editor.callback.blockPaste(event)\r\n\t *\r\n\t * Secont - uses Mutation observer.\r\n\t * Observer \"observe\" DOM changes and send changings to callback.\r\n\t * Callback gets changed node, not whole Block_content.\r\n\t * Inserted or changed node, which we've gotten have been cleared and replaced with diry node\r\n\t *\r\n\t * Method is used in editor.callback.blockPasteViaSanitize(event)\r\n\t *\r\n\t * @uses html-janitor\r\n\t * @example editor.callback.blockPasteViaSanitize(event), the second method.\r\n\t *\r\n\t */\n\t editor.listeners.add(block, 'paste', editor.callback.blockPasteCallback, false);\n\t\n\t editor.listeners.add(block, 'mouseup', editor.toolbar.inline.show, false);\n\t };\n\t\n\t /** getting all contenteditable elements */\n\t ui.saveInputs = function () {\n\t\n\t var redactor = editor.nodes.redactor;\n\t\n\t editor.state.inputs = [];\n\t\n\t /** Save all inputs in global variable state */\n\t var inputs = redactor.querySelectorAll('[contenteditable], input, textarea');\n\t\n\t Array.prototype.map.call(inputs, function (current) {\n\t\n\t if (!current.type || current.type == 'text' || current.type == 'textarea') {\n\t\n\t editor.state.inputs.push(current);\n\t }\n\t });\n\t };\n\t\n\t /**\r\n\t * Adds first initial block on empty redactor\r\n\t */\n\t ui.addInitialBlock = function () {\n\t\n\t var initialBlockType = editor.settings.initialBlockPlugin,\n\t initialBlock;\n\t\n\t if (!editor.tools[initialBlockType]) {\n\t\n\t editor.core.log('Plugin %o was not implemented and can\\'t be used as initial block', 'warn', initialBlockType);\n\t return;\n\t }\n\t\n\t initialBlock = editor.tools[initialBlockType].render();\n\t\n\t initialBlock.setAttribute('data-placeholder', 'Расскажите свою историю...');\n\t\n\t editor.content.insertBlock({\n\t type: initialBlockType,\n\t block: initialBlock\n\t });\n\t\n\t editor.content.workingNodeChanged(initialBlock);\n\t };\n\t\n\t ui.setInlineToolbarButtonBehaviour = function (button, type) {\n\t\n\t editor.listeners.add(button, 'mousedown', function (event) {\n\t\n\t editor.toolbar.inline.toolClicked(event, type);\n\t }, false);\n\t };\n\t\n\t return ui;\n\t}({});\n\n/***/ },\n/* 4 */\n/***/ function(module, exports) {\n\n\t'use strict';\n\t\n\t/**\r\n\t *\r\n\t * Codex.Editor Transport Module\r\n\t *\r\n\t * @copyright 2017 Codex-Team\r\n\t * @version 1.2.0\r\n\t */\n\t\n\tmodule.exports = function (transport) {\n\t\n\t var editor = codex.editor;\n\t\n\t /**\r\n\t * @private {Object} current XmlHttpRequest instance\r\n\t */\n\t var currentRequest = null;\n\t\n\t /**\r\n\t * @type {null} | {DOMElement} input - keeps input element in memory\r\n\t */\n\t transport.input = null;\n\t\n\t /**\r\n\t * @property {Object} arguments - keep plugin settings and defined callbacks\r\n\t */\n\t transport.arguments = null;\n\t\n\t /**\r\n\t * Prepares input element where will be files\r\n\t */\n\t transport.prepare = function () {\n\t\n\t var input = editor.draw.node('INPUT', '', { type: 'file' });\n\t\n\t editor.listeners.add(input, 'change', editor.transport.fileSelected);\n\t editor.transport.input = input;\n\t };\n\t\n\t /** Clear input when files is uploaded */\n\t transport.clearInput = function () {\n\t\n\t /** Remove old input */\n\t transport.input = null;\n\t\n\t /** Prepare new one */\n\t transport.prepare();\n\t };\n\t\n\t /**\r\n\t * Callback for file selection\r\n\t * @param {Event} event\r\n\t */\n\t transport.fileSelected = function () {\n\t\n\t var input = this,\n\t i,\n\t files = input.files,\n\t formData = new FormData();\n\t\n\t if (editor.transport.arguments.multiple === true) {\n\t\n\t for (i = 0; i < files.length; i++) {\n\t\n\t formData.append('files[]', files[i], files[i].name);\n\t }\n\t } else {\n\t\n\t formData.append('files', files[0], files[0].name);\n\t }\n\t\n\t currentRequest = editor.core.ajax({\n\t type: 'POST',\n\t data: formData,\n\t url: editor.transport.arguments.url,\n\t beforeSend: editor.transport.arguments.beforeSend,\n\t success: editor.transport.arguments.success,\n\t error: editor.transport.arguments.error,\n\t progress: editor.transport.arguments.progress\n\t });\n\t\n\t /** Clear input */\n\t transport.clearInput();\n\t };\n\t\n\t /**\r\n\t * Use plugin callbacks\r\n\t * @protected\r\n\t *\r\n\t * @param {Object} args - can have :\r\n\t * @param {String} args.url - fetch URL\r\n\t * @param {Function} args.beforeSend - function calls before sending ajax\r\n\t * @param {Function} args.success - success callback\r\n\t * @param {Function} args.error - on error handler\r\n\t * @param {Function} args.progress - xhr onprogress handler\r\n\t * @param {Boolean} args.multiple - allow select several files\r\n\t * @param {String} args.accept - adds accept attribute\r\n\t */\n\t transport.selectAndUpload = function (args) {\n\t\n\t transport.arguments = args;\n\t\n\t if (args.multiple === true) {\n\t\n\t transport.input.setAttribute('multiple', 'multiple');\n\t }\n\t\n\t if (args.accept) {\n\t\n\t transport.input.setAttribute('accept', args.accept);\n\t }\n\t\n\t transport.input.click();\n\t };\n\t\n\t transport.abort = function () {\n\t\n\t currentRequest.abort();\n\t\n\t currentRequest = null;\n\t };\n\t\n\t return transport;\n\t}({});\n\n/***/ },\n/* 5 */\n/***/ function(module, exports) {\n\n\t'use strict';\n\t\n\t/**\r\n\t * Codex Editor Renderer Module\r\n\t *\r\n\t * @author Codex Team\r\n\t * @version 1.0\r\n\t */\n\t\n\tmodule.exports = function (renderer) {\n\t\n\t var editor = codex.editor;\n\t\n\t /**\r\n\t * Asyncronously parses input JSON to redactor blocks\r\n\t */\n\t renderer.makeBlocksFromData = function () {\n\t\n\t /**\r\n\t * If redactor is empty, add first paragraph to start writing\r\n\t */\n\t if (editor.core.isEmpty(editor.state.blocks) || !editor.state.blocks.items.length) {\n\t\n\t editor.ui.addInitialBlock();\n\t return;\n\t }\n\t\n\t Promise.resolve()\n\t\n\t /** First, get JSON from state */\n\t .then(function () {\n\t\n\t return editor.state.blocks;\n\t })\n\t\n\t /** Then, start to iterate they */\n\t .then(editor.renderer.appendBlocks)\n\t\n\t /** Write log if something goes wrong */\n\t .catch(function (error) {\n\t\n\t editor.core.log('Error while parsing JSON: %o', 'error', error);\n\t });\n\t };\n\t\n\t /**\r\n\t * Parses JSON to blocks\r\n\t * @param {object} data\r\n\t * @return Primise -> nodeList\r\n\t */\n\t renderer.appendBlocks = function (data) {\n\t\n\t var blocks = data.items;\n\t\n\t /**\r\n\t * Sequence of one-by-one blocks appending\r\n\t * Uses to save blocks order after async-handler\r\n\t */\n\t var nodeSequence = Promise.resolve();\n\t\n\t for (var index = 0; index < blocks.length; index++) {\n\t\n\t /** Add node to sequence at specified index */\n\t editor.renderer.appendNodeAtIndex(nodeSequence, blocks, index);\n\t }\n\t };\n\t\n\t /**\r\n\t * Append node at specified index\r\n\t */\n\t renderer.appendNodeAtIndex = function (nodeSequence, blocks, index) {\n\t\n\t /** We need to append node to sequence */\n\t nodeSequence\n\t\n\t /** first, get node async-aware */\n\t .then(function () {\n\t\n\t return editor.renderer.getNodeAsync(blocks, index);\n\t })\n\t\n\t /**\r\n\t * second, compose editor-block from JSON object\r\n\t */\n\t .then(editor.renderer.createBlockFromData)\n\t\n\t /**\r\n\t * now insert block to redactor\r\n\t */\n\t .then(function (blockData) {\n\t\n\t /**\r\n\t * blockData has 'block', 'type' and 'stretched' information\r\n\t */\n\t editor.content.insertBlock(blockData);\n\t\n\t /** Pass created block to next step */\n\t return blockData.block;\n\t })\n\t\n\t /** Log if something wrong with node */\n\t .catch(function (error) {\n\t\n\t editor.core.log('Node skipped while parsing because %o', 'error', error);\n\t });\n\t };\n\t\n\t /**\r\n\t * Asynchronously returns block data from blocksList by index\r\n\t * @return Promise to node\r\n\t */\n\t renderer.getNodeAsync = function (blocksList, index) {\n\t\n\t return Promise.resolve().then(function () {\n\t\n\t return {\n\t tool: blocksList[index],\n\t position: index\n\t };\n\t });\n\t };\n\t\n\t /**\r\n\t * Creates editor block by JSON-data\r\n\t *\r\n\t * @uses render method of each plugin\r\n\t *\r\n\t * @param {Object} toolData.tool\r\n\t * { header : {\r\n\t * text: '',\r\n\t * type: 'H3', ...\r\n\t * }\r\n\t * }\r\n\t * @param {Number} toolData.position - index in input-blocks array\r\n\t * @return {Object} with type and Element\r\n\t */\n\t renderer.createBlockFromData = function (toolData) {\n\t\n\t /** New parser */\n\t var block,\n\t tool = toolData.tool,\n\t pluginName = tool.type;\n\t\n\t /** Get first key of object that stores plugin name */\n\t // for (var pluginName in blockData) break;\n\t\n\t /** Check for plugin existance */\n\t if (!editor.tools[pluginName]) {\n\t\n\t throw Error('Plugin \\xAB' + pluginName + '\\xBB not found');\n\t }\n\t\n\t /** Check for plugin having render method */\n\t if (typeof editor.tools[pluginName].render != 'function') {\n\t\n\t throw Error('Plugin \\xAB' + pluginName + '\\xBB must have \\xABrender\\xBB method');\n\t }\n\t\n\t if (editor.tools[pluginName].available === false) {\n\t\n\t block = editor.draw.unavailableBlock();\n\t\n\t block.innerHTML = editor.tools[pluginName].loadingMessage;\n\t\n\t /**\r\n\t * Saver will extract data from initial block data by position in array\r\n\t */\n\t block.dataset.inputPosition = toolData.position;\n\t } else {\n\t\n\t /** New Parser */\n\t block = editor.tools[pluginName].render(tool.data);\n\t }\n\t\n\t /** is first-level block stretched */\n\t var stretched = editor.tools[pluginName].isStretched || false;\n\t\n\t /** Retrun type and block */\n\t return {\n\t type: pluginName,\n\t block: block,\n\t stretched: stretched\n\t };\n\t };\n\t\n\t return renderer;\n\t}({});\n\n/***/ },\n/* 6 */\n/***/ function(module, exports) {\n\n\t'use strict';\n\t\n\t/**\r\n\t * Codex Editor Saver\r\n\t *\r\n\t * @author Codex Team\r\n\t * @version 1.0.2\r\n\t */\n\t\n\tmodule.exports = function (saver) {\n\t\n\t var editor = codex.editor;\n\t\n\t /**\r\n\t * Saves blocks\r\n\t * @private\r\n\t */\n\t saver.saveBlocks = function () {\n\t\n\t /** Save html content of redactor to memory */\n\t editor.state.html = editor.nodes.redactor.innerHTML;\n\t\n\t /** Empty jsonOutput state */\n\t editor.state.jsonOutput = [];\n\t\n\t Promise.resolve().then(function () {\n\t\n\t return editor.nodes.redactor.childNodes;\n\t })\n\t /** Making a sequence from separate blocks */\n\t .then(editor.saver.makeQueue).then(function () {\n\t // editor.nodes.textarea.innerHTML = editor.state.html;\n\t }).catch(function (error) {\n\t\n\t editor.core.log(error);\n\t });\n\t };\n\t\n\t saver.makeQueue = function (blocks) {\n\t\n\t var queue = Promise.resolve();\n\t\n\t for (var index = 0; index < blocks.length; index++) {\n\t\n\t /** Add node to sequence at specified index */\n\t editor.saver.getBlockData(queue, blocks, index);\n\t }\n\t };\n\t\n\t /** Gets every block and makes From Data */\n\t saver.getBlockData = function (queue, blocks, index) {\n\t\n\t queue.then(function () {\n\t\n\t return editor.saver.getNodeAsync(blocks, index);\n\t }).then(editor.saver.makeFormDataFromBlocks);\n\t };\n\t\n\t /**\r\n\t * Asynchronously returns block data from blocksList by index\r\n\t * @return Promise to node\r\n\t */\n\t saver.getNodeAsync = function (blocksList, index) {\n\t\n\t return Promise.resolve().then(function () {\n\t\n\t return blocksList[index];\n\t });\n\t };\n\t\n\t saver.makeFormDataFromBlocks = function (block) {\n\t\n\t var pluginName = block.dataset.tool;\n\t\n\t /** Check for plugin existance */\n\t if (!editor.tools[pluginName]) {\n\t\n\t throw Error('Plugin \\xAB' + pluginName + '\\xBB not found');\n\t }\n\t\n\t /** Check for plugin having render method */\n\t if (typeof editor.tools[pluginName].save != 'function') {\n\t\n\t throw Error('Plugin \\xAB' + pluginName + '\\xBB must have save method');\n\t }\n\t\n\t /** Result saver */\n\t var blockContent = block.childNodes[0],\n\t pluginsContent = blockContent.childNodes[0],\n\t savedData,\n\t position,\n\t output;\n\t\n\t /** If plugin wasn't available then return data from cache */\n\t if (editor.tools[pluginName].available === false) {\n\t\n\t position = pluginsContent.dataset.inputPosition;\n\t savedData = codex.editor.state.blocks.items[position].data;\n\t } else {\n\t\n\t savedData = editor.tools[pluginName].save(pluginsContent);\n\t\n\t if (editor.tools[pluginName].validate) {\n\t\n\t var result = editor.tools[pluginName].validate(savedData);\n\t\n\t /**\r\n\t * Do not allow invalid data\r\n\t */\n\t if (!result) return;\n\t }\n\t }\n\t\n\t output = {\n\t type: pluginName,\n\t data: savedData\n\t };\n\t\n\t editor.state.jsonOutput.push(output);\n\t };\n\t\n\t return saver;\n\t}({});\n\n/***/ },\n/* 7 */\n/***/ function(module, exports) {\n\n\t'use strict';\n\t\n\t/**\r\n\t * Codex Editor Content Module\r\n\t * Works with DOM\r\n\t *\r\n\t * @module Codex Editor content module\r\n\t *\r\n\t * @author Codex Team\r\n\t * @version 1.3.13\r\n\t *\r\n\t * @description Module works with Elements that have been appended to the main DOM\r\n\t */\n\t\n\tmodule.exports = function (content) {\n\t\n\t var editor = codex.editor;\n\t\n\t /**\r\n\t * Links to current active block\r\n\t * @type {null | Element}\r\n\t */\n\t content.currentNode = null;\n\t\n\t /**\r\n\t * clicked in redactor area\r\n\t * @type {null | Boolean}\r\n\t */\n\t content.editorAreaHightlighted = null;\n\t\n\t /**\r\n\t * @deprecated\r\n\t * Synchronizes redactor with original textarea\r\n\t */\n\t content.sync = function () {\n\t\n\t editor.core.log('syncing...');\n\t\n\t /**\r\n\t * Save redactor content to editor.state\r\n\t */\n\t editor.state.html = editor.nodes.redactor.innerHTML;\n\t };\n\t\n\t /**\r\n\t * Appends background to the block\r\n\t *\r\n\t * @description add CSS class to highlight visually first-level block area\r\n\t */\n\t content.markBlock = function () {\n\t\n\t editor.content.currentNode.classList.add(editor.ui.className.BLOCK_HIGHLIGHTED);\n\t };\n\t\n\t /**\r\n\t * Clear background\r\n\t *\r\n\t * @description clears styles that highlights block\r\n\t */\n\t content.clearMark = function () {\n\t\n\t if (editor.content.currentNode) {\n\t\n\t editor.content.currentNode.classList.remove(editor.ui.className.BLOCK_HIGHLIGHTED);\n\t }\n\t };\n\t\n\t /**\r\n\t * Finds first-level block\r\n\t *\r\n\t * @param {Element} node - selected or clicked in redactors area node\r\n\t * @protected\r\n\t *\r\n\t * @description looks for first-level block.\r\n\t * gets parent while node is not first-level\r\n\t */\n\t content.getFirstLevelBlock = function (node) {\n\t\n\t if (!editor.core.isDomNode(node)) {\n\t\n\t node = node.parentNode;\n\t }\n\t\n\t if (node === editor.nodes.redactor || node === document.body) {\n\t\n\t return null;\n\t } else {\n\t\n\t while (!node.classList.contains(editor.ui.className.BLOCK_CLASSNAME)) {\n\t\n\t node = node.parentNode;\n\t }\n\t\n\t return node;\n\t }\n\t };\n\t\n\t /**\r\n\t * Trigger this event when working node changed\r\n\t * @param {Element} targetNode - first-level of this node will be current\r\n\t * @protected\r\n\t *\r\n\t * @description If targetNode is first-level then we set it as current else we look for parents to find first-level\r\n\t */\n\t content.workingNodeChanged = function (targetNode) {\n\t\n\t /** Clear background from previous marked block before we change */\n\t editor.content.clearMark();\n\t\n\t if (!targetNode) {\n\t\n\t return;\n\t }\n\t\n\t content.currentNode = content.getFirstLevelBlock(targetNode);\n\t };\n\t\n\t /**\r\n\t * Replaces one redactor block with another\r\n\t * @protected\r\n\t * @param {Element} targetBlock - block to replace. Mostly currentNode.\r\n\t * @param {Element} newBlock\r\n\t * @param {string} newBlockType - type of new block; we need to store it to data-attribute\r\n\t *\r\n\t * [!] Function does not saves old block content.\r\n\t * You can get it manually and pass with newBlock.innerHTML\r\n\t */\n\t content.replaceBlock = function (targetBlock, newBlock) {\n\t\n\t if (!targetBlock || !newBlock) {\n\t\n\t editor.core.log('replaceBlock: missed params');\n\t return;\n\t }\n\t\n\t /** If target-block is not a frist-level block, then we iterate parents to find it */\n\t while (!targetBlock.classList.contains(editor.ui.className.BLOCK_CLASSNAME)) {\n\t\n\t targetBlock = targetBlock.parentNode;\n\t }\n\t\n\t /** Replacing */\n\t editor.nodes.redactor.replaceChild(newBlock, targetBlock);\n\t\n\t /**\r\n\t * Set new node as current\r\n\t */\n\t editor.content.workingNodeChanged(newBlock);\n\t\n\t /**\r\n\t * Add block handlers\r\n\t */\n\t editor.ui.addBlockHandlers(newBlock);\n\t\n\t /**\r\n\t * Save changes\r\n\t */\n\t editor.ui.saveInputs();\n\t };\n\t\n\t /**\r\n\t * @protected\r\n\t *\r\n\t * Inserts new block to redactor\r\n\t * Wrapps block into a DIV with BLOCK_CLASSNAME class\r\n\t *\r\n\t * @param blockData {object}\r\n\t * @param blockData.block {Element} element with block content\r\n\t * @param blockData.type {string} block plugin\r\n\t * @param needPlaceCaret {bool} pass true to set caret in new block\r\n\t *\r\n\t */\n\t content.insertBlock = function (blockData, needPlaceCaret) {\n\t\n\t var workingBlock = editor.content.currentNode,\n\t newBlockContent = blockData.block,\n\t blockType = blockData.type,\n\t isStretched = blockData.stretched;\n\t\n\t var newBlock = composeNewBlock_(newBlockContent, blockType, isStretched);\n\t\n\t if (workingBlock) {\n\t\n\t editor.core.insertAfter(workingBlock, newBlock);\n\t } else {\n\t\n\t /**\r\n\t * If redactor is empty, append as first child\r\n\t */\n\t editor.nodes.redactor.appendChild(newBlock);\n\t }\n\t\n\t /**\r\n\t * Block handler\r\n\t */\n\t editor.ui.addBlockHandlers(newBlock);\n\t\n\t /**\r\n\t * Set new node as current\r\n\t */\n\t editor.content.workingNodeChanged(newBlock);\n\t\n\t /**\r\n\t * Save changes\r\n\t */\n\t editor.ui.saveInputs();\n\t\n\t if (needPlaceCaret) {\n\t\n\t /**\r\n\t * If we don't know input index then we set default value -1\r\n\t */\n\t var currentInputIndex = editor.caret.getCurrentInputIndex() || -1;\n\t\n\t if (currentInputIndex == -1) {\n\t\n\t var editableElement = newBlock.querySelector('[contenteditable]'),\n\t emptyText = document.createTextNode('');\n\t\n\t editableElement.appendChild(emptyText);\n\t editor.caret.set(editableElement, 0, 0);\n\t\n\t editor.toolbar.move();\n\t editor.toolbar.showPlusButton();\n\t } else {\n\t\n\t if (currentInputIndex === editor.state.inputs.length - 1) return;\n\t\n\t /** Timeout for browsers execution */\n\t window.setTimeout(function () {\n\t\n\t /** Setting to the new input */\n\t editor.caret.setToNextBlock(currentInputIndex);\n\t editor.toolbar.move();\n\t editor.toolbar.open();\n\t }, 10);\n\t }\n\t }\n\t\n\t /**\r\n\t * Block is inserted, wait for new click that defined focusing on editors area\r\n\t * @type {boolean}\r\n\t */\n\t content.editorAreaHightlighted = false;\n\t };\n\t\n\t /**\r\n\t * Replaces blocks with saving content\r\n\t * @protected\r\n\t * @param {Element} noteToReplace\r\n\t * @param {Element} newNode\r\n\t * @param {Element} blockType\r\n\t */\n\t content.switchBlock = function (blockToReplace, newBlock, tool) {\n\t\n\t tool = tool || editor.content.currentNode.dataset.tool;\n\t var newBlockComposed = composeNewBlock_(newBlock, tool);\n\t\n\t /** Replacing */\n\t editor.content.replaceBlock(blockToReplace, newBlockComposed);\n\t\n\t /** Save new Inputs when block is changed */\n\t editor.ui.saveInputs();\n\t };\n\t\n\t /**\r\n\t * Iterates between child noted and looking for #text node on deepest level\r\n\t * @protected\r\n\t *\r\n\t * @param {Element} block - node where find\r\n\t * @param {int} postiton - starting postion\r\n\t * Example: childNodex.length to find from the end\r\n\t * or 0 to find from the start\r\n\t * @return {Text} block\r\n\t * @uses DFS\r\n\t */\n\t content.getDeepestTextNodeFromPosition = function (block, position) {\n\t\n\t /**\r\n\t * Clear Block from empty and useless spaces with trim.\r\n\t * Such nodes we should remove\r\n\t */\n\t var blockChilds = block.childNodes,\n\t index,\n\t node,\n\t text;\n\t\n\t for (index = 0; index < blockChilds.length; index++) {\n\t\n\t node = blockChilds[index];\n\t\n\t if (node.nodeType == editor.core.nodeTypes.TEXT) {\n\t\n\t text = node.textContent.trim();\n\t\n\t /** Text is empty. We should remove this child from node before we start DFS\r\n\t * decrease the quantity of childs.\r\n\t */\n\t if (text === '') {\n\t\n\t block.removeChild(node);\n\t position--;\n\t }\n\t }\n\t }\n\t\n\t if (block.childNodes.length === 0) {\n\t\n\t return document.createTextNode('');\n\t }\n\t\n\t /** Setting default position when we deleted all empty nodes */\n\t if (position < 0) position = 1;\n\t\n\t var lookingFromStart = false;\n\t\n\t /** For looking from START */\n\t if (position === 0) {\n\t\n\t lookingFromStart = true;\n\t position = 1;\n\t }\n\t\n\t while (position) {\n\t\n\t /** initial verticle of node. */\n\t if (lookingFromStart) {\n\t\n\t block = block.childNodes[0];\n\t } else {\n\t\n\t block = block.childNodes[position - 1];\n\t }\n\t\n\t if (block.nodeType == editor.core.nodeTypes.TAG) {\n\t\n\t position = block.childNodes.length;\n\t } else if (block.nodeType == editor.core.nodeTypes.TEXT) {\n\t\n\t position = 0;\n\t }\n\t }\n\t\n\t return block;\n\t };\n\t\n\t /**\r\n\t * @private\r\n\t * @param {Element} block - current plugins render\r\n\t * @param {String} tool - plugins name\r\n\t * @param {Boolean} isStretched - make stretched block or not\r\n\t *\r\n\t * @description adds necessary information to wrap new created block by first-level holder\r\n\t */\n\t var composeNewBlock_ = function composeNewBlock_(block, tool, isStretched) {\n\t\n\t var newBlock = editor.draw.node('DIV', editor.ui.className.BLOCK_CLASSNAME, {}),\n\t blockContent = editor.draw.node('DIV', editor.ui.className.BLOCK_CONTENT, {});\n\t\n\t blockContent.appendChild(block);\n\t newBlock.appendChild(blockContent);\n\t\n\t if (isStretched) {\n\t\n\t blockContent.classList.add(editor.ui.className.BLOCK_STRETCHED);\n\t }\n\t\n\t newBlock.dataset.tool = tool;\n\t return newBlock;\n\t };\n\t\n\t /**\r\n\t * Returns Range object of current selection\r\n\t * @protected\r\n\t */\n\t content.getRange = function () {\n\t\n\t var selection = window.getSelection().getRangeAt(0);\n\t\n\t return selection;\n\t };\n\t\n\t /**\r\n\t * Divides block in two blocks (after and before caret)\r\n\t *\r\n\t * @protected\r\n\t * @param {int} inputIndex - target input index\r\n\t *\r\n\t * @description splits current input content to the separate blocks\r\n\t * When enter is pressed among the words, that text will be splited.\r\n\t */\n\t content.splitBlock = function (inputIndex) {\n\t\n\t var selection = window.getSelection(),\n\t anchorNode = selection.anchorNode,\n\t anchorNodeText = anchorNode.textContent,\n\t caretOffset = selection.anchorOffset,\n\t textBeforeCaret,\n\t textNodeBeforeCaret,\n\t textAfterCaret,\n\t textNodeAfterCaret;\n\t\n\t var currentBlock = editor.content.currentNode.querySelector('[contentEditable]');\n\t\n\t textBeforeCaret = anchorNodeText.substring(0, caretOffset);\n\t textAfterCaret = anchorNodeText.substring(caretOffset);\n\t\n\t textNodeBeforeCaret = document.createTextNode(textBeforeCaret);\n\t\n\t if (textAfterCaret) {\n\t\n\t textNodeAfterCaret = document.createTextNode(textAfterCaret);\n\t }\n\t\n\t var previousChilds = [],\n\t nextChilds = [],\n\t reachedCurrent = false;\n\t\n\t if (textNodeAfterCaret) {\n\t\n\t nextChilds.push(textNodeAfterCaret);\n\t }\n\t\n\t for (var i = 0, child; !!(child = currentBlock.childNodes[i]); i++) {\n\t\n\t if (child != anchorNode) {\n\t\n\t if (!reachedCurrent) {\n\t\n\t previousChilds.push(child);\n\t } else {\n\t\n\t nextChilds.push(child);\n\t }\n\t } else {\n\t\n\t reachedCurrent = true;\n\t }\n\t }\n\t\n\t /** Clear current input */\n\t editor.state.inputs[inputIndex].innerHTML = '';\n\t\n\t /**\r\n\t * Append all childs founded before anchorNode\r\n\t */\n\t var previousChildsLength = previousChilds.length;\n\t\n\t for (i = 0; i < previousChildsLength; i++) {\n\t\n\t editor.state.inputs[inputIndex].appendChild(previousChilds[i]);\n\t }\n\t\n\t editor.state.inputs[inputIndex].appendChild(textNodeBeforeCaret);\n\t\n\t /**\r\n\t * Append text node which is after caret\r\n\t */\n\t var nextChildsLength = nextChilds.length,\n\t newNode = document.createElement('div');\n\t\n\t for (i = 0; i < nextChildsLength; i++) {\n\t\n\t newNode.appendChild(nextChilds[i]);\n\t }\n\t\n\t newNode = newNode.innerHTML;\n\t\n\t /** This type of block creates when enter is pressed */\n\t var NEW_BLOCK_TYPE = editor.settings.initialBlockPlugin;\n\t\n\t /**\r\n\t * Make new paragraph with text after caret\r\n\t */\n\t editor.content.insertBlock({\n\t type: NEW_BLOCK_TYPE,\n\t block: editor.tools[NEW_BLOCK_TYPE].render({\n\t text: newNode\n\t })\n\t }, true);\n\t };\n\t\n\t /**\r\n\t * Merges two blocks — current and target\r\n\t * If target index is not exist, then previous will be as target\r\n\t *\r\n\t * @protected\r\n\t * @param {int} currentInputIndex\r\n\t * @param {int} targetInputIndex\r\n\t *\r\n\t * @description gets two inputs indexes and merges into one\r\n\t */\n\t content.mergeBlocks = function (currentInputIndex, targetInputIndex) {\n\t\n\t /** If current input index is zero, then prevent method execution */\n\t if (currentInputIndex === 0) {\n\t\n\t return;\n\t }\n\t\n\t var targetInput,\n\t currentInputContent = editor.state.inputs[currentInputIndex].innerHTML;\n\t\n\t if (!targetInputIndex) {\n\t\n\t targetInput = editor.state.inputs[currentInputIndex - 1];\n\t } else {\n\t\n\t targetInput = editor.state.inputs[targetInputIndex];\n\t }\n\t\n\t targetInput.innerHTML += currentInputContent;\n\t };\n\t\n\t /**\r\n\t * Iterates all right siblings and parents, which has right siblings\r\n\t * while it does not reached the first-level block\r\n\t *\r\n\t * @param {Element} node\r\n\t * @return {boolean}\r\n\t */\n\t content.isLastNode = function (node) {\n\t\n\t // console.log('погнали перебор родителей');\n\t\n\t var allChecked = false;\n\t\n\t while (!allChecked) {\n\t\n\t // console.log('Смотрим на %o', node);\n\t // console.log('Проверим, пустые ли соседи справа');\n\t\n\t if (!allSiblingsEmpty_(node)) {\n\t\n\t // console.log('Есть непустые соседи. Узел не последний. Выходим.');\n\t return false;\n\t }\n\t\n\t node = node.parentNode;\n\t\n\t /**\r\n\t * Проверяем родителей до тех пор, пока не найдем блок первого уровня\r\n\t */\n\t if (node.classList.contains(editor.ui.className.BLOCK_CONTENT)) {\n\t\n\t allChecked = true;\n\t }\n\t }\n\t\n\t return true;\n\t };\n\t\n\t /**\r\n\t * Checks if all element right siblings is empty\r\n\t * @param node\r\n\t */\n\t var allSiblingsEmpty_ = function allSiblingsEmpty_(node) {\n\t\n\t /**\r\n\t * Нужно убедиться, что после пустого соседа ничего нет\r\n\t */\n\t var sibling = node.nextSibling;\n\t\n\t while (sibling) {\n\t\n\t if (sibling.textContent.length) {\n\t\n\t return false;\n\t }\n\t\n\t sibling = sibling.nextSibling;\n\t }\n\t\n\t return true;\n\t };\n\t\n\t /**\r\n\t * @public\r\n\t *\r\n\t * @param {string} htmlData - html content as string\r\n\t * @param {string} plainData - plain text\r\n\t * @return {string} - html content as string\r\n\t */\n\t content.wrapTextWithParagraphs = function (htmlData, plainData) {\n\t\n\t if (!htmlData) {\n\t\n\t return '

    ' + plainData.split('\\n\\n').join('

    ') + '

    ';\n\t }\n\t\n\t var wrapper = document.createElement('DIV'),\n\t newWrapper = document.createElement('DIV'),\n\t i,\n\t paragraph,\n\t firstLevelBlocks = ['DIV', 'P'],\n\t blockTyped,\n\t node;\n\t\n\t /**\r\n\t * Make HTML Element to Wrap Text\r\n\t * It allows us to work with input data as HTML content\r\n\t */\n\t wrapper.innerHTML = htmlData;\n\t paragraph = document.createElement('P');\n\t\n\t for (i = 0; i < wrapper.childNodes.length; i++) {\n\t\n\t node = wrapper.childNodes[i];\n\t\n\t blockTyped = firstLevelBlocks.indexOf(node.tagName) != -1;\n\t\n\t /**\r\n\t * If node is first-levet\r\n\t * we add this node to our new wrapper\r\n\t */\n\t if (blockTyped) {\n\t\n\t /**\r\n\t * If we had splitted inline nodes to paragraph before\r\n\t */\n\t if (paragraph.childNodes.length) {\n\t\n\t newWrapper.appendChild(paragraph.cloneNode(true));\n\t\n\t /** empty paragraph */\n\t paragraph = null;\n\t paragraph = document.createElement('P');\n\t }\n\t\n\t newWrapper.appendChild(node.cloneNode(true));\n\t } else {\n\t\n\t /** Collect all inline nodes to one as paragraph */\n\t paragraph.appendChild(node.cloneNode(true));\n\t\n\t /** if node is last we should append this node to paragraph and paragraph to new wrapper */\n\t if (i == wrapper.childNodes.length - 1) {\n\t\n\t newWrapper.appendChild(paragraph.cloneNode(true));\n\t }\n\t }\n\t }\n\t\n\t return newWrapper.innerHTML;\n\t };\n\t\n\t /**\r\n\t * Finds closest Contenteditable parent from Element\r\n\t * @param {Element} node element looking from\r\n\t * @return {Element} node contenteditable\r\n\t */\n\t content.getEditableParent = function (node) {\n\t\n\t while (node && node.contentEditable != 'true') {\n\t\n\t node = node.parentNode;\n\t }\n\t\n\t return node;\n\t };\n\t\n\t return content;\n\t}({});\n\n/***/ },\n/* 8 */\n/***/ function(module, exports, __webpack_require__) {\n\n\t'use strict';\n\t\n\t/**\r\n\t * Codex Editor toolbar module\r\n\t *\r\n\t * Contains:\r\n\t * - Inline toolbox\r\n\t * - Toolbox within plus button\r\n\t * - Settings section\r\n\t *\r\n\t * @author Codex Team\r\n\t * @version 1.0\r\n\t */\n\t\n\tmodule.exports = function (toolbar) {\n\t\n\t var editor = codex.editor;\n\t\n\t toolbar.settings = __webpack_require__(9);\n\t toolbar.inline = __webpack_require__(10);\n\t toolbar.toolbox = __webpack_require__(11);\n\t\n\t /**\r\n\t * Margin between focused node and toolbar\r\n\t */\n\t toolbar.defaultToolbarHeight = 49;\n\t\n\t toolbar.defaultOffset = 34;\n\t\n\t toolbar.opened = false;\n\t\n\t toolbar.current = null;\n\t\n\t /**\r\n\t * @protected\r\n\t */\n\t toolbar.open = function () {\n\t\n\t if (editor.hideToolbar) {\n\t\n\t return;\n\t }\n\t\n\t editor.nodes.toolbar.classList.add('opened');\n\t this.opened = true;\n\t };\n\t\n\t /**\r\n\t * @protected\r\n\t */\n\t toolbar.close = function () {\n\t\n\t editor.nodes.toolbar.classList.remove('opened');\n\t\n\t toolbar.opened = false;\n\t toolbar.current = null;\n\t\n\t for (var button in editor.nodes.toolbarButtons) {\n\t\n\t editor.nodes.toolbarButtons[button].classList.remove('selected');\n\t }\n\t\n\t /** Close toolbox when toolbar is not displayed */\n\t editor.toolbar.toolbox.close();\n\t editor.toolbar.settings.close();\n\t };\n\t\n\t toolbar.toggle = function () {\n\t\n\t if (!this.opened) {\n\t\n\t this.open();\n\t } else {\n\t\n\t this.close();\n\t }\n\t };\n\t\n\t toolbar.hidePlusButton = function () {\n\t\n\t editor.nodes.plusButton.classList.add('hide');\n\t };\n\t\n\t toolbar.showPlusButton = function () {\n\t\n\t editor.nodes.plusButton.classList.remove('hide');\n\t };\n\t\n\t /**\r\n\t * Moving toolbar to the specified node\r\n\t */\n\t toolbar.move = function () {\n\t\n\t /** Close Toolbox when we move toolbar */\n\t editor.toolbar.toolbox.close();\n\t\n\t if (!editor.content.currentNode) {\n\t\n\t return;\n\t }\n\t\n\t var newYCoordinate = editor.content.currentNode.offsetTop - editor.toolbar.defaultToolbarHeight / 2 + editor.toolbar.defaultOffset;\n\t\n\t editor.nodes.toolbar.style.transform = 'translate3D(0, ' + Math.floor(newYCoordinate) + 'px, 0)';\n\t\n\t /** Close trash actions */\n\t editor.toolbar.settings.hideRemoveActions();\n\t };\n\t\n\t return toolbar;\n\t}({});\n\n/***/ },\n/* 9 */\n/***/ function(module, exports) {\n\n\t'use strict';\n\t\n\t/**\r\n\t * Toolbar settings\r\n\t *\r\n\t * @version 1.0.4\r\n\t */\n\t\n\tmodule.exports = function (settings) {\n\t\n\t var editor = codex.editor;\n\t\n\t settings.opened = false;\n\t\n\t settings.setting = null;\n\t settings.actions = null;\n\t\n\t /**\r\n\t * Append and open settings\r\n\t */\n\t settings.open = function (toolType) {\n\t\n\t /**\r\n\t * Append settings content\r\n\t * It's stored in tool.settings\r\n\t */\n\t if (!editor.tools[toolType] || !editor.tools[toolType].makeSettings) {\n\t\n\t editor.core.log('Plugin \\xAB' + toolType + '\\xBB has no settings', 'warn');\n\t editor.nodes.pluginSettings.innerHTML = '\\u041F\\u043B\\u0430\\u0433\\u0438\\u043D \\xAB' + toolType + '\\xBB \\u043D\\u0435 \\u0438\\u043C\\u0435\\u0435\\u0442 \\u043D\\u0430\\u0441\\u0442\\u0440\\u043E\\u0435\\u043A';\n\t } else {\n\t\n\t /**\r\n\t * Draw settings block\r\n\t */\n\t var settingsBlock = editor.tools[toolType].makeSettings();\n\t\n\t editor.nodes.pluginSettings.appendChild(settingsBlock);\n\t }\n\t\n\t /** Open settings block */\n\t editor.nodes.blockSettings.classList.add('opened');\n\t this.opened = true;\n\t };\n\t\n\t /**\r\n\t * Close and clear settings\r\n\t */\n\t settings.close = function () {\n\t\n\t editor.nodes.blockSettings.classList.remove('opened');\n\t editor.nodes.pluginSettings.innerHTML = '';\n\t\n\t this.opened = false;\n\t };\n\t\n\t /**\r\n\t * @param {string} toolType - plugin type\r\n\t */\n\t settings.toggle = function (toolType) {\n\t\n\t if (!this.opened) {\n\t\n\t this.open(toolType);\n\t } else {\n\t\n\t this.close();\n\t }\n\t };\n\t\n\t /**\r\n\t * Here we will draw buttons and add listeners to components\r\n\t */\n\t settings.makeRemoveBlockButton = function () {\n\t\n\t var removeBlockWrapper = editor.draw.node('SPAN', 'ce-toolbar__remove-btn', {}),\n\t settingButton = editor.draw.node('SPAN', 'ce-toolbar__remove-setting', { innerHTML: '' }),\n\t actionWrapper = editor.draw.node('DIV', 'ce-toolbar__remove-confirmation', {}),\n\t confirmAction = editor.draw.node('DIV', 'ce-toolbar__remove-confirm', { textContent: 'Удалить блок' }),\n\t cancelAction = editor.draw.node('DIV', 'ce-toolbar__remove-cancel', { textContent: 'Отмена' });\n\t\n\t editor.listeners.add(settingButton, 'click', editor.toolbar.settings.removeButtonClicked, false);\n\t\n\t editor.listeners.add(confirmAction, 'click', editor.toolbar.settings.confirmRemovingRequest, false);\n\t\n\t editor.listeners.add(cancelAction, 'click', editor.toolbar.settings.cancelRemovingRequest, false);\n\t\n\t actionWrapper.appendChild(confirmAction);\n\t actionWrapper.appendChild(cancelAction);\n\t\n\t removeBlockWrapper.appendChild(settingButton);\n\t removeBlockWrapper.appendChild(actionWrapper);\n\t\n\t /** Save setting */\n\t editor.toolbar.settings.setting = settingButton;\n\t editor.toolbar.settings.actions = actionWrapper;\n\t\n\t return removeBlockWrapper;\n\t };\n\t\n\t settings.removeButtonClicked = function () {\n\t\n\t var action = editor.toolbar.settings.actions;\n\t\n\t if (action.classList.contains('opened')) {\n\t\n\t editor.toolbar.settings.hideRemoveActions();\n\t } else {\n\t\n\t editor.toolbar.settings.showRemoveActions();\n\t }\n\t\n\t editor.toolbar.toolbox.close();\n\t editor.toolbar.settings.close();\n\t };\n\t\n\t settings.cancelRemovingRequest = function () {\n\t\n\t editor.toolbar.settings.actions.classList.remove('opened');\n\t };\n\t\n\t settings.confirmRemovingRequest = function () {\n\t\n\t var currentBlock = editor.content.currentNode,\n\t firstLevelBlocksCount;\n\t\n\t currentBlock.remove();\n\t\n\t firstLevelBlocksCount = editor.nodes.redactor.childNodes.length;\n\t\n\t /**\r\n\t * If all blocks are removed\r\n\t */\n\t if (firstLevelBlocksCount === 0) {\n\t\n\t /** update currentNode variable */\n\t editor.content.currentNode = null;\n\t\n\t /** Inserting new empty initial block */\n\t editor.ui.addInitialBlock();\n\t }\n\t\n\t editor.ui.saveInputs();\n\t\n\t editor.toolbar.close();\n\t };\n\t\n\t settings.showRemoveActions = function () {\n\t\n\t editor.toolbar.settings.actions.classList.add('opened');\n\t };\n\t\n\t settings.hideRemoveActions = function () {\n\t\n\t editor.toolbar.settings.actions.classList.remove('opened');\n\t };\n\t\n\t return settings;\n\t}({});\n\n/***/ },\n/* 10 */\n/***/ function(module, exports) {\n\n\t'use strict';\n\t\n\t/**\r\n\t * Inline toolbar\r\n\t *\r\n\t * Contains from tools:\r\n\t * Bold, Italic, Underline and Anchor\r\n\t *\r\n\t * @author Codex Team\r\n\t * @version 1.0\r\n\t */\n\t\n\tmodule.exports = function (inline) {\n\t\n\t var editor = codex.editor;\n\t\n\t inline.buttonsOpened = null;\n\t inline.actionsOpened = null;\n\t inline.wrappersOffset = null;\n\t\n\t /**\r\n\t * saving selection that need for execCommand for styling\r\n\t *\r\n\t */\n\t inline.storedSelection = null;\n\t\n\t /**\r\n\t * @protected\r\n\t *\r\n\t * Open inline toobar\r\n\t */\n\t inline.show = function () {\n\t\n\t var currentNode = editor.content.currentNode,\n\t tool = currentNode.dataset.tool,\n\t plugin;\n\t\n\t /**\r\n\t * tool allowed to open inline toolbar\r\n\t */\n\t plugin = editor.tools[tool];\n\t\n\t if (!plugin.showInlineToolbar) return;\n\t\n\t var selectedText = inline.getSelectionText(),\n\t toolbar = editor.nodes.inlineToolbar.wrapper;\n\t\n\t if (selectedText.length > 0) {\n\t\n\t /** Move toolbar and open */\n\t editor.toolbar.inline.move();\n\t\n\t /** Open inline toolbar */\n\t toolbar.classList.add('opened');\n\t\n\t /** show buttons of inline toolbar */\n\t editor.toolbar.inline.showButtons();\n\t }\n\t };\n\t\n\t /**\r\n\t * @protected\r\n\t *\r\n\t * Closes inline toolbar\r\n\t */\n\t inline.close = function () {\n\t\n\t var toolbar = editor.nodes.inlineToolbar.wrapper;\n\t\n\t toolbar.classList.remove('opened');\n\t };\n\t\n\t /**\r\n\t * @private\r\n\t *\r\n\t * Moving toolbar\r\n\t */\n\t inline.move = function () {\n\t\n\t if (!this.wrappersOffset) {\n\t\n\t this.wrappersOffset = this.getWrappersOffset();\n\t }\n\t\n\t var coords = this.getSelectionCoords(),\n\t defaultOffset = 0,\n\t toolbar = editor.nodes.inlineToolbar.wrapper,\n\t newCoordinateX,\n\t newCoordinateY;\n\t\n\t if (toolbar.offsetHeight === 0) {\n\t\n\t defaultOffset = 40;\n\t }\n\t\n\t newCoordinateX = coords.x - this.wrappersOffset.left;\n\t newCoordinateY = coords.y + window.scrollY - this.wrappersOffset.top - defaultOffset - toolbar.offsetHeight;\n\t\n\t toolbar.style.transform = 'translate3D(' + Math.floor(newCoordinateX) + 'px, ' + Math.floor(newCoordinateY) + 'px, 0)';\n\t\n\t /** Close everything */\n\t editor.toolbar.inline.closeButtons();\n\t editor.toolbar.inline.closeAction();\n\t };\n\t\n\t /**\r\n\t * @private\r\n\t *\r\n\t * Tool Clicked\r\n\t */\n\t\n\t inline.toolClicked = function (event, type) {\n\t\n\t /**\r\n\t * For simple tools we use default browser function\r\n\t * For more complicated tools, we should write our own behavior\r\n\t */\n\t switch (type) {\n\t case 'createLink':\n\t editor.toolbar.inline.createLinkAction(event, type);break;\n\t default:\n\t editor.toolbar.inline.defaultToolAction(type);break;\n\t }\n\t\n\t /**\r\n\t * highlight buttons\r\n\t * after making some action\r\n\t */\n\t editor.nodes.inlineToolbar.buttons.childNodes.forEach(editor.toolbar.inline.hightlight);\n\t };\n\t\n\t /**\r\n\t * @private\r\n\t *\r\n\t * Saving wrappers offset in DOM\r\n\t */\n\t inline.getWrappersOffset = function () {\n\t\n\t var wrapper = editor.nodes.wrapper,\n\t offset = this.getOffset(wrapper);\n\t\n\t this.wrappersOffset = offset;\n\t return offset;\n\t };\n\t\n\t /**\r\n\t * @private\r\n\t *\r\n\t * Calculates offset of DOM element\r\n\t *\r\n\t * @param el\r\n\t * @returns {{top: number, left: number}}\r\n\t */\n\t inline.getOffset = function (el) {\n\t\n\t var _x = 0;\n\t var _y = 0;\n\t\n\t while (el && !isNaN(el.offsetLeft) && !isNaN(el.offsetTop)) {\n\t\n\t _x += el.offsetLeft + el.clientLeft;\n\t _y += el.offsetTop + el.clientTop;\n\t el = el.offsetParent;\n\t }\n\t return { top: _y, left: _x };\n\t };\n\t\n\t /**\r\n\t * @private\r\n\t *\r\n\t * Calculates position of selected text\r\n\t * @returns {{x: number, y: number}}\r\n\t */\n\t inline.getSelectionCoords = function () {\n\t\n\t var sel = document.selection,\n\t range;\n\t var x = 0,\n\t y = 0;\n\t\n\t if (sel) {\n\t\n\t if (sel.type != 'Control') {\n\t\n\t range = sel.createRange();\n\t range.collapse(true);\n\t x = range.boundingLeft;\n\t y = range.boundingTop;\n\t }\n\t } else if (window.getSelection) {\n\t\n\t sel = window.getSelection();\n\t\n\t if (sel.rangeCount) {\n\t\n\t range = sel.getRangeAt(0).cloneRange();\n\t if (range.getClientRects) {\n\t\n\t range.collapse(true);\n\t var rect = range.getClientRects()[0];\n\t\n\t if (!rect) {\n\t\n\t return;\n\t }\n\t\n\t x = rect.left;\n\t y = rect.top;\n\t }\n\t }\n\t }\n\t return { x: x, y: y };\n\t };\n\t\n\t /**\r\n\t * @private\r\n\t *\r\n\t * Returns selected text as String\r\n\t * @returns {string}\r\n\t */\n\t inline.getSelectionText = function () {\n\t\n\t var selectedText = '';\n\t\n\t // all modern browsers and IE9+\n\t if (window.getSelection) {\n\t\n\t selectedText = window.getSelection().toString();\n\t }\n\t\n\t return selectedText;\n\t };\n\t\n\t /** Opens buttons block */\n\t inline.showButtons = function () {\n\t\n\t var buttons = editor.nodes.inlineToolbar.buttons;\n\t\n\t buttons.classList.add('opened');\n\t\n\t editor.toolbar.inline.buttonsOpened = true;\n\t\n\t /** highlight buttons */\n\t editor.nodes.inlineToolbar.buttons.childNodes.forEach(editor.toolbar.inline.hightlight);\n\t };\n\t\n\t /** Makes buttons disappear */\n\t inline.closeButtons = function () {\n\t\n\t var buttons = editor.nodes.inlineToolbar.buttons;\n\t\n\t buttons.classList.remove('opened');\n\t\n\t editor.toolbar.inline.buttonsOpened = false;\n\t };\n\t\n\t /** Open buttons defined action if exist */\n\t inline.showActions = function () {\n\t\n\t var action = editor.nodes.inlineToolbar.actions;\n\t\n\t action.classList.add('opened');\n\t\n\t editor.toolbar.inline.actionsOpened = true;\n\t };\n\t\n\t /** Close actions block */\n\t inline.closeAction = function () {\n\t\n\t var action = editor.nodes.inlineToolbar.actions;\n\t\n\t action.innerHTML = '';\n\t action.classList.remove('opened');\n\t editor.toolbar.inline.actionsOpened = false;\n\t };\n\t\n\t /**\r\n\t * Callback for keydowns in inline toolbar \"Insert link...\" input\r\n\t */\n\t var inlineToolbarAnchorInputKeydown_ = function inlineToolbarAnchorInputKeydown_(event) {\n\t\n\t if (event.keyCode != editor.core.keys.ENTER) {\n\t\n\t return;\n\t }\n\t\n\t var editable = editor.content.currentNode,\n\t storedSelection = editor.toolbar.inline.storedSelection;\n\t\n\t editor.toolbar.inline.restoreSelection(editable, storedSelection);\n\t editor.toolbar.inline.setAnchor(this.value);\n\t\n\t /**\r\n\t * Preventing events that will be able to happen\r\n\t */\n\t event.preventDefault();\n\t event.stopImmediatePropagation();\n\t\n\t editor.toolbar.inline.clearRange();\n\t };\n\t\n\t /** Action for link creation or for setting anchor */\n\t inline.createLinkAction = function (event) {\n\t\n\t var isActive = this.isLinkActive();\n\t\n\t var editable = editor.content.currentNode,\n\t storedSelection = editor.toolbar.inline.saveSelection(editable);\n\t\n\t /** Save globally selection */\n\t editor.toolbar.inline.storedSelection = storedSelection;\n\t\n\t if (isActive) {\n\t\n\t /**\r\n\t * Changing stored selection. if we want to remove anchor from word\r\n\t * we should remove anchor from whole word, not only selected part.\r\n\t * The solution is than we get the length of current link\r\n\t * Change start position to - end of selection minus length of anchor\r\n\t */\n\t editor.toolbar.inline.restoreSelection(editable, storedSelection);\n\t\n\t editor.toolbar.inline.defaultToolAction('unlink');\n\t } else {\n\t\n\t /** Create input and close buttons */\n\t var action = editor.draw.inputForLink();\n\t\n\t editor.nodes.inlineToolbar.actions.appendChild(action);\n\t\n\t editor.toolbar.inline.closeButtons();\n\t editor.toolbar.inline.showActions();\n\t\n\t /**\r\n\t * focus to input\r\n\t * Solution: https://developer.mozilla.org/ru/docs/Web/API/HTMLElement/focus\r\n\t * Prevents event after showing input and when we need to focus an input which is in unexisted form\r\n\t */\n\t action.focus();\n\t event.preventDefault();\n\t\n\t /** Callback to link action */\n\t editor.listeners.add(action, 'keydown', inlineToolbarAnchorInputKeydown_, false);\n\t }\n\t };\n\t\n\t inline.isLinkActive = function () {\n\t\n\t var isActive = false;\n\t\n\t editor.nodes.inlineToolbar.buttons.childNodes.forEach(function (tool) {\n\t\n\t var dataType = tool.dataset.type;\n\t\n\t if (dataType == 'link' && tool.classList.contains('hightlighted')) {\n\t\n\t isActive = true;\n\t }\n\t });\n\t\n\t return isActive;\n\t };\n\t\n\t /** default action behavior of tool */\n\t inline.defaultToolAction = function (type) {\n\t\n\t document.execCommand(type, false, null);\n\t };\n\t\n\t /**\r\n\t * @private\r\n\t *\r\n\t * Sets URL\r\n\t *\r\n\t * @param {String} url - URL\r\n\t */\n\t inline.setAnchor = function (url) {\n\t\n\t document.execCommand('createLink', false, url);\n\t\n\t /** Close after URL inserting */\n\t editor.toolbar.inline.closeAction();\n\t };\n\t\n\t /**\r\n\t * @private\r\n\t *\r\n\t * Saves selection\r\n\t */\n\t inline.saveSelection = function (containerEl) {\n\t\n\t var range = window.getSelection().getRangeAt(0),\n\t preSelectionRange = range.cloneRange(),\n\t start;\n\t\n\t preSelectionRange.selectNodeContents(containerEl);\n\t preSelectionRange.setEnd(range.startContainer, range.startOffset);\n\t\n\t start = preSelectionRange.toString().length;\n\t\n\t return {\n\t start: start,\n\t end: start + range.toString().length\n\t };\n\t };\n\t\n\t /**\r\n\t * @private\r\n\t *\r\n\t * Sets to previous selection (Range)\r\n\t *\r\n\t * @param {Element} containerEl - editable element where we restore range\r\n\t * @param {Object} savedSel - range basic information to restore\r\n\t */\n\t inline.restoreSelection = function (containerEl, savedSel) {\n\t\n\t var range = document.createRange(),\n\t charIndex = 0;\n\t\n\t range.setStart(containerEl, 0);\n\t range.collapse(true);\n\t\n\t var nodeStack = [containerEl],\n\t node,\n\t foundStart = false,\n\t stop = false,\n\t nextCharIndex;\n\t\n\t while (!stop && (node = nodeStack.pop())) {\n\t\n\t if (node.nodeType == 3) {\n\t\n\t nextCharIndex = charIndex + node.length;\n\t\n\t if (!foundStart && savedSel.start >= charIndex && savedSel.start <= nextCharIndex) {\n\t\n\t range.setStart(node, savedSel.start - charIndex);\n\t foundStart = true;\n\t }\n\t if (foundStart && savedSel.end >= charIndex && savedSel.end <= nextCharIndex) {\n\t\n\t range.setEnd(node, savedSel.end - charIndex);\n\t stop = true;\n\t }\n\t charIndex = nextCharIndex;\n\t } else {\n\t\n\t var i = node.childNodes.length;\n\t\n\t while (i--) {\n\t\n\t nodeStack.push(node.childNodes[i]);\n\t }\n\t }\n\t }\n\t\n\t var sel = window.getSelection();\n\t\n\t sel.removeAllRanges();\n\t sel.addRange(range);\n\t };\n\t\n\t /**\r\n\t * @private\r\n\t *\r\n\t * Removes all ranges from window selection\r\n\t */\n\t inline.clearRange = function () {\n\t\n\t var selection = window.getSelection();\n\t\n\t selection.removeAllRanges();\n\t };\n\t\n\t /**\r\n\t * @private\r\n\t *\r\n\t * sets or removes hightlight\r\n\t */\n\t inline.hightlight = function (tool) {\n\t\n\t var dataType = tool.dataset.type;\n\t\n\t if (document.queryCommandState(dataType)) {\n\t\n\t editor.toolbar.inline.setButtonHighlighted(tool);\n\t } else {\n\t\n\t editor.toolbar.inline.removeButtonsHighLight(tool);\n\t }\n\t\n\t /**\r\n\t *\r\n\t * hightlight for anchors\r\n\t */\n\t var selection = window.getSelection(),\n\t tag = selection.anchorNode.parentNode;\n\t\n\t if (tag.tagName == 'A' && dataType == 'link') {\n\t\n\t editor.toolbar.inline.setButtonHighlighted(tool);\n\t }\n\t };\n\t\n\t /**\r\n\t * @private\r\n\t *\r\n\t * Mark button if text is already executed\r\n\t */\n\t inline.setButtonHighlighted = function (button) {\n\t\n\t button.classList.add('hightlighted');\n\t\n\t /** At link tool we also change icon */\n\t if (button.dataset.type == 'link') {\n\t\n\t var icon = button.childNodes[0];\n\t\n\t icon.classList.remove('ce-icon-link');\n\t icon.classList.add('ce-icon-unlink');\n\t }\n\t };\n\t\n\t /**\r\n\t * @private\r\n\t *\r\n\t * Removes hightlight\r\n\t */\n\t inline.removeButtonsHighLight = function (button) {\n\t\n\t button.classList.remove('hightlighted');\n\t\n\t /** At link tool we also change icon */\n\t if (button.dataset.type == 'link') {\n\t\n\t var icon = button.childNodes[0];\n\t\n\t icon.classList.remove('ce-icon-unlink');\n\t icon.classList.add('ce-icon-link');\n\t }\n\t };\n\t\n\t return inline;\n\t}({});\n\n/***/ },\n/* 11 */\n/***/ function(module, exports) {\n\n\t'use strict';\n\t\n\t/**\r\n\t * Codex Editor toolbox\r\n\t *\r\n\t * All tools be able to appended here\r\n\t *\r\n\t * @author Codex Team\r\n\t * @version 1.0\r\n\t */\n\t\n\tmodule.exports = function (toolbox) {\n\t\n\t var editor = codex.editor;\n\t\n\t toolbox.opened = false;\n\t\n\t /** Shows toolbox */\n\t toolbox.open = function () {\n\t\n\t /** Close setting if toolbox is opened */\n\t if (editor.toolbar.settings.opened) {\n\t\n\t editor.toolbar.settings.close();\n\t }\n\t\n\t /** display toolbox */\n\t editor.nodes.toolbox.classList.add('opened');\n\t\n\t /** Animate plus button */\n\t editor.nodes.plusButton.classList.add('clicked');\n\t\n\t /** toolbox state */\n\t editor.toolbar.toolbox.opened = true;\n\t };\n\t\n\t /** Closes toolbox */\n\t toolbox.close = function () {\n\t\n\t /** Makes toolbox disapear */\n\t editor.nodes.toolbox.classList.remove('opened');\n\t\n\t /** Rotate plus button */\n\t editor.nodes.plusButton.classList.remove('clicked');\n\t\n\t /** toolbox state */\n\t editor.toolbar.toolbox.opened = false;\n\t\n\t editor.toolbar.current = null;\n\t };\n\t\n\t toolbox.leaf = function () {\n\t\n\t var currentTool = editor.toolbar.current,\n\t tools = Object.keys(editor.tools),\n\t barButtons = editor.nodes.toolbarButtons,\n\t nextToolIndex = 0,\n\t toolToSelect = void 0,\n\t visibleTool = void 0,\n\t tool = void 0;\n\t\n\t if (!currentTool) {\n\t\n\t /** Get first tool from object*/\n\t for (tool in editor.tools) {\n\t\n\t if (editor.tools[tool].displayInToolbox) {\n\t\n\t break;\n\t }\n\t\n\t nextToolIndex++;\n\t }\n\t } else {\n\t\n\t nextToolIndex = (tools.indexOf(currentTool) + 1) % tools.length;\n\t visibleTool = tools[nextToolIndex];\n\t\n\t while (!editor.tools[visibleTool].displayInToolbox) {\n\t\n\t nextToolIndex = (nextToolIndex + 1) % tools.length;\n\t visibleTool = tools[nextToolIndex];\n\t }\n\t }\n\t\n\t toolToSelect = tools[nextToolIndex];\n\t\n\t for (var button in barButtons) {\n\t\n\t barButtons[button].classList.remove('selected');\n\t }\n\t\n\t barButtons[toolToSelect].classList.add('selected');\n\t editor.toolbar.current = toolToSelect;\n\t };\n\t\n\t /**\r\n\t * Transforming selected node type into selected toolbar element type\r\n\t * @param {event} event\r\n\t */\n\t toolbox.toolClicked = function (event) {\n\t\n\t /**\r\n\t * UNREPLACEBLE_TOOLS this types of tools are forbidden to replace even they are empty\r\n\t */\n\t var UNREPLACEBLE_TOOLS = ['image', 'link', 'list', 'instagram', 'twitter', 'embed'],\n\t tool = editor.tools[editor.toolbar.current],\n\t workingNode = editor.content.currentNode,\n\t currentInputIndex = editor.caret.inputIndex,\n\t newBlockContent,\n\t appendCallback,\n\t blockData;\n\t\n\t /** Make block from plugin */\n\t newBlockContent = tool.render();\n\t\n\t /** information about block */\n\t blockData = {\n\t block: newBlockContent,\n\t type: tool.type,\n\t stretched: false\n\t };\n\t\n\t if (workingNode && UNREPLACEBLE_TOOLS.indexOf(workingNode.dataset.tool) === -1 && workingNode.textContent.trim() === '') {\n\t\n\t /** Replace current block */\n\t editor.content.switchBlock(workingNode, newBlockContent, tool.type);\n\t } else {\n\t\n\t /** Insert new Block from plugin */\n\t editor.content.insertBlock(blockData);\n\t\n\t /** increase input index */\n\t currentInputIndex++;\n\t }\n\t\n\t /** Fire tool append callback */\n\t appendCallback = tool.appendCallback;\n\t\n\t if (appendCallback && typeof appendCallback == 'function') {\n\t\n\t appendCallback.call(event);\n\t }\n\t\n\t window.setTimeout(function () {\n\t\n\t /** Set caret to current block */\n\t editor.caret.setToBlock(currentInputIndex);\n\t }, 10);\n\t\n\t /**\r\n\t * Changing current Node\r\n\t */\n\t editor.content.workingNodeChanged();\n\t\n\t /**\r\n\t * Move toolbar when node is changed\r\n\t */\n\t editor.toolbar.move();\n\t };\n\t\n\t return toolbox;\n\t}({});\n\n/***/ },\n/* 12 */\n/***/ function(module, exports) {\n\n\t'use strict';\n\t\n\t/**\r\n\t * @module Codex Editor Callbacks module\r\n\t * @description Module works with editor added Elements\r\n\t *\r\n\t * @author Codex Team\r\n\t * @version 1.4.0\r\n\t */\n\t\n\tmodule.exports = function (callbacks) {\n\t\n\t var editor = codex.editor;\n\t\n\t /**\r\n\t * used by UI module\r\n\t * @description Routes all keydowns on document\r\n\t * @param {Object} event\r\n\t */\n\t callbacks.globalKeydown = function (event) {\n\t\n\t switch (event.keyCode) {\n\t case editor.core.keys.ENTER:\n\t enterKeyPressed_(event);break;\n\t }\n\t };\n\t\n\t /**\r\n\t * used by UI module\r\n\t * @description Routes all keydowns on redactors area\r\n\t * @param {Object} event\r\n\t */\n\t callbacks.redactorKeyDown = function (event) {\n\t\n\t switch (event.keyCode) {\n\t case editor.core.keys.TAB:\n\t tabKeyPressedOnRedactorsZone_(event);break;\n\t case editor.core.keys.ENTER:\n\t enterKeyPressedOnRedactorsZone_(event);break;\n\t case editor.core.keys.ESC:\n\t escapeKeyPressedOnRedactorsZone_(event);break;\n\t default:\n\t defaultKeyPressedOnRedactorsZone_(event);break;\n\t }\n\t };\n\t\n\t /**\r\n\t * used by UI module\r\n\t * @description Routes all keyup events\r\n\t * @param {Object} event\r\n\t */\n\t callbacks.globalKeyup = function (event) {\n\t\n\t switch (event.keyCode) {\n\t case editor.core.keys.UP:\n\t case editor.core.keys.LEFT:\n\t case editor.core.keys.RIGHT:\n\t case editor.core.keys.DOWN:\n\t arrowKeyPressed_(event);break;\n\t }\n\t };\n\t\n\t /**\r\n\t * @param {Object} event\r\n\t * @private\r\n\t *\r\n\t * Handles behaviour when tab pressed\r\n\t * @description if Content is empty show toolbox (if it is closed) or leaf tools\r\n\t * uses Toolbars toolbox module to handle the situation\r\n\t */\n\t var tabKeyPressedOnRedactorsZone_ = function tabKeyPressedOnRedactorsZone_(event) {\n\t\n\t /**\r\n\t * Wait for solution. Would like to know the behaviour\r\n\t * @todo Add spaces\r\n\t */\n\t event.preventDefault();\n\t\n\t var nativeInputs = editor.content.currentNode.querySelectorAll('textarea, input'),\n\t nativeInputsAreEmpty = true,\n\t textContentIsEmpty = !editor.content.currentNode.textContent.trim();\n\t\n\t Array.prototype.map.call(nativeInputs, function (input) {\n\t\n\t if (input.type == 'textarea' || input.type == 'text') {\n\t\n\t nativeInputsAreEmpty = nativeInputsAreEmpty && !input.value.trim();\n\t }\n\t });\n\t\n\t var blockIsEmpty = textContentIsEmpty && nativeInputsAreEmpty;\n\t\n\t if (!blockIsEmpty) {\n\t\n\t return;\n\t }\n\t\n\t if (!editor.toolbar.opened) {\n\t\n\t editor.toolbar.open();\n\t }\n\t\n\t if (editor.toolbar.opened && !editor.toolbar.toolbox.opened) {\n\t\n\t editor.toolbar.toolbox.open();\n\t } else {\n\t\n\t editor.toolbar.toolbox.leaf();\n\t }\n\t };\n\t\n\t /**\r\n\t * Handles global EnterKey Press\r\n\t * @see enterPressedOnBlock_\r\n\t * @param {Object} event\r\n\t */\n\t var enterKeyPressed_ = function enterKeyPressed_() {\n\t\n\t if (editor.content.editorAreaHightlighted) {\n\t\n\t /**\r\n\t * it means that we lose input index, saved index before is not correct\r\n\t * therefore we need to set caret when we insert new block\r\n\t */\n\t editor.caret.inputIndex = -1;\n\t\n\t enterPressedOnBlock_();\n\t }\n\t };\n\t\n\t /**\r\n\t * Callback for enter key pressing in first-level block area\r\n\t *\r\n\t * @param {Event} event\r\n\t * @private\r\n\t *\r\n\t * @description Inserts new block with initial type from settings\r\n\t */\n\t var enterPressedOnBlock_ = function enterPressedOnBlock_() {\n\t\n\t var NEW_BLOCK_TYPE = editor.settings.initialBlockPlugin;\n\t\n\t editor.content.insertBlock({\n\t type: NEW_BLOCK_TYPE,\n\t block: editor.tools[NEW_BLOCK_TYPE].render()\n\t }, true);\n\t\n\t editor.toolbar.move();\n\t editor.toolbar.open();\n\t };\n\t\n\t /**\r\n\t * ENTER key handler\r\n\t *\r\n\t * @param {Object} event\r\n\t * @private\r\n\t *\r\n\t * @description Makes new block with initial type from settings\r\n\t */\n\t var enterKeyPressedOnRedactorsZone_ = function enterKeyPressedOnRedactorsZone_(event) {\n\t\n\t if (event.target.contentEditable == 'true') {\n\t\n\t /** Update input index */\n\t editor.caret.saveCurrentInputIndex();\n\t }\n\t\n\t var currentInputIndex = editor.caret.getCurrentInputIndex() || 0,\n\t workingNode = editor.content.currentNode,\n\t tool = workingNode.dataset.tool,\n\t isEnterPressedOnToolbar = editor.toolbar.opened && editor.toolbar.current && event.target == editor.state.inputs[currentInputIndex];\n\t\n\t /** The list of tools which needs the default browser behaviour */\n\t var enableLineBreaks = editor.tools[tool].enableLineBreaks;\n\t\n\t /** This type of block creates when enter is pressed */\n\t var NEW_BLOCK_TYPE = editor.settings.initialBlockPlugin;\n\t\n\t /**\r\n\t * When toolbar is opened, select tool instead of making new paragraph\r\n\t */\n\t if (isEnterPressedOnToolbar) {\n\t\n\t event.preventDefault();\n\t\n\t editor.toolbar.toolbox.toolClicked(event);\n\t\n\t editor.toolbar.close();\n\t\n\t /**\r\n\t * Stop other listeners callback executions\r\n\t */\n\t event.stopPropagation();\n\t event.stopImmediatePropagation();\n\t\n\t return;\n\t }\n\t\n\t /**\r\n\t * Allow paragraph lineBreaks with shift enter\r\n\t * Or if shiftkey pressed and enter and enabledLineBreaks, the let new block creation\r\n\t */\n\t if (event.shiftKey || enableLineBreaks) {\n\t\n\t event.stopPropagation();\n\t event.stopImmediatePropagation();\n\t return;\n\t }\n\t\n\t var currentSelection = window.getSelection(),\n\t currentSelectedNode = currentSelection.anchorNode,\n\t caretAtTheEndOfText = editor.caret.position.atTheEnd(),\n\t isTextNodeHasParentBetweenContenteditable = false;\n\t\n\t /**\r\n\t * Allow making new

    in same block by SHIFT+ENTER and forbids to prevent default browser behaviour\r\n\t */\n\t if (event.shiftKey && !enableLineBreaks) {\n\t\n\t editor.callback.enterPressedOnBlock(editor.content.currentBlock, event);\n\t event.preventDefault();\n\t return;\n\t }\n\t\n\t /**\r\n\t * Workaround situation when caret at the Text node that has some wrapper Elements\r\n\t * Split block cant handle this.\r\n\t * We need to save default behavior\r\n\t */\n\t isTextNodeHasParentBetweenContenteditable = currentSelectedNode && currentSelectedNode.parentNode.contentEditable != 'true';\n\t\n\t /**\r\n\t * Split blocks when input has several nodes and caret placed in textNode\r\n\t */\n\t if (currentSelectedNode.nodeType == editor.core.nodeTypes.TEXT && !isTextNodeHasParentBetweenContenteditable && !caretAtTheEndOfText) {\n\t\n\t event.preventDefault();\n\t\n\t editor.core.log('Splitting Text node...');\n\t\n\t editor.content.splitBlock(currentInputIndex);\n\t\n\t /** Show plus button when next input after split is empty*/\n\t if (!editor.state.inputs[currentInputIndex + 1].textContent.trim()) {\n\t\n\t editor.toolbar.showPlusButton();\n\t }\n\t } else {\n\t\n\t var islastNode = editor.content.isLastNode(currentSelectedNode);\n\t\n\t if (islastNode && caretAtTheEndOfText) {\n\t\n\t event.preventDefault();\n\t event.stopPropagation();\n\t event.stopImmediatePropagation();\n\t\n\t editor.core.log('ENTER clicked in last textNode. Create new BLOCK');\n\t\n\t editor.content.insertBlock({\n\t type: NEW_BLOCK_TYPE,\n\t block: editor.tools[NEW_BLOCK_TYPE].render()\n\t }, true);\n\t\n\t editor.toolbar.move();\n\t editor.toolbar.open();\n\t\n\t /** Show plus button with empty block */\n\t editor.toolbar.showPlusButton();\n\t }\n\t }\n\t\n\t /** get all inputs after new appending block */\n\t editor.ui.saveInputs();\n\t };\n\t\n\t /**\r\n\t * Escape behaviour\r\n\t * @param event\r\n\t * @private\r\n\t *\r\n\t * @description Closes toolbox and toolbar. Prevents default behaviour\r\n\t */\n\t var escapeKeyPressedOnRedactorsZone_ = function escapeKeyPressedOnRedactorsZone_(event) {\n\t\n\t /** Close all toolbar */\n\t editor.toolbar.close();\n\t\n\t /** Close toolbox */\n\t editor.toolbar.toolbox.close();\n\t\n\t event.preventDefault();\n\t };\n\t\n\t /**\r\n\t * @param {Event} event\r\n\t * @private\r\n\t *\r\n\t * closes and moves toolbar\r\n\t */\n\t var arrowKeyPressed_ = function arrowKeyPressed_(event) {\n\t\n\t editor.content.workingNodeChanged();\n\t\n\t /* Closing toolbar */\n\t editor.toolbar.close();\n\t editor.toolbar.move();\n\t };\n\t\n\t /**\r\n\t * @private\r\n\t * @param {Event} event\r\n\t *\r\n\t * @description Closes all opened bars from toolbar.\r\n\t * If block is mark, clears highlightning\r\n\t */\n\t var defaultKeyPressedOnRedactorsZone_ = function defaultKeyPressedOnRedactorsZone_() {\n\t\n\t editor.toolbar.close();\n\t\n\t if (!editor.toolbar.inline.actionsOpened) {\n\t\n\t editor.toolbar.inline.close();\n\t editor.content.clearMark();\n\t }\n\t };\n\t\n\t /**\r\n\t * Handler when clicked on redactors area\r\n\t *\r\n\t * @protected\r\n\t * @param event\r\n\t *\r\n\t * @description Detects clicked area. If it is first-level block area, marks as detected and\r\n\t * on next enter press will be inserted new block\r\n\t * Otherwise, save carets position (input index) and put caret to the editable zone.\r\n\t *\r\n\t * @see detectWhenClickedOnFirstLevelBlockArea_\r\n\t *\r\n\t */\n\t callbacks.redactorClicked = function (event) {\n\t\n\t detectWhenClickedOnFirstLevelBlockArea_();\n\t\n\t editor.content.workingNodeChanged(event.target);\n\t editor.ui.saveInputs();\n\t\n\t var selectedText = editor.toolbar.inline.getSelectionText(),\n\t firstLevelBlock;\n\t\n\t /** If selection range took off, then we hide inline toolbar */\n\t if (selectedText.length === 0) {\n\t\n\t editor.toolbar.inline.close();\n\t }\n\t\n\t /** Update current input index in memory when caret focused into existed input */\n\t if (event.target.contentEditable == 'true') {\n\t\n\t editor.caret.saveCurrentInputIndex();\n\t }\n\t\n\t if (editor.content.currentNode === null) {\n\t\n\t /**\r\n\t * If inputs in redactor does not exits, then we put input index 0 not -1\r\n\t */\n\t var indexOfLastInput = editor.state.inputs.length > 0 ? editor.state.inputs.length - 1 : 0;\n\t\n\t /** If we have any inputs */\n\t if (editor.state.inputs.length) {\n\t\n\t /** getting firstlevel parent of input */\n\t firstLevelBlock = editor.content.getFirstLevelBlock(editor.state.inputs[indexOfLastInput]);\n\t }\n\t\n\t /** If input is empty, then we set caret to the last input */\n\t if (editor.state.inputs.length && editor.state.inputs[indexOfLastInput].textContent === '' && firstLevelBlock.dataset.tool == editor.settings.initialBlockPlugin) {\n\t\n\t editor.caret.setToBlock(indexOfLastInput);\n\t } else {\n\t\n\t /** Create new input when caret clicked in redactors area */\n\t var NEW_BLOCK_TYPE = editor.settings.initialBlockPlugin;\n\t\n\t editor.content.insertBlock({\n\t type: NEW_BLOCK_TYPE,\n\t block: editor.tools[NEW_BLOCK_TYPE].render()\n\t });\n\t\n\t /** If there is no inputs except inserted */\n\t if (editor.state.inputs.length === 1) {\n\t\n\t editor.caret.setToBlock(indexOfLastInput);\n\t } else {\n\t\n\t /** Set caret to this appended input */\n\t editor.caret.setToNextBlock(indexOfLastInput);\n\t }\n\t }\n\t } else {\n\t\n\t /** Close all panels */\n\t editor.toolbar.settings.close();\n\t editor.toolbar.toolbox.close();\n\t }\n\t\n\t /**\r\n\t * Move toolbar and open\r\n\t */\n\t editor.toolbar.move();\n\t editor.toolbar.open();\n\t\n\t var inputIsEmpty = !editor.content.currentNode.textContent.trim(),\n\t currentNodeType = editor.content.currentNode.dataset.tool,\n\t isInitialType = currentNodeType == editor.settings.initialBlockPlugin;\n\t\n\t /** Hide plus buttons */\n\t editor.toolbar.hidePlusButton();\n\t\n\t if (!inputIsEmpty) {\n\t\n\t /** Mark current block */\n\t editor.content.markBlock();\n\t }\n\t\n\t if (isInitialType && inputIsEmpty) {\n\t\n\t /** Show plus button */\n\t editor.toolbar.showPlusButton();\n\t }\n\t };\n\t\n\t /**\r\n\t * This method allows to define, is caret in contenteditable element or not.\r\n\t *\r\n\t * @private\r\n\t *\r\n\t * @description Otherwise, if we get TEXT node from range container, that will means we have input index.\r\n\t * In this case we use default browsers behaviour (if plugin allows that) or overwritten action.\r\n\t * Therefore, to be sure that we've clicked first-level block area, we should have currentNode, which always\r\n\t * specifies to the first-level block. Other cases we just ignore.\r\n\t */\n\t var detectWhenClickedOnFirstLevelBlockArea_ = function detectWhenClickedOnFirstLevelBlockArea_() {\n\t\n\t var selection = window.getSelection(),\n\t anchorNode = selection.anchorNode,\n\t flag = false;\n\t\n\t if (selection.rangeCount === 0) {\n\t\n\t editor.content.editorAreaHightlighted = true;\n\t } else {\n\t\n\t if (!editor.core.isDomNode(anchorNode)) {\n\t\n\t anchorNode = anchorNode.parentNode;\n\t }\n\t\n\t /** Already founded, without loop */\n\t if (anchorNode.contentEditable == 'true') {\n\t\n\t flag = true;\n\t }\n\t\n\t while (anchorNode.contentEditable != 'true') {\n\t\n\t anchorNode = anchorNode.parentNode;\n\t\n\t if (anchorNode.contentEditable == 'true') {\n\t\n\t flag = true;\n\t }\n\t\n\t if (anchorNode == document.body) {\n\t\n\t break;\n\t }\n\t }\n\t\n\t /** If editable element founded, flag is \"TRUE\", Therefore we return \"FALSE\" */\n\t editor.content.editorAreaHightlighted = !flag;\n\t }\n\t };\n\t\n\t /**\r\n\t * Toolbar button click handler\r\n\t *\r\n\t * @param {Object} event - cursor to the button\r\n\t * @protected\r\n\t *\r\n\t * @description gets current tool and calls render method\r\n\t */\n\t callbacks.toolbarButtonClicked = function (event) {\n\t\n\t var button = this;\n\t\n\t editor.toolbar.current = button.dataset.type;\n\t\n\t editor.toolbar.toolbox.toolClicked(event);\n\t editor.toolbar.close();\n\t };\n\t\n\t /**\r\n\t * Show or Hide toolbox when plus button is clicked\r\n\t */\n\t callbacks.plusButtonClicked = function () {\n\t\n\t if (!editor.nodes.toolbox.classList.contains('opened')) {\n\t\n\t editor.toolbar.toolbox.open();\n\t } else {\n\t\n\t editor.toolbar.toolbox.close();\n\t }\n\t };\n\t\n\t /**\r\n\t * Block handlers for KeyDown events\r\n\t *\r\n\t * @protected\r\n\t * @param {Object} event\r\n\t *\r\n\t * Handles keydowns on block\r\n\t * @see blockRightOrDownArrowPressed_\r\n\t * @see backspacePressed_\r\n\t * @see blockLeftOrUpArrowPressed_\r\n\t */\n\t callbacks.blockKeydown = function (event) {\n\t\n\t var block = event.target; // event.target is input\n\t\n\t switch (event.keyCode) {\n\t\n\t case editor.core.keys.DOWN:\n\t case editor.core.keys.RIGHT:\n\t blockRightOrDownArrowPressed_(event);\n\t break;\n\t\n\t case editor.core.keys.BACKSPACE:\n\t backspacePressed_(block, event);\n\t break;\n\t\n\t case editor.core.keys.UP:\n\t case editor.core.keys.LEFT:\n\t blockLeftOrUpArrowPressed_(event);\n\t break;\n\t\n\t }\n\t };\n\t\n\t /**\r\n\t * RIGHT or DOWN keydowns on block\r\n\t *\r\n\t * @param {Object} event\r\n\t * @private\r\n\t *\r\n\t * @description watches the selection and gets closest editable element.\r\n\t * Uses method getDeepestTextNodeFromPosition to get the last node of next block\r\n\t * Sets caret if it is contenteditable\r\n\t */\n\t var blockRightOrDownArrowPressed_ = function blockRightOrDownArrowPressed_(event) {\n\t\n\t var selection = window.getSelection(),\n\t inputs = editor.state.inputs,\n\t focusedNode = selection.anchorNode,\n\t focusedNodeHolder;\n\t\n\t /** Check for caret existance */\n\t if (!focusedNode) {\n\t\n\t return false;\n\t }\n\t\n\t /** Looking for closest (parent) contentEditable element of focused node */\n\t while (focusedNode.contentEditable != 'true') {\n\t\n\t focusedNodeHolder = focusedNode.parentNode;\n\t focusedNode = focusedNodeHolder;\n\t }\n\t\n\t /** Input index in DOM level */\n\t var editableElementIndex = 0;\n\t\n\t while (focusedNode != inputs[editableElementIndex]) {\n\t\n\t editableElementIndex++;\n\t }\n\t\n\t /**\r\n\t * Founded contentEditable element doesn't have childs\r\n\t * Or maybe New created block\r\n\t */\n\t if (!focusedNode.textContent) {\n\t\n\t editor.caret.setToNextBlock(editableElementIndex);\n\t return;\n\t }\n\t\n\t /**\r\n\t * Do nothing when caret doesn not reaches the end of last child\r\n\t */\n\t var caretInLastChild = false,\n\t caretAtTheEndOfText = false;\n\t\n\t var lastChild, deepestTextnode;\n\t\n\t lastChild = focusedNode.childNodes[focusedNode.childNodes.length - 1];\n\t\n\t if (editor.core.isDomNode(lastChild)) {\n\t\n\t deepestTextnode = editor.content.getDeepestTextNodeFromPosition(lastChild, lastChild.childNodes.length);\n\t } else {\n\t\n\t deepestTextnode = lastChild;\n\t }\n\t\n\t caretInLastChild = selection.anchorNode == deepestTextnode;\n\t caretAtTheEndOfText = deepestTextnode.length == selection.anchorOffset;\n\t\n\t if (!caretInLastChild || !caretAtTheEndOfText) {\n\t\n\t editor.core.log('arrow [down|right] : caret does not reached the end');\n\t return false;\n\t }\n\t\n\t editor.caret.setToNextBlock(editableElementIndex);\n\t };\n\t\n\t /**\r\n\t * LEFT or UP keydowns on block\r\n\t *\r\n\t * @param {Object} event\r\n\t * @private\r\n\t *\r\n\t * watches the selection and gets closest editable element.\r\n\t * Uses method getDeepestTextNodeFromPosition to get the last node of previous block\r\n\t * Sets caret if it is contenteditable\r\n\t *\r\n\t */\n\t var blockLeftOrUpArrowPressed_ = function blockLeftOrUpArrowPressed_(event) {\n\t\n\t var selection = window.getSelection(),\n\t inputs = editor.state.inputs,\n\t focusedNode = selection.anchorNode,\n\t focusedNodeHolder;\n\t\n\t /** Check for caret existance */\n\t if (!focusedNode) {\n\t\n\t return false;\n\t }\n\t\n\t /**\r\n\t * LEFT or UP not at the beginning\r\n\t */\n\t if (selection.anchorOffset !== 0) {\n\t\n\t return false;\n\t }\n\t\n\t /** Looking for parent contentEditable block */\n\t while (focusedNode.contentEditable != 'true') {\n\t\n\t focusedNodeHolder = focusedNode.parentNode;\n\t focusedNode = focusedNodeHolder;\n\t }\n\t\n\t /** Input index in DOM level */\n\t var editableElementIndex = 0;\n\t\n\t while (focusedNode != inputs[editableElementIndex]) {\n\t\n\t editableElementIndex++;\n\t }\n\t\n\t /**\r\n\t * Do nothing if caret is not at the beginning of first child\r\n\t */\n\t var caretInFirstChild = false,\n\t caretAtTheBeginning = false;\n\t\n\t var firstChild, deepestTextnode;\n\t\n\t /**\r\n\t * Founded contentEditable element doesn't have childs\r\n\t * Or maybe New created block\r\n\t */\n\t if (!focusedNode.textContent) {\n\t\n\t editor.caret.setToPreviousBlock(editableElementIndex);\n\t return;\n\t }\n\t\n\t firstChild = focusedNode.childNodes[0];\n\t\n\t if (editor.core.isDomNode(firstChild)) {\n\t\n\t deepestTextnode = editor.content.getDeepestTextNodeFromPosition(firstChild, 0);\n\t } else {\n\t\n\t deepestTextnode = firstChild;\n\t }\n\t\n\t caretInFirstChild = selection.anchorNode == deepestTextnode;\n\t caretAtTheBeginning = selection.anchorOffset === 0;\n\t\n\t if (caretInFirstChild && caretAtTheBeginning) {\n\t\n\t editor.caret.setToPreviousBlock(editableElementIndex);\n\t }\n\t };\n\t\n\t /**\r\n\t * Handles backspace keydown\r\n\t *\r\n\t * @param {Element} block\r\n\t * @param {Object} event\r\n\t * @private\r\n\t *\r\n\t * @description if block is empty, delete the block and set caret to the previous block\r\n\t * If block is not empty, try to merge two blocks - current and previous\r\n\t * But it we try'n to remove first block, then we should set caret to the next block, not previous.\r\n\t * If we removed the last block, create new one\r\n\t */\n\t var backspacePressed_ = function backspacePressed_(block, event) {\n\t\n\t var currentInputIndex = editor.caret.getCurrentInputIndex(),\n\t range,\n\t selectionLength,\n\t firstLevelBlocksCount;\n\t\n\t if (isNativeInput_(event.target)) {\n\t\n\t /** If input value is empty - remove block */\n\t if (event.target.value.trim() == '') {\n\t\n\t block.remove();\n\t } else {\n\t\n\t return;\n\t }\n\t }\n\t\n\t if (block.textContent.trim()) {\n\t\n\t range = editor.content.getRange();\n\t selectionLength = range.endOffset - range.startOffset;\n\t\n\t if (editor.caret.position.atStart() && !selectionLength && editor.state.inputs[currentInputIndex - 1]) {\n\t\n\t editor.content.mergeBlocks(currentInputIndex);\n\t } else {\n\t\n\t return;\n\t }\n\t }\n\t\n\t if (!selectionLength) {\n\t\n\t block.remove();\n\t }\n\t\n\t firstLevelBlocksCount = editor.nodes.redactor.childNodes.length;\n\t\n\t /**\r\n\t * If all blocks are removed\r\n\t */\n\t if (firstLevelBlocksCount === 0) {\n\t\n\t /** update currentNode variable */\n\t editor.content.currentNode = null;\n\t\n\t /** Inserting new empty initial block */\n\t editor.ui.addInitialBlock();\n\t\n\t /** Updating inputs state after deleting last block */\n\t editor.ui.saveInputs();\n\t\n\t /** Set to current appended block */\n\t window.setTimeout(function () {\n\t\n\t editor.caret.setToPreviousBlock(1);\n\t }, 10);\n\t } else {\n\t\n\t if (editor.caret.inputIndex !== 0) {\n\t\n\t /** Target block is not first */\n\t editor.caret.setToPreviousBlock(editor.caret.inputIndex);\n\t } else {\n\t\n\t /** If we try to delete first block */\n\t editor.caret.setToNextBlock(editor.caret.inputIndex);\n\t }\n\t }\n\t\n\t editor.toolbar.move();\n\t\n\t if (!editor.toolbar.opened) {\n\t\n\t editor.toolbar.open();\n\t }\n\t\n\t /** Updating inputs state */\n\t editor.ui.saveInputs();\n\t\n\t /** Prevent default browser behaviour */\n\t event.preventDefault();\n\t };\n\t\n\t /**\r\n\t * This method prevents default behaviour.\r\n\t *\r\n\t * @param {Object} event\r\n\t * @protected\r\n\t *\r\n\t * @description We get from clipboard pasted data, sanitize, make a fragment that contains of this sanitized nodes.\r\n\t * Firstly, we need to memorize the caret position. We can do that by getting the range of selection.\r\n\t * After all, we insert clear fragment into caret placed position. Then, we should move the caret to the last node\r\n\t */\n\t callbacks.blockPasteCallback = function (event) {\n\t\n\t /** If area is input or textarea then allow default behaviour */\n\t if (isNativeInput_(event.target)) {\n\t\n\t return;\n\t }\n\t\n\t /** Prevent default behaviour */\n\t event.preventDefault();\n\t\n\t var editableParent = editor.content.getEditableParent(event.target);\n\t\n\t /** Allow paste when event target placed in Editable element */\n\t if (!editableParent) {\n\t\n\t return;\n\t }\n\t\n\t /** get html pasted data - dirty data */\n\t var htmlData = event.clipboardData.getData('text/html'),\n\t plainData = event.clipboardData.getData('text/plain');\n\t\n\t /** Temporary DIV that is used to work with childs as arrays item */\n\t var div = editor.draw.node('DIV', '', {}),\n\t cleanData,\n\t wrappedData;\n\t\n\t /** Create fragment, that we paste to range after proccesing */\n\t cleanData = editor.sanitizer.clean(htmlData);\n\t\n\t /**\r\n\t * We wrap pasted text with

    tags to split it logically\r\n\t *\r\n\t * @type {string}\r\n\t */\n\t wrappedData = editor.content.wrapTextWithParagraphs(cleanData, plainData);\n\t div.innerHTML = wrappedData;\n\t\n\t var NEW_BLOCK_TYPE = editor.settings.initialBlockPlugin,\n\t currentBlockContent = editor.content.currentNode.firstChild.firstChild;\n\t\n\t /**\r\n\t * If there only one paragraph, just insert it\r\n\t */\n\t if (div.childNodes.length == 1) {\n\t\n\t editor.caret.insertNode(document.createTextNode(div.firstChild.innerHTML));\n\t return;\n\t }\n\t\n\t div.childNodes.forEach(function (paragraph, index) {\n\t\n\t /**\r\n\t * If there was no data in working node, replace it with first paragraph of pasted text\r\n\t */\n\t if (index == 0 && currentBlockContent.innerHTML.trim() === '') {\n\t\n\t editor.content.switchBlock(editor.content.currentNode, editor.tools[NEW_BLOCK_TYPE].render({\n\t text: paragraph.innerHTML\n\t }), NEW_BLOCK_TYPE);\n\t\n\t return;\n\t }\n\t\n\t editor.content.insertBlock({\n\t type: NEW_BLOCK_TYPE,\n\t block: editor.tools[NEW_BLOCK_TYPE].render({\n\t text: paragraph.innerHTML\n\t })\n\t });\n\t\n\t editor.caret.inputIndex++;\n\t });\n\t\n\t editor.caret.setToPreviousBlock(editor.caret.getCurrentInputIndex() + 1);\n\t };\n\t\n\t /**\r\n\t * used by UI module\r\n\t * Clicks on block settings button\r\n\t *\r\n\t * @param {Object} event\r\n\t * @protected\r\n\t * @description Opens toolbar settings\r\n\t */\n\t callbacks.showSettingsButtonClicked = function (event) {\n\t\n\t /**\r\n\t * Get type of current block\r\n\t * It uses to append settings from tool.settings property.\r\n\t * ...\r\n\t * Type is stored in data-type attribute on block\r\n\t */\n\t var currentToolType = editor.content.currentNode.dataset.tool;\n\t\n\t editor.toolbar.settings.toggle(currentToolType);\n\t\n\t /** Close toolbox when settings button is active */\n\t editor.toolbar.toolbox.close();\n\t editor.toolbar.settings.hideRemoveActions();\n\t };\n\t\n\t /**\r\n\t * Check block\r\n\t * @param target\r\n\t * @private\r\n\t *\r\n\t * @description Checks target is it native input\r\n\t */\n\t var isNativeInput_ = function isNativeInput_(target) {\n\t\n\t var nativeInputAreas = ['INPUT', 'TEXTAREA'];\n\t\n\t return nativeInputAreas.indexOf(target.tagName) != -1;\n\t };\n\t\n\t return callbacks;\n\t}({});\n\n/***/ },\n/* 13 */\n/***/ function(module, exports) {\n\n\t'use strict';\n\t\n\t/**\r\n\t * Codex Editor Draw module\r\n\t *\r\n\t * @author Codex Team\r\n\t * @version 1.0.\r\n\t */\n\t\n\tmodule.exports = function (draw) {\n\t\n\t /**\r\n\t * Base editor wrapper\r\n\t */\n\t draw.wrapper = function () {\n\t\n\t var wrapper = document.createElement('div');\n\t\n\t wrapper.className += 'codex-editor';\n\t\n\t return wrapper;\n\t };\n\t\n\t /**\r\n\t * Content-editable holder\r\n\t */\n\t draw.redactor = function () {\n\t\n\t var redactor = document.createElement('div');\n\t\n\t redactor.className += 'ce-redactor';\n\t\n\t return redactor;\n\t };\n\t\n\t draw.ceBlock = function () {\n\t\n\t var block = document.createElement('DIV');\n\t\n\t block.className += 'ce_block';\n\t\n\t return block;\n\t };\n\t\n\t /**\r\n\t * Empty toolbar with toggler\r\n\t */\n\t draw.toolbar = function () {\n\t\n\t var bar = document.createElement('div');\n\t\n\t bar.className += 'ce-toolbar';\n\t\n\t return bar;\n\t };\n\t\n\t draw.toolbarContent = function () {\n\t\n\t var wrapper = document.createElement('DIV');\n\t\n\t wrapper.classList.add('ce-toolbar__content');\n\t\n\t return wrapper;\n\t };\n\t\n\t /**\r\n\t * Inline toolbar\r\n\t */\n\t draw.inlineToolbar = function () {\n\t\n\t var bar = document.createElement('DIV');\n\t\n\t bar.className += 'ce-toolbar-inline';\n\t\n\t return bar;\n\t };\n\t\n\t /**\r\n\t * Wrapper for inline toobar buttons\r\n\t */\n\t draw.inlineToolbarButtons = function () {\n\t\n\t var wrapper = document.createElement('DIV');\n\t\n\t wrapper.className += 'ce-toolbar-inline__buttons';\n\t\n\t return wrapper;\n\t };\n\t\n\t /**\r\n\t * For some actions\r\n\t */\n\t draw.inlineToolbarActions = function () {\n\t\n\t var wrapper = document.createElement('DIV');\n\t\n\t wrapper.className += 'ce-toolbar-inline__actions';\n\t\n\t return wrapper;\n\t };\n\t\n\t draw.inputForLink = function () {\n\t\n\t var input = document.createElement('INPUT');\n\t\n\t input.type = 'input';\n\t input.className += 'inputForLink';\n\t input.placeholder = 'Вставьте ссылку ...';\n\t input.setAttribute('form', 'defaultForm');\n\t\n\t input.setAttribute('autofocus', 'autofocus');\n\t\n\t return input;\n\t };\n\t\n\t /**\r\n\t * @todo Desc\r\n\t */\n\t draw.blockButtons = function () {\n\t\n\t var block = document.createElement('div');\n\t\n\t block.className += 'ce-toolbar__actions';\n\t\n\t return block;\n\t };\n\t\n\t /**\r\n\t * Block settings panel\r\n\t */\n\t draw.blockSettings = function () {\n\t\n\t var settings = document.createElement('div');\n\t\n\t settings.className += 'ce-settings';\n\t\n\t return settings;\n\t };\n\t\n\t draw.defaultSettings = function () {\n\t\n\t var div = document.createElement('div');\n\t\n\t div.classList.add('ce-settings_default');\n\t\n\t return div;\n\t };\n\t\n\t draw.pluginsSettings = function () {\n\t\n\t var div = document.createElement('div');\n\t\n\t div.classList.add('ce-settings_plugin');\n\t\n\t return div;\n\t };\n\t\n\t draw.plusButton = function () {\n\t\n\t var button = document.createElement('span');\n\t\n\t button.className = 'ce-toolbar__plus';\n\t // button.innerHTML = '';\n\t\n\t return button;\n\t };\n\t\n\t /**\r\n\t * Settings button in toolbar\r\n\t */\n\t draw.settingsButton = function () {\n\t\n\t var toggler = document.createElement('span');\n\t\n\t toggler.className = 'ce-toolbar__settings-btn';\n\t\n\t /** Toggler button*/\n\t toggler.innerHTML = '';\n\t\n\t return toggler;\n\t };\n\t\n\t /**\r\n\t * Redactor tools wrapper\r\n\t */\n\t\n\t draw.toolbox = function () {\n\t\n\t var wrapper = document.createElement('div');\n\t\n\t wrapper.className = 'ce-toolbar__tools';\n\t\n\t return wrapper;\n\t };\n\t\n\t /**\r\n\t * @protected\r\n\t *\r\n\t * Draws tool buttons for toolbox\r\n\t *\r\n\t * @param {String} type\r\n\t * @param {String} classname\r\n\t * @returns {Element}\r\n\t */\n\t draw.toolbarButton = function (type, classname) {\n\t\n\t var button = document.createElement('li'),\n\t toolIcon = document.createElement('i'),\n\t toolTitle = document.createElement('span');\n\t\n\t button.dataset.type = type;\n\t button.setAttribute('title', type);\n\t\n\t toolIcon.classList.add(classname);\n\t toolTitle.classList.add('ce_toolbar_tools--title');\n\t\n\t button.appendChild(toolIcon);\n\t button.appendChild(toolTitle);\n\t\n\t return button;\n\t };\n\t\n\t /**\r\n\t * @protected\r\n\t *\r\n\t * Draws tools for inline toolbar\r\n\t *\r\n\t * @param {String} type\r\n\t * @param {String} classname\r\n\t */\n\t draw.toolbarButtonInline = function (type, classname) {\n\t\n\t var button = document.createElement('BUTTON'),\n\t toolIcon = document.createElement('I');\n\t\n\t button.type = 'button';\n\t button.dataset.type = type;\n\t toolIcon.classList.add(classname);\n\t\n\t button.appendChild(toolIcon);\n\t\n\t return button;\n\t };\n\t\n\t /**\r\n\t * Redactor block\r\n\t */\n\t draw.block = function (tagName, content) {\n\t\n\t var node = document.createElement(tagName);\n\t\n\t node.innerHTML = content || '';\n\t\n\t return node;\n\t };\n\t\n\t /**\r\n\t * Creates Node with passed tagName and className\r\n\t * @param {string} tagName\r\n\t * @param {string} className\r\n\t * @param {object} properties - allow to assign properties\r\n\t */\n\t draw.node = function (tagName, className, properties) {\n\t\n\t var el = document.createElement(tagName);\n\t\n\t if (className) el.className = className;\n\t\n\t if (properties) {\n\t\n\t for (var name in properties) {\n\t\n\t el[name] = properties[name];\n\t }\n\t }\n\t\n\t return el;\n\t };\n\t\n\t /**\r\n\t * Unavailable plugin block\r\n\t */\n\t draw.unavailableBlock = function () {\n\t\n\t var wrapper = document.createElement('DIV');\n\t\n\t wrapper.classList.add('cdx-unavailable-block');\n\t\n\t return wrapper;\n\t };\n\t\n\t return draw;\n\t}({});\n\n/***/ },\n/* 14 */\n/***/ function(module, exports) {\n\n\t'use strict';\n\t\n\t/**\r\n\t * Codex Editor Caret Module\r\n\t *\r\n\t * @author Codex Team\r\n\t * @version 1.0\r\n\t */\n\t\n\tmodule.exports = function (caret) {\n\t\n\t var editor = codex.editor;\n\t\n\t /**\r\n\t * @var {int} InputIndex - editable element in DOM\r\n\t */\n\t caret.inputIndex = null;\n\t\n\t /**\r\n\t * @var {int} offset - caret position in a text node.\r\n\t */\n\t caret.offset = null;\n\t\n\t /**\r\n\t * @var {int} focusedNodeIndex - we get index of child node from first-level block\r\n\t */\n\t caret.focusedNodeIndex = null;\n\t\n\t /**\r\n\t * Creates Document Range and sets caret to the element.\r\n\t * @protected\r\n\t * @uses caret.save — if you need to save caret position\r\n\t * @param {Element} el - Changed Node.\r\n\t */\n\t caret.set = function (el, index, offset) {\n\t\n\t offset = offset || caret.offset || 0;\n\t index = index || caret.focusedNodeIndex || 0;\n\t\n\t var childs = el.childNodes,\n\t nodeToSet;\n\t\n\t if (childs.length === 0) {\n\t\n\t nodeToSet = el;\n\t } else {\n\t\n\t nodeToSet = childs[index];\n\t }\n\t\n\t /** If Element is INPUT */\n\t if (el.contentEditable != 'true') {\n\t\n\t el.focus();\n\t return;\n\t }\n\t\n\t if (editor.core.isDomNode(nodeToSet)) {\n\t\n\t nodeToSet = editor.content.getDeepestTextNodeFromPosition(nodeToSet, nodeToSet.childNodes.length);\n\t }\n\t\n\t var range = document.createRange(),\n\t selection = window.getSelection();\n\t\n\t window.setTimeout(function () {\n\t\n\t range.setStart(nodeToSet, offset);\n\t range.setEnd(nodeToSet, offset);\n\t\n\t selection.removeAllRanges();\n\t selection.addRange(range);\n\t\n\t editor.caret.saveCurrentInputIndex();\n\t }, 20);\n\t };\n\t\n\t /**\r\n\t * @protected\r\n\t * Updates index of input and saves it in caret object\r\n\t */\n\t caret.saveCurrentInputIndex = function () {\n\t\n\t /** Index of Input that we paste sanitized content */\n\t var selection = window.getSelection(),\n\t inputs = editor.state.inputs,\n\t focusedNode = selection.anchorNode,\n\t focusedNodeHolder;\n\t\n\t if (!focusedNode) {\n\t\n\t return;\n\t }\n\t\n\t /** Looking for parent contentEditable block */\n\t while (focusedNode.contentEditable != 'true') {\n\t\n\t focusedNodeHolder = focusedNode.parentNode;\n\t focusedNode = focusedNodeHolder;\n\t }\n\t\n\t /** Input index in DOM level */\n\t var editableElementIndex = 0;\n\t\n\t while (focusedNode != inputs[editableElementIndex]) {\n\t\n\t editableElementIndex++;\n\t }\n\t\n\t caret.inputIndex = editableElementIndex;\n\t };\n\t\n\t /**\r\n\t * Returns current input index (caret object)\r\n\t */\n\t caret.getCurrentInputIndex = function () {\n\t\n\t return caret.inputIndex;\n\t };\n\t\n\t /**\r\n\t * @param {int} index - index of first-level block after that we set caret into next input\r\n\t */\n\t caret.setToNextBlock = function (index) {\n\t\n\t var inputs = editor.state.inputs,\n\t nextInput = inputs[index + 1];\n\t\n\t if (!nextInput) {\n\t\n\t editor.core.log('We are reached the end');\n\t return;\n\t }\n\t\n\t /**\r\n\t * When new Block created or deleted content of input\r\n\t * We should add some text node to set caret\r\n\t */\n\t if (!nextInput.childNodes.length) {\n\t\n\t var emptyTextElement = document.createTextNode('');\n\t\n\t nextInput.appendChild(emptyTextElement);\n\t }\n\t\n\t editor.caret.inputIndex = index + 1;\n\t editor.caret.set(nextInput, 0, 0);\n\t editor.content.workingNodeChanged(nextInput);\n\t };\n\t\n\t /**\r\n\t * @param {int} index - index of target input.\r\n\t * Sets caret to input with this index\r\n\t */\n\t caret.setToBlock = function (index) {\n\t\n\t var inputs = editor.state.inputs,\n\t targetInput = inputs[index];\n\t\n\t if (!targetInput) {\n\t\n\t return;\n\t }\n\t\n\t /**\r\n\t * When new Block created or deleted content of input\r\n\t * We should add some text node to set caret\r\n\t */\n\t if (!targetInput.childNodes.length) {\n\t\n\t var emptyTextElement = document.createTextNode('');\n\t\n\t targetInput.appendChild(emptyTextElement);\n\t }\n\t\n\t editor.caret.inputIndex = index;\n\t editor.caret.set(targetInput, 0, 0);\n\t editor.content.workingNodeChanged(targetInput);\n\t };\n\t\n\t /**\r\n\t * @param {int} index - index of input\r\n\t */\n\t caret.setToPreviousBlock = function (index) {\n\t\n\t index = index || 0;\n\t\n\t var inputs = editor.state.inputs,\n\t previousInput = inputs[index - 1],\n\t lastChildNode,\n\t lengthOfLastChildNode,\n\t emptyTextElement;\n\t\n\t if (!previousInput) {\n\t\n\t editor.core.log('We are reached first node');\n\t return;\n\t }\n\t\n\t lastChildNode = editor.content.getDeepestTextNodeFromPosition(previousInput, previousInput.childNodes.length);\n\t lengthOfLastChildNode = lastChildNode.length;\n\t\n\t /**\r\n\t * When new Block created or deleted content of input\r\n\t * We should add some text node to set caret\r\n\t */\n\t if (!previousInput.childNodes.length) {\n\t\n\t emptyTextElement = document.createTextNode('');\n\t previousInput.appendChild(emptyTextElement);\n\t }\n\t editor.caret.inputIndex = index - 1;\n\t editor.caret.set(previousInput, previousInput.childNodes.length - 1, lengthOfLastChildNode);\n\t editor.content.workingNodeChanged(inputs[index - 1]);\n\t };\n\t\n\t caret.position = {\n\t\n\t atStart: function atStart() {\n\t\n\t var selection = window.getSelection(),\n\t anchorOffset = selection.anchorOffset,\n\t anchorNode = selection.anchorNode,\n\t firstLevelBlock = editor.content.getFirstLevelBlock(anchorNode),\n\t pluginsRender = firstLevelBlock.childNodes[0];\n\t\n\t if (!editor.core.isDomNode(anchorNode)) {\n\t\n\t anchorNode = anchorNode.parentNode;\n\t }\n\t\n\t var isFirstNode = anchorNode === pluginsRender.childNodes[0],\n\t isOffsetZero = anchorOffset === 0;\n\t\n\t return isFirstNode && isOffsetZero;\n\t },\n\t\n\t atTheEnd: function atTheEnd() {\n\t\n\t var selection = window.getSelection(),\n\t anchorOffset = selection.anchorOffset,\n\t anchorNode = selection.anchorNode;\n\t\n\t /** Caret is at the end of input */\n\t return !anchorNode || !anchorNode.length || anchorOffset === anchorNode.length;\n\t }\n\t };\n\t\n\t /**\r\n\t * Inserts node at the caret location\r\n\t * @param node\r\n\t */\n\t caret.insertNode = function (node) {\n\t\n\t var selection, range;\n\t\n\t selection = window.getSelection();\n\t\n\t range = selection.getRangeAt(0);\n\t range.deleteContents();\n\t\n\t range.insertNode(node);\n\t\n\t range.setStartAfter(node);\n\t range.collapse(true);\n\t\n\t selection.removeAllRanges();\n\t selection.addRange(range);\n\t };\n\t\n\t return caret;\n\t}({});\n\n/***/ },\n/* 15 */\n/***/ function(module, exports) {\n\n\t'use strict';\n\t\n\t/**\r\n\t * Codex Editor Notification Module\r\n\t *\r\n\t * @author Codex Team\r\n\t * @version 1.0\r\n\t */\n\t\n\tmodule.exports = function (notifications) {\n\t\n\t var editor = codex.editor;\n\t\n\t var queue = [];\n\t\n\t var addToQueue = function addToQueue(settings) {\n\t\n\t queue.push(settings);\n\t\n\t var index = 0;\n\t\n\t while (index < queue.length && queue.length > 5) {\n\t\n\t if (queue[index].type == 'confirm' || queue[index].type == 'prompt') {\n\t\n\t index++;\n\t continue;\n\t }\n\t\n\t queue[index].close();\n\t queue.splice(index, 1);\n\t }\n\t };\n\t\n\t notifications.createHolder = function () {\n\t\n\t var holder = editor.draw.node('DIV', 'cdx-notifications-block');\n\t\n\t editor.nodes.notifications = document.body.appendChild(holder);\n\t\n\t return holder;\n\t };\n\t\n\t /**\r\n\t * Error notificator. Shows block with message\r\n\t * @protected\r\n\t */\n\t notifications.errorThrown = function (errorMsg, event) {\n\t\n\t editor.notifications.notification({ message: 'This action is not available currently', type: event.type });\n\t };\n\t\n\t /**\r\n\t *\r\n\t * Appends notification\r\n\t *\r\n\t * settings = {\r\n\t * type - notification type (reserved types: alert, confirm, prompt). Just add class 'cdx-notification-'+type\r\n\t * message - notification message\r\n\t * okMsg - confirm button text (default - 'Ok')\r\n\t * cancelBtn - cancel button text (default - 'Cancel'). Only for confirm and prompt types\r\n\t * confirm - function-handler for ok button click\r\n\t * cancel - function-handler for cancel button click. Only for confirm and prompt types\r\n\t * time - time (in seconds) after which notification will close (default - 10s)\r\n\t * }\r\n\t *\r\n\t * @param settings\r\n\t */\n\t notifications.notification = function (constructorSettings) {\n\t\n\t /** Private vars and methods */\n\t var notification = null,\n\t cancel = null,\n\t type = null,\n\t confirm = null,\n\t inputField = null;\n\t\n\t var confirmHandler = function confirmHandler() {\n\t\n\t close();\n\t\n\t if (typeof confirm !== 'function') {\n\t\n\t return;\n\t }\n\t\n\t if (type == 'prompt') {\n\t\n\t confirm(inputField.value);\n\t return;\n\t }\n\t\n\t confirm();\n\t };\n\t\n\t var cancelHandler = function cancelHandler() {\n\t\n\t close();\n\t\n\t if (typeof cancel !== 'function') {\n\t\n\t return;\n\t }\n\t\n\t cancel();\n\t };\n\t\n\t /** Public methods */\n\t function create(settings) {\n\t\n\t if (!(settings && settings.message)) {\n\t\n\t editor.core.log('Can\\'t create notification. Message is missed');\n\t return;\n\t }\n\t\n\t settings.type = settings.type || 'alert';\n\t settings.time = settings.time * 1000 || 10000;\n\t\n\t var wrapper = editor.draw.node('DIV', 'cdx-notification'),\n\t message = editor.draw.node('DIV', 'cdx-notification__message'),\n\t input = editor.draw.node('INPUT', 'cdx-notification__input'),\n\t okBtn = editor.draw.node('SPAN', 'cdx-notification__ok-btn'),\n\t cancelBtn = editor.draw.node('SPAN', 'cdx-notification__cancel-btn');\n\t\n\t message.textContent = settings.message;\n\t okBtn.textContent = settings.okMsg || 'ОК';\n\t cancelBtn.textContent = settings.cancelMsg || 'Отмена';\n\t\n\t editor.listeners.add(okBtn, 'click', confirmHandler);\n\t editor.listeners.add(cancelBtn, 'click', cancelHandler);\n\t\n\t wrapper.appendChild(message);\n\t\n\t if (settings.type == 'prompt') {\n\t\n\t wrapper.appendChild(input);\n\t }\n\t\n\t wrapper.appendChild(okBtn);\n\t\n\t if (settings.type == 'prompt' || settings.type == 'confirm') {\n\t\n\t wrapper.appendChild(cancelBtn);\n\t }\n\t\n\t wrapper.classList.add('cdx-notification-' + settings.type);\n\t wrapper.dataset.type = settings.type;\n\t\n\t notification = wrapper;\n\t type = settings.type;\n\t confirm = settings.confirm;\n\t cancel = settings.cancel;\n\t inputField = input;\n\t\n\t if (settings.type != 'prompt' && settings.type != 'confirm') {\n\t\n\t window.setTimeout(close, settings.time);\n\t }\n\t };\n\t\n\t function send() {\n\t\n\t editor.nodes.notifications.appendChild(notification);\n\t inputField.focus();\n\t\n\t editor.nodes.notifications.classList.add('cdx-notification__notification-appending');\n\t\n\t window.setTimeout(function () {\n\t\n\t editor.nodes.notifications.classList.remove('cdx-notification__notification-appending');\n\t }, 100);\n\t\n\t addToQueue({ type: type, close: close });\n\t };\n\t\n\t function close() {\n\t\n\t notification.remove();\n\t };\n\t\n\t if (constructorSettings) {\n\t\n\t create(constructorSettings);\n\t send();\n\t }\n\t\n\t return {\n\t create: create,\n\t send: send,\n\t close: close\n\t };\n\t };\n\t\n\t notifications.clear = function () {\n\t\n\t editor.nodes.notifications.innerHTML = '';\n\t queue = [];\n\t };\n\t\n\t return notifications;\n\t}({});\n\n/***/ },\n/* 16 */\n/***/ function(module, exports) {\n\n\t\"use strict\";\n\t\n\t/**\r\n\t * Codex Editor Parser Module\r\n\t *\r\n\t * @author Codex Team\r\n\t * @version 1.1\r\n\t */\n\t\n\tmodule.exports = function (parser) {\n\t\n\t var editor = codex.editor;\n\t\n\t /** inserting text */\n\t parser.insertPastedContent = function (blockType, tag) {\n\t\n\t editor.content.insertBlock({\n\t type: blockType.type,\n\t block: blockType.render({\n\t text: tag.innerHTML\n\t })\n\t });\n\t };\n\t\n\t /**\r\n\t * Check DOM node for display style: separated block or child-view\r\n\t */\n\t parser.isFirstLevelBlock = function (node) {\n\t\n\t return node.nodeType == editor.core.nodeTypes.TAG && node.classList.contains(editor.ui.className.BLOCK_CLASSNAME);\n\t };\n\t\n\t return parser;\n\t}({});\n\n/***/ },\n/* 17 */\n/***/ function(module, exports, __webpack_require__) {\n\n\t'use strict';\n\t\n\t/**\r\n\t * Codex Sanitizer\r\n\t */\n\t\n\tmodule.exports = function (sanitizer) {\n\t\n\t /** HTML Janitor library */\n\t var janitor = __webpack_require__(18);\n\t\n\t /** Codex Editor */\n\t var editor = codex.editor;\n\t\n\t sanitizer.prepare = function () {\n\t\n\t if (editor.settings.sanitizer && !editor.core.isEmpty(editor.settings.sanitizer)) {\n\t\n\t Config.CUSTOM = editor.settings.sanitizer;\n\t }\n\t };\n\t\n\t /**\r\n\t * Basic config\r\n\t */\n\t var Config = {\n\t\n\t /** User configuration */\n\t CUSTOM: null,\n\t\n\t BASIC: {\n\t\n\t tags: {\n\t p: {},\n\t a: {\n\t href: true,\n\t target: '_blank',\n\t rel: 'nofollow'\n\t },\n\t i: {},\n\t b: {},\n\t strong: {},\n\t em: {},\n\t span: {}\n\t }\n\t }\n\t };\n\t\n\t sanitizer.Config = Config;\n\t\n\t /**\r\n\t *\r\n\t * @param userCustomConfig\r\n\t * @returns {*}\r\n\t * @private\r\n\t *\r\n\t * @description If developer uses editor's API, then he can customize sane restrictions.\r\n\t * Or, sane config can be defined globally in editors initialization. That config will be used everywhere\r\n\t * At least, if there is no config overrides, that API uses BASIC Default configation\r\n\t */\n\t var init_ = function init_(userCustomConfig) {\n\t\n\t var configuration = userCustomConfig || Config.CUSTOM || Config.BASIC;\n\t\n\t return new janitor(configuration);\n\t };\n\t\n\t /**\r\n\t * Cleans string from unwanted tags\r\n\t * @protected\r\n\t * @param {String} dirtyString - taint string\r\n\t * @param {Object} customConfig - allowed tags\r\n\t */\n\t sanitizer.clean = function (dirtyString, customConfig) {\n\t\n\t var janitorInstance = init_(customConfig);\n\t\n\t return janitorInstance.clean(dirtyString);\n\t };\n\t\n\t return sanitizer;\n\t}({});\n\n/***/ },\n/* 18 */\n/***/ function(module, exports, __webpack_require__) {\n\n\tvar __WEBPACK_AMD_DEFINE_FACTORY__, __WEBPACK_AMD_DEFINE_RESULT__;(function (root, factory) {\n\t if (true) {\n\t !(__WEBPACK_AMD_DEFINE_FACTORY__ = (factory), __WEBPACK_AMD_DEFINE_RESULT__ = (typeof __WEBPACK_AMD_DEFINE_FACTORY__ === 'function' ? (__WEBPACK_AMD_DEFINE_FACTORY__.call(exports, __webpack_require__, exports, module)) : __WEBPACK_AMD_DEFINE_FACTORY__), __WEBPACK_AMD_DEFINE_RESULT__ !== undefined && (module.exports = __WEBPACK_AMD_DEFINE_RESULT__));\n\t } else if (typeof exports === 'object') {\n\t module.exports = factory();\n\t } else {\n\t root.HTMLJanitor = factory();\n\t }\n\t}(this, function () {\n\t\n\t /**\n\t * @param {Object} config.tags Dictionary of allowed tags.\n\t * @param {boolean} config.keepNestedBlockElements Default false.\n\t */\n\t function HTMLJanitor(config) {\n\t\n\t var tagDefinitions = config['tags'];\n\t var tags = Object.keys(tagDefinitions);\n\t\n\t var validConfigValues = tags\n\t .map(function(k) { return typeof tagDefinitions[k]; })\n\t .every(function(type) { return type === 'object' || type === 'boolean' || type === 'function'; });\n\t\n\t if(!validConfigValues) {\n\t throw new Error(\"The configuration was invalid\");\n\t }\n\t\n\t this.config = config;\n\t }\n\t\n\t // TODO: not exhaustive?\n\t var blockElementNames = ['P', 'LI', 'TD', 'TH', 'DIV', 'H1', 'H2', 'H3', 'H4', 'H5', 'H6', 'PRE'];\n\t function isBlockElement(node) {\n\t return blockElementNames.indexOf(node.nodeName) !== -1;\n\t }\n\t\n\t var inlineElementNames = ['A', 'B', 'STRONG', 'I', 'EM', 'SUB', 'SUP', 'U', 'STRIKE'];\n\t function isInlineElement(node) {\n\t return inlineElementNames.indexOf(node.nodeName) !== -1;\n\t }\n\t\n\t HTMLJanitor.prototype.clean = function (html) {\n\t var sandbox = document.createElement('div');\n\t sandbox.innerHTML = html;\n\t\n\t this._sanitize(sandbox);\n\t\n\t return sandbox.innerHTML;\n\t };\n\t\n\t HTMLJanitor.prototype._sanitize = function (parentNode) {\n\t var treeWalker = createTreeWalker(parentNode);\n\t var node = treeWalker.firstChild();\n\t if (!node) { return; }\n\t\n\t do {\n\t // Ignore nodes that have already been sanitized\n\t if (node._sanitized) {\n\t continue;\n\t }\n\t\n\t if (node.nodeType === Node.TEXT_NODE) {\n\t // If this text node is just whitespace and the previous or next element\n\t // sibling is a block element, remove it\n\t // N.B.: This heuristic could change. Very specific to a bug with\n\t // `contenteditable` in Firefox: http://jsbin.com/EyuKase/1/edit?js,output\n\t // FIXME: make this an option?\n\t if (node.data.trim() === ''\n\t && ((node.previousElementSibling && isBlockElement(node.previousElementSibling))\n\t || (node.nextElementSibling && isBlockElement(node.nextElementSibling)))) {\n\t parentNode.removeChild(node);\n\t this._sanitize(parentNode);\n\t break;\n\t } else {\n\t continue;\n\t }\n\t }\n\t\n\t // Remove all comments\n\t if (node.nodeType === Node.COMMENT_NODE) {\n\t parentNode.removeChild(node);\n\t this._sanitize(parentNode);\n\t break;\n\t }\n\t\n\t var isInline = isInlineElement(node);\n\t var containsBlockElement;\n\t if (isInline) {\n\t containsBlockElement = Array.prototype.some.call(node.childNodes, isBlockElement);\n\t }\n\t\n\t // Block elements should not be nested (e.g.

  • ...); if\n\t // they are, we want to unwrap the inner block element.\n\t var isNotTopContainer = !! parentNode.parentNode;\n\t var isNestedBlockElement =\n\t isBlockElement(parentNode) &&\n\t isBlockElement(node) &&\n\t isNotTopContainer;\n\t\n\t var nodeName = node.nodeName.toLowerCase();\n\t\n\t var allowedAttrs = getAllowedAttrs(this.config, nodeName, node);\n\t\n\t var isInvalid = isInline && containsBlockElement;\n\t\n\t // Drop tag entirely according to the whitelist *and* if the markup\n\t // is invalid.\n\t if (isInvalid || shouldRejectNode(node, allowedAttrs)\n\t || (!this.config.keepNestedBlockElements && isNestedBlockElement)) {\n\t // Do not keep the inner text of SCRIPT/STYLE elements.\n\t if (! (node.nodeName === 'SCRIPT' || node.nodeName === 'STYLE')) {\n\t while (node.childNodes.length > 0) {\n\t parentNode.insertBefore(node.childNodes[0], node);\n\t }\n\t }\n\t parentNode.removeChild(node);\n\t\n\t this._sanitize(parentNode);\n\t break;\n\t }\n\t\n\t // Sanitize attributes\n\t for (var a = 0; a < node.attributes.length; a += 1) {\n\t var attr = node.attributes[a];\n\t\n\t if (shouldRejectAttr(attr, allowedAttrs, node)) {\n\t node.removeAttribute(attr.name);\n\t // Shift the array to continue looping.\n\t a = a - 1;\n\t }\n\t }\n\t\n\t // Sanitize children\n\t this._sanitize(node);\n\t\n\t // Mark node as sanitized so it's ignored in future runs\n\t node._sanitized = true;\n\t } while ((node = treeWalker.nextSibling()));\n\t };\n\t\n\t function createTreeWalker(node) {\n\t return document.createTreeWalker(node,\n\t NodeFilter.SHOW_TEXT | NodeFilter.SHOW_ELEMENT | NodeFilter.SHOW_COMMENT,\n\t null, false);\n\t }\n\t\n\t function getAllowedAttrs(config, nodeName, node){\n\t if (typeof config.tags[nodeName] === 'function') {\n\t return config.tags[nodeName](node);\n\t } else {\n\t return config.tags[nodeName];\n\t }\n\t }\n\t\n\t function shouldRejectNode(node, allowedAttrs){\n\t if (typeof allowedAttrs === 'undefined') {\n\t return true;\n\t } else if (typeof allowedAttrs === 'boolean') {\n\t return !allowedAttrs;\n\t }\n\t\n\t return false;\n\t }\n\t\n\t function shouldRejectAttr(attr, allowedAttrs, node){\n\t var attrName = attr.name.toLowerCase();\n\t\n\t if (allowedAttrs === true){\n\t return false;\n\t } else if (typeof allowedAttrs[attrName] === 'function'){\n\t return !allowedAttrs[attrName](attr.value, node);\n\t } else if (typeof allowedAttrs[attrName] === 'undefined'){\n\t return true;\n\t } else if (allowedAttrs[attrName] === false) {\n\t return true;\n\t } else if (typeof allowedAttrs[attrName] === 'string') {\n\t return (allowedAttrs[attrName] !== attr.value);\n\t }\n\t\n\t return false;\n\t }\n\t\n\t return HTMLJanitor;\n\t\n\t}));\n\n\n/***/ },\n/* 19 */\n/***/ function(module, exports) {\n\n\t\"use strict\";\n\t\n\t/**\r\n\t * Codex Editor Listeners module\r\n\t *\r\n\t * @author Codex Team\r\n\t * @version 1.0\r\n\t */\n\t\n\t/**\r\n\t * Module-decorator for event listeners assignment\r\n\t */\n\tmodule.exports = function (listeners) {\n\t\n\t var allListeners = [];\n\t\n\t /**\r\n\t * Search methods\r\n\t *\r\n\t * byElement, byType and byHandler returns array of suitable listeners\r\n\t * one and all takes element, eventType, and handler and returns first (all) suitable listener\r\n\t *\r\n\t */\n\t listeners.search = function () {\n\t\n\t var byElement = function byElement(element, context) {\n\t\n\t var listenersOnElement = [];\n\t\n\t context = context || allListeners;\n\t\n\t for (var i = 0; i < context.length; i++) {\n\t\n\t var listener = context[i];\n\t\n\t if (listener.element === element) {\n\t\n\t listenersOnElement.push(listener);\n\t }\n\t }\n\t\n\t return listenersOnElement;\n\t };\n\t\n\t var byType = function byType(eventType, context) {\n\t\n\t var listenersWithType = [];\n\t\n\t context = context || allListeners;\n\t\n\t for (var i = 0; i < context.length; i++) {\n\t\n\t var listener = context[i];\n\t\n\t if (listener.type === eventType) {\n\t\n\t listenersWithType.push(listener);\n\t }\n\t }\n\t\n\t return listenersWithType;\n\t };\n\t\n\t var byHandler = function byHandler(handler, context) {\n\t\n\t var listenersWithHandler = [];\n\t\n\t context = context || allListeners;\n\t\n\t for (var i = 0; i < context.length; i++) {\n\t\n\t var listener = context[i];\n\t\n\t if (listener.handler === handler) {\n\t\n\t listenersWithHandler.push(listener);\n\t }\n\t }\n\t\n\t return listenersWithHandler;\n\t };\n\t\n\t var one = function one(element, eventType, handler) {\n\t\n\t var result = allListeners;\n\t\n\t if (element) result = byElement(element, result);\n\t\n\t if (eventType) result = byType(eventType, result);\n\t\n\t if (handler) result = byHandler(handler, result);\n\t\n\t return result[0];\n\t };\n\t\n\t var all = function all(element, eventType, handler) {\n\t\n\t var result = allListeners;\n\t\n\t if (element) result = byElement(element, result);\n\t\n\t if (eventType) result = byType(eventType, result);\n\t\n\t if (handler) result = byHandler(handler, result);\n\t\n\t return result;\n\t };\n\t\n\t return {\n\t byElement: byElement,\n\t byType: byType,\n\t byHandler: byHandler,\n\t one: one,\n\t all: all\n\t };\n\t }();\n\t\n\t listeners.add = function (element, eventType, handler, isCapture) {\n\t\n\t element.addEventListener(eventType, handler, isCapture);\n\t\n\t var data = {\n\t element: element,\n\t type: eventType,\n\t handler: handler\n\t };\n\t\n\t var alreadyAddedListener = listeners.search.one(element, eventType, handler);\n\t\n\t if (!alreadyAddedListener) {\n\t\n\t allListeners.push(data);\n\t }\n\t };\n\t\n\t listeners.remove = function (element, eventType, handler) {\n\t\n\t element.removeEventListener(eventType, handler);\n\t\n\t var existingListeners = listeners.search.all(element, eventType, handler);\n\t\n\t for (var i = 0; i < existingListeners.length; i++) {\n\t\n\t var index = allListeners.indexOf(existingListeners[i]);\n\t\n\t if (index > 0) {\n\t\n\t allListeners.splice(index, 1);\n\t }\n\t }\n\t };\n\t\n\t listeners.removeAll = function () {\n\t\n\t allListeners.map(function (current) {\n\t\n\t listeners.remove(current.element, current.type, current.handler);\n\t });\n\t };\n\t\n\t listeners.get = function (element, eventType, handler) {\n\t\n\t return listeners.search.all(element, eventType, handler);\n\t };\n\t\n\t return listeners;\n\t}({});\n\n/***/ },\n/* 20 */\n/***/ function(module, exports) {\n\n\t'use strict';\n\t\n\tvar _typeof = typeof Symbol === \"function\" && typeof Symbol.iterator === \"symbol\" ? function (obj) { return typeof obj; } : function (obj) { return obj && typeof Symbol === \"function\" && obj.constructor === Symbol && obj !== Symbol.prototype ? \"symbol\" : typeof obj; };\n\t\n\t/**\r\n\t * Codex Editor Destroyer module\r\n\t *\r\n\t * @auhor Codex Team\r\n\t * @version 1.0\r\n\t */\n\t\n\tmodule.exports = function (destroyer) {\n\t\n\t var editor = codex.editor;\n\t\n\t destroyer.removeNodes = function () {\n\t\n\t editor.nodes.wrapper.remove();\n\t editor.nodes.notifications.remove();\n\t };\n\t\n\t destroyer.destroyPlugins = function () {\n\t\n\t for (var tool in editor.tools) {\n\t\n\t if (typeof editor.tools[tool].destroy === 'function') {\n\t\n\t editor.tools[tool].destroy();\n\t }\n\t }\n\t };\n\t\n\t destroyer.destroyScripts = function () {\n\t\n\t var scripts = document.getElementsByTagName('SCRIPT');\n\t\n\t for (var i = 0; i < scripts.length; i++) {\n\t\n\t if (scripts[i].id.indexOf(editor.scriptPrefix) + 1) {\n\t\n\t scripts[i].remove();\n\t i--;\n\t }\n\t }\n\t };\n\t\n\t /**\r\n\t * Delete editor data from webpage.\r\n\t * You should send settings argument with boolean flags:\r\n\t * @param settings.ui- remove redactor event listeners and DOM nodes\r\n\t * @param settings.scripts - remove redactor scripts from DOM\r\n\t * @param settings.plugins - remove plugin's objects\r\n\t * @param settings.core - remove editor core. You can remove core only if UI and scripts flags is true\r\n\t * }\r\n\t *\r\n\t */\n\t destroyer.destroy = function (settings) {\n\t\n\t if (!settings || (typeof settings === 'undefined' ? 'undefined' : _typeof(settings)) !== 'object') {\n\t\n\t return;\n\t }\n\t\n\t if (settings.ui) {\n\t\n\t destroyer.removeNodes();\n\t editor.listeners.removeAll();\n\t }\n\t\n\t if (settings.scripts) {\n\t\n\t destroyer.destroyScripts();\n\t }\n\t\n\t if (settings.plugins) {\n\t\n\t destroyer.destroyPlugins();\n\t }\n\t\n\t if (settings.ui && settings.scripts && settings.core) {\n\t\n\t delete codex.editor;\n\t }\n\t };\n\t\n\t return destroyer;\n\t}({});\n\n/***/ },\n/* 21 */\n/***/ function(module, exports) {\n\n\t'use strict';\n\t\n\t/**\r\n\t * Codex Editor Paste module\r\n\t *\r\n\t * @author Codex Team\r\n\t * @version 1.1.1\r\n\t */\n\t\n\tmodule.exports = function (paste) {\n\t\n\t var editor = codex.editor;\n\t\n\t var patterns = [];\n\t\n\t paste.prepare = function () {\n\t\n\t var tools = editor.tools;\n\t\n\t for (var tool in tools) {\n\t\n\t if (!tools[tool].renderOnPastePatterns || !Array.isArray(tools[tool].renderOnPastePatterns)) {\n\t\n\t continue;\n\t }\n\t\n\t tools[tool].renderOnPastePatterns.map(function (pattern) {\n\t\n\t patterns.push(pattern);\n\t });\n\t }\n\t\n\t return Promise.resolve();\n\t };\n\t\n\t /**\r\n\t * Saves data\r\n\t * @param event\r\n\t */\n\t paste.pasted = function (event) {\n\t\n\t var clipBoardData = event.clipboardData || window.clipboardData,\n\t content = clipBoardData.getData('Text');\n\t\n\t var result = analize(content);\n\t\n\t if (result) {\n\t\n\t event.preventDefault();\n\t event.stopImmediatePropagation();\n\t }\n\t\n\t return result;\n\t };\n\t\n\t /**\r\n\t * Analizes pated string and calls necessary method\r\n\t */\n\t\n\t var analize = function analize(string) {\n\t\n\t var result = false,\n\t content = editor.content.currentNode,\n\t plugin = content.dataset.tool;\n\t\n\t patterns.map(function (pattern) {\n\t\n\t var execArray = pattern.regex.exec(string),\n\t match = execArray && execArray[0];\n\t\n\t if (match && match === string.trim()) {\n\t\n\t /** current block is not empty */\n\t if (content.textContent.trim() && plugin == editor.settings.initialBlockPlugin) {\n\t\n\t pasteToNewBlock_();\n\t }\n\t\n\t pattern.callback(string, pattern);\n\t result = true;\n\t }\n\t });\n\t\n\t return result;\n\t };\n\t\n\t var pasteToNewBlock_ = function pasteToNewBlock_() {\n\t\n\t /** Create new initial block */\n\t editor.content.insertBlock({\n\t\n\t type: editor.settings.initialBlockPlugin,\n\t block: editor.tools[editor.settings.initialBlockPlugin].render({\n\t text: ''\n\t })\n\t\n\t }, false);\n\t };\n\t\n\t return paste;\n\t}({});\n\n/***/ }\n/******/ ]);\n\n\n// WEBPACK FOOTER //\n// codex-editor.js"," \t// The module cache\n \tvar installedModules = {};\n\n \t// The require function\n \tfunction __webpack_require__(moduleId) {\n\n \t\t// Check if module is in cache\n \t\tif(installedModules[moduleId])\n \t\t\treturn installedModules[moduleId].exports;\n\n \t\t// Create a new module (and put it into the cache)\n \t\tvar module = installedModules[moduleId] = {\n \t\t\texports: {},\n \t\t\tid: moduleId,\n \t\t\tloaded: false\n \t\t};\n\n \t\t// Execute the module function\n \t\tmodules[moduleId].call(module.exports, module, module.exports, __webpack_require__);\n\n \t\t// Flag the module as loaded\n \t\tmodule.loaded = true;\n\n \t\t// Return the exports of the module\n \t\treturn module.exports;\n \t}\n\n\n \t// expose the modules object (__webpack_modules__)\n \t__webpack_require__.m = modules;\n\n \t// expose the module cache\n \t__webpack_require__.c = installedModules;\n\n \t// __webpack_public_path__\n \t__webpack_require__.p = \"\";\n\n \t// Load entry module and return exports\n \treturn __webpack_require__(0);\n\n\n\n// WEBPACK FOOTER //\n// webpack/bootstrap b4a1ef56ed45877046f1","/**\r\n *\r\n * Codex Editor\r\n *\r\n * @author Codex Team\r\n */\r\n\r\nmodule.exports = (function (editor) {\r\n\r\n 'use strict';\r\n\r\n editor.version = VERSION;\r\n editor.scriptPrefix = 'cdx-script-';\r\n\r\n var init = function () {\r\n\r\n editor.core = require('./modules/core');\r\n editor.tools = require('./modules/tools');\r\n editor.ui = require('./modules/ui');\r\n editor.transport = require('./modules/transport');\r\n editor.renderer = require('./modules/renderer');\r\n editor.saver = require('./modules/saver');\r\n editor.content = require('./modules/content');\r\n editor.toolbar = require('./modules/toolbar/toolbar');\r\n editor.callback = require('./modules/callbacks');\r\n editor.draw = require('./modules/draw');\r\n editor.caret = require('./modules/caret');\r\n editor.notifications = require('./modules/notifications');\r\n editor.parser = require('./modules/parser');\r\n editor.sanitizer = require('./modules/sanitizer');\r\n editor.listeners = require('./modules/listeners');\r\n editor.destroyer = require('./modules/destroyer');\r\n editor.paste = require('./modules/paste');\r\n\r\n };\r\n\r\n /**\r\n * @public\r\n * holds initial settings\r\n */\r\n editor.settings = {\r\n tools : ['paragraph', 'header', 'picture', 'list', 'quote', 'code', 'twitter', 'instagram', 'smile'],\r\n holderId : 'codex-editor',\r\n\r\n // Type of block showing on empty editor\r\n initialBlockPlugin: 'paragraph'\r\n };\r\n\r\n /**\r\n * public\r\n *\r\n * Static nodes\r\n */\r\n editor.nodes = {\r\n holder : null,\r\n wrapper : null,\r\n toolbar : null,\r\n inlineToolbar : {\r\n wrapper : null,\r\n buttons : null,\r\n actions : null\r\n },\r\n toolbox : null,\r\n notifications : null,\r\n plusButton : null,\r\n showSettingsButton: null,\r\n showTrashButton : null,\r\n blockSettings : null,\r\n pluginSettings : null,\r\n defaultSettings : null,\r\n toolbarButtons : {}, // { type : DomEl, ... }\r\n redactor : null\r\n };\r\n\r\n /**\r\n * @public\r\n *\r\n * Output state\r\n */\r\n editor.state = {\r\n jsonOutput : [],\r\n blocks : [],\r\n inputs : []\r\n };\r\n\r\n /**\r\n * @public\r\n * Editor plugins\r\n */\r\n editor.tools = {};\r\n\r\n /**\r\n * Initialization\r\n * @uses Promise cEditor.core.prepare\r\n * @param {Object} userSettings\r\n * @param {Array} userSettings.tools list of plugins\r\n * @param {String} userSettings.holderId Element's id to append editor\r\n *\r\n * Load user defined tools\r\n * Tools must contain this important objects :\r\n * @param {String} type - this is a type of plugin. It can be used as plugin name\r\n * @param {String} iconClassname - this a icon in toolbar\r\n * @param {Object} make - what should plugin do, when it is clicked\r\n * @param {Object} appendCallback - callback after clicking\r\n * @param {Element} settings - what settings does it have\r\n * @param {Object} render - plugin get JSON, and should return HTML\r\n * @param {Object} save - plugin gets HTML content, returns JSON\r\n * @param {Boolean} displayInToolbox - will be displayed in toolbox. Default value is TRUE\r\n * @param {Boolean} enableLineBreaks - inserts new block or break lines. Default value is FALSE\r\n *\r\n * @example\r\n * - type : 'header',\r\n * - iconClassname : 'ce-icon-header',\r\n * - make : headerTool.make,\r\n * - appendCallback : headerTool.appendCallback,\r\n * - settings : headerTool.makeSettings(),\r\n * - render : headerTool.render,\r\n * - save : headerTool.save,\r\n * - displayInToolbox : true,\r\n * - enableLineBreaks : false\r\n */\r\n editor.start = function (userSettings) {\r\n\r\n init();\r\n\r\n editor.core.prepare(userSettings)\r\n\r\n // If all ok, make UI, bind events and parse initial-content\r\n .then(editor.ui.prepare)\r\n .then(editor.tools.prepare)\r\n .then(editor.sanitizer.prepare)\r\n .then(editor.paste.prepare)\r\n .then(editor.transport.prepare)\r\n .then(editor.renderer.makeBlocksFromData)\r\n .then(editor.ui.saveInputs)\r\n .catch(function (error) {\r\n\r\n editor.core.log('Initialization failed with error: %o', 'warn', error);\r\n\r\n });\r\n\r\n };\r\n\r\n return editor;\r\n\r\n})({});\r\n\n\n\n// WEBPACK FOOTER //\n// ./codex.js","/**\r\n * Codex Editor Core\r\n *\r\n * @author Codex Team\r\n * @version 1.1.3\r\n */\r\n\r\nmodule.exports = (function (core) {\r\n\r\n let editor = codex.editor;\r\n\r\n /**\r\n * @public\r\n *\r\n * Editor preparing method\r\n * @return Promise\r\n */\r\n core.prepare = function (userSettings) {\r\n\r\n return new Promise(function (resolve, reject) {\r\n\r\n if ( userSettings ) {\r\n\r\n editor.settings.tools = userSettings.tools || editor.settings.tools;\r\n\r\n }\r\n\r\n if (userSettings.data) {\r\n\r\n editor.state.blocks = userSettings.data;\r\n\r\n }\r\n\r\n if (userSettings.initialBlockPlugin) {\r\n\r\n editor.settings.initialBlockPlugin = userSettings.initialBlockPlugin;\r\n\r\n }\r\n\r\n if (userSettings.sanitizer) {\r\n\r\n editor.settings.sanitizer = userSettings.sanitizer;\r\n\r\n }\r\n\r\n editor.hideToolbar = userSettings.hideToolbar;\r\n\r\n editor.nodes.holder = document.getElementById(userSettings.holderId || editor.settings.holderId);\r\n\r\n if (typeof editor.nodes.holder === undefined || editor.nodes.holder === null) {\r\n\r\n reject(Error(\"Holder wasn't found by ID: #\" + userSettings.holderId));\r\n\r\n } else {\r\n\r\n resolve();\r\n\r\n }\r\n\r\n });\r\n\r\n };\r\n\r\n /**\r\n * Logging method\r\n * @param type = ['log', 'info', 'warn']\r\n */\r\n core.log = function (msg, type, arg) {\r\n\r\n type = type || 'log';\r\n\r\n if (!arg) {\r\n\r\n arg = msg || 'undefined';\r\n msg = '[codex-editor]: %o';\r\n\r\n } else {\r\n\r\n msg = '[codex-editor]: ' + msg;\r\n\r\n }\r\n\r\n try{\r\n\r\n if ( 'console' in window && window.console[ type ] ) {\r\n\r\n if ( arg ) window.console[ type ]( msg, arg );\r\n else window.console[ type ]( msg );\r\n\r\n }\r\n\r\n }catch(e) {}\r\n\r\n };\r\n\r\n /**\r\n * @protected\r\n *\r\n * Helper for insert one element after another\r\n */\r\n core.insertAfter = function (target, element) {\r\n\r\n target.parentNode.insertBefore(element, target.nextSibling);\r\n\r\n };\r\n\r\n /**\r\n * @const\r\n *\r\n * Readable DOM-node types map\r\n */\r\n core.nodeTypes = {\r\n TAG : 1,\r\n TEXT : 3,\r\n COMMENT : 8\r\n };\r\n\r\n /**\r\n * @const\r\n * Readable keys map\r\n */\r\n core.keys = { BACKSPACE: 8, TAB: 9, ENTER: 13, SHIFT: 16, CTRL: 17, ALT: 18, ESC: 27, SPACE: 32, LEFT: 37, UP: 38, DOWN: 40, RIGHT: 39, DELETE: 46, META: 91 };\r\n\r\n /**\r\n * @protected\r\n *\r\n * Check object for DOM node\r\n */\r\n core.isDomNode = function (el) {\r\n\r\n return el && typeof el === 'object' && el.nodeType && el.nodeType == this.nodeTypes.TAG;\r\n\r\n };\r\n\r\n /**\r\n * Checks passed object for emptiness\r\n * @require ES5 - Object.keys\r\n * @param {object}\r\n */\r\n core.isEmpty = function ( obj ) {\r\n\r\n return Object.keys(obj).length === 0;\r\n\r\n };\r\n\r\n /**\r\n * Native Ajax\r\n * @param {String} settings.url - request URL\r\n * @param {function} settings.beforeSend - returned value will be passed as context to the Success, Error and Progress callbacks\r\n * @param {function} settings.success\r\n * @param {function} settings.progress\r\n */\r\n core.ajax = function (settings) {\r\n\r\n if (!settings || !settings.url) {\r\n\r\n return;\r\n\r\n }\r\n\r\n var XMLHTTP = window.XMLHttpRequest ? new XMLHttpRequest() : new ActiveXObject('Microsoft.XMLHTTP'),\r\n encodedString,\r\n isFormData,\r\n prop;\r\n\r\n\r\n settings.async = true;\r\n settings.type = settings.type || 'GET';\r\n settings.data = settings.data || '';\r\n settings['content-type'] = settings['content-type'] || 'application/json; charset=utf-8';\r\n\r\n if (settings.type == 'GET' && settings.data) {\r\n\r\n settings.url = /\\?/.test(settings.url) ? settings.url + '&' + settings.data : settings.url + '?' + settings.data;\r\n\r\n } else {\r\n\r\n encodedString = '';\r\n for(prop in settings.data) {\r\n\r\n encodedString += (prop + '=' + encodeURIComponent(settings.data[prop]) + '&');\r\n\r\n }\r\n\r\n }\r\n\r\n if (settings.withCredentials) {\r\n\r\n XMLHTTP.withCredentials = true;\r\n\r\n }\r\n\r\n /**\r\n * Value returned in beforeSend funtion will be passed as context to the other response callbacks\r\n * If beforeSend returns false, AJAX will be blocked\r\n */\r\n let responseContext,\r\n beforeSendResult;\r\n\r\n if (typeof settings.beforeSend === 'function') {\r\n\r\n beforeSendResult = settings.beforeSend.call();\r\n\r\n if (beforeSendResult === false) {\r\n\r\n return;\r\n\r\n }\r\n\r\n }\r\n\r\n XMLHTTP.open( settings.type, settings.url, settings.async );\r\n\r\n /**\r\n * If we send FormData, we need no content-type header\r\n */\r\n isFormData = isFormData_(settings.data);\r\n\r\n if (!isFormData) {\r\n\r\n if (settings.type !== 'POST') {\r\n\r\n XMLHTTP.setRequestHeader('Content-type', settings['content-type']);\r\n\r\n } else {\r\n\r\n XMLHTTP.setRequestHeader('Content-type', 'application/x-www-form-urlencoded');\r\n\r\n }\r\n\r\n }\r\n\r\n XMLHTTP.setRequestHeader('X-Requested-With', 'XMLHttpRequest');\r\n\r\n responseContext = beforeSendResult || XMLHTTP;\r\n\r\n if (typeof settings.progress === 'function') {\r\n\r\n XMLHTTP.upload.onprogress = settings.progress.bind(responseContext);\r\n\r\n }\r\n\r\n XMLHTTP.onreadystatechange = function () {\r\n\r\n if (XMLHTTP.readyState === 4) {\r\n\r\n if (XMLHTTP.status === 200) {\r\n\r\n if (typeof settings.success === 'function') {\r\n\r\n settings.success.call(responseContext, XMLHTTP.responseText);\r\n\r\n }\r\n\r\n } else {\r\n\r\n if (typeof settings.error === 'function') {\r\n\r\n settings.error.call(responseContext, XMLHTTP.responseText, XMLHTTP.status);\r\n\r\n }\r\n\r\n }\r\n\r\n }\r\n\r\n };\r\n\r\n if (isFormData) {\r\n\r\n // Sending FormData\r\n XMLHTTP.send(settings.data);\r\n\r\n } else {\r\n\r\n // POST requests\r\n XMLHTTP.send(encodedString);\r\n\r\n }\r\n\r\n return XMLHTTP;\r\n\r\n };\r\n\r\n /**\r\n * Appends script to head of document\r\n * @return Promise\r\n */\r\n core.importScript = function (scriptPath, instanceName) {\r\n\r\n return new Promise(function (resolve, reject) {\r\n\r\n let script;\r\n\r\n /** Script is already loaded */\r\n if ( !instanceName ) {\r\n\r\n reject('Instance name is missed');\r\n\r\n } else if ( document.getElementById(editor.scriptPrefix + instanceName) ) {\r\n\r\n resolve(scriptPath);\r\n\r\n }\r\n\r\n script = document.createElement('SCRIPT');\r\n script.async = true;\r\n script.defer = true;\r\n script.id = editor.scriptPrefix + instanceName;\r\n\r\n script.onload = function () {\r\n\r\n resolve(scriptPath);\r\n\r\n };\r\n\r\n script.onerror = function () {\r\n\r\n reject(scriptPath);\r\n\r\n };\r\n\r\n script.src = scriptPath;\r\n document.head.appendChild(script);\r\n\r\n });\r\n\r\n };\r\n\r\n /**\r\n * Function for checking is it FormData object to send.\r\n * @param {Object} object to check\r\n * @return boolean\r\n */\r\n var isFormData_ = function (object) {\r\n\r\n return object instanceof FormData;\r\n\r\n };\r\n\r\n return core;\r\n\r\n})({});\r\n\n\n\n// WEBPACK FOOTER //\n// ./modules/core.js","/**\r\n* Module working with plugins\r\n*/\r\nmodule.exports = (function () {\r\n\r\n let editor = codex.editor;\r\n\r\n /**\r\n * Initialize plugins before using\r\n * Ex. Load scripts or call some internal methods\r\n * @return Promise\r\n */\r\n function prepare() {\r\n\r\n return new Promise(function (resolve_, reject_) {\r\n\r\n Promise.resolve()\r\n\r\n /**\r\n * Compose a sequence of plugins that requires preparation\r\n */\r\n .then(function () {\r\n\r\n let pluginsRequiresPreparation = [],\r\n allPlugins = editor.tools;\r\n\r\n for ( let pluginName in allPlugins ) {\r\n\r\n let plugin = allPlugins[pluginName];\r\n\r\n if (plugin.prepare && typeof plugin.prepare != 'function' || !plugin.prepare) {\r\n\r\n continue;\r\n\r\n }\r\n\r\n pluginsRequiresPreparation.push(plugin);\r\n\r\n }\r\n\r\n /**\r\n * If no one passed plugins requires preparation, finish prepare() and go ahead\r\n */\r\n if (!pluginsRequiresPreparation.length) {\r\n\r\n resolve_();\r\n\r\n }\r\n\r\n return pluginsRequiresPreparation;\r\n\r\n })\r\n\r\n /** Wait plugins while they prepares */\r\n .then(waitAllPluginsPreparation_)\r\n\r\n .then(function () {\r\n\r\n editor.core.log('Plugins loaded', 'info');\r\n resolve_();\r\n\r\n }).catch(function (error) {\r\n\r\n reject_(error);\r\n\r\n });\r\n\r\n });\r\n\r\n }\r\n\r\n /**\r\n * @param {array} plugins - list of tools that requires preparation\r\n * @return {Promise} resolved while all plugins will be ready or failed\r\n */\r\n function waitAllPluginsPreparation_(plugins) {\r\n\r\n /**\r\n * @calls allPluginsProcessed__ when all plugins prepared or failed\r\n */\r\n return new Promise (function (allPluginsProcessed__) {\r\n\r\n /**\r\n * pluck each element from queue\r\n * First, send resolved Promise as previous value\r\n * Each plugins \"prepare\" method returns a Promise, that's why\r\n * reduce current element will not be able to continue while can't get\r\n * a resolved Promise\r\n *\r\n * If last plugin is \"prepared\" then go to the next stage of initialization\r\n */\r\n plugins.reduce(function (previousValue, plugin, iteration) {\r\n\r\n return previousValue.then(function () {\r\n\r\n /**\r\n * Wait till plugins prepared\r\n * @calls pluginIsReady__ when plugin is ready or failed\r\n */\r\n return new Promise ( function (pluginIsReady__) {\r\n\r\n callPluginsPrepareMethod_( plugin )\r\n\r\n .then( pluginIsReady__ )\r\n .then( function () {\r\n\r\n plugin.available = true;\r\n\r\n })\r\n\r\n .catch(function (error) {\r\n\r\n editor.core.log(`Plugin «${plugin.type}» was not loaded. Preparation failed because %o`, 'warn', error);\r\n plugin.available = false;\r\n plugin.loadingMessage = error;\r\n\r\n /** Go ahead even some plugin has problems */\r\n pluginIsReady__();\r\n\r\n })\r\n\r\n .then(function () {\r\n\r\n /** If last plugin has problems then just ignore and continue */\r\n if (iteration == plugins.length - 1) {\r\n\r\n allPluginsProcessed__();\r\n\r\n }\r\n\r\n });\r\n\r\n });\r\n\r\n });\r\n\r\n }, Promise.resolve() );\r\n\r\n });\r\n\r\n }\r\n\r\n var callPluginsPrepareMethod_ = function (plugin) {\r\n\r\n return plugin.prepare( plugin.config || {} );\r\n\r\n };\r\n\r\n return {\r\n prepare: prepare\r\n };\r\n\r\n}());\n\n\n// WEBPACK FOOTER //\n// ./modules/tools.js","/**\r\n * Codex Editor UI module\r\n *\r\n * @author Codex Team\r\n * @version 1.2.0\r\n */\r\n\r\nmodule.exports = (function (ui) {\r\n\r\n let editor = codex.editor;\r\n\r\n /**\r\n * Basic editor classnames\r\n */\r\n ui.className = {\r\n\r\n /**\r\n * @const {string} BLOCK_CLASSNAME - redactor blocks name\r\n */\r\n BLOCK_CLASSNAME : 'ce-block',\r\n\r\n /**\r\n * @const {String} wrapper for plugins content\r\n */\r\n BLOCK_CONTENT : 'ce-block__content',\r\n\r\n /**\r\n * @const {String} BLOCK_STRETCHED - makes block stretched\r\n */\r\n BLOCK_STRETCHED : 'ce-block--stretched',\r\n\r\n /**\r\n * @const {String} BLOCK_HIGHLIGHTED - adds background\r\n */\r\n BLOCK_HIGHLIGHTED : 'ce-block--focused',\r\n\r\n /**\r\n * @const {String} - for all default settings\r\n */\r\n SETTINGS_ITEM : 'ce-settings__item'\r\n\r\n };\r\n\r\n /**\r\n * @protected\r\n *\r\n * Making main interface\r\n */\r\n ui.prepare = function () {\r\n\r\n return new Promise(function (resolve) {\r\n\r\n let wrapper = editor.draw.wrapper(),\r\n redactor = editor.draw.redactor(),\r\n toolbar = makeToolBar_();\r\n\r\n wrapper.appendChild(toolbar);\r\n wrapper.appendChild(redactor);\r\n\r\n /** Save created ui-elements to static nodes state */\r\n editor.nodes.wrapper = wrapper;\r\n editor.nodes.redactor = redactor;\r\n\r\n /** Append editor wrapper with redactor zone into holder */\r\n editor.nodes.holder.appendChild(wrapper);\r\n\r\n resolve();\r\n\r\n })\r\n\r\n /** Add toolbox tools */\r\n .then(addTools_)\r\n\r\n /** Make container for inline toolbar */\r\n .then(makeInlineToolbar_)\r\n\r\n /** Add inline toolbar tools */\r\n .then(addInlineToolbarTools_)\r\n\r\n /** Draw wrapper for notifications */\r\n .then(makeNotificationHolder_)\r\n\r\n /** Add eventlisteners to redactor elements */\r\n .then(bindEvents_)\r\n\r\n .catch( function () {\r\n\r\n editor.core.log(\"Can't draw editor interface\");\r\n\r\n });\r\n\r\n };\r\n\r\n /**\r\n * @private\r\n * Draws inline toolbar zone\r\n */\r\n var makeInlineToolbar_ = function () {\r\n\r\n var container = editor.draw.inlineToolbar();\r\n\r\n /** Append to redactor new inline block */\r\n editor.nodes.inlineToolbar.wrapper = container;\r\n\r\n /** Draw toolbar buttons */\r\n editor.nodes.inlineToolbar.buttons = editor.draw.inlineToolbarButtons();\r\n\r\n /** Buttons action or settings */\r\n editor.nodes.inlineToolbar.actions = editor.draw.inlineToolbarActions();\r\n\r\n /** Append to inline toolbar buttons as part of it */\r\n editor.nodes.inlineToolbar.wrapper.appendChild(editor.nodes.inlineToolbar.buttons);\r\n editor.nodes.inlineToolbar.wrapper.appendChild(editor.nodes.inlineToolbar.actions);\r\n\r\n editor.nodes.wrapper.appendChild(editor.nodes.inlineToolbar.wrapper);\r\n\r\n };\r\n\r\n var makeToolBar_ = function () {\r\n\r\n let toolbar = editor.draw.toolbar(),\r\n blockButtons = makeToolbarSettings_(),\r\n toolbarContent = makeToolbarContent_();\r\n\r\n /** Appending first-level block buttons */\r\n toolbar.appendChild(blockButtons);\r\n\r\n /** Append toolbarContent to toolbar */\r\n toolbar.appendChild(toolbarContent);\r\n\r\n /** Make toolbar global */\r\n editor.nodes.toolbar = toolbar;\r\n\r\n return toolbar;\r\n\r\n };\r\n\r\n var makeToolbarContent_ = function () {\r\n\r\n let toolbarContent = editor.draw.toolbarContent(),\r\n toolbox = editor.draw.toolbox(),\r\n plusButton = editor.draw.plusButton();\r\n\r\n /** Append plus button */\r\n toolbarContent.appendChild(plusButton);\r\n\r\n /** Appending toolbar tools */\r\n toolbarContent.appendChild(toolbox);\r\n\r\n /** Make Toolbox and plusButton global */\r\n editor.nodes.toolbox = toolbox;\r\n editor.nodes.plusButton = plusButton;\r\n\r\n return toolbarContent;\r\n\r\n };\r\n\r\n var makeToolbarSettings_ = function () {\r\n\r\n let blockSettings = editor.draw.blockSettings(),\r\n blockButtons = editor.draw.blockButtons(),\r\n defaultSettings = editor.draw.defaultSettings(),\r\n showSettingsButton = editor.draw.settingsButton(),\r\n showTrashButton = editor.toolbar.settings.makeRemoveBlockButton(),\r\n pluginSettings = editor.draw.pluginsSettings();\r\n\r\n /** Add default and plugins settings */\r\n blockSettings.appendChild(pluginSettings);\r\n blockSettings.appendChild(defaultSettings);\r\n\r\n /**\r\n * Make blocks buttons\r\n * This block contains settings button and remove block button\r\n */\r\n blockButtons.appendChild(showSettingsButton);\r\n blockButtons.appendChild(showTrashButton);\r\n blockButtons.appendChild(blockSettings);\r\n\r\n /** Make BlockSettings, PluginSettings, DefaultSettings global */\r\n editor.nodes.blockSettings = blockSettings;\r\n editor.nodes.pluginSettings = pluginSettings;\r\n editor.nodes.defaultSettings = defaultSettings;\r\n editor.nodes.showSettingsButton = showSettingsButton;\r\n editor.nodes.showTrashButton = showTrashButton;\r\n\r\n return blockButtons;\r\n\r\n };\r\n\r\n /** Draw notifications holder */\r\n var makeNotificationHolder_ = function () {\r\n\r\n /** Append block with notifications to the document */\r\n editor.nodes.notifications = editor.notifications.createHolder();\r\n\r\n };\r\n\r\n /**\r\n * @private\r\n * Append tools passed in editor.tools\r\n */\r\n var addTools_ = function () {\r\n\r\n var tool,\r\n toolName,\r\n toolButton;\r\n\r\n for ( toolName in editor.settings.tools ) {\r\n\r\n tool = editor.settings.tools[toolName];\r\n\r\n editor.tools[toolName] = tool;\r\n\r\n if (!tool.iconClassname && tool.displayInToolbox) {\r\n\r\n editor.core.log('Toolbar icon classname missed. Tool %o skipped', 'warn', toolName);\r\n continue;\r\n\r\n }\r\n\r\n if (typeof tool.render != 'function') {\r\n\r\n editor.core.log('render method missed. Tool %o skipped', 'warn', toolName);\r\n continue;\r\n\r\n }\r\n\r\n if (!tool.displayInToolbox) {\r\n\r\n continue;\r\n\r\n } else {\r\n\r\n /** if tools is for toolbox */\r\n toolButton = editor.draw.toolbarButton(toolName, tool.iconClassname);\r\n\r\n editor.nodes.toolbox.appendChild(toolButton);\r\n\r\n editor.nodes.toolbarButtons[toolName] = toolButton;\r\n\r\n }\r\n\r\n }\r\n\r\n };\r\n\r\n var addInlineToolbarTools_ = function () {\r\n\r\n var tools = {\r\n\r\n bold: {\r\n icon : 'ce-icon-bold',\r\n command : 'bold'\r\n },\r\n\r\n italic: {\r\n icon : 'ce-icon-italic',\r\n command : 'italic'\r\n },\r\n\r\n underline: {\r\n icon : 'ce-icon-underline',\r\n command : 'underline'\r\n },\r\n\r\n link: {\r\n icon : 'ce-icon-link',\r\n command : 'createLink'\r\n }\r\n };\r\n\r\n var toolButton,\r\n tool;\r\n\r\n for(var name in tools) {\r\n\r\n tool = tools[name];\r\n\r\n toolButton = editor.draw.toolbarButtonInline(name, tool.icon);\r\n\r\n editor.nodes.inlineToolbar.buttons.appendChild(toolButton);\r\n /**\r\n * Add callbacks to this buttons\r\n */\r\n editor.ui.setInlineToolbarButtonBehaviour(toolButton, tool.command);\r\n\r\n }\r\n\r\n };\r\n\r\n /**\r\n * @private\r\n * Bind editor UI events\r\n */\r\n var bindEvents_ = function () {\r\n\r\n editor.core.log('ui.bindEvents fired', 'info');\r\n\r\n // window.addEventListener('error', function (errorMsg, url, lineNumber) {\r\n // editor.notifications.errorThrown(errorMsg, event);\r\n // }, false );\r\n\r\n /** All keydowns on Document */\r\n editor.listeners.add(document, 'keydown', editor.callback.globalKeydown, false);\r\n\r\n /** All keydowns on Redactor zone */\r\n editor.listeners.add(editor.nodes.redactor, 'keydown', editor.callback.redactorKeyDown, false);\r\n\r\n /** All keydowns on Document */\r\n editor.listeners.add(document, 'keyup', editor.callback.globalKeyup, false );\r\n\r\n /**\r\n * Mouse click to radactor\r\n */\r\n editor.listeners.add(editor.nodes.redactor, 'click', editor.callback.redactorClicked, false );\r\n\r\n /**\r\n * Clicks to the Plus button\r\n */\r\n editor.listeners.add(editor.nodes.plusButton, 'click', editor.callback.plusButtonClicked, false);\r\n\r\n /**\r\n * Clicks to SETTINGS button in toolbar\r\n */\r\n editor.listeners.add(editor.nodes.showSettingsButton, 'click', editor.callback.showSettingsButtonClicked, false );\r\n\r\n /** Bind click listeners on toolbar buttons */\r\n for (var button in editor.nodes.toolbarButtons) {\r\n\r\n editor.listeners.add(editor.nodes.toolbarButtons[button], 'click', editor.callback.toolbarButtonClicked, false);\r\n\r\n }\r\n\r\n };\r\n\r\n ui.addBlockHandlers = function (block) {\r\n\r\n if (!block) return;\r\n\r\n /**\r\n * Block keydowns\r\n */\r\n editor.listeners.add(block, 'keydown', editor.callback.blockKeydown, false);\r\n\r\n /**\r\n * Pasting content from another source\r\n * We have two type of sanitization\r\n * First - uses deep-first search algorithm to get sub nodes,\r\n * sanitizes whole Block_content and replaces cleared nodes\r\n * This method is deprecated\r\n * Method is used in editor.callback.blockPaste(event)\r\n *\r\n * Secont - uses Mutation observer.\r\n * Observer \"observe\" DOM changes and send changings to callback.\r\n * Callback gets changed node, not whole Block_content.\r\n * Inserted or changed node, which we've gotten have been cleared and replaced with diry node\r\n *\r\n * Method is used in editor.callback.blockPasteViaSanitize(event)\r\n *\r\n * @uses html-janitor\r\n * @example editor.callback.blockPasteViaSanitize(event), the second method.\r\n *\r\n */\r\n editor.listeners.add(block, 'paste', editor.callback.blockPasteCallback, false);\r\n\r\n editor.listeners.add(block, 'mouseup', editor.toolbar.inline.show, false);\r\n\r\n };\r\n\r\n /** getting all contenteditable elements */\r\n ui.saveInputs = function () {\r\n\r\n var redactor = editor.nodes.redactor;\r\n\r\n editor.state.inputs = [];\r\n\r\n /** Save all inputs in global variable state */\r\n var inputs = redactor.querySelectorAll('[contenteditable], input, textarea');\r\n\r\n Array.prototype.map.call(inputs, function (current) {\r\n\r\n if (!current.type || current.type == 'text' || current.type == 'textarea') {\r\n\r\n editor.state.inputs.push(current);\r\n\r\n }\r\n\r\n });\r\n\r\n };\r\n\r\n /**\r\n * Adds first initial block on empty redactor\r\n */\r\n ui.addInitialBlock = function () {\r\n\r\n var initialBlockType = editor.settings.initialBlockPlugin,\r\n initialBlock;\r\n\r\n if ( !editor.tools[initialBlockType] ) {\r\n\r\n editor.core.log('Plugin %o was not implemented and can\\'t be used as initial block', 'warn', initialBlockType);\r\n return;\r\n\r\n }\r\n\r\n initialBlock = editor.tools[initialBlockType].render();\r\n\r\n initialBlock.setAttribute('data-placeholder', 'Расскажите свою историю...');\r\n\r\n editor.content.insertBlock({\r\n type : initialBlockType,\r\n block : initialBlock\r\n });\r\n\r\n editor.content.workingNodeChanged(initialBlock);\r\n\r\n };\r\n\r\n ui.setInlineToolbarButtonBehaviour = function (button, type) {\r\n\r\n editor.listeners.add(button, 'mousedown', function (event) {\r\n\r\n editor.toolbar.inline.toolClicked(event, type);\r\n\r\n }, false);\r\n\r\n };\r\n\r\n return ui;\r\n\r\n})({});\r\n\n\n\n// WEBPACK FOOTER //\n// ./modules/ui.js","/**\r\n *\r\n * Codex.Editor Transport Module\r\n *\r\n * @copyright 2017 Codex-Team\r\n * @version 1.2.0\r\n */\r\n\r\nmodule.exports = (function (transport) {\r\n\r\n let editor = codex.editor;\r\n\r\n\r\n /**\r\n * @private {Object} current XmlHttpRequest instance\r\n */\r\n var currentRequest = null;\r\n\r\n\r\n /**\r\n * @type {null} | {DOMElement} input - keeps input element in memory\r\n */\r\n transport.input = null;\r\n\r\n /**\r\n * @property {Object} arguments - keep plugin settings and defined callbacks\r\n */\r\n transport.arguments = null;\r\n\r\n /**\r\n * Prepares input element where will be files\r\n */\r\n transport.prepare = function () {\r\n\r\n let input = editor.draw.node( 'INPUT', '', { type : 'file' } );\r\n\r\n editor.listeners.add(input, 'change', editor.transport.fileSelected);\r\n editor.transport.input = input;\r\n\r\n };\r\n\r\n /** Clear input when files is uploaded */\r\n transport.clearInput = function () {\r\n\r\n /** Remove old input */\r\n transport.input = null;\r\n\r\n /** Prepare new one */\r\n transport.prepare();\r\n\r\n };\r\n\r\n /**\r\n * Callback for file selection\r\n * @param {Event} event\r\n */\r\n transport.fileSelected = function () {\r\n\r\n var input = this,\r\n i,\r\n files = input.files,\r\n formData = new FormData();\r\n\r\n if (editor.transport.arguments.multiple === true) {\r\n\r\n for ( i = 0; i < files.length; i++) {\r\n\r\n formData.append('files[]', files[i], files[i].name);\r\n\r\n }\r\n\r\n } else {\r\n\r\n formData.append('files', files[0], files[0].name);\r\n\r\n }\r\n\r\n currentRequest = editor.core.ajax({\r\n type : 'POST',\r\n data : formData,\r\n url : editor.transport.arguments.url,\r\n beforeSend : editor.transport.arguments.beforeSend,\r\n success : editor.transport.arguments.success,\r\n error : editor.transport.arguments.error,\r\n progress : editor.transport.arguments.progress\r\n });\r\n\r\n /** Clear input */\r\n transport.clearInput();\r\n\r\n };\r\n\r\n /**\r\n * Use plugin callbacks\r\n * @protected\r\n *\r\n * @param {Object} args - can have :\r\n * @param {String} args.url - fetch URL\r\n * @param {Function} args.beforeSend - function calls before sending ajax\r\n * @param {Function} args.success - success callback\r\n * @param {Function} args.error - on error handler\r\n * @param {Function} args.progress - xhr onprogress handler\r\n * @param {Boolean} args.multiple - allow select several files\r\n * @param {String} args.accept - adds accept attribute\r\n */\r\n transport.selectAndUpload = function (args) {\r\n\r\n transport.arguments = args;\r\n\r\n if ( args.multiple === true) {\r\n\r\n transport.input.setAttribute('multiple', 'multiple');\r\n\r\n }\r\n\r\n if ( args.accept ) {\r\n\r\n transport.input.setAttribute('accept', args.accept);\r\n\r\n }\r\n\r\n transport.input.click();\r\n\r\n };\r\n\r\n transport.abort = function () {\r\n\r\n currentRequest.abort();\r\n\r\n currentRequest = null;\r\n\r\n };\r\n\r\n return transport;\r\n\r\n})({});\n\n\n// WEBPACK FOOTER //\n// ./modules/transport.js","/**\r\n * Codex Editor Renderer Module\r\n *\r\n * @author Codex Team\r\n * @version 1.0\r\n */\r\n\r\nmodule.exports = (function (renderer) {\r\n\r\n let editor = codex.editor;\r\n\r\n /**\r\n * Asyncronously parses input JSON to redactor blocks\r\n */\r\n renderer.makeBlocksFromData = function () {\r\n\r\n /**\r\n * If redactor is empty, add first paragraph to start writing\r\n */\r\n if (editor.core.isEmpty(editor.state.blocks) || !editor.state.blocks.items.length) {\r\n\r\n editor.ui.addInitialBlock();\r\n return;\r\n\r\n }\r\n\r\n Promise.resolve()\r\n\r\n /** First, get JSON from state */\r\n .then(function () {\r\n\r\n return editor.state.blocks;\r\n\r\n })\r\n\r\n /** Then, start to iterate they */\r\n .then(editor.renderer.appendBlocks)\r\n\r\n /** Write log if something goes wrong */\r\n .catch(function (error) {\r\n\r\n editor.core.log('Error while parsing JSON: %o', 'error', error);\r\n\r\n });\r\n\r\n };\r\n\r\n /**\r\n * Parses JSON to blocks\r\n * @param {object} data\r\n * @return Primise -> nodeList\r\n */\r\n renderer.appendBlocks = function (data) {\r\n\r\n var blocks = data.items;\r\n\r\n /**\r\n * Sequence of one-by-one blocks appending\r\n * Uses to save blocks order after async-handler\r\n */\r\n var nodeSequence = Promise.resolve();\r\n\r\n for (var index = 0; index < blocks.length ; index++ ) {\r\n\r\n /** Add node to sequence at specified index */\r\n editor.renderer.appendNodeAtIndex(nodeSequence, blocks, index);\r\n\r\n }\r\n\r\n };\r\n\r\n /**\r\n * Append node at specified index\r\n */\r\n renderer.appendNodeAtIndex = function (nodeSequence, blocks, index) {\r\n\r\n /** We need to append node to sequence */\r\n nodeSequence\r\n\r\n /** first, get node async-aware */\r\n .then(function () {\r\n\r\n return editor.renderer.getNodeAsync(blocks, index);\r\n\r\n })\r\n\r\n /**\r\n * second, compose editor-block from JSON object\r\n */\r\n .then(editor.renderer.createBlockFromData)\r\n\r\n /**\r\n * now insert block to redactor\r\n */\r\n .then(function (blockData) {\r\n\r\n /**\r\n * blockData has 'block', 'type' and 'stretched' information\r\n */\r\n editor.content.insertBlock(blockData);\r\n\r\n /** Pass created block to next step */\r\n return blockData.block;\r\n\r\n })\r\n\r\n /** Log if something wrong with node */\r\n .catch(function (error) {\r\n\r\n editor.core.log('Node skipped while parsing because %o', 'error', error);\r\n\r\n });\r\n\r\n };\r\n\r\n /**\r\n * Asynchronously returns block data from blocksList by index\r\n * @return Promise to node\r\n */\r\n renderer.getNodeAsync = function (blocksList, index) {\r\n\r\n return Promise.resolve().then(function () {\r\n\r\n return {\r\n tool : blocksList[index],\r\n position : index\r\n };\r\n\r\n });\r\n\r\n };\r\n\r\n /**\r\n * Creates editor block by JSON-data\r\n *\r\n * @uses render method of each plugin\r\n *\r\n * @param {Object} toolData.tool\r\n * { header : {\r\n * text: '',\r\n * type: 'H3', ...\r\n * }\r\n * }\r\n * @param {Number} toolData.position - index in input-blocks array\r\n * @return {Object} with type and Element\r\n */\r\n renderer.createBlockFromData = function ( toolData ) {\r\n\r\n /** New parser */\r\n var block,\r\n tool = toolData.tool,\r\n pluginName = tool.type;\r\n\r\n /** Get first key of object that stores plugin name */\r\n // for (var pluginName in blockData) break;\r\n\r\n /** Check for plugin existance */\r\n if (!editor.tools[pluginName]) {\r\n\r\n throw Error(`Plugin «${pluginName}» not found`);\r\n\r\n }\r\n\r\n /** Check for plugin having render method */\r\n if (typeof editor.tools[pluginName].render != 'function') {\r\n\r\n throw Error(`Plugin «${pluginName}» must have «render» method`);\r\n\r\n }\r\n\r\n if ( editor.tools[pluginName].available === false ) {\r\n\r\n block = editor.draw.unavailableBlock();\r\n\r\n block.innerHTML = editor.tools[pluginName].loadingMessage;\r\n\r\n /**\r\n * Saver will extract data from initial block data by position in array\r\n */\r\n block.dataset.inputPosition = toolData.position;\r\n\r\n } else {\r\n\r\n /** New Parser */\r\n block = editor.tools[pluginName].render(tool.data);\r\n\r\n }\r\n\r\n /** is first-level block stretched */\r\n var stretched = editor.tools[pluginName].isStretched || false;\r\n\r\n /** Retrun type and block */\r\n return {\r\n type : pluginName,\r\n block : block,\r\n stretched : stretched\r\n };\r\n\r\n };\r\n\r\n return renderer;\r\n\r\n})({});\n\n\n// WEBPACK FOOTER //\n// ./modules/renderer.js","/**\r\n * Codex Editor Saver\r\n *\r\n * @author Codex Team\r\n * @version 1.0.2\r\n */\r\n\r\nmodule.exports = (function (saver) {\r\n\r\n let editor = codex.editor;\r\n\r\n /**\r\n * Saves blocks\r\n * @private\r\n */\r\n saver.saveBlocks = function () {\r\n\r\n /** Save html content of redactor to memory */\r\n editor.state.html = editor.nodes.redactor.innerHTML;\r\n\r\n /** Empty jsonOutput state */\r\n editor.state.jsonOutput = [];\r\n\r\n Promise.resolve()\r\n\r\n .then(function () {\r\n\r\n return editor.nodes.redactor.childNodes;\r\n\r\n })\r\n /** Making a sequence from separate blocks */\r\n .then(editor.saver.makeQueue)\r\n\r\n .then(function () {\r\n // editor.nodes.textarea.innerHTML = editor.state.html;\r\n })\r\n\r\n .catch( function (error) {\r\n\r\n editor.core.log(error);\r\n\r\n });\r\n\r\n };\r\n\r\n saver.makeQueue = function (blocks) {\r\n\r\n var queue = Promise.resolve();\r\n\r\n for(var index = 0; index < blocks.length; index++) {\r\n\r\n /** Add node to sequence at specified index */\r\n editor.saver.getBlockData(queue, blocks, index);\r\n\r\n }\r\n\r\n };\r\n\r\n /** Gets every block and makes From Data */\r\n saver.getBlockData = function (queue, blocks, index) {\r\n\r\n queue.then(function () {\r\n\r\n return editor.saver.getNodeAsync(blocks, index);\r\n\r\n })\r\n\r\n .then(editor.saver.makeFormDataFromBlocks);\r\n\r\n };\r\n\r\n\r\n /**\r\n * Asynchronously returns block data from blocksList by index\r\n * @return Promise to node\r\n */\r\n saver.getNodeAsync = function (blocksList, index) {\r\n\r\n return Promise.resolve().then(function () {\r\n\r\n return blocksList[index];\r\n\r\n });\r\n\r\n };\r\n\r\n saver.makeFormDataFromBlocks = function (block) {\r\n\r\n var pluginName = block.dataset.tool;\r\n\r\n /** Check for plugin existance */\r\n if (!editor.tools[pluginName]) {\r\n\r\n throw Error(`Plugin «${pluginName}» not found`);\r\n\r\n }\r\n\r\n /** Check for plugin having render method */\r\n if (typeof editor.tools[pluginName].save != 'function') {\r\n\r\n throw Error(`Plugin «${pluginName}» must have save method`);\r\n\r\n }\r\n\r\n /** Result saver */\r\n var blockContent = block.childNodes[0],\r\n pluginsContent = blockContent.childNodes[0],\r\n savedData,\r\n position,\r\n output;\r\n\r\n /** If plugin wasn't available then return data from cache */\r\n if ( editor.tools[pluginName].available === false ) {\r\n\r\n position = pluginsContent.dataset.inputPosition;\r\n savedData = codex.editor.state.blocks.items[position].data;\r\n\r\n } else {\r\n\r\n savedData = editor.tools[pluginName].save(pluginsContent);\r\n\r\n if (editor.tools[pluginName].validate) {\r\n\r\n var result = editor.tools[pluginName].validate(savedData);\r\n\r\n /**\r\n * Do not allow invalid data\r\n */\r\n if (!result)\r\n return;\r\n\r\n }\r\n\r\n }\r\n\r\n output = {\r\n type : pluginName,\r\n data : savedData\r\n };\r\n\r\n editor.state.jsonOutput.push(output);\r\n\r\n };\r\n\r\n return saver;\r\n\r\n})({});\n\n\n// WEBPACK FOOTER //\n// ./modules/saver.js","/**\r\n * Codex Editor Content Module\r\n * Works with DOM\r\n *\r\n * @module Codex Editor content module\r\n *\r\n * @author Codex Team\r\n * @version 1.3.13\r\n *\r\n * @description Module works with Elements that have been appended to the main DOM\r\n */\r\n\r\nmodule.exports = (function (content) {\r\n\r\n let editor = codex.editor;\r\n\r\n /**\r\n * Links to current active block\r\n * @type {null | Element}\r\n */\r\n content.currentNode = null;\r\n\r\n /**\r\n * clicked in redactor area\r\n * @type {null | Boolean}\r\n */\r\n content.editorAreaHightlighted = null;\r\n\r\n /**\r\n * @deprecated\r\n * Synchronizes redactor with original textarea\r\n */\r\n content.sync = function () {\r\n\r\n editor.core.log('syncing...');\r\n\r\n /**\r\n * Save redactor content to editor.state\r\n */\r\n editor.state.html = editor.nodes.redactor.innerHTML;\r\n\r\n };\r\n\r\n /**\r\n * Appends background to the block\r\n *\r\n * @description add CSS class to highlight visually first-level block area\r\n */\r\n content.markBlock = function () {\r\n\r\n editor.content.currentNode.classList.add(editor.ui.className.BLOCK_HIGHLIGHTED);\r\n\r\n };\r\n\r\n /**\r\n * Clear background\r\n *\r\n * @description clears styles that highlights block\r\n */\r\n content.clearMark = function () {\r\n\r\n if (editor.content.currentNode) {\r\n\r\n editor.content.currentNode.classList.remove(editor.ui.className.BLOCK_HIGHLIGHTED);\r\n\r\n }\r\n\r\n };\r\n\r\n /**\r\n * Finds first-level block\r\n *\r\n * @param {Element} node - selected or clicked in redactors area node\r\n * @protected\r\n *\r\n * @description looks for first-level block.\r\n * gets parent while node is not first-level\r\n */\r\n content.getFirstLevelBlock = function (node) {\r\n\r\n if (!editor.core.isDomNode(node)) {\r\n\r\n node = node.parentNode;\r\n\r\n }\r\n\r\n if (node === editor.nodes.redactor || node === document.body) {\r\n\r\n return null;\r\n\r\n } else {\r\n\r\n while(!node.classList.contains(editor.ui.className.BLOCK_CLASSNAME)) {\r\n\r\n node = node.parentNode;\r\n\r\n }\r\n\r\n return node;\r\n\r\n }\r\n\r\n };\r\n\r\n /**\r\n * Trigger this event when working node changed\r\n * @param {Element} targetNode - first-level of this node will be current\r\n * @protected\r\n *\r\n * @description If targetNode is first-level then we set it as current else we look for parents to find first-level\r\n */\r\n content.workingNodeChanged = function (targetNode) {\r\n\r\n /** Clear background from previous marked block before we change */\r\n editor.content.clearMark();\r\n\r\n if (!targetNode) {\r\n\r\n return;\r\n\r\n }\r\n\r\n content.currentNode = content.getFirstLevelBlock(targetNode);\r\n\r\n };\r\n\r\n /**\r\n * Replaces one redactor block with another\r\n * @protected\r\n * @param {Element} targetBlock - block to replace. Mostly currentNode.\r\n * @param {Element} newBlock\r\n * @param {string} newBlockType - type of new block; we need to store it to data-attribute\r\n *\r\n * [!] Function does not saves old block content.\r\n * You can get it manually and pass with newBlock.innerHTML\r\n */\r\n content.replaceBlock = function (targetBlock, newBlock) {\r\n\r\n if (!targetBlock || !newBlock) {\r\n\r\n editor.core.log('replaceBlock: missed params');\r\n return;\r\n\r\n }\r\n\r\n /** If target-block is not a frist-level block, then we iterate parents to find it */\r\n while(!targetBlock.classList.contains(editor.ui.className.BLOCK_CLASSNAME)) {\r\n\r\n targetBlock = targetBlock.parentNode;\r\n\r\n }\r\n\r\n /** Replacing */\r\n editor.nodes.redactor.replaceChild(newBlock, targetBlock);\r\n\r\n /**\r\n * Set new node as current\r\n */\r\n editor.content.workingNodeChanged(newBlock);\r\n\r\n /**\r\n * Add block handlers\r\n */\r\n editor.ui.addBlockHandlers(newBlock);\r\n\r\n /**\r\n * Save changes\r\n */\r\n editor.ui.saveInputs();\r\n\r\n };\r\n\r\n /**\r\n * @protected\r\n *\r\n * Inserts new block to redactor\r\n * Wrapps block into a DIV with BLOCK_CLASSNAME class\r\n *\r\n * @param blockData {object}\r\n * @param blockData.block {Element} element with block content\r\n * @param blockData.type {string} block plugin\r\n * @param needPlaceCaret {bool} pass true to set caret in new block\r\n *\r\n */\r\n content.insertBlock = function ( blockData, needPlaceCaret ) {\r\n\r\n var workingBlock = editor.content.currentNode,\r\n newBlockContent = blockData.block,\r\n blockType = blockData.type,\r\n isStretched = blockData.stretched;\r\n\r\n var newBlock = composeNewBlock_(newBlockContent, blockType, isStretched);\r\n\r\n if (workingBlock) {\r\n\r\n editor.core.insertAfter(workingBlock, newBlock);\r\n\r\n } else {\r\n\r\n /**\r\n * If redactor is empty, append as first child\r\n */\r\n editor.nodes.redactor.appendChild(newBlock);\r\n\r\n }\r\n\r\n /**\r\n * Block handler\r\n */\r\n editor.ui.addBlockHandlers(newBlock);\r\n\r\n /**\r\n * Set new node as current\r\n */\r\n editor.content.workingNodeChanged(newBlock);\r\n\r\n /**\r\n * Save changes\r\n */\r\n editor.ui.saveInputs();\r\n\r\n\r\n if ( needPlaceCaret ) {\r\n\r\n /**\r\n * If we don't know input index then we set default value -1\r\n */\r\n var currentInputIndex = editor.caret.getCurrentInputIndex() || -1;\r\n\r\n\r\n if (currentInputIndex == -1) {\r\n\r\n\r\n var editableElement = newBlock.querySelector('[contenteditable]'),\r\n emptyText = document.createTextNode('');\r\n\r\n editableElement.appendChild(emptyText);\r\n editor.caret.set(editableElement, 0, 0);\r\n\r\n editor.toolbar.move();\r\n editor.toolbar.showPlusButton();\r\n\r\n\r\n } else {\r\n\r\n if (currentInputIndex === editor.state.inputs.length - 1)\r\n return;\r\n\r\n /** Timeout for browsers execution */\r\n window.setTimeout(function () {\r\n\r\n /** Setting to the new input */\r\n editor.caret.setToNextBlock(currentInputIndex);\r\n editor.toolbar.move();\r\n editor.toolbar.open();\r\n\r\n }, 10);\r\n\r\n }\r\n\r\n }\r\n\r\n /**\r\n * Block is inserted, wait for new click that defined focusing on editors area\r\n * @type {boolean}\r\n */\r\n content.editorAreaHightlighted = false;\r\n\r\n };\r\n\r\n /**\r\n * Replaces blocks with saving content\r\n * @protected\r\n * @param {Element} noteToReplace\r\n * @param {Element} newNode\r\n * @param {Element} blockType\r\n */\r\n content.switchBlock = function (blockToReplace, newBlock, tool) {\r\n\r\n tool = tool || editor.content.currentNode.dataset.tool;\r\n var newBlockComposed = composeNewBlock_(newBlock, tool);\r\n\r\n /** Replacing */\r\n editor.content.replaceBlock(blockToReplace, newBlockComposed);\r\n\r\n /** Save new Inputs when block is changed */\r\n editor.ui.saveInputs();\r\n\r\n };\r\n\r\n /**\r\n * Iterates between child noted and looking for #text node on deepest level\r\n * @protected\r\n *\r\n * @param {Element} block - node where find\r\n * @param {int} postiton - starting postion\r\n * Example: childNodex.length to find from the end\r\n * or 0 to find from the start\r\n * @return {Text} block\r\n * @uses DFS\r\n */\r\n content.getDeepestTextNodeFromPosition = function (block, position) {\r\n\r\n /**\r\n * Clear Block from empty and useless spaces with trim.\r\n * Such nodes we should remove\r\n */\r\n var blockChilds = block.childNodes,\r\n index,\r\n node,\r\n text;\r\n\r\n for(index = 0; index < blockChilds.length; index++) {\r\n\r\n node = blockChilds[index];\r\n\r\n if (node.nodeType == editor.core.nodeTypes.TEXT) {\r\n\r\n text = node.textContent.trim();\r\n\r\n /** Text is empty. We should remove this child from node before we start DFS\r\n * decrease the quantity of childs.\r\n */\r\n if (text === '') {\r\n\r\n block.removeChild(node);\r\n position--;\r\n\r\n }\r\n\r\n }\r\n\r\n }\r\n\r\n if (block.childNodes.length === 0) {\r\n\r\n return document.createTextNode('');\r\n\r\n }\r\n\r\n /** Setting default position when we deleted all empty nodes */\r\n if ( position < 0 )\r\n position = 1;\r\n\r\n var lookingFromStart = false;\r\n\r\n /** For looking from START */\r\n if (position === 0) {\r\n\r\n lookingFromStart = true;\r\n position = 1;\r\n\r\n }\r\n\r\n while ( position ) {\r\n\r\n /** initial verticle of node. */\r\n if ( lookingFromStart ) {\r\n\r\n block = block.childNodes[0];\r\n\r\n } else {\r\n\r\n block = block.childNodes[position - 1];\r\n\r\n }\r\n\r\n if ( block.nodeType == editor.core.nodeTypes.TAG ) {\r\n\r\n position = block.childNodes.length;\r\n\r\n } else if (block.nodeType == editor.core.nodeTypes.TEXT ) {\r\n\r\n position = 0;\r\n\r\n }\r\n\r\n }\r\n\r\n return block;\r\n\r\n };\r\n\r\n /**\r\n * @private\r\n * @param {Element} block - current plugins render\r\n * @param {String} tool - plugins name\r\n * @param {Boolean} isStretched - make stretched block or not\r\n *\r\n * @description adds necessary information to wrap new created block by first-level holder\r\n */\r\n var composeNewBlock_ = function (block, tool, isStretched) {\r\n\r\n var newBlock = editor.draw.node('DIV', editor.ui.className.BLOCK_CLASSNAME, {}),\r\n blockContent = editor.draw.node('DIV', editor.ui.className.BLOCK_CONTENT, {});\r\n\r\n blockContent.appendChild(block);\r\n newBlock.appendChild(blockContent);\r\n\r\n if (isStretched) {\r\n\r\n blockContent.classList.add(editor.ui.className.BLOCK_STRETCHED);\r\n\r\n }\r\n\r\n newBlock.dataset.tool = tool;\r\n return newBlock;\r\n\r\n };\r\n\r\n /**\r\n * Returns Range object of current selection\r\n * @protected\r\n */\r\n content.getRange = function () {\r\n\r\n var selection = window.getSelection().getRangeAt(0);\r\n\r\n return selection;\r\n\r\n };\r\n\r\n /**\r\n * Divides block in two blocks (after and before caret)\r\n *\r\n * @protected\r\n * @param {int} inputIndex - target input index\r\n *\r\n * @description splits current input content to the separate blocks\r\n * When enter is pressed among the words, that text will be splited.\r\n */\r\n content.splitBlock = function (inputIndex) {\r\n\r\n var selection = window.getSelection(),\r\n anchorNode = selection.anchorNode,\r\n anchorNodeText = anchorNode.textContent,\r\n caretOffset = selection.anchorOffset,\r\n textBeforeCaret,\r\n textNodeBeforeCaret,\r\n textAfterCaret,\r\n textNodeAfterCaret;\r\n\r\n var currentBlock = editor.content.currentNode.querySelector('[contentEditable]');\r\n\r\n\r\n textBeforeCaret = anchorNodeText.substring(0, caretOffset);\r\n textAfterCaret = anchorNodeText.substring(caretOffset);\r\n\r\n textNodeBeforeCaret = document.createTextNode(textBeforeCaret);\r\n\r\n if (textAfterCaret) {\r\n\r\n textNodeAfterCaret = document.createTextNode(textAfterCaret);\r\n\r\n }\r\n\r\n var previousChilds = [],\r\n nextChilds = [],\r\n reachedCurrent = false;\r\n\r\n if (textNodeAfterCaret) {\r\n\r\n nextChilds.push(textNodeAfterCaret);\r\n\r\n }\r\n\r\n for ( var i = 0, child; !!(child = currentBlock.childNodes[i]); i++) {\r\n\r\n if ( child != anchorNode ) {\r\n\r\n if ( !reachedCurrent ) {\r\n\r\n previousChilds.push(child);\r\n\r\n } else {\r\n\r\n nextChilds.push(child);\r\n\r\n }\r\n\r\n } else {\r\n\r\n reachedCurrent = true;\r\n\r\n }\r\n\r\n }\r\n\r\n /** Clear current input */\r\n editor.state.inputs[inputIndex].innerHTML = '';\r\n\r\n /**\r\n * Append all childs founded before anchorNode\r\n */\r\n var previousChildsLength = previousChilds.length;\r\n\r\n for(i = 0; i < previousChildsLength; i++) {\r\n\r\n editor.state.inputs[inputIndex].appendChild(previousChilds[i]);\r\n\r\n }\r\n\r\n editor.state.inputs[inputIndex].appendChild(textNodeBeforeCaret);\r\n\r\n /**\r\n * Append text node which is after caret\r\n */\r\n var nextChildsLength = nextChilds.length,\r\n newNode = document.createElement('div');\r\n\r\n for(i = 0; i < nextChildsLength; i++) {\r\n\r\n newNode.appendChild(nextChilds[i]);\r\n\r\n }\r\n\r\n newNode = newNode.innerHTML;\r\n\r\n /** This type of block creates when enter is pressed */\r\n var NEW_BLOCK_TYPE = editor.settings.initialBlockPlugin;\r\n\r\n /**\r\n * Make new paragraph with text after caret\r\n */\r\n editor.content.insertBlock({\r\n type : NEW_BLOCK_TYPE,\r\n block : editor.tools[NEW_BLOCK_TYPE].render({\r\n text : newNode\r\n })\r\n }, true );\r\n\r\n };\r\n\r\n /**\r\n * Merges two blocks — current and target\r\n * If target index is not exist, then previous will be as target\r\n *\r\n * @protected\r\n * @param {int} currentInputIndex\r\n * @param {int} targetInputIndex\r\n *\r\n * @description gets two inputs indexes and merges into one\r\n */\r\n content.mergeBlocks = function (currentInputIndex, targetInputIndex) {\r\n\r\n /** If current input index is zero, then prevent method execution */\r\n if (currentInputIndex === 0) {\r\n\r\n return;\r\n\r\n }\r\n\r\n var targetInput,\r\n currentInputContent = editor.state.inputs[currentInputIndex].innerHTML;\r\n\r\n if (!targetInputIndex) {\r\n\r\n targetInput = editor.state.inputs[currentInputIndex - 1];\r\n\r\n } else {\r\n\r\n targetInput = editor.state.inputs[targetInputIndex];\r\n\r\n }\r\n\r\n targetInput.innerHTML += currentInputContent;\r\n\r\n };\r\n\r\n /**\r\n * Iterates all right siblings and parents, which has right siblings\r\n * while it does not reached the first-level block\r\n *\r\n * @param {Element} node\r\n * @return {boolean}\r\n */\r\n content.isLastNode = function (node) {\r\n\r\n // console.log('погнали перебор родителей');\r\n\r\n var allChecked = false;\r\n\r\n while ( !allChecked ) {\r\n\r\n // console.log('Смотрим на %o', node);\r\n // console.log('Проверим, пустые ли соседи справа');\r\n\r\n if ( !allSiblingsEmpty_(node) ) {\r\n\r\n // console.log('Есть непустые соседи. Узел не последний. Выходим.');\r\n return false;\r\n\r\n }\r\n\r\n node = node.parentNode;\r\n\r\n /**\r\n * Проверяем родителей до тех пор, пока не найдем блок первого уровня\r\n */\r\n if ( node.classList.contains(editor.ui.className.BLOCK_CONTENT) ) {\r\n\r\n allChecked = true;\r\n\r\n }\r\n\r\n }\r\n\r\n return true;\r\n\r\n };\r\n\r\n /**\r\n * Checks if all element right siblings is empty\r\n * @param node\r\n */\r\n var allSiblingsEmpty_ = function (node) {\r\n\r\n /**\r\n * Нужно убедиться, что после пустого соседа ничего нет\r\n */\r\n var sibling = node.nextSibling;\r\n\r\n while ( sibling ) {\r\n\r\n if (sibling.textContent.length) {\r\n\r\n return false;\r\n\r\n }\r\n\r\n sibling = sibling.nextSibling;\r\n\r\n }\r\n\r\n return true;\r\n\r\n };\r\n\r\n /**\r\n * @public\r\n *\r\n * @param {string} htmlData - html content as string\r\n * @param {string} plainData - plain text\r\n * @return {string} - html content as string\r\n */\r\n content.wrapTextWithParagraphs = function (htmlData, plainData) {\r\n\r\n if (!htmlData) {\r\n\r\n return '

    ' + plainData.split('\\n\\n').join('

    ') + '

    ';\r\n\r\n }\r\n\r\n var wrapper = document.createElement('DIV'),\r\n newWrapper = document.createElement('DIV'),\r\n i,\r\n paragraph,\r\n firstLevelBlocks = ['DIV', 'P'],\r\n blockTyped,\r\n node;\r\n\r\n /**\r\n * Make HTML Element to Wrap Text\r\n * It allows us to work with input data as HTML content\r\n */\r\n wrapper.innerHTML = htmlData;\r\n paragraph = document.createElement('P');\r\n\r\n for (i = 0; i < wrapper.childNodes.length; i++) {\r\n\r\n node = wrapper.childNodes[i];\r\n\r\n blockTyped = firstLevelBlocks.indexOf(node.tagName) != -1;\r\n\r\n /**\r\n * If node is first-levet\r\n * we add this node to our new wrapper\r\n */\r\n if ( blockTyped ) {\r\n\r\n /**\r\n * If we had splitted inline nodes to paragraph before\r\n */\r\n if ( paragraph.childNodes.length ) {\r\n\r\n newWrapper.appendChild(paragraph.cloneNode(true));\r\n\r\n /** empty paragraph */\r\n paragraph = null;\r\n paragraph = document.createElement('P');\r\n\r\n }\r\n\r\n newWrapper.appendChild(node.cloneNode(true));\r\n\r\n } else {\r\n\r\n /** Collect all inline nodes to one as paragraph */\r\n paragraph.appendChild(node.cloneNode(true));\r\n\r\n /** if node is last we should append this node to paragraph and paragraph to new wrapper */\r\n if ( i == wrapper.childNodes.length - 1 ) {\r\n\r\n newWrapper.appendChild(paragraph.cloneNode(true));\r\n\r\n }\r\n\r\n }\r\n\r\n }\r\n\r\n return newWrapper.innerHTML;\r\n\r\n };\r\n\r\n /**\r\n * Finds closest Contenteditable parent from Element\r\n * @param {Element} node element looking from\r\n * @return {Element} node contenteditable\r\n */\r\n content.getEditableParent = function (node) {\r\n\r\n while (node && node.contentEditable != 'true') {\r\n\r\n node = node.parentNode;\r\n\r\n }\r\n\r\n return node;\r\n\r\n };\r\n\r\n return content;\r\n\r\n})({});\n\n\n// WEBPACK FOOTER //\n// ./modules/content.js","/**\r\n * Codex Editor toolbar module\r\n *\r\n * Contains:\r\n * - Inline toolbox\r\n * - Toolbox within plus button\r\n * - Settings section\r\n *\r\n * @author Codex Team\r\n * @version 1.0\r\n */\r\n\r\nmodule.exports = (function (toolbar) {\r\n\r\n let editor = codex.editor;\r\n\r\n toolbar.settings = require('./settings');\r\n toolbar.inline = require('./inline');\r\n toolbar.toolbox = require('./toolbox');\r\n\r\n /**\r\n * Margin between focused node and toolbar\r\n */\r\n toolbar.defaultToolbarHeight = 49;\r\n\r\n toolbar.defaultOffset = 34;\r\n\r\n toolbar.opened = false;\r\n\r\n toolbar.current = null;\r\n\r\n /**\r\n * @protected\r\n */\r\n toolbar.open = function () {\r\n\r\n if (editor.hideToolbar) {\r\n\r\n return;\r\n\r\n }\r\n\r\n editor.nodes.toolbar.classList.add('opened');\r\n this.opened = true;\r\n\r\n };\r\n\r\n /**\r\n * @protected\r\n */\r\n toolbar.close = function () {\r\n\r\n editor.nodes.toolbar.classList.remove('opened');\r\n\r\n toolbar.opened = false;\r\n toolbar.current = null;\r\n\r\n for (var button in editor.nodes.toolbarButtons) {\r\n\r\n editor.nodes.toolbarButtons[button].classList.remove('selected');\r\n\r\n }\r\n\r\n /** Close toolbox when toolbar is not displayed */\r\n editor.toolbar.toolbox.close();\r\n editor.toolbar.settings.close();\r\n\r\n };\r\n\r\n toolbar.toggle = function () {\r\n\r\n if ( !this.opened ) {\r\n\r\n this.open();\r\n\r\n } else {\r\n\r\n this.close();\r\n\r\n }\r\n\r\n };\r\n\r\n toolbar.hidePlusButton = function () {\r\n\r\n editor.nodes.plusButton.classList.add('hide');\r\n\r\n };\r\n\r\n toolbar.showPlusButton = function () {\r\n\r\n editor.nodes.plusButton.classList.remove('hide');\r\n\r\n };\r\n\r\n /**\r\n * Moving toolbar to the specified node\r\n */\r\n toolbar.move = function () {\r\n\r\n /** Close Toolbox when we move toolbar */\r\n editor.toolbar.toolbox.close();\r\n\r\n if (!editor.content.currentNode) {\r\n\r\n return;\r\n\r\n }\r\n\r\n var newYCoordinate = editor.content.currentNode.offsetTop - (editor.toolbar.defaultToolbarHeight / 2) + editor.toolbar.defaultOffset;\r\n\r\n editor.nodes.toolbar.style.transform = `translate3D(0, ${Math.floor(newYCoordinate)}px, 0)`;\r\n\r\n /** Close trash actions */\r\n editor.toolbar.settings.hideRemoveActions();\r\n\r\n };\r\n\r\n return toolbar;\r\n\r\n})({});\n\n\n// WEBPACK FOOTER //\n// ./modules/toolbar/toolbar.js","/**\r\n * Toolbar settings\r\n *\r\n * @version 1.0.4\r\n */\r\n\r\nmodule.exports = (function (settings) {\r\n\r\n let editor = codex.editor;\r\n\r\n settings.opened = false;\r\n\r\n settings.setting = null;\r\n settings.actions = null;\r\n\r\n /**\r\n * Append and open settings\r\n */\r\n settings.open = function (toolType) {\r\n\r\n /**\r\n * Append settings content\r\n * It's stored in tool.settings\r\n */\r\n if (!editor.tools[toolType] || !editor.tools[toolType].makeSettings ) {\r\n\r\n editor.core.log(`Plugin «${toolType}» has no settings`, 'warn');\r\n editor.nodes.pluginSettings.innerHTML = `Плагин «${toolType}» не имеет настроек`;\r\n\r\n } else {\r\n\r\n /**\r\n * Draw settings block\r\n */\r\n var settingsBlock = editor.tools[toolType].makeSettings();\r\n\r\n editor.nodes.pluginSettings.appendChild(settingsBlock);\r\n\r\n }\r\n\r\n /** Open settings block */\r\n editor.nodes.blockSettings.classList.add('opened');\r\n this.opened = true;\r\n\r\n };\r\n\r\n /**\r\n * Close and clear settings\r\n */\r\n settings.close = function () {\r\n\r\n editor.nodes.blockSettings.classList.remove('opened');\r\n editor.nodes.pluginSettings.innerHTML = '';\r\n\r\n this.opened = false;\r\n\r\n };\r\n\r\n /**\r\n * @param {string} toolType - plugin type\r\n */\r\n settings.toggle = function ( toolType ) {\r\n\r\n if ( !this.opened ) {\r\n\r\n this.open(toolType);\r\n\r\n } else {\r\n\r\n this.close();\r\n\r\n }\r\n\r\n };\r\n\r\n /**\r\n * Here we will draw buttons and add listeners to components\r\n */\r\n settings.makeRemoveBlockButton = function () {\r\n\r\n var removeBlockWrapper = editor.draw.node('SPAN', 'ce-toolbar__remove-btn', {}),\r\n settingButton = editor.draw.node('SPAN', 'ce-toolbar__remove-setting', { innerHTML : '' }),\r\n actionWrapper = editor.draw.node('DIV', 'ce-toolbar__remove-confirmation', {}),\r\n confirmAction = editor.draw.node('DIV', 'ce-toolbar__remove-confirm', { textContent : 'Удалить блок' }),\r\n cancelAction = editor.draw.node('DIV', 'ce-toolbar__remove-cancel', { textContent : 'Отмена' });\r\n\r\n editor.listeners.add(settingButton, 'click', editor.toolbar.settings.removeButtonClicked, false);\r\n\r\n editor.listeners.add(confirmAction, 'click', editor.toolbar.settings.confirmRemovingRequest, false);\r\n\r\n editor.listeners.add(cancelAction, 'click', editor.toolbar.settings.cancelRemovingRequest, false);\r\n\r\n actionWrapper.appendChild(confirmAction);\r\n actionWrapper.appendChild(cancelAction);\r\n\r\n removeBlockWrapper.appendChild(settingButton);\r\n removeBlockWrapper.appendChild(actionWrapper);\r\n\r\n /** Save setting */\r\n editor.toolbar.settings.setting = settingButton;\r\n editor.toolbar.settings.actions = actionWrapper;\r\n\r\n return removeBlockWrapper;\r\n\r\n };\r\n\r\n settings.removeButtonClicked = function () {\r\n\r\n var action = editor.toolbar.settings.actions;\r\n\r\n if (action.classList.contains('opened')) {\r\n\r\n editor.toolbar.settings.hideRemoveActions();\r\n\r\n } else {\r\n\r\n editor.toolbar.settings.showRemoveActions();\r\n\r\n }\r\n\r\n editor.toolbar.toolbox.close();\r\n editor.toolbar.settings.close();\r\n\r\n };\r\n\r\n settings.cancelRemovingRequest = function () {\r\n\r\n editor.toolbar.settings.actions.classList.remove('opened');\r\n\r\n };\r\n\r\n settings.confirmRemovingRequest = function () {\r\n\r\n var currentBlock = editor.content.currentNode,\r\n firstLevelBlocksCount;\r\n\r\n currentBlock.remove();\r\n\r\n firstLevelBlocksCount = editor.nodes.redactor.childNodes.length;\r\n\r\n /**\r\n * If all blocks are removed\r\n */\r\n if (firstLevelBlocksCount === 0) {\r\n\r\n /** update currentNode variable */\r\n editor.content.currentNode = null;\r\n\r\n /** Inserting new empty initial block */\r\n editor.ui.addInitialBlock();\r\n\r\n }\r\n\r\n editor.ui.saveInputs();\r\n\r\n editor.toolbar.close();\r\n\r\n };\r\n\r\n settings.showRemoveActions = function () {\r\n\r\n editor.toolbar.settings.actions.classList.add('opened');\r\n\r\n };\r\n\r\n settings.hideRemoveActions = function () {\r\n\r\n editor.toolbar.settings.actions.classList.remove('opened');\r\n\r\n };\r\n\r\n return settings;\r\n\r\n})({});\n\n\n// WEBPACK FOOTER //\n// ./modules/toolbar/settings.js","/**\r\n * Inline toolbar\r\n *\r\n * Contains from tools:\r\n * Bold, Italic, Underline and Anchor\r\n *\r\n * @author Codex Team\r\n * @version 1.0\r\n */\r\n\r\nmodule.exports = (function (inline) {\r\n\r\n let editor = codex.editor;\r\n\r\n inline.buttonsOpened = null;\r\n inline.actionsOpened = null;\r\n inline.wrappersOffset = null;\r\n\r\n /**\r\n * saving selection that need for execCommand for styling\r\n *\r\n */\r\n inline.storedSelection = null;\r\n\r\n /**\r\n * @protected\r\n *\r\n * Open inline toobar\r\n */\r\n inline.show = function () {\r\n\r\n var currentNode = editor.content.currentNode,\r\n tool = currentNode.dataset.tool,\r\n plugin;\r\n\r\n /**\r\n * tool allowed to open inline toolbar\r\n */\r\n plugin = editor.tools[tool];\r\n\r\n if (!plugin.showInlineToolbar)\r\n return;\r\n\r\n var selectedText = inline.getSelectionText(),\r\n toolbar = editor.nodes.inlineToolbar.wrapper;\r\n\r\n if (selectedText.length > 0) {\r\n\r\n /** Move toolbar and open */\r\n editor.toolbar.inline.move();\r\n\r\n /** Open inline toolbar */\r\n toolbar.classList.add('opened');\r\n\r\n /** show buttons of inline toolbar */\r\n editor.toolbar.inline.showButtons();\r\n\r\n }\r\n\r\n };\r\n\r\n /**\r\n * @protected\r\n *\r\n * Closes inline toolbar\r\n */\r\n inline.close = function () {\r\n\r\n var toolbar = editor.nodes.inlineToolbar.wrapper;\r\n\r\n toolbar.classList.remove('opened');\r\n\r\n };\r\n\r\n /**\r\n * @private\r\n *\r\n * Moving toolbar\r\n */\r\n inline.move = function () {\r\n\r\n if (!this.wrappersOffset) {\r\n\r\n this.wrappersOffset = this.getWrappersOffset();\r\n\r\n }\r\n\r\n var coords = this.getSelectionCoords(),\r\n defaultOffset = 0,\r\n toolbar = editor.nodes.inlineToolbar.wrapper,\r\n newCoordinateX,\r\n newCoordinateY;\r\n\r\n if (toolbar.offsetHeight === 0) {\r\n\r\n defaultOffset = 40;\r\n\r\n }\r\n\r\n newCoordinateX = coords.x - this.wrappersOffset.left;\r\n newCoordinateY = coords.y + window.scrollY - this.wrappersOffset.top - defaultOffset - toolbar.offsetHeight;\r\n\r\n toolbar.style.transform = `translate3D(${Math.floor(newCoordinateX)}px, ${Math.floor(newCoordinateY)}px, 0)`;\r\n\r\n /** Close everything */\r\n editor.toolbar.inline.closeButtons();\r\n editor.toolbar.inline.closeAction();\r\n\r\n };\r\n\r\n /**\r\n * @private\r\n *\r\n * Tool Clicked\r\n */\r\n\r\n inline.toolClicked = function (event, type) {\r\n\r\n /**\r\n * For simple tools we use default browser function\r\n * For more complicated tools, we should write our own behavior\r\n */\r\n switch (type) {\r\n case 'createLink' : editor.toolbar.inline.createLinkAction(event, type); break;\r\n default : editor.toolbar.inline.defaultToolAction(type); break;\r\n }\r\n\r\n /**\r\n * highlight buttons\r\n * after making some action\r\n */\r\n editor.nodes.inlineToolbar.buttons.childNodes.forEach(editor.toolbar.inline.hightlight);\r\n\r\n };\r\n\r\n /**\r\n * @private\r\n *\r\n * Saving wrappers offset in DOM\r\n */\r\n inline.getWrappersOffset = function () {\r\n\r\n var wrapper = editor.nodes.wrapper,\r\n offset = this.getOffset(wrapper);\r\n\r\n this.wrappersOffset = offset;\r\n return offset;\r\n\r\n };\r\n\r\n /**\r\n * @private\r\n *\r\n * Calculates offset of DOM element\r\n *\r\n * @param el\r\n * @returns {{top: number, left: number}}\r\n */\r\n inline.getOffset = function ( el ) {\r\n\r\n var _x = 0;\r\n var _y = 0;\r\n\r\n while( el && !isNaN( el.offsetLeft ) && !isNaN( el.offsetTop ) ) {\r\n\r\n _x += (el.offsetLeft + el.clientLeft);\r\n _y += (el.offsetTop + el.clientTop);\r\n el = el.offsetParent;\r\n\r\n }\r\n return { top: _y, left: _x };\r\n\r\n };\r\n\r\n /**\r\n * @private\r\n *\r\n * Calculates position of selected text\r\n * @returns {{x: number, y: number}}\r\n */\r\n inline.getSelectionCoords = function () {\r\n\r\n var sel = document.selection, range;\r\n var x = 0, y = 0;\r\n\r\n if (sel) {\r\n\r\n if (sel.type != 'Control') {\r\n\r\n range = sel.createRange();\r\n range.collapse(true);\r\n x = range.boundingLeft;\r\n y = range.boundingTop;\r\n\r\n }\r\n\r\n } else if (window.getSelection) {\r\n\r\n sel = window.getSelection();\r\n\r\n if (sel.rangeCount) {\r\n\r\n range = sel.getRangeAt(0).cloneRange();\r\n if (range.getClientRects) {\r\n\r\n range.collapse(true);\r\n var rect = range.getClientRects()[0];\r\n\r\n if (!rect) {\r\n\r\n return;\r\n\r\n }\r\n\r\n x = rect.left;\r\n y = rect.top;\r\n\r\n }\r\n\r\n }\r\n\r\n }\r\n return { x: x, y: y };\r\n\r\n };\r\n\r\n /**\r\n * @private\r\n *\r\n * Returns selected text as String\r\n * @returns {string}\r\n */\r\n inline.getSelectionText = function () {\r\n\r\n var selectedText = '';\r\n\r\n // all modern browsers and IE9+\r\n if (window.getSelection) {\r\n\r\n selectedText = window.getSelection().toString();\r\n\r\n }\r\n\r\n return selectedText;\r\n\r\n };\r\n\r\n /** Opens buttons block */\r\n inline.showButtons = function () {\r\n\r\n var buttons = editor.nodes.inlineToolbar.buttons;\r\n\r\n buttons.classList.add('opened');\r\n\r\n editor.toolbar.inline.buttonsOpened = true;\r\n\r\n /** highlight buttons */\r\n editor.nodes.inlineToolbar.buttons.childNodes.forEach(editor.toolbar.inline.hightlight);\r\n\r\n };\r\n\r\n /** Makes buttons disappear */\r\n inline.closeButtons = function () {\r\n\r\n var buttons = editor.nodes.inlineToolbar.buttons;\r\n\r\n buttons.classList.remove('opened');\r\n\r\n editor.toolbar.inline.buttonsOpened = false;\r\n\r\n };\r\n\r\n /** Open buttons defined action if exist */\r\n inline.showActions = function () {\r\n\r\n var action = editor.nodes.inlineToolbar.actions;\r\n\r\n action.classList.add('opened');\r\n\r\n editor.toolbar.inline.actionsOpened = true;\r\n\r\n };\r\n\r\n /** Close actions block */\r\n inline.closeAction = function () {\r\n\r\n var action = editor.nodes.inlineToolbar.actions;\r\n\r\n action.innerHTML = '';\r\n action.classList.remove('opened');\r\n editor.toolbar.inline.actionsOpened = false;\r\n\r\n };\r\n\r\n\r\n /**\r\n * Callback for keydowns in inline toolbar \"Insert link...\" input\r\n */\r\n let inlineToolbarAnchorInputKeydown_ = function (event) {\r\n\r\n if (event.keyCode != editor.core.keys.ENTER) {\r\n\r\n return;\r\n\r\n }\r\n\r\n let editable = editor.content.currentNode,\r\n storedSelection = editor.toolbar.inline.storedSelection;\r\n\r\n editor.toolbar.inline.restoreSelection(editable, storedSelection);\r\n editor.toolbar.inline.setAnchor(this.value);\r\n\r\n /**\r\n * Preventing events that will be able to happen\r\n */\r\n event.preventDefault();\r\n event.stopImmediatePropagation();\r\n\r\n editor.toolbar.inline.clearRange();\r\n\r\n };\r\n\r\n /** Action for link creation or for setting anchor */\r\n inline.createLinkAction = function (event) {\r\n\r\n var isActive = this.isLinkActive();\r\n\r\n var editable = editor.content.currentNode,\r\n storedSelection = editor.toolbar.inline.saveSelection(editable);\r\n\r\n /** Save globally selection */\r\n editor.toolbar.inline.storedSelection = storedSelection;\r\n\r\n if (isActive) {\r\n\r\n\r\n /**\r\n * Changing stored selection. if we want to remove anchor from word\r\n * we should remove anchor from whole word, not only selected part.\r\n * The solution is than we get the length of current link\r\n * Change start position to - end of selection minus length of anchor\r\n */\r\n editor.toolbar.inline.restoreSelection(editable, storedSelection);\r\n\r\n editor.toolbar.inline.defaultToolAction('unlink');\r\n\r\n } else {\r\n\r\n /** Create input and close buttons */\r\n var action = editor.draw.inputForLink();\r\n\r\n editor.nodes.inlineToolbar.actions.appendChild(action);\r\n\r\n editor.toolbar.inline.closeButtons();\r\n editor.toolbar.inline.showActions();\r\n\r\n /**\r\n * focus to input\r\n * Solution: https://developer.mozilla.org/ru/docs/Web/API/HTMLElement/focus\r\n * Prevents event after showing input and when we need to focus an input which is in unexisted form\r\n */\r\n action.focus();\r\n event.preventDefault();\r\n\r\n /** Callback to link action */\r\n editor.listeners.add(action, 'keydown', inlineToolbarAnchorInputKeydown_, false);\r\n\r\n }\r\n\r\n };\r\n\r\n inline.isLinkActive = function () {\r\n\r\n var isActive = false;\r\n\r\n editor.nodes.inlineToolbar.buttons.childNodes.forEach(function (tool) {\r\n\r\n var dataType = tool.dataset.type;\r\n\r\n if (dataType == 'link' && tool.classList.contains('hightlighted')) {\r\n\r\n isActive = true;\r\n\r\n }\r\n\r\n });\r\n\r\n return isActive;\r\n\r\n };\r\n\r\n /** default action behavior of tool */\r\n inline.defaultToolAction = function (type) {\r\n\r\n document.execCommand(type, false, null);\r\n\r\n };\r\n\r\n /**\r\n * @private\r\n *\r\n * Sets URL\r\n *\r\n * @param {String} url - URL\r\n */\r\n inline.setAnchor = function (url) {\r\n\r\n document.execCommand('createLink', false, url);\r\n\r\n /** Close after URL inserting */\r\n editor.toolbar.inline.closeAction();\r\n\r\n };\r\n\r\n /**\r\n * @private\r\n *\r\n * Saves selection\r\n */\r\n inline.saveSelection = function (containerEl) {\r\n\r\n var range = window.getSelection().getRangeAt(0),\r\n preSelectionRange = range.cloneRange(),\r\n start;\r\n\r\n preSelectionRange.selectNodeContents(containerEl);\r\n preSelectionRange.setEnd(range.startContainer, range.startOffset);\r\n\r\n start = preSelectionRange.toString().length;\r\n\r\n return {\r\n start: start,\r\n end: start + range.toString().length\r\n };\r\n\r\n };\r\n\r\n /**\r\n * @private\r\n *\r\n * Sets to previous selection (Range)\r\n *\r\n * @param {Element} containerEl - editable element where we restore range\r\n * @param {Object} savedSel - range basic information to restore\r\n */\r\n inline.restoreSelection = function (containerEl, savedSel) {\r\n\r\n var range = document.createRange(),\r\n charIndex = 0;\r\n\r\n range.setStart(containerEl, 0);\r\n range.collapse(true);\r\n\r\n var nodeStack = [ containerEl ],\r\n node,\r\n foundStart = false,\r\n stop = false,\r\n nextCharIndex;\r\n\r\n while (!stop && (node = nodeStack.pop())) {\r\n\r\n if (node.nodeType == 3) {\r\n\r\n nextCharIndex = charIndex + node.length;\r\n\r\n if (!foundStart && savedSel.start >= charIndex && savedSel.start <= nextCharIndex) {\r\n\r\n range.setStart(node, savedSel.start - charIndex);\r\n foundStart = true;\r\n\r\n }\r\n if (foundStart && savedSel.end >= charIndex && savedSel.end <= nextCharIndex) {\r\n\r\n range.setEnd(node, savedSel.end - charIndex);\r\n stop = true;\r\n\r\n }\r\n charIndex = nextCharIndex;\r\n\r\n } else {\r\n\r\n var i = node.childNodes.length;\r\n\r\n while (i--) {\r\n\r\n nodeStack.push(node.childNodes[i]);\r\n\r\n }\r\n\r\n }\r\n\r\n }\r\n\r\n var sel = window.getSelection();\r\n\r\n sel.removeAllRanges();\r\n sel.addRange(range);\r\n\r\n };\r\n\r\n /**\r\n * @private\r\n *\r\n * Removes all ranges from window selection\r\n */\r\n inline.clearRange = function () {\r\n\r\n var selection = window.getSelection();\r\n\r\n selection.removeAllRanges();\r\n\r\n };\r\n\r\n /**\r\n * @private\r\n *\r\n * sets or removes hightlight\r\n */\r\n inline.hightlight = function (tool) {\r\n\r\n var dataType = tool.dataset.type;\r\n\r\n if (document.queryCommandState(dataType)) {\r\n\r\n editor.toolbar.inline.setButtonHighlighted(tool);\r\n\r\n } else {\r\n\r\n editor.toolbar.inline.removeButtonsHighLight(tool);\r\n\r\n }\r\n\r\n /**\r\n *\r\n * hightlight for anchors\r\n */\r\n var selection = window.getSelection(),\r\n tag = selection.anchorNode.parentNode;\r\n\r\n if (tag.tagName == 'A' && dataType == 'link') {\r\n\r\n editor.toolbar.inline.setButtonHighlighted(tool);\r\n\r\n }\r\n\r\n };\r\n\r\n /**\r\n * @private\r\n *\r\n * Mark button if text is already executed\r\n */\r\n inline.setButtonHighlighted = function (button) {\r\n\r\n button.classList.add('hightlighted');\r\n\r\n /** At link tool we also change icon */\r\n if (button.dataset.type == 'link') {\r\n\r\n var icon = button.childNodes[0];\r\n\r\n icon.classList.remove('ce-icon-link');\r\n icon.classList.add('ce-icon-unlink');\r\n\r\n }\r\n\r\n };\r\n\r\n /**\r\n * @private\r\n *\r\n * Removes hightlight\r\n */\r\n inline.removeButtonsHighLight = function (button) {\r\n\r\n button.classList.remove('hightlighted');\r\n\r\n /** At link tool we also change icon */\r\n if (button.dataset.type == 'link') {\r\n\r\n var icon = button.childNodes[0];\r\n\r\n icon.classList.remove('ce-icon-unlink');\r\n icon.classList.add('ce-icon-link');\r\n\r\n }\r\n\r\n };\r\n\r\n\r\n return inline;\r\n\r\n})({});\n\n\n// WEBPACK FOOTER //\n// ./modules/toolbar/inline.js","/**\r\n * Codex Editor toolbox\r\n *\r\n * All tools be able to appended here\r\n *\r\n * @author Codex Team\r\n * @version 1.0\r\n */\r\n\r\nmodule.exports = (function (toolbox) {\r\n\r\n let editor = codex.editor;\r\n\r\n toolbox.opened = false;\r\n\r\n /** Shows toolbox */\r\n toolbox.open = function () {\r\n\r\n /** Close setting if toolbox is opened */\r\n if (editor.toolbar.settings.opened) {\r\n\r\n editor.toolbar.settings.close();\r\n\r\n }\r\n\r\n /** display toolbox */\r\n editor.nodes.toolbox.classList.add('opened');\r\n\r\n /** Animate plus button */\r\n editor.nodes.plusButton.classList.add('clicked');\r\n\r\n /** toolbox state */\r\n editor.toolbar.toolbox.opened = true;\r\n\r\n };\r\n\r\n /** Closes toolbox */\r\n toolbox.close = function () {\r\n\r\n /** Makes toolbox disapear */\r\n editor.nodes.toolbox.classList.remove('opened');\r\n\r\n /** Rotate plus button */\r\n editor.nodes.plusButton.classList.remove('clicked');\r\n\r\n /** toolbox state */\r\n editor.toolbar.toolbox.opened = false;\r\n\r\n editor.toolbar.current = null;\r\n\r\n };\r\n\r\n toolbox.leaf = function () {\r\n\r\n let currentTool = editor.toolbar.current,\r\n tools = Object.keys(editor.tools),\r\n barButtons = editor.nodes.toolbarButtons,\r\n nextToolIndex = 0,\r\n toolToSelect,\r\n visibleTool,\r\n tool;\r\n\r\n if ( !currentTool ) {\r\n\r\n /** Get first tool from object*/\r\n for(tool in editor.tools) {\r\n\r\n if (editor.tools[tool].displayInToolbox) {\r\n\r\n break;\r\n\r\n }\r\n\r\n nextToolIndex ++;\r\n\r\n }\r\n\r\n } else {\r\n\r\n nextToolIndex = (tools.indexOf(currentTool) + 1) % tools.length;\r\n visibleTool = tools[nextToolIndex];\r\n\r\n while (!editor.tools[visibleTool].displayInToolbox) {\r\n\r\n nextToolIndex = (nextToolIndex + 1) % tools.length;\r\n visibleTool = tools[nextToolIndex];\r\n\r\n }\r\n\r\n }\r\n\r\n toolToSelect = tools[nextToolIndex];\r\n\r\n for ( var button in barButtons ) {\r\n\r\n barButtons[button].classList.remove('selected');\r\n\r\n }\r\n\r\n barButtons[toolToSelect].classList.add('selected');\r\n editor.toolbar.current = toolToSelect;\r\n\r\n };\r\n\r\n /**\r\n * Transforming selected node type into selected toolbar element type\r\n * @param {event} event\r\n */\r\n toolbox.toolClicked = function (event) {\r\n\r\n /**\r\n * UNREPLACEBLE_TOOLS this types of tools are forbidden to replace even they are empty\r\n */\r\n var UNREPLACEBLE_TOOLS = ['image', 'link', 'list', 'instagram', 'twitter', 'embed'],\r\n tool = editor.tools[editor.toolbar.current],\r\n workingNode = editor.content.currentNode,\r\n currentInputIndex = editor.caret.inputIndex,\r\n newBlockContent,\r\n appendCallback,\r\n blockData;\r\n\r\n /** Make block from plugin */\r\n newBlockContent = tool.render();\r\n\r\n /** information about block */\r\n blockData = {\r\n block : newBlockContent,\r\n type : tool.type,\r\n stretched : false\r\n };\r\n\r\n if (\r\n workingNode &&\r\n UNREPLACEBLE_TOOLS.indexOf(workingNode.dataset.tool) === -1 &&\r\n workingNode.textContent.trim() === ''\r\n ) {\r\n\r\n /** Replace current block */\r\n editor.content.switchBlock(workingNode, newBlockContent, tool.type);\r\n\r\n } else {\r\n\r\n /** Insert new Block from plugin */\r\n editor.content.insertBlock(blockData);\r\n\r\n /** increase input index */\r\n currentInputIndex++;\r\n\r\n }\r\n\r\n /** Fire tool append callback */\r\n appendCallback = tool.appendCallback;\r\n\r\n if (appendCallback && typeof appendCallback == 'function') {\r\n\r\n appendCallback.call(event);\r\n\r\n }\r\n\r\n window.setTimeout(function () {\r\n\r\n /** Set caret to current block */\r\n editor.caret.setToBlock(currentInputIndex);\r\n\r\n }, 10);\r\n\r\n\r\n /**\r\n * Changing current Node\r\n */\r\n editor.content.workingNodeChanged();\r\n\r\n /**\r\n * Move toolbar when node is changed\r\n */\r\n editor.toolbar.move();\r\n\r\n };\r\n\r\n return toolbox;\r\n\r\n})({});\n\n\n// WEBPACK FOOTER //\n// ./modules/toolbar/toolbox.js","/**\r\n * @module Codex Editor Callbacks module\r\n * @description Module works with editor added Elements\r\n *\r\n * @author Codex Team\r\n * @version 1.4.0\r\n */\r\n\r\nmodule.exports = (function (callbacks) {\r\n\r\n let editor = codex.editor;\r\n\r\n /**\r\n * used by UI module\r\n * @description Routes all keydowns on document\r\n * @param {Object} event\r\n */\r\n callbacks.globalKeydown = function (event) {\r\n\r\n switch (event.keyCode) {\r\n case editor.core.keys.ENTER : enterKeyPressed_(event); break;\r\n }\r\n\r\n };\r\n\r\n /**\r\n * used by UI module\r\n * @description Routes all keydowns on redactors area\r\n * @param {Object} event\r\n */\r\n callbacks.redactorKeyDown = function (event) {\r\n\r\n switch (event.keyCode) {\r\n case editor.core.keys.TAB : tabKeyPressedOnRedactorsZone_(event); break;\r\n case editor.core.keys.ENTER : enterKeyPressedOnRedactorsZone_(event); break;\r\n case editor.core.keys.ESC : escapeKeyPressedOnRedactorsZone_(event); break;\r\n default : defaultKeyPressedOnRedactorsZone_(event); break;\r\n }\r\n\r\n };\r\n\r\n /**\r\n * used by UI module\r\n * @description Routes all keyup events\r\n * @param {Object} event\r\n */\r\n callbacks.globalKeyup = function (event) {\r\n\r\n switch (event.keyCode) {\r\n case editor.core.keys.UP :\r\n case editor.core.keys.LEFT :\r\n case editor.core.keys.RIGHT :\r\n case editor.core.keys.DOWN : arrowKeyPressed_(event); break;\r\n }\r\n\r\n };\r\n\r\n /**\r\n * @param {Object} event\r\n * @private\r\n *\r\n * Handles behaviour when tab pressed\r\n * @description if Content is empty show toolbox (if it is closed) or leaf tools\r\n * uses Toolbars toolbox module to handle the situation\r\n */\r\n var tabKeyPressedOnRedactorsZone_ = function (event) {\r\n\r\n /**\r\n * Wait for solution. Would like to know the behaviour\r\n * @todo Add spaces\r\n */\r\n event.preventDefault();\r\n\r\n var nativeInputs = editor.content.currentNode.querySelectorAll('textarea, input'),\r\n nativeInputsAreEmpty = true,\r\n textContentIsEmpty = !editor.content.currentNode.textContent.trim();\r\n\r\n Array.prototype.map.call(nativeInputs, function (input) {\r\n\r\n if (input.type == 'textarea' || input.type == 'text') {\r\n\r\n nativeInputsAreEmpty = nativeInputsAreEmpty && !input.value.trim();\r\n\r\n }\r\n\r\n });\r\n\r\n var blockIsEmpty = textContentIsEmpty && nativeInputsAreEmpty;\r\n\r\n if (!blockIsEmpty) {\r\n\r\n return;\r\n\r\n }\r\n\r\n if ( !editor.toolbar.opened ) {\r\n\r\n editor.toolbar.open();\r\n\r\n }\r\n\r\n if (editor.toolbar.opened && !editor.toolbar.toolbox.opened) {\r\n\r\n editor.toolbar.toolbox.open();\r\n\r\n } else {\r\n\r\n editor.toolbar.toolbox.leaf();\r\n\r\n }\r\n\r\n };\r\n\r\n /**\r\n * Handles global EnterKey Press\r\n * @see enterPressedOnBlock_\r\n * @param {Object} event\r\n */\r\n var enterKeyPressed_ = function () {\r\n\r\n if (editor.content.editorAreaHightlighted) {\r\n\r\n /**\r\n * it means that we lose input index, saved index before is not correct\r\n * therefore we need to set caret when we insert new block\r\n */\r\n editor.caret.inputIndex = -1;\r\n\r\n enterPressedOnBlock_();\r\n\r\n }\r\n\r\n };\r\n\r\n /**\r\n * Callback for enter key pressing in first-level block area\r\n *\r\n * @param {Event} event\r\n * @private\r\n *\r\n * @description Inserts new block with initial type from settings\r\n */\r\n var enterPressedOnBlock_ = function () {\r\n\r\n var NEW_BLOCK_TYPE = editor.settings.initialBlockPlugin;\r\n\r\n editor.content.insertBlock({\r\n type : NEW_BLOCK_TYPE,\r\n block : editor.tools[NEW_BLOCK_TYPE].render()\r\n }, true );\r\n\r\n editor.toolbar.move();\r\n editor.toolbar.open();\r\n\r\n };\r\n\r\n\r\n /**\r\n * ENTER key handler\r\n *\r\n * @param {Object} event\r\n * @private\r\n *\r\n * @description Makes new block with initial type from settings\r\n */\r\n var enterKeyPressedOnRedactorsZone_ = function (event) {\r\n\r\n if (event.target.contentEditable == 'true') {\r\n\r\n /** Update input index */\r\n editor.caret.saveCurrentInputIndex();\r\n\r\n }\r\n\r\n var currentInputIndex = editor.caret.getCurrentInputIndex() || 0,\r\n workingNode = editor.content.currentNode,\r\n tool = workingNode.dataset.tool,\r\n isEnterPressedOnToolbar = editor.toolbar.opened &&\r\n editor.toolbar.current &&\r\n event.target == editor.state.inputs[currentInputIndex];\r\n\r\n /** The list of tools which needs the default browser behaviour */\r\n var enableLineBreaks = editor.tools[tool].enableLineBreaks;\r\n\r\n /** This type of block creates when enter is pressed */\r\n var NEW_BLOCK_TYPE = editor.settings.initialBlockPlugin;\r\n\r\n /**\r\n * When toolbar is opened, select tool instead of making new paragraph\r\n */\r\n if ( isEnterPressedOnToolbar ) {\r\n\r\n event.preventDefault();\r\n\r\n editor.toolbar.toolbox.toolClicked(event);\r\n\r\n editor.toolbar.close();\r\n\r\n /**\r\n * Stop other listeners callback executions\r\n */\r\n event.stopPropagation();\r\n event.stopImmediatePropagation();\r\n\r\n return;\r\n\r\n }\r\n\r\n /**\r\n * Allow paragraph lineBreaks with shift enter\r\n * Or if shiftkey pressed and enter and enabledLineBreaks, the let new block creation\r\n */\r\n if ( event.shiftKey || enableLineBreaks ) {\r\n\r\n event.stopPropagation();\r\n event.stopImmediatePropagation();\r\n return;\r\n\r\n }\r\n\r\n var currentSelection = window.getSelection(),\r\n currentSelectedNode = currentSelection.anchorNode,\r\n caretAtTheEndOfText = editor.caret.position.atTheEnd(),\r\n isTextNodeHasParentBetweenContenteditable = false;\r\n\r\n /**\r\n * Allow making new

    in same block by SHIFT+ENTER and forbids to prevent default browser behaviour\r\n */\r\n if ( event.shiftKey && !enableLineBreaks ) {\r\n\r\n editor.callback.enterPressedOnBlock(editor.content.currentBlock, event);\r\n event.preventDefault();\r\n return;\r\n\r\n }\r\n\r\n /**\r\n * Workaround situation when caret at the Text node that has some wrapper Elements\r\n * Split block cant handle this.\r\n * We need to save default behavior\r\n */\r\n isTextNodeHasParentBetweenContenteditable = currentSelectedNode && currentSelectedNode.parentNode.contentEditable != 'true';\r\n\r\n /**\r\n * Split blocks when input has several nodes and caret placed in textNode\r\n */\r\n if (\r\n currentSelectedNode.nodeType == editor.core.nodeTypes.TEXT &&\r\n !isTextNodeHasParentBetweenContenteditable &&\r\n !caretAtTheEndOfText\r\n ) {\r\n\r\n event.preventDefault();\r\n\r\n editor.core.log('Splitting Text node...');\r\n\r\n editor.content.splitBlock(currentInputIndex);\r\n\r\n /** Show plus button when next input after split is empty*/\r\n if (!editor.state.inputs[currentInputIndex + 1].textContent.trim()) {\r\n\r\n editor.toolbar.showPlusButton();\r\n\r\n }\r\n\r\n } else {\r\n\r\n var islastNode = editor.content.isLastNode(currentSelectedNode);\r\n\r\n if ( islastNode && caretAtTheEndOfText ) {\r\n\r\n event.preventDefault();\r\n event.stopPropagation();\r\n event.stopImmediatePropagation();\r\n\r\n editor.core.log('ENTER clicked in last textNode. Create new BLOCK');\r\n\r\n editor.content.insertBlock({\r\n type: NEW_BLOCK_TYPE,\r\n block: editor.tools[NEW_BLOCK_TYPE].render()\r\n }, true);\r\n\r\n editor.toolbar.move();\r\n editor.toolbar.open();\r\n\r\n /** Show plus button with empty block */\r\n editor.toolbar.showPlusButton();\r\n\r\n }\r\n\r\n }\r\n\r\n /** get all inputs after new appending block */\r\n editor.ui.saveInputs();\r\n\r\n };\r\n\r\n /**\r\n * Escape behaviour\r\n * @param event\r\n * @private\r\n *\r\n * @description Closes toolbox and toolbar. Prevents default behaviour\r\n */\r\n var escapeKeyPressedOnRedactorsZone_ = function (event) {\r\n\r\n /** Close all toolbar */\r\n editor.toolbar.close();\r\n\r\n /** Close toolbox */\r\n editor.toolbar.toolbox.close();\r\n\r\n event.preventDefault();\r\n\r\n };\r\n\r\n /**\r\n * @param {Event} event\r\n * @private\r\n *\r\n * closes and moves toolbar\r\n */\r\n var arrowKeyPressed_ = function (event) {\r\n\r\n editor.content.workingNodeChanged();\r\n\r\n /* Closing toolbar */\r\n editor.toolbar.close();\r\n editor.toolbar.move();\r\n\r\n };\r\n\r\n /**\r\n * @private\r\n * @param {Event} event\r\n *\r\n * @description Closes all opened bars from toolbar.\r\n * If block is mark, clears highlightning\r\n */\r\n var defaultKeyPressedOnRedactorsZone_ = function () {\r\n\r\n editor.toolbar.close();\r\n\r\n if (!editor.toolbar.inline.actionsOpened) {\r\n\r\n editor.toolbar.inline.close();\r\n editor.content.clearMark();\r\n\r\n }\r\n\r\n };\r\n\r\n /**\r\n * Handler when clicked on redactors area\r\n *\r\n * @protected\r\n * @param event\r\n *\r\n * @description Detects clicked area. If it is first-level block area, marks as detected and\r\n * on next enter press will be inserted new block\r\n * Otherwise, save carets position (input index) and put caret to the editable zone.\r\n *\r\n * @see detectWhenClickedOnFirstLevelBlockArea_\r\n *\r\n */\r\n callbacks.redactorClicked = function (event) {\r\n\r\n detectWhenClickedOnFirstLevelBlockArea_();\r\n\r\n editor.content.workingNodeChanged(event.target);\r\n editor.ui.saveInputs();\r\n\r\n var selectedText = editor.toolbar.inline.getSelectionText(),\r\n firstLevelBlock;\r\n\r\n /** If selection range took off, then we hide inline toolbar */\r\n if (selectedText.length === 0) {\r\n\r\n editor.toolbar.inline.close();\r\n\r\n }\r\n\r\n /** Update current input index in memory when caret focused into existed input */\r\n if (event.target.contentEditable == 'true') {\r\n\r\n editor.caret.saveCurrentInputIndex();\r\n\r\n }\r\n\r\n if (editor.content.currentNode === null) {\r\n\r\n /**\r\n * If inputs in redactor does not exits, then we put input index 0 not -1\r\n */\r\n var indexOfLastInput = editor.state.inputs.length > 0 ? editor.state.inputs.length - 1 : 0;\r\n\r\n /** If we have any inputs */\r\n if (editor.state.inputs.length) {\r\n\r\n /** getting firstlevel parent of input */\r\n firstLevelBlock = editor.content.getFirstLevelBlock(editor.state.inputs[indexOfLastInput]);\r\n\r\n }\r\n\r\n /** If input is empty, then we set caret to the last input */\r\n if (editor.state.inputs.length && editor.state.inputs[indexOfLastInput].textContent === '' && firstLevelBlock.dataset.tool == editor.settings.initialBlockPlugin) {\r\n\r\n editor.caret.setToBlock(indexOfLastInput);\r\n\r\n } else {\r\n\r\n /** Create new input when caret clicked in redactors area */\r\n var NEW_BLOCK_TYPE = editor.settings.initialBlockPlugin;\r\n\r\n editor.content.insertBlock({\r\n type : NEW_BLOCK_TYPE,\r\n block : editor.tools[NEW_BLOCK_TYPE].render()\r\n });\r\n\r\n /** If there is no inputs except inserted */\r\n if (editor.state.inputs.length === 1) {\r\n\r\n editor.caret.setToBlock(indexOfLastInput);\r\n\r\n } else {\r\n\r\n /** Set caret to this appended input */\r\n editor.caret.setToNextBlock(indexOfLastInput);\r\n\r\n }\r\n\r\n }\r\n\r\n } else {\r\n\r\n /** Close all panels */\r\n editor.toolbar.settings.close();\r\n editor.toolbar.toolbox.close();\r\n\r\n }\r\n\r\n /**\r\n * Move toolbar and open\r\n */\r\n editor.toolbar.move();\r\n editor.toolbar.open();\r\n\r\n var inputIsEmpty = !editor.content.currentNode.textContent.trim(),\r\n currentNodeType = editor.content.currentNode.dataset.tool,\r\n isInitialType = currentNodeType == editor.settings.initialBlockPlugin;\r\n\r\n\r\n /** Hide plus buttons */\r\n editor.toolbar.hidePlusButton();\r\n\r\n if (!inputIsEmpty) {\r\n\r\n /** Mark current block */\r\n editor.content.markBlock();\r\n\r\n }\r\n\r\n if ( isInitialType && inputIsEmpty ) {\r\n\r\n /** Show plus button */\r\n editor.toolbar.showPlusButton();\r\n\r\n }\r\n\r\n\r\n };\r\n\r\n /**\r\n * This method allows to define, is caret in contenteditable element or not.\r\n *\r\n * @private\r\n *\r\n * @description Otherwise, if we get TEXT node from range container, that will means we have input index.\r\n * In this case we use default browsers behaviour (if plugin allows that) or overwritten action.\r\n * Therefore, to be sure that we've clicked first-level block area, we should have currentNode, which always\r\n * specifies to the first-level block. Other cases we just ignore.\r\n */\r\n var detectWhenClickedOnFirstLevelBlockArea_ = function () {\r\n\r\n var selection = window.getSelection(),\r\n anchorNode = selection.anchorNode,\r\n flag = false;\r\n\r\n if (selection.rangeCount === 0) {\r\n\r\n editor.content.editorAreaHightlighted = true;\r\n\r\n } else {\r\n\r\n if (!editor.core.isDomNode(anchorNode)) {\r\n\r\n anchorNode = anchorNode.parentNode;\r\n\r\n }\r\n\r\n /** Already founded, without loop */\r\n if (anchorNode.contentEditable == 'true') {\r\n\r\n flag = true;\r\n\r\n }\r\n\r\n while (anchorNode.contentEditable != 'true') {\r\n\r\n anchorNode = anchorNode.parentNode;\r\n\r\n if (anchorNode.contentEditable == 'true') {\r\n\r\n flag = true;\r\n\r\n }\r\n\r\n if (anchorNode == document.body) {\r\n\r\n break;\r\n\r\n }\r\n\r\n }\r\n\r\n /** If editable element founded, flag is \"TRUE\", Therefore we return \"FALSE\" */\r\n editor.content.editorAreaHightlighted = !flag;\r\n\r\n }\r\n\r\n };\r\n\r\n /**\r\n * Toolbar button click handler\r\n *\r\n * @param {Object} event - cursor to the button\r\n * @protected\r\n *\r\n * @description gets current tool and calls render method\r\n */\r\n callbacks.toolbarButtonClicked = function (event) {\r\n\r\n var button = this;\r\n\r\n editor.toolbar.current = button.dataset.type;\r\n\r\n editor.toolbar.toolbox.toolClicked(event);\r\n editor.toolbar.close();\r\n\r\n };\r\n\r\n /**\r\n * Show or Hide toolbox when plus button is clicked\r\n */\r\n callbacks.plusButtonClicked = function () {\r\n\r\n if (!editor.nodes.toolbox.classList.contains('opened')) {\r\n\r\n editor.toolbar.toolbox.open();\r\n\r\n } else {\r\n\r\n editor.toolbar.toolbox.close();\r\n\r\n }\r\n\r\n };\r\n\r\n /**\r\n * Block handlers for KeyDown events\r\n *\r\n * @protected\r\n * @param {Object} event\r\n *\r\n * Handles keydowns on block\r\n * @see blockRightOrDownArrowPressed_\r\n * @see backspacePressed_\r\n * @see blockLeftOrUpArrowPressed_\r\n */\r\n callbacks.blockKeydown = function (event) {\r\n\r\n let block = event.target; // event.target is input\r\n\r\n switch (event.keyCode) {\r\n\r\n case editor.core.keys.DOWN:\r\n case editor.core.keys.RIGHT:\r\n blockRightOrDownArrowPressed_(event);\r\n break;\r\n\r\n case editor.core.keys.BACKSPACE:\r\n backspacePressed_(block, event);\r\n break;\r\n\r\n case editor.core.keys.UP:\r\n case editor.core.keys.LEFT:\r\n blockLeftOrUpArrowPressed_(event);\r\n break;\r\n\r\n }\r\n\r\n };\r\n\r\n /**\r\n * RIGHT or DOWN keydowns on block\r\n *\r\n * @param {Object} event\r\n * @private\r\n *\r\n * @description watches the selection and gets closest editable element.\r\n * Uses method getDeepestTextNodeFromPosition to get the last node of next block\r\n * Sets caret if it is contenteditable\r\n */\r\n var blockRightOrDownArrowPressed_ = function (event) {\r\n\r\n var selection = window.getSelection(),\r\n inputs = editor.state.inputs,\r\n focusedNode = selection.anchorNode,\r\n focusedNodeHolder;\r\n\r\n /** Check for caret existance */\r\n if (!focusedNode) {\r\n\r\n return false;\r\n\r\n }\r\n\r\n /** Looking for closest (parent) contentEditable element of focused node */\r\n while (focusedNode.contentEditable != 'true') {\r\n\r\n focusedNodeHolder = focusedNode.parentNode;\r\n focusedNode = focusedNodeHolder;\r\n\r\n }\r\n\r\n /** Input index in DOM level */\r\n var editableElementIndex = 0;\r\n\r\n while (focusedNode != inputs[editableElementIndex]) {\r\n\r\n editableElementIndex ++;\r\n\r\n }\r\n\r\n /**\r\n * Founded contentEditable element doesn't have childs\r\n * Or maybe New created block\r\n */\r\n if (!focusedNode.textContent) {\r\n\r\n editor.caret.setToNextBlock(editableElementIndex);\r\n return;\r\n\r\n }\r\n\r\n /**\r\n * Do nothing when caret doesn not reaches the end of last child\r\n */\r\n var caretInLastChild = false,\r\n caretAtTheEndOfText = false;\r\n\r\n var lastChild,\r\n deepestTextnode;\r\n\r\n lastChild = focusedNode.childNodes[focusedNode.childNodes.length - 1 ];\r\n\r\n if (editor.core.isDomNode(lastChild)) {\r\n\r\n deepestTextnode = editor.content.getDeepestTextNodeFromPosition(lastChild, lastChild.childNodes.length);\r\n\r\n } else {\r\n\r\n deepestTextnode = lastChild;\r\n\r\n }\r\n\r\n caretInLastChild = selection.anchorNode == deepestTextnode;\r\n caretAtTheEndOfText = deepestTextnode.length == selection.anchorOffset;\r\n\r\n if ( !caretInLastChild || !caretAtTheEndOfText ) {\r\n\r\n editor.core.log('arrow [down|right] : caret does not reached the end');\r\n return false;\r\n\r\n }\r\n\r\n editor.caret.setToNextBlock(editableElementIndex);\r\n\r\n };\r\n\r\n /**\r\n * LEFT or UP keydowns on block\r\n *\r\n * @param {Object} event\r\n * @private\r\n *\r\n * watches the selection and gets closest editable element.\r\n * Uses method getDeepestTextNodeFromPosition to get the last node of previous block\r\n * Sets caret if it is contenteditable\r\n *\r\n */\r\n var blockLeftOrUpArrowPressed_ = function (event) {\r\n\r\n var selection = window.getSelection(),\r\n inputs = editor.state.inputs,\r\n focusedNode = selection.anchorNode,\r\n focusedNodeHolder;\r\n\r\n /** Check for caret existance */\r\n if (!focusedNode) {\r\n\r\n return false;\r\n\r\n }\r\n\r\n /**\r\n * LEFT or UP not at the beginning\r\n */\r\n if ( selection.anchorOffset !== 0) {\r\n\r\n return false;\r\n\r\n }\r\n\r\n /** Looking for parent contentEditable block */\r\n while (focusedNode.contentEditable != 'true') {\r\n\r\n focusedNodeHolder = focusedNode.parentNode;\r\n focusedNode = focusedNodeHolder;\r\n\r\n }\r\n\r\n /** Input index in DOM level */\r\n var editableElementIndex = 0;\r\n\r\n while (focusedNode != inputs[editableElementIndex]) {\r\n\r\n editableElementIndex ++;\r\n\r\n }\r\n\r\n /**\r\n * Do nothing if caret is not at the beginning of first child\r\n */\r\n var caretInFirstChild = false,\r\n caretAtTheBeginning = false;\r\n\r\n var firstChild,\r\n deepestTextnode;\r\n\r\n /**\r\n * Founded contentEditable element doesn't have childs\r\n * Or maybe New created block\r\n */\r\n if (!focusedNode.textContent) {\r\n\r\n editor.caret.setToPreviousBlock(editableElementIndex);\r\n return;\r\n\r\n }\r\n\r\n firstChild = focusedNode.childNodes[0];\r\n\r\n if (editor.core.isDomNode(firstChild)) {\r\n\r\n deepestTextnode = editor.content.getDeepestTextNodeFromPosition(firstChild, 0);\r\n\r\n } else {\r\n\r\n deepestTextnode = firstChild;\r\n\r\n }\r\n\r\n caretInFirstChild = selection.anchorNode == deepestTextnode;\r\n caretAtTheBeginning = selection.anchorOffset === 0;\r\n\r\n if ( caretInFirstChild && caretAtTheBeginning ) {\r\n\r\n editor.caret.setToPreviousBlock(editableElementIndex);\r\n\r\n }\r\n\r\n };\r\n\r\n /**\r\n * Handles backspace keydown\r\n *\r\n * @param {Element} block\r\n * @param {Object} event\r\n * @private\r\n *\r\n * @description if block is empty, delete the block and set caret to the previous block\r\n * If block is not empty, try to merge two blocks - current and previous\r\n * But it we try'n to remove first block, then we should set caret to the next block, not previous.\r\n * If we removed the last block, create new one\r\n */\r\n var backspacePressed_ = function (block, event) {\r\n\r\n var currentInputIndex = editor.caret.getCurrentInputIndex(),\r\n range,\r\n selectionLength,\r\n firstLevelBlocksCount;\r\n\r\n if (isNativeInput_(event.target)) {\r\n\r\n /** If input value is empty - remove block */\r\n if (event.target.value.trim() == '') {\r\n\r\n block.remove();\r\n\r\n } else {\r\n\r\n return;\r\n\r\n }\r\n\r\n }\r\n\r\n if (block.textContent.trim()) {\r\n\r\n range = editor.content.getRange();\r\n selectionLength = range.endOffset - range.startOffset;\r\n\r\n if (editor.caret.position.atStart() && !selectionLength && editor.state.inputs[currentInputIndex - 1]) {\r\n\r\n editor.content.mergeBlocks(currentInputIndex);\r\n\r\n } else {\r\n\r\n return;\r\n\r\n }\r\n\r\n }\r\n\r\n if (!selectionLength) {\r\n\r\n block.remove();\r\n\r\n }\r\n\r\n\r\n firstLevelBlocksCount = editor.nodes.redactor.childNodes.length;\r\n\r\n /**\r\n * If all blocks are removed\r\n */\r\n if (firstLevelBlocksCount === 0) {\r\n\r\n /** update currentNode variable */\r\n editor.content.currentNode = null;\r\n\r\n /** Inserting new empty initial block */\r\n editor.ui.addInitialBlock();\r\n\r\n /** Updating inputs state after deleting last block */\r\n editor.ui.saveInputs();\r\n\r\n /** Set to current appended block */\r\n window.setTimeout(function () {\r\n\r\n editor.caret.setToPreviousBlock(1);\r\n\r\n }, 10);\r\n\r\n } else {\r\n\r\n if (editor.caret.inputIndex !== 0) {\r\n\r\n /** Target block is not first */\r\n editor.caret.setToPreviousBlock(editor.caret.inputIndex);\r\n\r\n } else {\r\n\r\n /** If we try to delete first block */\r\n editor.caret.setToNextBlock(editor.caret.inputIndex);\r\n\r\n }\r\n\r\n }\r\n\r\n editor.toolbar.move();\r\n\r\n if (!editor.toolbar.opened) {\r\n\r\n editor.toolbar.open();\r\n\r\n }\r\n\r\n /** Updating inputs state */\r\n editor.ui.saveInputs();\r\n\r\n /** Prevent default browser behaviour */\r\n event.preventDefault();\r\n\r\n };\r\n\r\n /**\r\n * This method prevents default behaviour.\r\n *\r\n * @param {Object} event\r\n * @protected\r\n *\r\n * @description We get from clipboard pasted data, sanitize, make a fragment that contains of this sanitized nodes.\r\n * Firstly, we need to memorize the caret position. We can do that by getting the range of selection.\r\n * After all, we insert clear fragment into caret placed position. Then, we should move the caret to the last node\r\n */\r\n callbacks.blockPasteCallback = function (event) {\r\n\r\n /** If area is input or textarea then allow default behaviour */\r\n if ( isNativeInput_(event.target) ) {\r\n\r\n return;\r\n\r\n }\r\n\r\n /** Prevent default behaviour */\r\n event.preventDefault();\r\n\r\n var editableParent = editor.content.getEditableParent(event.target);\r\n\r\n /** Allow paste when event target placed in Editable element */\r\n if (!editableParent) {\r\n\r\n return;\r\n\r\n }\r\n\r\n /** get html pasted data - dirty data */\r\n var htmlData = event.clipboardData.getData('text/html'),\r\n plainData = event.clipboardData.getData('text/plain');\r\n\r\n\r\n /** Temporary DIV that is used to work with childs as arrays item */\r\n var div = editor.draw.node('DIV', '', {}),\r\n cleanData,\r\n wrappedData;\r\n\r\n /** Create fragment, that we paste to range after proccesing */\r\n cleanData = editor.sanitizer.clean(htmlData);\r\n\r\n\r\n /**\r\n * We wrap pasted text with

    tags to split it logically\r\n *\r\n * @type {string}\r\n */\r\n wrappedData = editor.content.wrapTextWithParagraphs(cleanData, plainData);\r\n div.innerHTML = wrappedData;\r\n\r\n var NEW_BLOCK_TYPE = editor.settings.initialBlockPlugin,\r\n currentBlockContent = editor.content.currentNode.firstChild.firstChild;\r\n\r\n /**\r\n * If there only one paragraph, just insert it\r\n */\r\n if (div.childNodes.length == 1) {\r\n\r\n editor.caret.insertNode(document.createTextNode(div.firstChild.innerHTML));\r\n return;\r\n\r\n }\r\n\r\n div.childNodes.forEach(function (paragraph, index) {\r\n\r\n\r\n /**\r\n * If there was no data in working node, replace it with first paragraph of pasted text\r\n */\r\n if (index == 0 && currentBlockContent.innerHTML.trim() === '') {\r\n\r\n editor.content.switchBlock(editor.content.currentNode, editor.tools[NEW_BLOCK_TYPE].render({\r\n text : paragraph.innerHTML\r\n }), NEW_BLOCK_TYPE);\r\n\r\n return;\r\n\r\n }\r\n\r\n editor.content.insertBlock({\r\n type : NEW_BLOCK_TYPE,\r\n block : editor.tools[NEW_BLOCK_TYPE].render({\r\n text : paragraph.innerHTML\r\n })\r\n });\r\n\r\n editor.caret.inputIndex++;\r\n\r\n });\r\n\r\n editor.caret.setToPreviousBlock(editor.caret.getCurrentInputIndex() + 1);\r\n\r\n\r\n };\r\n\r\n /**\r\n * used by UI module\r\n * Clicks on block settings button\r\n *\r\n * @param {Object} event\r\n * @protected\r\n * @description Opens toolbar settings\r\n */\r\n callbacks.showSettingsButtonClicked = function (event) {\r\n\r\n /**\r\n * Get type of current block\r\n * It uses to append settings from tool.settings property.\r\n * ...\r\n * Type is stored in data-type attribute on block\r\n */\r\n var currentToolType = editor.content.currentNode.dataset.tool;\r\n\r\n editor.toolbar.settings.toggle(currentToolType);\r\n\r\n /** Close toolbox when settings button is active */\r\n editor.toolbar.toolbox.close();\r\n editor.toolbar.settings.hideRemoveActions();\r\n\r\n };\r\n\r\n /**\r\n * Check block\r\n * @param target\r\n * @private\r\n *\r\n * @description Checks target is it native input\r\n */\r\n var isNativeInput_ = function (target) {\r\n\r\n var nativeInputAreas = ['INPUT', 'TEXTAREA'];\r\n\r\n return (nativeInputAreas.indexOf(target.tagName) != -1);\r\n\r\n };\r\n\r\n return callbacks;\r\n\r\n})({});\n\n\n// WEBPACK FOOTER //\n// ./modules/callbacks.js","/**\r\n * Codex Editor Draw module\r\n *\r\n * @author Codex Team\r\n * @version 1.0.\r\n */\r\n\r\nmodule.exports = (function (draw) {\r\n\r\n /**\r\n * Base editor wrapper\r\n */\r\n draw.wrapper = function () {\r\n\r\n var wrapper = document.createElement('div');\r\n\r\n wrapper.className += 'codex-editor';\r\n\r\n return wrapper;\r\n\r\n };\r\n\r\n /**\r\n * Content-editable holder\r\n */\r\n draw.redactor = function () {\r\n\r\n var redactor = document.createElement('div');\r\n\r\n redactor.className += 'ce-redactor';\r\n\r\n return redactor;\r\n\r\n };\r\n\r\n draw.ceBlock = function () {\r\n\r\n var block = document.createElement('DIV');\r\n\r\n block.className += 'ce_block';\r\n\r\n return block;\r\n\r\n };\r\n\r\n /**\r\n * Empty toolbar with toggler\r\n */\r\n draw.toolbar = function () {\r\n\r\n var bar = document.createElement('div');\r\n\r\n bar.className += 'ce-toolbar';\r\n\r\n return bar;\r\n\r\n };\r\n\r\n draw.toolbarContent = function () {\r\n\r\n var wrapper = document.createElement('DIV');\r\n\r\n wrapper.classList.add('ce-toolbar__content');\r\n\r\n return wrapper;\r\n\r\n };\r\n\r\n /**\r\n * Inline toolbar\r\n */\r\n draw.inlineToolbar = function () {\r\n\r\n var bar = document.createElement('DIV');\r\n\r\n bar.className += 'ce-toolbar-inline';\r\n\r\n return bar;\r\n\r\n };\r\n\r\n /**\r\n * Wrapper for inline toobar buttons\r\n */\r\n draw.inlineToolbarButtons = function () {\r\n\r\n var wrapper = document.createElement('DIV');\r\n\r\n wrapper.className += 'ce-toolbar-inline__buttons';\r\n\r\n return wrapper;\r\n\r\n };\r\n\r\n /**\r\n * For some actions\r\n */\r\n draw.inlineToolbarActions = function () {\r\n\r\n var wrapper = document.createElement('DIV');\r\n\r\n wrapper.className += 'ce-toolbar-inline__actions';\r\n\r\n return wrapper;\r\n\r\n };\r\n\r\n draw.inputForLink = function () {\r\n\r\n var input = document.createElement('INPUT');\r\n\r\n input.type = 'input';\r\n input.className += 'inputForLink';\r\n input.placeholder = 'Вставьте ссылку ...';\r\n input.setAttribute('form', 'defaultForm');\r\n\r\n input.setAttribute('autofocus', 'autofocus');\r\n\r\n return input;\r\n\r\n };\r\n\r\n /**\r\n * @todo Desc\r\n */\r\n draw.blockButtons = function () {\r\n\r\n var block = document.createElement('div');\r\n\r\n block.className += 'ce-toolbar__actions';\r\n\r\n return block;\r\n\r\n };\r\n\r\n /**\r\n * Block settings panel\r\n */\r\n draw.blockSettings = function () {\r\n\r\n var settings = document.createElement('div');\r\n\r\n settings.className += 'ce-settings';\r\n\r\n return settings;\r\n\r\n };\r\n\r\n draw.defaultSettings = function () {\r\n\r\n var div = document.createElement('div');\r\n\r\n div.classList.add('ce-settings_default');\r\n\r\n return div;\r\n\r\n };\r\n\r\n draw.pluginsSettings = function () {\r\n\r\n var div = document.createElement('div');\r\n\r\n div.classList.add('ce-settings_plugin');\r\n\r\n return div;\r\n\r\n };\r\n\r\n draw.plusButton = function () {\r\n\r\n var button = document.createElement('span');\r\n\r\n button.className = 'ce-toolbar__plus';\r\n // button.innerHTML = '';\r\n\r\n return button;\r\n\r\n };\r\n\r\n /**\r\n * Settings button in toolbar\r\n */\r\n draw.settingsButton = function () {\r\n\r\n var toggler = document.createElement('span');\r\n\r\n toggler.className = 'ce-toolbar__settings-btn';\r\n\r\n /** Toggler button*/\r\n toggler.innerHTML = '';\r\n\r\n return toggler;\r\n\r\n };\r\n\r\n /**\r\n * Redactor tools wrapper\r\n */\r\n\r\n draw.toolbox = function () {\r\n\r\n var wrapper = document.createElement('div');\r\n\r\n wrapper.className = 'ce-toolbar__tools';\r\n\r\n return wrapper;\r\n\r\n };\r\n\r\n /**\r\n * @protected\r\n *\r\n * Draws tool buttons for toolbox\r\n *\r\n * @param {String} type\r\n * @param {String} classname\r\n * @returns {Element}\r\n */\r\n draw.toolbarButton = function (type, classname) {\r\n\r\n var button = document.createElement('li'),\r\n toolIcon = document.createElement('i'),\r\n toolTitle = document.createElement('span');\r\n\r\n button.dataset.type = type;\r\n button.setAttribute('title', type);\r\n\r\n toolIcon.classList.add(classname);\r\n toolTitle.classList.add('ce_toolbar_tools--title');\r\n\r\n\r\n button.appendChild(toolIcon);\r\n button.appendChild(toolTitle);\r\n\r\n return button;\r\n\r\n };\r\n\r\n /**\r\n * @protected\r\n *\r\n * Draws tools for inline toolbar\r\n *\r\n * @param {String} type\r\n * @param {String} classname\r\n */\r\n draw.toolbarButtonInline = function (type, classname) {\r\n\r\n var button = document.createElement('BUTTON'),\r\n toolIcon = document.createElement('I');\r\n\r\n button.type = 'button';\r\n button.dataset.type = type;\r\n toolIcon.classList.add(classname);\r\n\r\n button.appendChild(toolIcon);\r\n\r\n return button;\r\n\r\n };\r\n\r\n /**\r\n * Redactor block\r\n */\r\n draw.block = function (tagName, content) {\r\n\r\n var node = document.createElement(tagName);\r\n\r\n node.innerHTML = content || '';\r\n\r\n return node;\r\n\r\n };\r\n\r\n /**\r\n * Creates Node with passed tagName and className\r\n * @param {string} tagName\r\n * @param {string} className\r\n * @param {object} properties - allow to assign properties\r\n */\r\n draw.node = function ( tagName, className, properties ) {\r\n\r\n var el = document.createElement( tagName );\r\n\r\n if ( className ) el.className = className;\r\n\r\n if ( properties ) {\r\n\r\n for (var name in properties) {\r\n\r\n el[name] = properties[name];\r\n\r\n }\r\n\r\n }\r\n\r\n return el;\r\n\r\n };\r\n\r\n /**\r\n * Unavailable plugin block\r\n */\r\n draw.unavailableBlock = function () {\r\n\r\n var wrapper = document.createElement('DIV');\r\n\r\n wrapper.classList.add('cdx-unavailable-block');\r\n\r\n return wrapper;\r\n\r\n };\r\n\r\n return draw;\r\n\r\n})({});\n\n\n// WEBPACK FOOTER //\n// ./modules/draw.js","/**\r\n * Codex Editor Caret Module\r\n *\r\n * @author Codex Team\r\n * @version 1.0\r\n */\r\n\r\nmodule.exports = (function (caret) {\r\n\r\n let editor = codex.editor;\r\n\r\n /**\r\n * @var {int} InputIndex - editable element in DOM\r\n */\r\n caret.inputIndex = null;\r\n\r\n /**\r\n * @var {int} offset - caret position in a text node.\r\n */\r\n caret.offset = null;\r\n\r\n /**\r\n * @var {int} focusedNodeIndex - we get index of child node from first-level block\r\n */\r\n caret.focusedNodeIndex = null;\r\n\r\n /**\r\n * Creates Document Range and sets caret to the element.\r\n * @protected\r\n * @uses caret.save — if you need to save caret position\r\n * @param {Element} el - Changed Node.\r\n */\r\n caret.set = function ( el, index, offset) {\r\n\r\n offset = offset || caret.offset || 0;\r\n index = index || caret.focusedNodeIndex || 0;\r\n\r\n var childs = el.childNodes,\r\n nodeToSet;\r\n\r\n if ( childs.length === 0 ) {\r\n\r\n nodeToSet = el;\r\n\r\n } else {\r\n\r\n nodeToSet = childs[index];\r\n\r\n }\r\n\r\n /** If Element is INPUT */\r\n if (el.contentEditable != 'true') {\r\n\r\n el.focus();\r\n return;\r\n\r\n }\r\n\r\n if (editor.core.isDomNode(nodeToSet)) {\r\n\r\n nodeToSet = editor.content.getDeepestTextNodeFromPosition(nodeToSet, nodeToSet.childNodes.length);\r\n\r\n }\r\n\r\n var range = document.createRange(),\r\n selection = window.getSelection();\r\n\r\n window.setTimeout(function () {\r\n\r\n range.setStart(nodeToSet, offset);\r\n range.setEnd(nodeToSet, offset);\r\n\r\n selection.removeAllRanges();\r\n selection.addRange(range);\r\n\r\n editor.caret.saveCurrentInputIndex();\r\n\r\n }, 20);\r\n\r\n };\r\n\r\n /**\r\n * @protected\r\n * Updates index of input and saves it in caret object\r\n */\r\n caret.saveCurrentInputIndex = function () {\r\n\r\n /** Index of Input that we paste sanitized content */\r\n var selection = window.getSelection(),\r\n inputs = editor.state.inputs,\r\n focusedNode = selection.anchorNode,\r\n focusedNodeHolder;\r\n\r\n if (!focusedNode) {\r\n\r\n return;\r\n\r\n }\r\n\r\n /** Looking for parent contentEditable block */\r\n while (focusedNode.contentEditable != 'true') {\r\n\r\n focusedNodeHolder = focusedNode.parentNode;\r\n focusedNode = focusedNodeHolder;\r\n\r\n }\r\n\r\n /** Input index in DOM level */\r\n var editableElementIndex = 0;\r\n\r\n while (focusedNode != inputs[editableElementIndex]) {\r\n\r\n editableElementIndex ++;\r\n\r\n }\r\n\r\n caret.inputIndex = editableElementIndex;\r\n\r\n };\r\n\r\n /**\r\n * Returns current input index (caret object)\r\n */\r\n caret.getCurrentInputIndex = function () {\r\n\r\n return caret.inputIndex;\r\n\r\n };\r\n\r\n /**\r\n * @param {int} index - index of first-level block after that we set caret into next input\r\n */\r\n caret.setToNextBlock = function (index) {\r\n\r\n var inputs = editor.state.inputs,\r\n nextInput = inputs[index + 1];\r\n\r\n if (!nextInput) {\r\n\r\n editor.core.log('We are reached the end');\r\n return;\r\n\r\n }\r\n\r\n /**\r\n * When new Block created or deleted content of input\r\n * We should add some text node to set caret\r\n */\r\n if (!nextInput.childNodes.length) {\r\n\r\n var emptyTextElement = document.createTextNode('');\r\n\r\n nextInput.appendChild(emptyTextElement);\r\n\r\n }\r\n\r\n editor.caret.inputIndex = index + 1;\r\n editor.caret.set(nextInput, 0, 0);\r\n editor.content.workingNodeChanged(nextInput);\r\n\r\n };\r\n\r\n /**\r\n * @param {int} index - index of target input.\r\n * Sets caret to input with this index\r\n */\r\n caret.setToBlock = function (index) {\r\n\r\n var inputs = editor.state.inputs,\r\n targetInput = inputs[index];\r\n\r\n if ( !targetInput ) {\r\n\r\n return;\r\n\r\n }\r\n\r\n /**\r\n * When new Block created or deleted content of input\r\n * We should add some text node to set caret\r\n */\r\n if (!targetInput.childNodes.length) {\r\n\r\n var emptyTextElement = document.createTextNode('');\r\n\r\n targetInput.appendChild(emptyTextElement);\r\n\r\n }\r\n\r\n editor.caret.inputIndex = index;\r\n editor.caret.set(targetInput, 0, 0);\r\n editor.content.workingNodeChanged(targetInput);\r\n\r\n };\r\n\r\n /**\r\n * @param {int} index - index of input\r\n */\r\n caret.setToPreviousBlock = function (index) {\r\n\r\n index = index || 0;\r\n\r\n var inputs = editor.state.inputs,\r\n previousInput = inputs[index - 1],\r\n lastChildNode,\r\n lengthOfLastChildNode,\r\n emptyTextElement;\r\n\r\n\r\n if (!previousInput) {\r\n\r\n editor.core.log('We are reached first node');\r\n return;\r\n\r\n }\r\n\r\n lastChildNode = editor.content.getDeepestTextNodeFromPosition(previousInput, previousInput.childNodes.length);\r\n lengthOfLastChildNode = lastChildNode.length;\r\n\r\n /**\r\n * When new Block created or deleted content of input\r\n * We should add some text node to set caret\r\n */\r\n if (!previousInput.childNodes.length) {\r\n\r\n emptyTextElement = document.createTextNode('');\r\n previousInput.appendChild(emptyTextElement);\r\n\r\n }\r\n editor.caret.inputIndex = index - 1;\r\n editor.caret.set(previousInput, previousInput.childNodes.length - 1, lengthOfLastChildNode);\r\n editor.content.workingNodeChanged(inputs[index - 1]);\r\n\r\n };\r\n\r\n caret.position = {\r\n\r\n atStart : function () {\r\n\r\n var selection = window.getSelection(),\r\n anchorOffset = selection.anchorOffset,\r\n anchorNode = selection.anchorNode,\r\n firstLevelBlock = editor.content.getFirstLevelBlock(anchorNode),\r\n pluginsRender = firstLevelBlock.childNodes[0];\r\n\r\n if (!editor.core.isDomNode(anchorNode)) {\r\n\r\n anchorNode = anchorNode.parentNode;\r\n\r\n }\r\n\r\n var isFirstNode = anchorNode === pluginsRender.childNodes[0],\r\n isOffsetZero = anchorOffset === 0;\r\n\r\n return isFirstNode && isOffsetZero;\r\n\r\n },\r\n\r\n atTheEnd : function () {\r\n\r\n var selection = window.getSelection(),\r\n anchorOffset = selection.anchorOffset,\r\n anchorNode = selection.anchorNode;\r\n\r\n /** Caret is at the end of input */\r\n return !anchorNode || !anchorNode.length || anchorOffset === anchorNode.length;\r\n\r\n }\r\n };\r\n\r\n\r\n /**\r\n * Inserts node at the caret location\r\n * @param node\r\n */\r\n caret.insertNode = function (node) {\r\n\r\n var selection, range;\r\n\r\n selection = window.getSelection();\r\n\r\n range = selection.getRangeAt(0);\r\n range.deleteContents();\r\n\r\n range.insertNode(node);\r\n\r\n range.setStartAfter(node);\r\n range.collapse(true);\r\n\r\n selection.removeAllRanges();\r\n selection.addRange(range);\r\n\r\n\r\n };\r\n\r\n return caret;\r\n\r\n})({});\n\n\n// WEBPACK FOOTER //\n// ./modules/caret.js","/**\r\n * Codex Editor Notification Module\r\n *\r\n * @author Codex Team\r\n * @version 1.0\r\n */\r\n\r\nmodule.exports = (function (notifications) {\r\n\r\n let editor = codex.editor;\r\n\r\n var queue = [];\r\n\r\n var addToQueue = function (settings) {\r\n\r\n queue.push(settings);\r\n\r\n var index = 0;\r\n\r\n while ( index < queue.length && queue.length > 5) {\r\n\r\n if (queue[index].type == 'confirm' || queue[index].type == 'prompt') {\r\n\r\n index++;\r\n continue;\r\n\r\n }\r\n\r\n queue[index].close();\r\n queue.splice(index, 1);\r\n\r\n }\r\n\r\n };\r\n\r\n notifications.createHolder = function () {\r\n\r\n var holder = editor.draw.node('DIV', 'cdx-notifications-block');\r\n\r\n editor.nodes.notifications = document.body.appendChild(holder);\r\n\r\n return holder;\r\n\r\n };\r\n\r\n\r\n /**\r\n * Error notificator. Shows block with message\r\n * @protected\r\n */\r\n notifications.errorThrown = function (errorMsg, event) {\r\n\r\n editor.notifications.notification({message: 'This action is not available currently', type: event.type});\r\n\r\n };\r\n\r\n /**\r\n *\r\n * Appends notification\r\n *\r\n * settings = {\r\n * type - notification type (reserved types: alert, confirm, prompt). Just add class 'cdx-notification-'+type\r\n * message - notification message\r\n * okMsg - confirm button text (default - 'Ok')\r\n * cancelBtn - cancel button text (default - 'Cancel'). Only for confirm and prompt types\r\n * confirm - function-handler for ok button click\r\n * cancel - function-handler for cancel button click. Only for confirm and prompt types\r\n * time - time (in seconds) after which notification will close (default - 10s)\r\n * }\r\n *\r\n * @param settings\r\n */\r\n notifications.notification = function (constructorSettings) {\r\n\r\n /** Private vars and methods */\r\n var notification = null,\r\n cancel = null,\r\n type = null,\r\n confirm = null,\r\n inputField = null;\r\n\r\n var confirmHandler = function () {\r\n\r\n close();\r\n\r\n if (typeof confirm !== 'function' ) {\r\n\r\n return;\r\n\r\n }\r\n\r\n if (type == 'prompt') {\r\n\r\n confirm(inputField.value);\r\n return;\r\n\r\n }\r\n\r\n confirm();\r\n\r\n };\r\n\r\n var cancelHandler = function () {\r\n\r\n close();\r\n\r\n if (typeof cancel !== 'function' ) {\r\n\r\n return;\r\n\r\n }\r\n\r\n cancel();\r\n\r\n };\r\n\r\n\r\n /** Public methods */\r\n function create(settings) {\r\n\r\n if (!(settings && settings.message)) {\r\n\r\n editor.core.log('Can\\'t create notification. Message is missed');\r\n return;\r\n\r\n }\r\n\r\n settings.type = settings.type || 'alert';\r\n settings.time = settings.time*1000 || 10000;\r\n\r\n var wrapper = editor.draw.node('DIV', 'cdx-notification'),\r\n message = editor.draw.node('DIV', 'cdx-notification__message'),\r\n input = editor.draw.node('INPUT', 'cdx-notification__input'),\r\n okBtn = editor.draw.node('SPAN', 'cdx-notification__ok-btn'),\r\n cancelBtn = editor.draw.node('SPAN', 'cdx-notification__cancel-btn');\r\n\r\n message.textContent = settings.message;\r\n okBtn.textContent = settings.okMsg || 'ОК';\r\n cancelBtn.textContent = settings.cancelMsg || 'Отмена';\r\n\r\n editor.listeners.add(okBtn, 'click', confirmHandler);\r\n editor.listeners.add(cancelBtn, 'click', cancelHandler);\r\n\r\n wrapper.appendChild(message);\r\n\r\n if (settings.type == 'prompt') {\r\n\r\n wrapper.appendChild(input);\r\n\r\n }\r\n\r\n wrapper.appendChild(okBtn);\r\n\r\n if (settings.type == 'prompt' || settings.type == 'confirm') {\r\n\r\n wrapper.appendChild(cancelBtn);\r\n\r\n }\r\n\r\n wrapper.classList.add('cdx-notification-' + settings.type);\r\n wrapper.dataset.type = settings.type;\r\n\r\n notification = wrapper;\r\n type = settings.type;\r\n confirm = settings.confirm;\r\n cancel = settings.cancel;\r\n inputField = input;\r\n\r\n if (settings.type != 'prompt' && settings.type != 'confirm') {\r\n\r\n window.setTimeout(close, settings.time);\r\n\r\n }\r\n\r\n };\r\n\r\n function send() {\r\n\r\n editor.nodes.notifications.appendChild(notification);\r\n inputField.focus();\r\n\r\n editor.nodes.notifications.classList.add('cdx-notification__notification-appending');\r\n\r\n window.setTimeout(function () {\r\n\r\n editor.nodes.notifications.classList.remove('cdx-notification__notification-appending');\r\n\r\n }, 100);\r\n\r\n addToQueue({type: type, close: close});\r\n\r\n };\r\n\r\n function close() {\r\n\r\n notification.remove();\r\n\r\n };\r\n\r\n\r\n if (constructorSettings) {\r\n\r\n create(constructorSettings);\r\n send();\r\n\r\n }\r\n\r\n return {\r\n create: create,\r\n send: send,\r\n close: close\r\n };\r\n\r\n };\r\n\r\n notifications.clear = function () {\r\n\r\n editor.nodes.notifications.innerHTML = '';\r\n queue = [];\r\n\r\n };\r\n\r\n return notifications;\r\n\r\n})({});\n\n\n// WEBPACK FOOTER //\n// ./modules/notifications.js","/**\r\n * Codex Editor Parser Module\r\n *\r\n * @author Codex Team\r\n * @version 1.1\r\n */\r\n\r\nmodule.exports = (function (parser) {\r\n\r\n let editor = codex.editor;\r\n\r\n /** inserting text */\r\n parser.insertPastedContent = function (blockType, tag) {\r\n\r\n editor.content.insertBlock({\r\n type : blockType.type,\r\n block : blockType.render({\r\n text : tag.innerHTML\r\n })\r\n });\r\n\r\n };\r\n\r\n /**\r\n * Check DOM node for display style: separated block or child-view\r\n */\r\n parser.isFirstLevelBlock = function (node) {\r\n\r\n return node.nodeType == editor.core.nodeTypes.TAG &&\r\n node.classList.contains(editor.ui.className.BLOCK_CLASSNAME);\r\n\r\n };\r\n\r\n return parser;\r\n\r\n})({});\r\n\n\n\n// WEBPACK FOOTER //\n// ./modules/parser.js","/**\r\n * Codex Sanitizer\r\n */\r\n\r\nmodule.exports = (function (sanitizer) {\r\n\r\n /** HTML Janitor library */\r\n let janitor = require('html-janitor');\r\n\r\n /** Codex Editor */\r\n let editor = codex.editor;\r\n\r\n sanitizer.prepare = function () {\r\n\r\n if (editor.settings.sanitizer && !editor.core.isEmpty(editor.settings.sanitizer)) {\r\n\r\n Config.CUSTOM = editor.settings.sanitizer;\r\n\r\n }\r\n\r\n };\r\n\r\n /**\r\n * Basic config\r\n */\r\n var Config = {\r\n\r\n /** User configuration */\r\n CUSTOM : null,\r\n\r\n BASIC : {\r\n\r\n tags: {\r\n p: {},\r\n a: {\r\n href: true,\r\n target: '_blank',\r\n rel: 'nofollow'\r\n },\r\n i: {},\r\n b: {},\r\n strong: {},\r\n em: {},\r\n span: {}\r\n }\r\n }\r\n };\r\n\r\n sanitizer.Config = Config;\r\n\r\n /**\r\n *\r\n * @param userCustomConfig\r\n * @returns {*}\r\n * @private\r\n *\r\n * @description If developer uses editor's API, then he can customize sane restrictions.\r\n * Or, sane config can be defined globally in editors initialization. That config will be used everywhere\r\n * At least, if there is no config overrides, that API uses BASIC Default configation\r\n */\r\n let init_ = function (userCustomConfig) {\r\n\r\n let configuration = userCustomConfig || Config.CUSTOM || Config.BASIC;\r\n\r\n return new janitor(configuration);\r\n\r\n };\r\n\r\n /**\r\n * Cleans string from unwanted tags\r\n * @protected\r\n * @param {String} dirtyString - taint string\r\n * @param {Object} customConfig - allowed tags\r\n */\r\n sanitizer.clean = function (dirtyString, customConfig) {\r\n\r\n let janitorInstance = init_(customConfig);\r\n\r\n return janitorInstance.clean(dirtyString);\r\n\r\n };\r\n\r\n return sanitizer;\r\n\r\n})({});\n\n\n// WEBPACK FOOTER //\n// ./modules/sanitizer.js","(function (root, factory) {\n if (typeof define === 'function' && define.amd) {\n define('html-janitor', factory);\n } else if (typeof exports === 'object') {\n module.exports = factory();\n } else {\n root.HTMLJanitor = factory();\n }\n}(this, function () {\n\n /**\n * @param {Object} config.tags Dictionary of allowed tags.\n * @param {boolean} config.keepNestedBlockElements Default false.\n */\n function HTMLJanitor(config) {\n\n var tagDefinitions = config['tags'];\n var tags = Object.keys(tagDefinitions);\n\n var validConfigValues = tags\n .map(function(k) { return typeof tagDefinitions[k]; })\n .every(function(type) { return type === 'object' || type === 'boolean' || type === 'function'; });\n\n if(!validConfigValues) {\n throw new Error(\"The configuration was invalid\");\n }\n\n this.config = config;\n }\n\n // TODO: not exhaustive?\n var blockElementNames = ['P', 'LI', 'TD', 'TH', 'DIV', 'H1', 'H2', 'H3', 'H4', 'H5', 'H6', 'PRE'];\n function isBlockElement(node) {\n return blockElementNames.indexOf(node.nodeName) !== -1;\n }\n\n var inlineElementNames = ['A', 'B', 'STRONG', 'I', 'EM', 'SUB', 'SUP', 'U', 'STRIKE'];\n function isInlineElement(node) {\n return inlineElementNames.indexOf(node.nodeName) !== -1;\n }\n\n HTMLJanitor.prototype.clean = function (html) {\n var sandbox = document.createElement('div');\n sandbox.innerHTML = html;\n\n this._sanitize(sandbox);\n\n return sandbox.innerHTML;\n };\n\n HTMLJanitor.prototype._sanitize = function (parentNode) {\n var treeWalker = createTreeWalker(parentNode);\n var node = treeWalker.firstChild();\n if (!node) { return; }\n\n do {\n // Ignore nodes that have already been sanitized\n if (node._sanitized) {\n continue;\n }\n\n if (node.nodeType === Node.TEXT_NODE) {\n // If this text node is just whitespace and the previous or next element\n // sibling is a block element, remove it\n // N.B.: This heuristic could change. Very specific to a bug with\n // `contenteditable` in Firefox: http://jsbin.com/EyuKase/1/edit?js,output\n // FIXME: make this an option?\n if (node.data.trim() === ''\n && ((node.previousElementSibling && isBlockElement(node.previousElementSibling))\n || (node.nextElementSibling && isBlockElement(node.nextElementSibling)))) {\n parentNode.removeChild(node);\n this._sanitize(parentNode);\n break;\n } else {\n continue;\n }\n }\n\n // Remove all comments\n if (node.nodeType === Node.COMMENT_NODE) {\n parentNode.removeChild(node);\n this._sanitize(parentNode);\n break;\n }\n\n var isInline = isInlineElement(node);\n var containsBlockElement;\n if (isInline) {\n containsBlockElement = Array.prototype.some.call(node.childNodes, isBlockElement);\n }\n\n // Block elements should not be nested (e.g.

  • ...); if\n // they are, we want to unwrap the inner block element.\n var isNotTopContainer = !! parentNode.parentNode;\n var isNestedBlockElement =\n isBlockElement(parentNode) &&\n isBlockElement(node) &&\n isNotTopContainer;\n\n var nodeName = node.nodeName.toLowerCase();\n\n var allowedAttrs = getAllowedAttrs(this.config, nodeName, node);\n\n var isInvalid = isInline && containsBlockElement;\n\n // Drop tag entirely according to the whitelist *and* if the markup\n // is invalid.\n if (isInvalid || shouldRejectNode(node, allowedAttrs)\n || (!this.config.keepNestedBlockElements && isNestedBlockElement)) {\n // Do not keep the inner text of SCRIPT/STYLE elements.\n if (! (node.nodeName === 'SCRIPT' || node.nodeName === 'STYLE')) {\n while (node.childNodes.length > 0) {\n parentNode.insertBefore(node.childNodes[0], node);\n }\n }\n parentNode.removeChild(node);\n\n this._sanitize(parentNode);\n break;\n }\n\n // Sanitize attributes\n for (var a = 0; a < node.attributes.length; a += 1) {\n var attr = node.attributes[a];\n\n if (shouldRejectAttr(attr, allowedAttrs, node)) {\n node.removeAttribute(attr.name);\n // Shift the array to continue looping.\n a = a - 1;\n }\n }\n\n // Sanitize children\n this._sanitize(node);\n\n // Mark node as sanitized so it's ignored in future runs\n node._sanitized = true;\n } while ((node = treeWalker.nextSibling()));\n };\n\n function createTreeWalker(node) {\n return document.createTreeWalker(node,\n NodeFilter.SHOW_TEXT | NodeFilter.SHOW_ELEMENT | NodeFilter.SHOW_COMMENT,\n null, false);\n }\n\n function getAllowedAttrs(config, nodeName, node){\n if (typeof config.tags[nodeName] === 'function') {\n return config.tags[nodeName](node);\n } else {\n return config.tags[nodeName];\n }\n }\n\n function shouldRejectNode(node, allowedAttrs){\n if (typeof allowedAttrs === 'undefined') {\n return true;\n } else if (typeof allowedAttrs === 'boolean') {\n return !allowedAttrs;\n }\n\n return false;\n }\n\n function shouldRejectAttr(attr, allowedAttrs, node){\n var attrName = attr.name.toLowerCase();\n\n if (allowedAttrs === true){\n return false;\n } else if (typeof allowedAttrs[attrName] === 'function'){\n return !allowedAttrs[attrName](attr.value, node);\n } else if (typeof allowedAttrs[attrName] === 'undefined'){\n return true;\n } else if (allowedAttrs[attrName] === false) {\n return true;\n } else if (typeof allowedAttrs[attrName] === 'string') {\n return (allowedAttrs[attrName] !== attr.value);\n }\n\n return false;\n }\n\n return HTMLJanitor;\n\n}));\n\n\n\n//////////////////\n// WEBPACK FOOTER\n// ./~/html-janitor/src/html-janitor.js\n// module id = 18\n// module chunks = 0","/**\r\n * Codex Editor Listeners module\r\n *\r\n * @author Codex Team\r\n * @version 1.0\r\n */\r\n\r\n/**\r\n * Module-decorator for event listeners assignment\r\n */\r\nmodule.exports = function (listeners) {\r\n\r\n var allListeners = [];\r\n\r\n /**\r\n * Search methods\r\n *\r\n * byElement, byType and byHandler returns array of suitable listeners\r\n * one and all takes element, eventType, and handler and returns first (all) suitable listener\r\n *\r\n */\r\n listeners.search = function () {\r\n\r\n var byElement = function (element, context) {\r\n\r\n var listenersOnElement = [];\r\n\r\n context = context || allListeners;\r\n\r\n for (var i = 0; i < context.length; i++) {\r\n\r\n var listener = context[i];\r\n\r\n if (listener.element === element) {\r\n\r\n listenersOnElement.push(listener);\r\n\r\n }\r\n\r\n }\r\n\r\n return listenersOnElement;\r\n\r\n };\r\n\r\n var byType = function (eventType, context) {\r\n\r\n var listenersWithType = [];\r\n\r\n context = context || allListeners;\r\n\r\n for (var i = 0; i < context.length; i++) {\r\n\r\n var listener = context[i];\r\n\r\n if (listener.type === eventType) {\r\n\r\n listenersWithType.push(listener);\r\n\r\n }\r\n\r\n }\r\n\r\n return listenersWithType;\r\n\r\n };\r\n\r\n var byHandler = function (handler, context) {\r\n\r\n var listenersWithHandler = [];\r\n\r\n context = context || allListeners;\r\n\r\n for (var i = 0; i < context.length; i++) {\r\n\r\n var listener = context[i];\r\n\r\n if (listener.handler === handler) {\r\n\r\n listenersWithHandler.push(listener);\r\n\r\n }\r\n\r\n }\r\n\r\n return listenersWithHandler;\r\n\r\n };\r\n\r\n var one = function (element, eventType, handler) {\r\n\r\n var result = allListeners;\r\n\r\n if (element)\r\n result = byElement(element, result);\r\n\r\n if (eventType)\r\n result = byType(eventType, result);\r\n\r\n if (handler)\r\n result = byHandler(handler, result);\r\n\r\n return result[0];\r\n\r\n };\r\n\r\n var all = function (element, eventType, handler) {\r\n\r\n var result = allListeners;\r\n\r\n if (element)\r\n result = byElement(element, result);\r\n\r\n if (eventType)\r\n result = byType(eventType, result);\r\n\r\n if (handler)\r\n result = byHandler(handler, result);\r\n\r\n return result;\r\n\r\n };\r\n\r\n return {\r\n byElement : byElement,\r\n byType : byType,\r\n byHandler : byHandler,\r\n one : one,\r\n all : all\r\n };\r\n\r\n }();\r\n\r\n listeners.add = function (element, eventType, handler, isCapture) {\r\n\r\n element.addEventListener(eventType, handler, isCapture);\r\n\r\n var data = {\r\n element: element,\r\n type: eventType,\r\n handler: handler\r\n };\r\n\r\n var alreadyAddedListener = listeners.search.one(element, eventType, handler);\r\n\r\n if (!alreadyAddedListener) {\r\n\r\n allListeners.push(data);\r\n\r\n }\r\n\r\n };\r\n\r\n listeners.remove = function (element, eventType, handler) {\r\n\r\n element.removeEventListener(eventType, handler);\r\n\r\n var existingListeners = listeners.search.all(element, eventType, handler);\r\n\r\n for (var i = 0; i < existingListeners.length; i++) {\r\n\r\n var index = allListeners.indexOf(existingListeners[i]);\r\n\r\n if (index > 0) {\r\n\r\n allListeners.splice(index, 1);\r\n\r\n }\r\n\r\n }\r\n\r\n };\r\n\r\n listeners.removeAll = function () {\r\n\r\n allListeners.map(function (current) {\r\n\r\n listeners.remove(current.element, current.type, current.handler);\r\n\r\n });\r\n\r\n };\r\n\r\n listeners.get = function (element, eventType, handler) {\r\n\r\n return listeners.search.all(element, eventType, handler);\r\n\r\n };\r\n\r\n return listeners;\r\n\r\n}({});\n\n\n// WEBPACK FOOTER //\n// ./modules/listeners.js","/**\r\n * Codex Editor Destroyer module\r\n *\r\n * @auhor Codex Team\r\n * @version 1.0\r\n */\r\n\r\nmodule.exports = function (destroyer) {\r\n\r\n let editor = codex.editor;\r\n\r\n destroyer.removeNodes = function () {\r\n\r\n editor.nodes.wrapper.remove();\r\n editor.nodes.notifications.remove();\r\n\r\n };\r\n\r\n destroyer.destroyPlugins = function () {\r\n\r\n for (var tool in editor.tools) {\r\n\r\n if (typeof editor.tools[tool].destroy === 'function') {\r\n\r\n editor.tools[tool].destroy();\r\n\r\n }\r\n\r\n }\r\n\r\n };\r\n\r\n destroyer.destroyScripts = function () {\r\n\r\n var scripts = document.getElementsByTagName('SCRIPT');\r\n\r\n for (var i = 0; i < scripts.length; i++) {\r\n\r\n if (scripts[i].id.indexOf(editor.scriptPrefix) + 1) {\r\n\r\n scripts[i].remove();\r\n i--;\r\n\r\n }\r\n\r\n }\r\n\r\n };\r\n\r\n\r\n /**\r\n * Delete editor data from webpage.\r\n * You should send settings argument with boolean flags:\r\n * @param settings.ui- remove redactor event listeners and DOM nodes\r\n * @param settings.scripts - remove redactor scripts from DOM\r\n * @param settings.plugins - remove plugin's objects\r\n * @param settings.core - remove editor core. You can remove core only if UI and scripts flags is true\r\n * }\r\n *\r\n */\r\n destroyer.destroy = function (settings) {\r\n\r\n if (!settings || typeof settings !== 'object') {\r\n\r\n return;\r\n\r\n }\r\n\r\n if (settings.ui) {\r\n\r\n destroyer.removeNodes();\r\n editor.listeners.removeAll();\r\n\r\n }\r\n\r\n if (settings.scripts) {\r\n\r\n destroyer.destroyScripts();\r\n\r\n }\r\n\r\n if (settings.plugins) {\r\n\r\n destroyer.destroyPlugins();\r\n\r\n }\r\n\r\n if (settings.ui && settings.scripts && settings.core) {\r\n\r\n delete codex.editor;\r\n\r\n }\r\n\r\n };\r\n\r\n return destroyer;\r\n\r\n}({});\n\n\n// WEBPACK FOOTER //\n// ./modules/destroyer.js","/**\r\n * Codex Editor Paste module\r\n *\r\n * @author Codex Team\r\n * @version 1.1.1\r\n */\r\n\r\nmodule.exports = function (paste) {\r\n\r\n let editor = codex.editor;\r\n\r\n var patterns = [];\r\n\r\n paste.prepare = function () {\r\n\r\n var tools = editor.tools;\r\n\r\n for (var tool in tools) {\r\n\r\n if (!tools[tool].renderOnPastePatterns || !Array.isArray(tools[tool].renderOnPastePatterns)) {\r\n\r\n continue;\r\n\r\n }\r\n\r\n tools[tool].renderOnPastePatterns.map(function (pattern) {\r\n\r\n\r\n patterns.push(pattern);\r\n\r\n });\r\n\r\n }\r\n\r\n return Promise.resolve();\r\n\r\n };\r\n\r\n /**\r\n * Saves data\r\n * @param event\r\n */\r\n paste.pasted = function (event) {\r\n\r\n var clipBoardData = event.clipboardData || window.clipboardData,\r\n content = clipBoardData.getData('Text');\r\n\r\n var result = analize(content);\r\n\r\n if (result) {\r\n\r\n event.preventDefault();\r\n event.stopImmediatePropagation();\r\n\r\n }\r\n\r\n return result;\r\n\r\n };\r\n\r\n /**\r\n * Analizes pated string and calls necessary method\r\n */\r\n\r\n var analize = function (string) {\r\n\r\n var result = false,\r\n content = editor.content.currentNode,\r\n plugin = content.dataset.tool;\r\n\r\n patterns.map( function (pattern) {\r\n\r\n var execArray = pattern.regex.exec(string),\r\n match = execArray && execArray[0];\r\n\r\n if ( match && match === string.trim()) {\r\n\r\n /** current block is not empty */\r\n if ( content.textContent.trim() && plugin == editor.settings.initialBlockPlugin ) {\r\n\r\n pasteToNewBlock_();\r\n\r\n }\r\n\r\n pattern.callback(string, pattern);\r\n result = true;\r\n\r\n }\r\n\r\n });\r\n\r\n return result;\r\n\r\n };\r\n\r\n var pasteToNewBlock_ = function () {\r\n\r\n /** Create new initial block */\r\n editor.content.insertBlock({\r\n\r\n type : editor.settings.initialBlockPlugin,\r\n block : editor.tools[editor.settings.initialBlockPlugin].render({\r\n text : ''\r\n })\r\n\r\n }, false);\r\n\r\n };\r\n\r\n return paste;\r\n\r\n}({});\n\n\n// WEBPACK FOOTER //\n// ./modules/paste.js"],"sourceRoot":""} \ No newline at end of file diff --git a/modules/callbacks.js b/modules/callbacks.js index 79f3cb9c..c0b70530 100644 --- a/modules/callbacks.js +++ b/modules/callbacks.js @@ -917,8 +917,7 @@ module.exports = (function (callbacks) { /** Prevent default behaviour */ event.preventDefault(); - var editableParent = editor.content.getEditableParent(event.target), - currentNode = editor.content.currentNode; + var editableParent = editor.content.getEditableParent(event.target); /** Allow paste when event target placed in Editable element */ if (!editableParent) { @@ -931,66 +930,67 @@ module.exports = (function (callbacks) { var htmlData = event.clipboardData.getData('text/html'), plainData = event.clipboardData.getData('text/plain'); + /** Temporary DIV that is used to work with childs as arrays item */ var div = editor.draw.node('DIV', '', {}), cleanData, - fragment; + wrappedData; /** Create fragment, that we paste to range after proccesing */ - fragment = document.createDocumentFragment(); + cleanData = editor.sanitizer.clean(htmlData); - if ( htmlData.trim() != '' ) { - - cleanData = editor.sanitizer.clean(htmlData); - div.innerHTML = cleanData; - - } else { - - div.innerText = plainData.toString(); - - } - - var node, lastNode; /** - * and fill in fragment + * We wrap pasted text with

    tags to split it logically + * + * @type {string} */ - while (( node = div.firstChild) ) { + wrappedData = editor.content.wrapTextWithParagraphs(cleanData, plainData); + div.innerHTML = wrappedData; - lastNode = fragment.appendChild(node); - - } - - - if (editor.tools[currentNode.dataset.tool].allowRenderOnPaste) { - - if (editor.paste.pasted(event)) return; - - } + var NEW_BLOCK_TYPE = editor.settings.initialBlockPlugin, + currentBlockContent = editor.content.currentNode.firstChild.firstChild; /** - * work with selection and range + * If there only one paragraph, just insert it */ - var selection, range; + if (div.childNodes.length == 1) { - selection = window.getSelection(); - - range = selection.getRangeAt(0); - range.deleteContents(); - - range.insertNode(fragment); - - /** Preserve the selection */ - if (lastNode) { - - range = range.cloneRange(); - range.setStartAfter(lastNode); - range.collapse(true); - selection.removeAllRanges(); - selection.addRange(range); + editor.caret.insertNode(document.createTextNode(div.firstChild.innerHTML)); + return; } + div.childNodes.forEach(function (paragraph, index) { + + + /** + * If there was no data in working node, replace it with first paragraph of pasted text + */ + if (index == 0 && currentBlockContent.innerHTML.trim() === '') { + + editor.content.switchBlock(editor.content.currentNode, editor.tools[NEW_BLOCK_TYPE].render({ + text : paragraph.innerHTML + }), NEW_BLOCK_TYPE); + + return; + + } + + editor.content.insertBlock({ + type : NEW_BLOCK_TYPE, + block : editor.tools[NEW_BLOCK_TYPE].render({ + text : paragraph.innerHTML + }) + }); + + editor.caret.inputIndex++; + + }); + + editor.caret.setToPreviousBlock(editor.caret.getCurrentInputIndex() + 1); + + }; /** diff --git a/modules/caret.js b/modules/caret.js index 53ef9902..eb6c7a37 100644 --- a/modules/caret.js +++ b/modules/caret.js @@ -268,6 +268,31 @@ module.exports = (function (caret) { } }; + + /** + * Inserts node at the caret location + * @param node + */ + caret.insertNode = function (node) { + + var selection, range; + + selection = window.getSelection(); + + range = selection.getRangeAt(0); + range.deleteContents(); + + range.insertNode(node); + + range.setStartAfter(node); + range.collapse(true); + + selection.removeAllRanges(); + selection.addRange(range); + + + }; + return caret; })({}); \ No newline at end of file diff --git a/modules/content.js b/modules/content.js index 1f7b8032..4513a878 100644 --- a/modules/content.js +++ b/modules/content.js @@ -120,7 +120,7 @@ module.exports = (function (content) { } - this.currentNode = this.getFirstLevelBlock(targetNode); + content.currentNode = content.getFirstLevelBlock(targetNode); }; @@ -639,10 +639,17 @@ module.exports = (function (content) { /** * @public * - * @param [String] htmlString - html content as string + * @param {string} htmlData - html content as string + * @param {string} plainData - plain text * @return {string} - html content as string */ - content.wrapTextWithParagraphs = function (htmlString) { + content.wrapTextWithParagraphs = function (htmlData, plainData) { + + if (!htmlData) { + + return '

    ' + plainData.split('\n\n').join('

    ') + '

    '; + + } var wrapper = document.createElement('DIV'), newWrapper = document.createElement('DIV'), @@ -656,7 +663,7 @@ module.exports = (function (content) { * Make HTML Element to Wrap Text * It allows us to work with input data as HTML content */ - wrapper.innerHTML = htmlString; + wrapper.innerHTML = htmlData; paragraph = document.createElement('P'); for (i = 0; i < wrapper.childNodes.length; i++) { From e027488fa853d4294c889fc958a6ac1358a8e437 Mon Sep 17 00:00:00 2001 From: George Berezhnoy Date: Mon, 24 Apr 2017 02:04:38 +0300 Subject: [PATCH 02/12] Update version --- codex-editor.js | 2 +- codex-editor.js.map | 2 +- package.json | 2 +- 3 files changed, 3 insertions(+), 3 deletions(-) diff --git a/codex-editor.js b/codex-editor.js index cb254ccc..6cef92d1 100644 --- a/codex-editor.js +++ b/codex-editor.js @@ -1,3 +1,3 @@ -var codex=codex||{};codex.editor=function(e){function t(o){if(n[o])return n[o].exports;var r=n[o]={exports:{},id:o,loaded:!1};return e[o].call(r.exports,r,r.exports,t),r.loaded=!0,r.exports}var n={};return t.m=e,t.c=n,t.p="",t(0)}([function(e,t,n){"use strict";e.exports=function(e){e.version="1.6.3",e.scriptPrefix="cdx-script-";var t=function(){e.core=n(1),e.tools=n(2),e.ui=n(3),e.transport=n(4),e.renderer=n(5),e.saver=n(6),e.content=n(7),e.toolbar=n(8),e.callback=n(12),e.draw=n(13),e.caret=n(14),e.notifications=n(15),e.parser=n(16),e.sanitizer=n(17),e.listeners=n(19),e.destroyer=n(20),e.paste=n(21)};return e.settings={tools:["paragraph","header","picture","list","quote","code","twitter","instagram","smile"],holderId:"codex-editor",initialBlockPlugin:"paragraph"},e.nodes={holder:null,wrapper:null,toolbar:null,inlineToolbar:{wrapper:null,buttons:null,actions:null},toolbox:null,notifications:null,plusButton:null,showSettingsButton:null,showTrashButton:null,blockSettings:null,pluginSettings:null,defaultSettings:null,toolbarButtons:{},redactor:null},e.state={jsonOutput:[],blocks:[],inputs:[]},e.tools={},e.start=function(n){t(),e.core.prepare(n).then(e.ui.prepare).then(e.tools.prepare).then(e.sanitizer.prepare).then(e.paste.prepare).then(e.transport.prepare).then(e.renderer.makeBlocksFromData).then(e.ui.saveInputs).catch(function(t){e.core.log("Initialization failed with error: %o","warn",t)})},e}({})},function(e,t){"use strict";var n="function"==typeof Symbol&&"symbol"==typeof Symbol.iterator?function(e){return typeof e}:function(e){return e&&"function"==typeof Symbol&&e.constructor===Symbol&&e!==Symbol.prototype?"symbol":typeof e};e.exports=function(e){var t=codex.editor;e.prepare=function(e){return new Promise(function(o,r){e&&(t.settings.tools=e.tools||t.settings.tools),e.data&&(t.state.blocks=e.data),e.initialBlockPlugin&&(t.settings.initialBlockPlugin=e.initialBlockPlugin),e.sanitizer&&(t.settings.sanitizer=e.sanitizer),t.hideToolbar=e.hideToolbar,t.nodes.holder=document.getElementById(e.holderId||t.settings.holderId),void 0===n(t.nodes.holder)||null===t.nodes.holder?r(Error("Holder wasn't found by ID: #"+e.holderId)):o()})},e.log=function(e,t,n){t=t||"log",n?e="[codex-editor]: "+e:(n=e||"undefined",e="[codex-editor]: %o");try{"console"in window&&window.console[t]&&(n?window.console[t](e,n):window.console[t](e))}catch(e){}},e.insertAfter=function(e,t){e.parentNode.insertBefore(t,e.nextSibling)},e.nodeTypes={TAG:1,TEXT:3,COMMENT:8},e.keys={BACKSPACE:8,TAB:9,ENTER:13,SHIFT:16,CTRL:17,ALT:18,ESC:27,SPACE:32,LEFT:37,UP:38,DOWN:40,RIGHT:39,DELETE:46,META:91},e.isDomNode=function(e){return e&&"object"===("undefined"==typeof e?"undefined":n(e))&&e.nodeType&&e.nodeType==this.nodeTypes.TAG},e.isEmpty=function(e){return 0===Object.keys(e).length},e.ajax=function(e){if(e&&e.url){var t,n,r,i=window.XMLHttpRequest?new XMLHttpRequest:new ActiveXObject("Microsoft.XMLHTTP");if(e.async=!0,e.type=e.type||"GET",e.data=e.data||"",e["content-type"]=e["content-type"]||"application/json; charset=utf-8","GET"==e.type&&e.data)e.url=/\?/.test(e.url)?e.url+"&"+e.data:e.url+"?"+e.data;else{t="";for(r in e.data)t+=r+"="+encodeURIComponent(e.data[r])+"&"}e.withCredentials&&(i.withCredentials=!0);var a=void 0,s=void 0;if("function"!=typeof e.beforeSend||(s=e.beforeSend.call(),s!==!1))return i.open(e.type,e.url,e.async),n=o(e.data),n||("POST"!==e.type?i.setRequestHeader("Content-type",e["content-type"]):i.setRequestHeader("Content-type","application/x-www-form-urlencoded")),i.setRequestHeader("X-Requested-With","XMLHttpRequest"),a=s||i,"function"==typeof e.progress&&(i.upload.onprogress=e.progress.bind(a)),i.onreadystatechange=function(){4===i.readyState&&(200===i.status?"function"==typeof e.success&&e.success.call(a,i.responseText):"function"==typeof e.error&&e.error.call(a,i.responseText,i.status))},n?i.send(e.data):i.send(t),i}},e.importScript=function(e,n){return new Promise(function(o,r){var i=void 0;n?document.getElementById(t.scriptPrefix+n)&&o(e):r("Instance name is missed"),i=document.createElement("SCRIPT"),i.async=!0,i.defer=!0,i.id=t.scriptPrefix+n,i.onload=function(){o(e)},i.onerror=function(){r(e)},i.src=e,document.head.appendChild(i)})};var o=function(e){return e instanceof FormData};return e}({})},function(e,t){"use strict";e.exports=function(){function e(){return new Promise(function(e,o){Promise.resolve().then(function(){var t=[],o=n.tools;for(var r in o){var i=o[r];i.prepare&&"function"!=typeof i.prepare||!i.prepare||t.push(i)}return t.length||e(),t}).then(t).then(function(){n.core.log("Plugins loaded","info"),e()}).catch(function(e){o(e)})})}function t(e){return new Promise(function(t){e.reduce(function(r,i,a){return r.then(function(){return new Promise(function(r){o(i).then(r).then(function(){i.available=!0}).catch(function(e){n.core.log("Plugin «"+i.type+"» was not loaded. Preparation failed because %o","warn",e),i.available=!1,i.loadingMessage=e,r()}).then(function(){a==e.length-1&&t()})})})},Promise.resolve())})}var n=codex.editor,o=function(e){return e.prepare(e.config||{})};return{prepare:e}}()},function(e,t){"use strict";e.exports=function(e){var t=codex.editor;e.className={BLOCK_CLASSNAME:"ce-block",BLOCK_CONTENT:"ce-block__content",BLOCK_STRETCHED:"ce-block--stretched",BLOCK_HIGHLIGHTED:"ce-block--focused",SETTINGS_ITEM:"ce-settings__item"},e.prepare=function(){return new Promise(function(e){var n=t.draw.wrapper(),r=t.draw.redactor(),i=o();n.appendChild(i),n.appendChild(r),t.nodes.wrapper=n,t.nodes.redactor=r,t.nodes.holder.appendChild(n),e()}).then(s).then(n).then(c).then(a).then(l).catch(function(){t.core.log("Can't draw editor interface")})};var n=function(){var e=t.draw.inlineToolbar();t.nodes.inlineToolbar.wrapper=e,t.nodes.inlineToolbar.buttons=t.draw.inlineToolbarButtons(),t.nodes.inlineToolbar.actions=t.draw.inlineToolbarActions(),t.nodes.inlineToolbar.wrapper.appendChild(t.nodes.inlineToolbar.buttons),t.nodes.inlineToolbar.wrapper.appendChild(t.nodes.inlineToolbar.actions),t.nodes.wrapper.appendChild(t.nodes.inlineToolbar.wrapper)},o=function(){var e=t.draw.toolbar(),n=i(),o=r();return e.appendChild(n),e.appendChild(o),t.nodes.toolbar=e,e},r=function(){var e=t.draw.toolbarContent(),n=t.draw.toolbox(),o=t.draw.plusButton();return e.appendChild(o),e.appendChild(n),t.nodes.toolbox=n,t.nodes.plusButton=o,e},i=function(){var e=t.draw.blockSettings(),n=t.draw.blockButtons(),o=t.draw.defaultSettings(),r=t.draw.settingsButton(),i=t.toolbar.settings.makeRemoveBlockButton(),a=t.draw.pluginsSettings();return e.appendChild(a),e.appendChild(o),n.appendChild(r),n.appendChild(i),n.appendChild(e),t.nodes.blockSettings=e,t.nodes.pluginSettings=a,t.nodes.defaultSettings=o,t.nodes.showSettingsButton=r,t.nodes.showTrashButton=i,n},a=function(){t.nodes.notifications=t.notifications.createHolder()},s=function(){var e,n,o;for(n in t.settings.tools)e=t.settings.tools[n],t.tools[n]=e,e.iconClassname||!e.displayInToolbox?"function"==typeof e.render?e.displayInToolbox&&(o=t.draw.toolbarButton(n,e.iconClassname),t.nodes.toolbox.appendChild(o),t.nodes.toolbarButtons[n]=o):t.core.log("render method missed. Tool %o skipped","warn",n):t.core.log("Toolbar icon classname missed. Tool %o skipped","warn",n)},c=function(){var e,n,o={bold:{icon:"ce-icon-bold",command:"bold"},italic:{icon:"ce-icon-italic",command:"italic"},underline:{icon:"ce-icon-underline",command:"underline"},link:{icon:"ce-icon-link",command:"createLink"}};for(var r in o)n=o[r],e=t.draw.toolbarButtonInline(r,n.icon),t.nodes.inlineToolbar.buttons.appendChild(e),t.ui.setInlineToolbarButtonBehaviour(e,n.command)},l=function(){t.core.log("ui.bindEvents fired","info"),t.listeners.add(document,"keydown",t.callback.globalKeydown,!1),t.listeners.add(t.nodes.redactor,"keydown",t.callback.redactorKeyDown,!1),t.listeners.add(document,"keyup",t.callback.globalKeyup,!1),t.listeners.add(t.nodes.redactor,"click",t.callback.redactorClicked,!1),t.listeners.add(t.nodes.plusButton,"click",t.callback.plusButtonClicked,!1),t.listeners.add(t.nodes.showSettingsButton,"click",t.callback.showSettingsButtonClicked,!1);for(var e in t.nodes.toolbarButtons)t.listeners.add(t.nodes.toolbarButtons[e],"click",t.callback.toolbarButtonClicked,!1)};return e.addBlockHandlers=function(e){e&&(t.listeners.add(e,"keydown",t.callback.blockKeydown,!1),t.listeners.add(e,"paste",t.callback.blockPasteCallback,!1),t.listeners.add(e,"mouseup",t.toolbar.inline.show,!1))},e.saveInputs=function(){var e=t.nodes.redactor;t.state.inputs=[];var n=e.querySelectorAll("[contenteditable], input, textarea");Array.prototype.map.call(n,function(e){e.type&&"text"!=e.type&&"textarea"!=e.type||t.state.inputs.push(e)})},e.addInitialBlock=function(){var e,n=t.settings.initialBlockPlugin;return t.tools[n]?(e=t.tools[n].render(),e.setAttribute("data-placeholder","Расскажите свою историю..."),t.content.insertBlock({type:n,block:e}),void t.content.workingNodeChanged(e)):void t.core.log("Plugin %o was not implemented and can't be used as initial block","warn",n)},e.setInlineToolbarButtonBehaviour=function(e,n){t.listeners.add(e,"mousedown",function(e){t.toolbar.inline.toolClicked(e,n)},!1)},e}({})},function(e,t){"use strict";e.exports=function(e){var t=codex.editor,n=null;return e.input=null,e.arguments=null,e.prepare=function(){var e=t.draw.node("INPUT","",{type:"file"});t.listeners.add(e,"change",t.transport.fileSelected),t.transport.input=e},e.clearInput=function(){e.input=null,e.prepare()},e.fileSelected=function(){var o,r=this,i=r.files,a=new FormData;if(t.transport.arguments.multiple===!0)for(o=0;o"+t.split("\n\n").join("

    ")+"

    ";var n,o,r,i,a=document.createElement("DIV"),s=document.createElement("DIV"),c=["DIV","P"];for(a.innerHTML=e,o=document.createElement("P"),n=0;n'}),o=t.draw.node("DIV","ce-toolbar__remove-confirmation",{}),r=t.draw.node("DIV","ce-toolbar__remove-confirm",{textContent:"Удалить блок"}),i=t.draw.node("DIV","ce-toolbar__remove-cancel",{textContent:"Отмена"});return t.listeners.add(n,"click",t.toolbar.settings.removeButtonClicked,!1),t.listeners.add(r,"click",t.toolbar.settings.confirmRemovingRequest,!1),t.listeners.add(i,"click",t.toolbar.settings.cancelRemovingRequest,!1),o.appendChild(r),o.appendChild(i),e.appendChild(n),e.appendChild(o),t.toolbar.settings.setting=n,t.toolbar.settings.actions=o,e},e.removeButtonClicked=function(){var e=t.toolbar.settings.actions;e.classList.contains("opened")?t.toolbar.settings.hideRemoveActions():t.toolbar.settings.showRemoveActions(),t.toolbar.toolbox.close(),t.toolbar.settings.close()},e.cancelRemovingRequest=function(){t.toolbar.settings.actions.classList.remove("opened")},e.confirmRemovingRequest=function(){var e,n=t.content.currentNode;n.remove(),e=t.nodes.redactor.childNodes.length,0===e&&(t.content.currentNode=null,t.ui.addInitialBlock()),t.ui.saveInputs(),t.toolbar.close()},e.showRemoveActions=function(){t.toolbar.settings.actions.classList.add("opened")},e.hideRemoveActions=function(){t.toolbar.settings.actions.classList.remove("opened")},e}({})},function(e,t){"use strict";e.exports=function(e){var t=codex.editor;e.buttonsOpened=null,e.actionsOpened=null,e.wrappersOffset=null,e.storedSelection=null,e.show=function(){var n,o=t.content.currentNode,r=o.dataset.tool;if(n=t.tools[r],n.showInlineToolbar){var i=e.getSelectionText(),a=t.nodes.inlineToolbar.wrapper;i.length>0&&(t.toolbar.inline.move(),a.classList.add("opened"),t.toolbar.inline.showButtons())}},e.close=function(){var e=t.nodes.inlineToolbar.wrapper;e.classList.remove("opened")},e.move=function(){this.wrappersOffset||(this.wrappersOffset=this.getWrappersOffset());var e,n,o=this.getSelectionCoords(),r=0,i=t.nodes.inlineToolbar.wrapper;0===i.offsetHeight&&(r=40),e=o.x-this.wrappersOffset.left,n=o.y+window.scrollY-this.wrappersOffset.top-r-i.offsetHeight,i.style.transform="translate3D("+Math.floor(e)+"px, "+Math.floor(n)+"px, 0)",t.toolbar.inline.closeButtons(),t.toolbar.inline.closeAction()},e.toolClicked=function(e,n){switch(n){case"createLink":t.toolbar.inline.createLinkAction(e,n);break;default:t.toolbar.inline.defaultToolAction(n)}t.nodes.inlineToolbar.buttons.childNodes.forEach(t.toolbar.inline.hightlight)},e.getWrappersOffset=function(){var e=t.nodes.wrapper,n=this.getOffset(e);return this.wrappersOffset=n,n},e.getOffset=function(e){for(var t=0,n=0;e&&!isNaN(e.offsetLeft)&&!isNaN(e.offsetTop);)t+=e.offsetLeft+e.clientLeft,n+=e.offsetTop+e.clientTop,e=e.offsetParent;return{top:n,left:t}},e.getSelectionCoords=function(){var e,t=document.selection,n=0,o=0;if(t)"Control"!=t.type&&(e=t.createRange(),e.collapse(!0),n=e.boundingLeft,o=e.boundingTop);else if(window.getSelection&&(t=window.getSelection(),t.rangeCount&&(e=t.getRangeAt(0).cloneRange(),e.getClientRects))){e.collapse(!0);var r=e.getClientRects()[0];if(!r)return;n=r.left,o=r.top}return{x:n,y:o}},e.getSelectionText=function(){var e="";return window.getSelection&&(e=window.getSelection().toString()),e},e.showButtons=function(){var e=t.nodes.inlineToolbar.buttons;e.classList.add("opened"),t.toolbar.inline.buttonsOpened=!0,t.nodes.inlineToolbar.buttons.childNodes.forEach(t.toolbar.inline.hightlight)},e.closeButtons=function(){var e=t.nodes.inlineToolbar.buttons;e.classList.remove("opened"),t.toolbar.inline.buttonsOpened=!1},e.showActions=function(){var e=t.nodes.inlineToolbar.actions;e.classList.add("opened"),t.toolbar.inline.actionsOpened=!0},e.closeAction=function(){var e=t.nodes.inlineToolbar.actions;e.innerHTML="",e.classList.remove("opened"),t.toolbar.inline.actionsOpened=!1};var n=function(e){if(e.keyCode==t.core.keys.ENTER){var n=t.content.currentNode,o=t.toolbar.inline.storedSelection;t.toolbar.inline.restoreSelection(n,o),t.toolbar.inline.setAnchor(this.value),e.preventDefault(),e.stopImmediatePropagation(),t.toolbar.inline.clearRange()}};return e.createLinkAction=function(e){var o=this.isLinkActive(),r=t.content.currentNode,i=t.toolbar.inline.saveSelection(r);if(t.toolbar.inline.storedSelection=i,o)t.toolbar.inline.restoreSelection(r,i),t.toolbar.inline.defaultToolAction("unlink");else{var a=t.draw.inputForLink();t.nodes.inlineToolbar.actions.appendChild(a),t.toolbar.inline.closeButtons(),t.toolbar.inline.showActions(),a.focus(),e.preventDefault(),t.listeners.add(a,"keydown",n,!1)}},e.isLinkActive=function(){var e=!1;return t.nodes.inlineToolbar.buttons.childNodes.forEach(function(t){var n=t.dataset.type;"link"==n&&t.classList.contains("hightlighted")&&(e=!0)}),e},e.defaultToolAction=function(e){document.execCommand(e,!1,null)},e.setAnchor=function(e){document.execCommand("createLink",!1,e),t.toolbar.inline.closeAction()},e.saveSelection=function(e){var t,n=window.getSelection().getRangeAt(0),o=n.cloneRange();return o.selectNodeContents(e),o.setEnd(n.startContainer,n.startOffset),t=o.toString().length,{start:t,end:t+n.toString().length}},e.restoreSelection=function(e,t){var n=document.createRange(),o=0;n.setStart(e,0),n.collapse(!0);for(var r,i,a=[e],s=!1,c=!1;!c&&(r=a.pop());)if(3==r.nodeType)i=o+r.length,!s&&t.start>=o&&t.start<=i&&(n.setStart(r,t.start-o),s=!0),s&&t.end>=o&&t.end<=i&&(n.setEnd(r,t.end-o),c=!0),o=i;else for(var l=r.childNodes.length;l--;)a.push(r.childNodes[l]);var d=window.getSelection();d.removeAllRanges(),d.addRange(n)},e.clearRange=function(){var e=window.getSelection();e.removeAllRanges()},e.hightlight=function(e){var n=e.dataset.type;document.queryCommandState(n)?t.toolbar.inline.setButtonHighlighted(e):t.toolbar.inline.removeButtonsHighLight(e);var o=window.getSelection(),r=o.anchorNode.parentNode;"A"==r.tagName&&"link"==n&&t.toolbar.inline.setButtonHighlighted(e)},e.setButtonHighlighted=function(e){if(e.classList.add("hightlighted"),"link"==e.dataset.type){var t=e.childNodes[0];t.classList.remove("ce-icon-link"),t.classList.add("ce-icon-unlink")}},e.removeButtonsHighLight=function(e){if(e.classList.remove("hightlighted"),"link"==e.dataset.type){var t=e.childNodes[0];t.classList.remove("ce-icon-unlink"),t.classList.add("ce-icon-link")}},e}({})},function(e,t){"use strict";e.exports=function(e){var t=codex.editor;return e.opened=!1,e.open=function(){t.toolbar.settings.opened&&t.toolbar.settings.close(),t.nodes.toolbox.classList.add("opened"),t.nodes.plusButton.classList.add("clicked"),t.toolbar.toolbox.opened=!0},e.close=function(){t.nodes.toolbox.classList.remove("opened"),t.nodes.plusButton.classList.remove("clicked"),t.toolbar.toolbox.opened=!1,t.toolbar.current=null},e.leaf=function(){var e=t.toolbar.current,n=Object.keys(t.tools),o=t.nodes.toolbarButtons,r=0,i=void 0,a=void 0,s=void 0;if(e)for(r=(n.indexOf(e)+1)%n.length,a=n[r];!t.tools[a].displayInToolbox;)r=(r+1)%n.length,a=n[r];else for(s in t.tools){if(t.tools[s].displayInToolbox)break;r++}i=n[r];for(var c in o)o[c].classList.remove("selected");o[i].classList.add("selected"),t.toolbar.current=i},e.toolClicked=function(e){var n,o,r,i=["image","link","list","instagram","twitter","embed"],a=t.tools[t.toolbar.current],s=t.content.currentNode,c=t.caret.inputIndex;n=a.render(),r={block:n,type:a.type,stretched:!1},s&&i.indexOf(s.dataset.tool)===-1&&""===s.textContent.trim()?t.content.switchBlock(s,n,a.type):(t.content.insertBlock(r),c++),o=a.appendCallback,o&&"function"==typeof o&&o.call(e),window.setTimeout(function(){t.caret.setToBlock(c)},10),t.content.workingNodeChanged(),t.toolbar.move()},e}({})},function(e,t){"use strict";e.exports=function(e){var t=codex.editor;e.globalKeydown=function(e){switch(e.keyCode){case t.core.keys.ENTER:o(e)}},e.redactorKeyDown=function(e){switch(e.keyCode){case t.core.keys.TAB:n(e);break;case t.core.keys.ENTER:i(e);break;case t.core.keys.ESC:a(e);break;default:c(e)}},e.globalKeyup=function(e){switch(e.keyCode){case t.core.keys.UP:case t.core.keys.LEFT:case t.core.keys.RIGHT:case t.core.keys.DOWN:s(e)}};var n=function(e){e.preventDefault();var n=t.content.currentNode.querySelectorAll("textarea, input"),o=!0,r=!t.content.currentNode.textContent.trim();Array.prototype.map.call(n,function(e){"textarea"!=e.type&&"text"!=e.type||(o=o&&!e.value.trim())});var i=r&&o;i&&(t.toolbar.opened||t.toolbar.open(),t.toolbar.opened&&!t.toolbar.toolbox.opened?t.toolbar.toolbox.open():t.toolbar.toolbox.leaf())},o=function(){t.content.editorAreaHightlighted&&(t.caret.inputIndex=-1,r())},r=function(){var e=t.settings.initialBlockPlugin;t.content.insertBlock({type:e,block:t.tools[e].render()},!0),t.toolbar.move(),t.toolbar.open()},i=function(e){"true"==e.target.contentEditable&&t.caret.saveCurrentInputIndex();var n=t.caret.getCurrentInputIndex()||0,o=t.content.currentNode,r=o.dataset.tool,i=t.toolbar.opened&&t.toolbar.current&&e.target==t.state.inputs[n],a=t.tools[r].enableLineBreaks,s=t.settings.initialBlockPlugin;if(i)return e.preventDefault(),t.toolbar.toolbox.toolClicked(e),t.toolbar.close(),e.stopPropagation(),void e.stopImmediatePropagation();if(e.shiftKey||a)return e.stopPropagation(),void e.stopImmediatePropagation();var c=window.getSelection(),l=c.anchorNode,d=t.caret.position.atTheEnd(),u=!1;if(e.shiftKey&&!a)return t.callback.enterPressedOnBlock(t.content.currentBlock,e),void e.preventDefault();if(u=l&&"true"!=l.parentNode.contentEditable,l.nodeType!=t.core.nodeTypes.TEXT||u||d){var p=t.content.isLastNode(l);p&&d&&(e.preventDefault(),e.stopPropagation(),e.stopImmediatePropagation(),t.core.log("ENTER clicked in last textNode. Create new BLOCK"),t.content.insertBlock({type:s,block:t.tools[s].render()},!0),t.toolbar.move(),t.toolbar.open(),t.toolbar.showPlusButton())}else e.preventDefault(),t.core.log("Splitting Text node..."),t.content.splitBlock(n),t.state.inputs[n+1].textContent.trim()||t.toolbar.showPlusButton();t.ui.saveInputs()},a=function(e){t.toolbar.close(),t.toolbar.toolbox.close(),e.preventDefault()},s=function(e){t.content.workingNodeChanged(),t.toolbar.close(),t.toolbar.move()},c=function(){t.toolbar.close(),t.toolbar.inline.actionsOpened||(t.toolbar.inline.close(),t.content.clearMark())};e.redactorClicked=function(e){l(),t.content.workingNodeChanged(e.target),t.ui.saveInputs();var n,o=t.toolbar.inline.getSelectionText();if(0===o.length&&t.toolbar.inline.close(),"true"==e.target.contentEditable&&t.caret.saveCurrentInputIndex(),null===t.content.currentNode){var r=t.state.inputs.length>0?t.state.inputs.length-1:0;if(t.state.inputs.length&&(n=t.content.getFirstLevelBlock(t.state.inputs[r])),t.state.inputs.length&&""===t.state.inputs[r].textContent&&n.dataset.tool==t.settings.initialBlockPlugin)t.caret.setToBlock(r);else{var i=t.settings.initialBlockPlugin;t.content.insertBlock({type:i,block:t.tools[i].render()}),1===t.state.inputs.length?t.caret.setToBlock(r):t.caret.setToNextBlock(r)}}else t.toolbar.settings.close(),t.toolbar.toolbox.close();t.toolbar.move(),t.toolbar.open();var a=!t.content.currentNode.textContent.trim(),s=t.content.currentNode.dataset.tool,c=s==t.settings.initialBlockPlugin;t.toolbar.hidePlusButton(),a||t.content.markBlock(),c&&a&&t.toolbar.showPlusButton()};var l=function(){var e=window.getSelection(),n=e.anchorNode,o=!1;if(0===e.rangeCount)t.content.editorAreaHightlighted=!0;else{for(t.core.isDomNode(n)||(n=n.parentNode),"true"==n.contentEditable&&(o=!0);"true"!=n.contentEditable&&(n=n.parentNode,"true"==n.contentEditable&&(o=!0),n!=document.body););t.content.editorAreaHightlighted=!o}};e.toolbarButtonClicked=function(e){var n=this;t.toolbar.current=n.dataset.type,t.toolbar.toolbox.toolClicked(e),t.toolbar.close()},e.plusButtonClicked=function(){t.nodes.toolbox.classList.contains("opened")?t.toolbar.toolbox.close():t.toolbar.toolbox.open()},e.blockKeydown=function(e){var n=e.target;switch(e.keyCode){case t.core.keys.DOWN:case t.core.keys.RIGHT:d(e);break;case t.core.keys.BACKSPACE:p(n,e);break;case t.core.keys.UP:case t.core.keys.LEFT:u(e)}};var d=function(e){var n,o=window.getSelection(),r=t.state.inputs,i=o.anchorNode;if(!i)return!1;for(;"true"!=i.contentEditable;)n=i.parentNode,i=n;for(var a=0;i!=r[a];)a++;if(!i.textContent)return void t.caret.setToNextBlock(a);var s,c,l=!1,d=!1;return s=i.childNodes[i.childNodes.length-1],c=t.core.isDomNode(s)?t.content.getDeepestTextNodeFromPosition(s,s.childNodes.length):s,l=o.anchorNode==c,d=c.length==o.anchorOffset,l&&d?void t.caret.setToNextBlock(a):(t.core.log("arrow [down|right] : caret does not reached the end"),!1)},u=function(e){var n,o=window.getSelection(),r=t.state.inputs,i=o.anchorNode;if(!i)return!1;if(0!==o.anchorOffset)return!1;for(;"true"!=i.contentEditable;)n=i.parentNode,i=n;for(var a=0;i!=r[a];)a++;var s,c,l=!1,d=!1;return i.textContent?(s=i.childNodes[0],c=t.core.isDomNode(s)?t.content.getDeepestTextNodeFromPosition(s,0):s,l=o.anchorNode==c,d=0===o.anchorOffset,void(l&&d&&t.caret.setToPreviousBlock(a))):void t.caret.setToPreviousBlock(a)},p=function(e,n){var o,r,i,a=t.caret.getCurrentInputIndex();if(f(n.target)){if(""!=n.target.value.trim())return;e.remove()}if(e.textContent.trim()){if(o=t.content.getRange(),r=o.endOffset-o.startOffset,!t.caret.position.atStart()||r||!t.state.inputs[a-1])return;t.content.mergeBlocks(a)}r||e.remove(),i=t.nodes.redactor.childNodes.length,0===i?(t.content.currentNode=null,t.ui.addInitialBlock(),t.ui.saveInputs(),window.setTimeout(function(){t.caret.setToPreviousBlock(1)},10)):0!==t.caret.inputIndex?t.caret.setToPreviousBlock(t.caret.inputIndex):t.caret.setToNextBlock(t.caret.inputIndex), +var codex=codex||{};codex.editor=function(e){function t(o){if(n[o])return n[o].exports;var r=n[o]={exports:{},id:o,loaded:!1};return e[o].call(r.exports,r,r.exports,t),r.loaded=!0,r.exports}var n={};return t.m=e,t.c=n,t.p="",t(0)}([function(e,t,n){"use strict";e.exports=function(e){e.version="1.6.4",e.scriptPrefix="cdx-script-";var t=function(){e.core=n(1),e.tools=n(2),e.ui=n(3),e.transport=n(4),e.renderer=n(5),e.saver=n(6),e.content=n(7),e.toolbar=n(8),e.callback=n(12),e.draw=n(13),e.caret=n(14),e.notifications=n(15),e.parser=n(16),e.sanitizer=n(17),e.listeners=n(19),e.destroyer=n(20),e.paste=n(21)};return e.settings={tools:["paragraph","header","picture","list","quote","code","twitter","instagram","smile"],holderId:"codex-editor",initialBlockPlugin:"paragraph"},e.nodes={holder:null,wrapper:null,toolbar:null,inlineToolbar:{wrapper:null,buttons:null,actions:null},toolbox:null,notifications:null,plusButton:null,showSettingsButton:null,showTrashButton:null,blockSettings:null,pluginSettings:null,defaultSettings:null,toolbarButtons:{},redactor:null},e.state={jsonOutput:[],blocks:[],inputs:[]},e.tools={},e.start=function(n){t(),e.core.prepare(n).then(e.ui.prepare).then(e.tools.prepare).then(e.sanitizer.prepare).then(e.paste.prepare).then(e.transport.prepare).then(e.renderer.makeBlocksFromData).then(e.ui.saveInputs).catch(function(t){e.core.log("Initialization failed with error: %o","warn",t)})},e}({})},function(e,t){"use strict";var n="function"==typeof Symbol&&"symbol"==typeof Symbol.iterator?function(e){return typeof e}:function(e){return e&&"function"==typeof Symbol&&e.constructor===Symbol&&e!==Symbol.prototype?"symbol":typeof e};e.exports=function(e){var t=codex.editor;e.prepare=function(e){return new Promise(function(o,r){e&&(t.settings.tools=e.tools||t.settings.tools),e.data&&(t.state.blocks=e.data),e.initialBlockPlugin&&(t.settings.initialBlockPlugin=e.initialBlockPlugin),e.sanitizer&&(t.settings.sanitizer=e.sanitizer),t.hideToolbar=e.hideToolbar,t.nodes.holder=document.getElementById(e.holderId||t.settings.holderId),void 0===n(t.nodes.holder)||null===t.nodes.holder?r(Error("Holder wasn't found by ID: #"+e.holderId)):o()})},e.log=function(e,t,n){t=t||"log",n?e="[codex-editor]: "+e:(n=e||"undefined",e="[codex-editor]: %o");try{"console"in window&&window.console[t]&&(n?window.console[t](e,n):window.console[t](e))}catch(e){}},e.insertAfter=function(e,t){e.parentNode.insertBefore(t,e.nextSibling)},e.nodeTypes={TAG:1,TEXT:3,COMMENT:8},e.keys={BACKSPACE:8,TAB:9,ENTER:13,SHIFT:16,CTRL:17,ALT:18,ESC:27,SPACE:32,LEFT:37,UP:38,DOWN:40,RIGHT:39,DELETE:46,META:91},e.isDomNode=function(e){return e&&"object"===("undefined"==typeof e?"undefined":n(e))&&e.nodeType&&e.nodeType==this.nodeTypes.TAG},e.isEmpty=function(e){return 0===Object.keys(e).length},e.ajax=function(e){if(e&&e.url){var t,n,r,i=window.XMLHttpRequest?new XMLHttpRequest:new ActiveXObject("Microsoft.XMLHTTP");if(e.async=!0,e.type=e.type||"GET",e.data=e.data||"",e["content-type"]=e["content-type"]||"application/json; charset=utf-8","GET"==e.type&&e.data)e.url=/\?/.test(e.url)?e.url+"&"+e.data:e.url+"?"+e.data;else{t="";for(r in e.data)t+=r+"="+encodeURIComponent(e.data[r])+"&"}e.withCredentials&&(i.withCredentials=!0);var a=void 0,s=void 0;if("function"!=typeof e.beforeSend||(s=e.beforeSend.call(),s!==!1))return i.open(e.type,e.url,e.async),n=o(e.data),n||("POST"!==e.type?i.setRequestHeader("Content-type",e["content-type"]):i.setRequestHeader("Content-type","application/x-www-form-urlencoded")),i.setRequestHeader("X-Requested-With","XMLHttpRequest"),a=s||i,"function"==typeof e.progress&&(i.upload.onprogress=e.progress.bind(a)),i.onreadystatechange=function(){4===i.readyState&&(200===i.status?"function"==typeof e.success&&e.success.call(a,i.responseText):"function"==typeof e.error&&e.error.call(a,i.responseText,i.status))},n?i.send(e.data):i.send(t),i}},e.importScript=function(e,n){return new Promise(function(o,r){var i=void 0;n?document.getElementById(t.scriptPrefix+n)&&o(e):r("Instance name is missed"),i=document.createElement("SCRIPT"),i.async=!0,i.defer=!0,i.id=t.scriptPrefix+n,i.onload=function(){o(e)},i.onerror=function(){r(e)},i.src=e,document.head.appendChild(i)})};var o=function(e){return e instanceof FormData};return e}({})},function(e,t){"use strict";e.exports=function(){function e(){return new Promise(function(e,o){Promise.resolve().then(function(){var t=[],o=n.tools;for(var r in o){var i=o[r];i.prepare&&"function"!=typeof i.prepare||!i.prepare||t.push(i)}return t.length||e(),t}).then(t).then(function(){n.core.log("Plugins loaded","info"),e()}).catch(function(e){o(e)})})}function t(e){return new Promise(function(t){e.reduce(function(r,i,a){return r.then(function(){return new Promise(function(r){o(i).then(r).then(function(){i.available=!0}).catch(function(e){n.core.log("Plugin «"+i.type+"» was not loaded. Preparation failed because %o","warn",e),i.available=!1,i.loadingMessage=e,r()}).then(function(){a==e.length-1&&t()})})})},Promise.resolve())})}var n=codex.editor,o=function(e){return e.prepare(e.config||{})};return{prepare:e}}()},function(e,t){"use strict";e.exports=function(e){var t=codex.editor;e.className={BLOCK_CLASSNAME:"ce-block",BLOCK_CONTENT:"ce-block__content",BLOCK_STRETCHED:"ce-block--stretched",BLOCK_HIGHLIGHTED:"ce-block--focused",SETTINGS_ITEM:"ce-settings__item"},e.prepare=function(){return new Promise(function(e){var n=t.draw.wrapper(),r=t.draw.redactor(),i=o();n.appendChild(i),n.appendChild(r),t.nodes.wrapper=n,t.nodes.redactor=r,t.nodes.holder.appendChild(n),e()}).then(s).then(n).then(c).then(a).then(l).catch(function(){t.core.log("Can't draw editor interface")})};var n=function(){var e=t.draw.inlineToolbar();t.nodes.inlineToolbar.wrapper=e,t.nodes.inlineToolbar.buttons=t.draw.inlineToolbarButtons(),t.nodes.inlineToolbar.actions=t.draw.inlineToolbarActions(),t.nodes.inlineToolbar.wrapper.appendChild(t.nodes.inlineToolbar.buttons),t.nodes.inlineToolbar.wrapper.appendChild(t.nodes.inlineToolbar.actions),t.nodes.wrapper.appendChild(t.nodes.inlineToolbar.wrapper)},o=function(){var e=t.draw.toolbar(),n=i(),o=r();return e.appendChild(n),e.appendChild(o),t.nodes.toolbar=e,e},r=function(){var e=t.draw.toolbarContent(),n=t.draw.toolbox(),o=t.draw.plusButton();return e.appendChild(o),e.appendChild(n),t.nodes.toolbox=n,t.nodes.plusButton=o,e},i=function(){var e=t.draw.blockSettings(),n=t.draw.blockButtons(),o=t.draw.defaultSettings(),r=t.draw.settingsButton(),i=t.toolbar.settings.makeRemoveBlockButton(),a=t.draw.pluginsSettings();return e.appendChild(a),e.appendChild(o),n.appendChild(r),n.appendChild(i),n.appendChild(e),t.nodes.blockSettings=e,t.nodes.pluginSettings=a,t.nodes.defaultSettings=o,t.nodes.showSettingsButton=r,t.nodes.showTrashButton=i,n},a=function(){t.nodes.notifications=t.notifications.createHolder()},s=function(){var e,n,o;for(n in t.settings.tools)e=t.settings.tools[n],t.tools[n]=e,e.iconClassname||!e.displayInToolbox?"function"==typeof e.render?e.displayInToolbox&&(o=t.draw.toolbarButton(n,e.iconClassname),t.nodes.toolbox.appendChild(o),t.nodes.toolbarButtons[n]=o):t.core.log("render method missed. Tool %o skipped","warn",n):t.core.log("Toolbar icon classname missed. Tool %o skipped","warn",n)},c=function(){var e,n,o={bold:{icon:"ce-icon-bold",command:"bold"},italic:{icon:"ce-icon-italic",command:"italic"},underline:{icon:"ce-icon-underline",command:"underline"},link:{icon:"ce-icon-link",command:"createLink"}};for(var r in o)n=o[r],e=t.draw.toolbarButtonInline(r,n.icon),t.nodes.inlineToolbar.buttons.appendChild(e),t.ui.setInlineToolbarButtonBehaviour(e,n.command)},l=function(){t.core.log("ui.bindEvents fired","info"),t.listeners.add(document,"keydown",t.callback.globalKeydown,!1),t.listeners.add(t.nodes.redactor,"keydown",t.callback.redactorKeyDown,!1),t.listeners.add(document,"keyup",t.callback.globalKeyup,!1),t.listeners.add(t.nodes.redactor,"click",t.callback.redactorClicked,!1),t.listeners.add(t.nodes.plusButton,"click",t.callback.plusButtonClicked,!1),t.listeners.add(t.nodes.showSettingsButton,"click",t.callback.showSettingsButtonClicked,!1);for(var e in t.nodes.toolbarButtons)t.listeners.add(t.nodes.toolbarButtons[e],"click",t.callback.toolbarButtonClicked,!1)};return e.addBlockHandlers=function(e){e&&(t.listeners.add(e,"keydown",t.callback.blockKeydown,!1),t.listeners.add(e,"paste",t.callback.blockPasteCallback,!1),t.listeners.add(e,"mouseup",t.toolbar.inline.show,!1))},e.saveInputs=function(){var e=t.nodes.redactor;t.state.inputs=[];var n=e.querySelectorAll("[contenteditable], input, textarea");Array.prototype.map.call(n,function(e){e.type&&"text"!=e.type&&"textarea"!=e.type||t.state.inputs.push(e)})},e.addInitialBlock=function(){var e,n=t.settings.initialBlockPlugin;return t.tools[n]?(e=t.tools[n].render(),e.setAttribute("data-placeholder","Расскажите свою историю..."),t.content.insertBlock({type:n,block:e}),void t.content.workingNodeChanged(e)):void t.core.log("Plugin %o was not implemented and can't be used as initial block","warn",n)},e.setInlineToolbarButtonBehaviour=function(e,n){t.listeners.add(e,"mousedown",function(e){t.toolbar.inline.toolClicked(e,n)},!1)},e}({})},function(e,t){"use strict";e.exports=function(e){var t=codex.editor,n=null;return e.input=null,e.arguments=null,e.prepare=function(){var e=t.draw.node("INPUT","",{type:"file"});t.listeners.add(e,"change",t.transport.fileSelected),t.transport.input=e},e.clearInput=function(){e.input=null,e.prepare()},e.fileSelected=function(){var o,r=this,i=r.files,a=new FormData;if(t.transport.arguments.multiple===!0)for(o=0;o"+t.split("\n\n").join("

    ")+"

    ";var n,o,r,i,a=document.createElement("DIV"),s=document.createElement("DIV"),c=["DIV","P"];for(a.innerHTML=e,o=document.createElement("P"),n=0;n'}),o=t.draw.node("DIV","ce-toolbar__remove-confirmation",{}),r=t.draw.node("DIV","ce-toolbar__remove-confirm",{textContent:"Удалить блок"}),i=t.draw.node("DIV","ce-toolbar__remove-cancel",{textContent:"Отмена"});return t.listeners.add(n,"click",t.toolbar.settings.removeButtonClicked,!1),t.listeners.add(r,"click",t.toolbar.settings.confirmRemovingRequest,!1),t.listeners.add(i,"click",t.toolbar.settings.cancelRemovingRequest,!1),o.appendChild(r),o.appendChild(i),e.appendChild(n),e.appendChild(o),t.toolbar.settings.setting=n,t.toolbar.settings.actions=o,e},e.removeButtonClicked=function(){var e=t.toolbar.settings.actions;e.classList.contains("opened")?t.toolbar.settings.hideRemoveActions():t.toolbar.settings.showRemoveActions(),t.toolbar.toolbox.close(),t.toolbar.settings.close()},e.cancelRemovingRequest=function(){t.toolbar.settings.actions.classList.remove("opened")},e.confirmRemovingRequest=function(){var e,n=t.content.currentNode;n.remove(),e=t.nodes.redactor.childNodes.length,0===e&&(t.content.currentNode=null,t.ui.addInitialBlock()),t.ui.saveInputs(),t.toolbar.close()},e.showRemoveActions=function(){t.toolbar.settings.actions.classList.add("opened")},e.hideRemoveActions=function(){t.toolbar.settings.actions.classList.remove("opened")},e}({})},function(e,t){"use strict";e.exports=function(e){var t=codex.editor;e.buttonsOpened=null,e.actionsOpened=null,e.wrappersOffset=null,e.storedSelection=null,e.show=function(){var n,o=t.content.currentNode,r=o.dataset.tool;if(n=t.tools[r],n.showInlineToolbar){var i=e.getSelectionText(),a=t.nodes.inlineToolbar.wrapper;i.length>0&&(t.toolbar.inline.move(),a.classList.add("opened"),t.toolbar.inline.showButtons())}},e.close=function(){var e=t.nodes.inlineToolbar.wrapper;e.classList.remove("opened")},e.move=function(){this.wrappersOffset||(this.wrappersOffset=this.getWrappersOffset());var e,n,o=this.getSelectionCoords(),r=0,i=t.nodes.inlineToolbar.wrapper;0===i.offsetHeight&&(r=40),e=o.x-this.wrappersOffset.left,n=o.y+window.scrollY-this.wrappersOffset.top-r-i.offsetHeight,i.style.transform="translate3D("+Math.floor(e)+"px, "+Math.floor(n)+"px, 0)",t.toolbar.inline.closeButtons(),t.toolbar.inline.closeAction()},e.toolClicked=function(e,n){switch(n){case"createLink":t.toolbar.inline.createLinkAction(e,n);break;default:t.toolbar.inline.defaultToolAction(n)}t.nodes.inlineToolbar.buttons.childNodes.forEach(t.toolbar.inline.hightlight)},e.getWrappersOffset=function(){var e=t.nodes.wrapper,n=this.getOffset(e);return this.wrappersOffset=n,n},e.getOffset=function(e){for(var t=0,n=0;e&&!isNaN(e.offsetLeft)&&!isNaN(e.offsetTop);)t+=e.offsetLeft+e.clientLeft,n+=e.offsetTop+e.clientTop,e=e.offsetParent;return{top:n,left:t}},e.getSelectionCoords=function(){var e,t=document.selection,n=0,o=0;if(t)"Control"!=t.type&&(e=t.createRange(),e.collapse(!0),n=e.boundingLeft,o=e.boundingTop);else if(window.getSelection&&(t=window.getSelection(),t.rangeCount&&(e=t.getRangeAt(0).cloneRange(),e.getClientRects))){e.collapse(!0);var r=e.getClientRects()[0];if(!r)return;n=r.left,o=r.top}return{x:n,y:o}},e.getSelectionText=function(){var e="";return window.getSelection&&(e=window.getSelection().toString()),e},e.showButtons=function(){var e=t.nodes.inlineToolbar.buttons;e.classList.add("opened"),t.toolbar.inline.buttonsOpened=!0,t.nodes.inlineToolbar.buttons.childNodes.forEach(t.toolbar.inline.hightlight)},e.closeButtons=function(){var e=t.nodes.inlineToolbar.buttons;e.classList.remove("opened"),t.toolbar.inline.buttonsOpened=!1},e.showActions=function(){var e=t.nodes.inlineToolbar.actions;e.classList.add("opened"),t.toolbar.inline.actionsOpened=!0},e.closeAction=function(){var e=t.nodes.inlineToolbar.actions;e.innerHTML="",e.classList.remove("opened"),t.toolbar.inline.actionsOpened=!1};var n=function(e){if(e.keyCode==t.core.keys.ENTER){var n=t.content.currentNode,o=t.toolbar.inline.storedSelection;t.toolbar.inline.restoreSelection(n,o),t.toolbar.inline.setAnchor(this.value),e.preventDefault(),e.stopImmediatePropagation(),t.toolbar.inline.clearRange()}};return e.createLinkAction=function(e){var o=this.isLinkActive(),r=t.content.currentNode,i=t.toolbar.inline.saveSelection(r);if(t.toolbar.inline.storedSelection=i,o)t.toolbar.inline.restoreSelection(r,i),t.toolbar.inline.defaultToolAction("unlink");else{var a=t.draw.inputForLink();t.nodes.inlineToolbar.actions.appendChild(a),t.toolbar.inline.closeButtons(),t.toolbar.inline.showActions(),a.focus(),e.preventDefault(),t.listeners.add(a,"keydown",n,!1)}},e.isLinkActive=function(){var e=!1;return t.nodes.inlineToolbar.buttons.childNodes.forEach(function(t){var n=t.dataset.type;"link"==n&&t.classList.contains("hightlighted")&&(e=!0)}),e},e.defaultToolAction=function(e){document.execCommand(e,!1,null)},e.setAnchor=function(e){document.execCommand("createLink",!1,e),t.toolbar.inline.closeAction()},e.saveSelection=function(e){var t,n=window.getSelection().getRangeAt(0),o=n.cloneRange();return o.selectNodeContents(e),o.setEnd(n.startContainer,n.startOffset),t=o.toString().length,{start:t,end:t+n.toString().length}},e.restoreSelection=function(e,t){var n=document.createRange(),o=0;n.setStart(e,0),n.collapse(!0);for(var r,i,a=[e],s=!1,c=!1;!c&&(r=a.pop());)if(3==r.nodeType)i=o+r.length,!s&&t.start>=o&&t.start<=i&&(n.setStart(r,t.start-o),s=!0),s&&t.end>=o&&t.end<=i&&(n.setEnd(r,t.end-o),c=!0),o=i;else for(var l=r.childNodes.length;l--;)a.push(r.childNodes[l]);var d=window.getSelection();d.removeAllRanges(),d.addRange(n)},e.clearRange=function(){var e=window.getSelection();e.removeAllRanges()},e.hightlight=function(e){var n=e.dataset.type;document.queryCommandState(n)?t.toolbar.inline.setButtonHighlighted(e):t.toolbar.inline.removeButtonsHighLight(e);var o=window.getSelection(),r=o.anchorNode.parentNode;"A"==r.tagName&&"link"==n&&t.toolbar.inline.setButtonHighlighted(e)},e.setButtonHighlighted=function(e){if(e.classList.add("hightlighted"),"link"==e.dataset.type){var t=e.childNodes[0];t.classList.remove("ce-icon-link"),t.classList.add("ce-icon-unlink")}},e.removeButtonsHighLight=function(e){if(e.classList.remove("hightlighted"),"link"==e.dataset.type){var t=e.childNodes[0];t.classList.remove("ce-icon-unlink"),t.classList.add("ce-icon-link")}},e}({})},function(e,t){"use strict";e.exports=function(e){var t=codex.editor;return e.opened=!1,e.open=function(){t.toolbar.settings.opened&&t.toolbar.settings.close(),t.nodes.toolbox.classList.add("opened"),t.nodes.plusButton.classList.add("clicked"),t.toolbar.toolbox.opened=!0},e.close=function(){t.nodes.toolbox.classList.remove("opened"),t.nodes.plusButton.classList.remove("clicked"),t.toolbar.toolbox.opened=!1,t.toolbar.current=null},e.leaf=function(){var e=t.toolbar.current,n=Object.keys(t.tools),o=t.nodes.toolbarButtons,r=0,i=void 0,a=void 0,s=void 0;if(e)for(r=(n.indexOf(e)+1)%n.length,a=n[r];!t.tools[a].displayInToolbox;)r=(r+1)%n.length,a=n[r];else for(s in t.tools){if(t.tools[s].displayInToolbox)break;r++}i=n[r];for(var c in o)o[c].classList.remove("selected");o[i].classList.add("selected"),t.toolbar.current=i},e.toolClicked=function(e){var n,o,r,i=["image","link","list","instagram","twitter","embed"],a=t.tools[t.toolbar.current],s=t.content.currentNode,c=t.caret.inputIndex;n=a.render(),r={block:n,type:a.type,stretched:!1},s&&i.indexOf(s.dataset.tool)===-1&&""===s.textContent.trim()?t.content.switchBlock(s,n,a.type):(t.content.insertBlock(r),c++),o=a.appendCallback,o&&"function"==typeof o&&o.call(e),window.setTimeout(function(){t.caret.setToBlock(c)},10),t.content.workingNodeChanged(),t.toolbar.move()},e}({})},function(e,t){"use strict";e.exports=function(e){var t=codex.editor;e.globalKeydown=function(e){switch(e.keyCode){case t.core.keys.ENTER:o(e)}},e.redactorKeyDown=function(e){switch(e.keyCode){case t.core.keys.TAB:n(e);break;case t.core.keys.ENTER:i(e);break;case t.core.keys.ESC:a(e);break;default:c(e)}},e.globalKeyup=function(e){switch(e.keyCode){case t.core.keys.UP:case t.core.keys.LEFT:case t.core.keys.RIGHT:case t.core.keys.DOWN:s(e)}};var n=function(e){e.preventDefault();var n=t.content.currentNode.querySelectorAll("textarea, input"),o=!0,r=!t.content.currentNode.textContent.trim();Array.prototype.map.call(n,function(e){"textarea"!=e.type&&"text"!=e.type||(o=o&&!e.value.trim())});var i=r&&o;i&&(t.toolbar.opened||t.toolbar.open(),t.toolbar.opened&&!t.toolbar.toolbox.opened?t.toolbar.toolbox.open():t.toolbar.toolbox.leaf())},o=function(){t.content.editorAreaHightlighted&&(t.caret.inputIndex=-1,r())},r=function(){var e=t.settings.initialBlockPlugin;t.content.insertBlock({type:e,block:t.tools[e].render()},!0),t.toolbar.move(),t.toolbar.open()},i=function(e){"true"==e.target.contentEditable&&t.caret.saveCurrentInputIndex();var n=t.caret.getCurrentInputIndex()||0,o=t.content.currentNode,r=o.dataset.tool,i=t.toolbar.opened&&t.toolbar.current&&e.target==t.state.inputs[n],a=t.tools[r].enableLineBreaks,s=t.settings.initialBlockPlugin;if(i)return e.preventDefault(),t.toolbar.toolbox.toolClicked(e),t.toolbar.close(),e.stopPropagation(),void e.stopImmediatePropagation();if(e.shiftKey||a)return e.stopPropagation(),void e.stopImmediatePropagation();var c=window.getSelection(),l=c.anchorNode,d=t.caret.position.atTheEnd(),u=!1;if(e.shiftKey&&!a)return t.callback.enterPressedOnBlock(t.content.currentBlock,e),void e.preventDefault();if(u=l&&"true"!=l.parentNode.contentEditable,l.nodeType!=t.core.nodeTypes.TEXT||u||d){var p=t.content.isLastNode(l);p&&d&&(e.preventDefault(),e.stopPropagation(),e.stopImmediatePropagation(),t.core.log("ENTER clicked in last textNode. Create new BLOCK"),t.content.insertBlock({type:s,block:t.tools[s].render()},!0),t.toolbar.move(),t.toolbar.open(),t.toolbar.showPlusButton())}else e.preventDefault(),t.core.log("Splitting Text node..."),t.content.splitBlock(n),t.state.inputs[n+1].textContent.trim()||t.toolbar.showPlusButton();t.ui.saveInputs()},a=function(e){t.toolbar.close(),t.toolbar.toolbox.close(),e.preventDefault()},s=function(e){t.content.workingNodeChanged(),t.toolbar.close(),t.toolbar.move()},c=function(){t.toolbar.close(),t.toolbar.inline.actionsOpened||(t.toolbar.inline.close(),t.content.clearMark())};e.redactorClicked=function(e){l(),t.content.workingNodeChanged(e.target),t.ui.saveInputs();var n,o=t.toolbar.inline.getSelectionText();if(0===o.length&&t.toolbar.inline.close(),"true"==e.target.contentEditable&&t.caret.saveCurrentInputIndex(),null===t.content.currentNode){var r=t.state.inputs.length>0?t.state.inputs.length-1:0;if(t.state.inputs.length&&(n=t.content.getFirstLevelBlock(t.state.inputs[r])),t.state.inputs.length&&""===t.state.inputs[r].textContent&&n.dataset.tool==t.settings.initialBlockPlugin)t.caret.setToBlock(r);else{var i=t.settings.initialBlockPlugin;t.content.insertBlock({type:i,block:t.tools[i].render()}),1===t.state.inputs.length?t.caret.setToBlock(r):t.caret.setToNextBlock(r)}}else t.toolbar.settings.close(),t.toolbar.toolbox.close();t.toolbar.move(),t.toolbar.open();var a=!t.content.currentNode.textContent.trim(),s=t.content.currentNode.dataset.tool,c=s==t.settings.initialBlockPlugin;t.toolbar.hidePlusButton(),a||t.content.markBlock(),c&&a&&t.toolbar.showPlusButton()};var l=function(){var e=window.getSelection(),n=e.anchorNode,o=!1;if(0===e.rangeCount)t.content.editorAreaHightlighted=!0;else{for(t.core.isDomNode(n)||(n=n.parentNode),"true"==n.contentEditable&&(o=!0);"true"!=n.contentEditable&&(n=n.parentNode,"true"==n.contentEditable&&(o=!0),n!=document.body););t.content.editorAreaHightlighted=!o}};e.toolbarButtonClicked=function(e){var n=this;t.toolbar.current=n.dataset.type,t.toolbar.toolbox.toolClicked(e),t.toolbar.close()},e.plusButtonClicked=function(){t.nodes.toolbox.classList.contains("opened")?t.toolbar.toolbox.close():t.toolbar.toolbox.open()},e.blockKeydown=function(e){var n=e.target;switch(e.keyCode){case t.core.keys.DOWN:case t.core.keys.RIGHT:d(e);break;case t.core.keys.BACKSPACE:p(n,e);break;case t.core.keys.UP:case t.core.keys.LEFT:u(e)}};var d=function(e){var n,o=window.getSelection(),r=t.state.inputs,i=o.anchorNode;if(!i)return!1;for(;"true"!=i.contentEditable;)n=i.parentNode,i=n;for(var a=0;i!=r[a];)a++;if(!i.textContent)return void t.caret.setToNextBlock(a);var s,c,l=!1,d=!1;return s=i.childNodes[i.childNodes.length-1],c=t.core.isDomNode(s)?t.content.getDeepestTextNodeFromPosition(s,s.childNodes.length):s,l=o.anchorNode==c,d=c.length==o.anchorOffset,l&&d?void t.caret.setToNextBlock(a):(t.core.log("arrow [down|right] : caret does not reached the end"),!1)},u=function(e){var n,o=window.getSelection(),r=t.state.inputs,i=o.anchorNode;if(!i)return!1;if(0!==o.anchorOffset)return!1;for(;"true"!=i.contentEditable;)n=i.parentNode,i=n;for(var a=0;i!=r[a];)a++;var s,c,l=!1,d=!1;return i.textContent?(s=i.childNodes[0],c=t.core.isDomNode(s)?t.content.getDeepestTextNodeFromPosition(s,0):s,l=o.anchorNode==c,d=0===o.anchorOffset,void(l&&d&&t.caret.setToPreviousBlock(a))):void t.caret.setToPreviousBlock(a)},p=function(e,n){var o,r,i,a=t.caret.getCurrentInputIndex();if(f(n.target)){if(""!=n.target.value.trim())return;e.remove()}if(e.textContent.trim()){if(o=t.content.getRange(),r=o.endOffset-o.startOffset,!t.caret.position.atStart()||r||!t.state.inputs[a-1])return;t.content.mergeBlocks(a)}r||e.remove(),i=t.nodes.redactor.childNodes.length,0===i?(t.content.currentNode=null,t.ui.addInitialBlock(),t.ui.saveInputs(),window.setTimeout(function(){t.caret.setToPreviousBlock(1)},10)):0!==t.caret.inputIndex?t.caret.setToPreviousBlock(t.caret.inputIndex):t.caret.setToNextBlock(t.caret.inputIndex), t.toolbar.move(),t.toolbar.opened||t.toolbar.open(),t.ui.saveInputs(),n.preventDefault()};e.blockPasteCallback=function(e){if(!f(e.target)){e.preventDefault();var n=t.content.getEditableParent(e.target);if(n){var o,r,i=e.clipboardData.getData("text/html"),a=e.clipboardData.getData("text/plain"),s=t.draw.node("DIV","",{});o=t.sanitizer.clean(i),r=t.content.wrapTextWithParagraphs(o,a),s.innerHTML=r;var c=t.settings.initialBlockPlugin,l=t.content.currentNode.firstChild.firstChild;if(1==s.childNodes.length)return void t.caret.insertNode(document.createTextNode(s.firstChild.innerHTML));s.childNodes.forEach(function(e,n){return 0==n&&""===l.innerHTML.trim()?void t.content.switchBlock(t.content.currentNode,t.tools[c].render({text:e.innerHTML}),c):(t.content.insertBlock({type:c,block:t.tools[c].render({text:e.innerHTML})}),void t.caret.inputIndex++)}),t.caret.setToPreviousBlock(t.caret.getCurrentInputIndex()+1)}}},e.showSettingsButtonClicked=function(e){var n=t.content.currentNode.dataset.tool;t.toolbar.settings.toggle(n),t.toolbar.toolbox.close(),t.toolbar.settings.hideRemoveActions()};var f=function(e){var t=["INPUT","TEXTAREA"];return t.indexOf(e.tagName)!=-1};return e}({})},function(e,t){"use strict";e.exports=function(e){return e.wrapper=function(){var e=document.createElement("div");return e.className+="codex-editor",e},e.redactor=function(){var e=document.createElement("div");return e.className+="ce-redactor",e},e.ceBlock=function(){var e=document.createElement("DIV");return e.className+="ce_block",e},e.toolbar=function(){var e=document.createElement("div");return e.className+="ce-toolbar",e},e.toolbarContent=function(){var e=document.createElement("DIV");return e.classList.add("ce-toolbar__content"),e},e.inlineToolbar=function(){var e=document.createElement("DIV");return e.className+="ce-toolbar-inline",e},e.inlineToolbarButtons=function(){var e=document.createElement("DIV");return e.className+="ce-toolbar-inline__buttons",e},e.inlineToolbarActions=function(){var e=document.createElement("DIV");return e.className+="ce-toolbar-inline__actions",e},e.inputForLink=function(){var e=document.createElement("INPUT");return e.type="input",e.className+="inputForLink",e.placeholder="Вставьте ссылку ...",e.setAttribute("form","defaultForm"),e.setAttribute("autofocus","autofocus"),e},e.blockButtons=function(){var e=document.createElement("div");return e.className+="ce-toolbar__actions",e},e.blockSettings=function(){var e=document.createElement("div");return e.className+="ce-settings",e},e.defaultSettings=function(){var e=document.createElement("div");return e.classList.add("ce-settings_default"),e},e.pluginsSettings=function(){var e=document.createElement("div");return e.classList.add("ce-settings_plugin"),e},e.plusButton=function(){var e=document.createElement("span");return e.className="ce-toolbar__plus",e},e.settingsButton=function(){var e=document.createElement("span");return e.className="ce-toolbar__settings-btn",e.innerHTML='',e},e.toolbox=function(){var e=document.createElement("div");return e.className="ce-toolbar__tools",e},e.toolbarButton=function(e,t){var n=document.createElement("li"),o=document.createElement("i"),r=document.createElement("span");return n.dataset.type=e,n.setAttribute("title",e),o.classList.add(t),r.classList.add("ce_toolbar_tools--title"),n.appendChild(o),n.appendChild(r),n},e.toolbarButtonInline=function(e,t){var n=document.createElement("BUTTON"),o=document.createElement("I");return n.type="button",n.dataset.type=e,o.classList.add(t),n.appendChild(o),n},e.block=function(e,t){var n=document.createElement(e);return n.innerHTML=t||"",n},e.node=function(e,t,n){var o=document.createElement(e);if(t&&(o.className=t),n)for(var r in n)o[r]=n[r];return o},e.unavailableBlock=function(){var e=document.createElement("DIV");return e.classList.add("cdx-unavailable-block"),e},e}({})},function(e,t){"use strict";e.exports=function(e){var t=codex.editor;return e.inputIndex=null,e.offset=null,e.focusedNodeIndex=null,e.set=function(n,o,r){r=r||e.offset||0,o=o||e.focusedNodeIndex||0;var i,a=n.childNodes;if(i=0===a.length?n:a[o],"true"!=n.contentEditable)return void n.focus();t.core.isDomNode(i)&&(i=t.content.getDeepestTextNodeFromPosition(i,i.childNodes.length));var s=document.createRange(),c=window.getSelection();window.setTimeout(function(){s.setStart(i,r),s.setEnd(i,r),c.removeAllRanges(),c.addRange(s),t.caret.saveCurrentInputIndex()},20)},e.saveCurrentInputIndex=function(){var n,o=window.getSelection(),r=t.state.inputs,i=o.anchorNode;if(i){for(;"true"!=i.contentEditable;)n=i.parentNode,i=n;for(var a=0;i!=r[a];)a++;e.inputIndex=a}},e.getCurrentInputIndex=function(){return e.inputIndex},e.setToNextBlock=function(e){var n=t.state.inputs,o=n[e+1];if(!o)return void t.core.log("We are reached the end");if(!o.childNodes.length){var r=document.createTextNode("");o.appendChild(r)}t.caret.inputIndex=e+1,t.caret.set(o,0,0),t.content.workingNodeChanged(o)},e.setToBlock=function(e){var n=t.state.inputs,o=n[e];if(o){if(!o.childNodes.length){var r=document.createTextNode("");o.appendChild(r)}t.caret.inputIndex=e,t.caret.set(o,0,0),t.content.workingNodeChanged(o)}},e.setToPreviousBlock=function(e){e=e||0;var n,o,r,i=t.state.inputs,a=i[e-1];return a?(n=t.content.getDeepestTextNodeFromPosition(a,a.childNodes.length),o=n.length,a.childNodes.length||(r=document.createTextNode(""),a.appendChild(r)),t.caret.inputIndex=e-1,t.caret.set(a,a.childNodes.length-1,o),void t.content.workingNodeChanged(i[e-1])):void t.core.log("We are reached first node")},e.position={atStart:function(){var e=window.getSelection(),n=e.anchorOffset,o=e.anchorNode,r=t.content.getFirstLevelBlock(o),i=r.childNodes[0];t.core.isDomNode(o)||(o=o.parentNode);var a=o===i.childNodes[0],s=0===n;return a&&s},atTheEnd:function(){var e=window.getSelection(),t=e.anchorOffset,n=e.anchorNode;return!n||!n.length||t===n.length}},e.insertNode=function(e){var t,n;t=window.getSelection(),n=t.getRangeAt(0),n.deleteContents(),n.insertNode(e),n.setStartAfter(e),n.collapse(!0),t.removeAllRanges(),t.addRange(n)},e}({})},function(e,t){"use strict";e.exports=function(e){var t=codex.editor,n=[],o=function(e){n.push(e);for(var t=0;t5;)"confirm"!=n[t].type&&"prompt"!=n[t].type?(n[t].close(),n.splice(t,1)):t++};return e.createHolder=function(){var e=t.draw.node("DIV","cdx-notifications-block");return t.nodes.notifications=document.body.appendChild(e),e},e.errorThrown=function(e,n){t.notifications.notification({message:"This action is not available currently",type:n.type})},e.notification=function(e){function n(e){if(!e||!e.message)return void t.core.log("Can't create notification. Message is missed");e.type=e.type||"alert",e.time=1e3*e.time||1e4;var n=t.draw.node("DIV","cdx-notification"),o=t.draw.node("DIV","cdx-notification__message"),r=t.draw.node("INPUT","cdx-notification__input"),f=t.draw.node("SPAN","cdx-notification__ok-btn"),g=t.draw.node("SPAN","cdx-notification__cancel-btn");o.textContent=e.message,f.textContent=e.okMsg||"ОК",g.textContent=e.cancelMsg||"Отмена",t.listeners.add(f,"click",u),t.listeners.add(g,"click",p),n.appendChild(o),"prompt"==e.type&&n.appendChild(r),n.appendChild(f),"prompt"!=e.type&&"confirm"!=e.type||n.appendChild(g),n.classList.add("cdx-notification-"+e.type),n.dataset.type=e.type,a=n,c=e.type,l=e.confirm,s=e.cancel,d=r,"prompt"!=e.type&&"confirm"!=e.type&&window.setTimeout(i,e.time)}function r(){t.nodes.notifications.appendChild(a),d.focus(),t.nodes.notifications.classList.add("cdx-notification__notification-appending"),window.setTimeout(function(){t.nodes.notifications.classList.remove("cdx-notification__notification-appending")},100),o({type:c,close:i})}function i(){a.remove()}var a=null,s=null,c=null,l=null,d=null,u=function(){if(i(),"function"==typeof l)return"prompt"==c?void l(d.value):void l()},p=function(){i(),"function"==typeof s&&s()};return e&&(n(e),r()),{create:n,send:r,close:i}},e.clear=function(){t.nodes.notifications.innerHTML="",n=[]},e}({})},function(e,t){"use strict";e.exports=function(e){var t=codex.editor;return e.insertPastedContent=function(e,n){t.content.insertBlock({type:e.type,block:e.render({text:n.innerHTML})})},e.isFirstLevelBlock=function(e){return e.nodeType==t.core.nodeTypes.TAG&&e.classList.contains(t.ui.className.BLOCK_CLASSNAME)},e}({})},function(e,t,n){"use strict";e.exports=function(e){var t=n(18),o=codex.editor;e.prepare=function(){o.settings.sanitizer&&!o.core.isEmpty(o.settings.sanitizer)&&(r.CUSTOM=o.settings.sanitizer)};var r={CUSTOM:null,BASIC:{tags:{p:{},a:{href:!0,target:"_blank",rel:"nofollow"},i:{},b:{},strong:{},em:{},span:{}}}};e.Config=r;var i=function(e){var n=e||r.CUSTOM||r.BASIC;return new t(n)};return e.clean=function(e,t){var n=i(t);return n.clean(e)},e}({})},function(e,t,n){var o,r;!function(i,a){o=a,r="function"==typeof o?o.call(t,n,t,e):o,!(void 0!==r&&(e.exports=r))}(this,function(){function e(e){var t=e.tags,n=Object.keys(t),o=n.map(function(e){return typeof t[e]}).every(function(e){return"object"===e||"boolean"===e||"function"===e});if(!o)throw new Error("The configuration was invalid");this.config=e}function t(e){return s.indexOf(e.nodeName)!==-1}function n(e){return c.indexOf(e.nodeName)!==-1}function o(e){return document.createTreeWalker(e,NodeFilter.SHOW_TEXT|NodeFilter.SHOW_ELEMENT|NodeFilter.SHOW_COMMENT,null,!1)}function r(e,t,n){return"function"==typeof e.tags[t]?e.tags[t](n):e.tags[t]}function i(e,t){return"undefined"==typeof t||"boolean"==typeof t&&!t}function a(e,t,n){var o=e.name.toLowerCase();return t!==!0&&("function"==typeof t[o]?!t[o](e.value,n):"undefined"==typeof t[o]||(t[o]===!1||"string"==typeof t[o]&&t[o]!==e.value))}var s=["P","LI","TD","TH","DIV","H1","H2","H3","H4","H5","H6","PRE"],c=["A","B","STRONG","I","EM","SUB","SUP","U","STRIKE"];return e.prototype.clean=function(e){var t=document.createElement("div");return t.innerHTML=e,this._sanitize(t),t.innerHTML},e.prototype._sanitize=function(e){var s=o(e),c=s.firstChild();if(c)do if(!c._sanitized)if(c.nodeType!==Node.TEXT_NODE){if(c.nodeType===Node.COMMENT_NODE){e.removeChild(c),this._sanitize(e);break}var l,d=n(c);d&&(l=Array.prototype.some.call(c.childNodes,t));var u=!!e.parentNode,p=t(e)&&t(c)&&u,f=c.nodeName.toLowerCase(),g=r(this.config,f,c),h=d&&l;if(h||i(c,g)||!this.config.keepNestedBlockElements&&p){if("SCRIPT"!==c.nodeName&&"STYLE"!==c.nodeName)for(;c.childNodes.length>0;)e.insertBefore(c.childNodes[0],c);e.removeChild(c),this._sanitize(e);break}for(var v=0;v0&&t.splice(s,1)}},e.removeAll=function(){t.map(function(t){e.remove(t.element,t.type,t.handler)})},e.get=function(t,n,o){return e.search.all(t,n,o)},e}({})},function(e,t){"use strict";var n="function"==typeof Symbol&&"symbol"==typeof Symbol.iterator?function(e){return typeof e}:function(e){return e&&"function"==typeof Symbol&&e.constructor===Symbol&&e!==Symbol.prototype?"symbol":typeof e};e.exports=function(e){var t=codex.editor;return e.removeNodes=function(){t.nodes.wrapper.remove(),t.nodes.notifications.remove()},e.destroyPlugins=function(){for(var e in t.tools)"function"==typeof t.tools[e].destroy&&t.tools[e].destroy()},e.destroyScripts=function(){for(var e=document.getElementsByTagName("SCRIPT"),n=0;n nodeList\r\n\t */\n\t renderer.appendBlocks = function (data) {\n\t\n\t var blocks = data.items;\n\t\n\t /**\r\n\t * Sequence of one-by-one blocks appending\r\n\t * Uses to save blocks order after async-handler\r\n\t */\n\t var nodeSequence = Promise.resolve();\n\t\n\t for (var index = 0; index < blocks.length; index++) {\n\t\n\t /** Add node to sequence at specified index */\n\t editor.renderer.appendNodeAtIndex(nodeSequence, blocks, index);\n\t }\n\t };\n\t\n\t /**\r\n\t * Append node at specified index\r\n\t */\n\t renderer.appendNodeAtIndex = function (nodeSequence, blocks, index) {\n\t\n\t /** We need to append node to sequence */\n\t nodeSequence\n\t\n\t /** first, get node async-aware */\n\t .then(function () {\n\t\n\t return editor.renderer.getNodeAsync(blocks, index);\n\t })\n\t\n\t /**\r\n\t * second, compose editor-block from JSON object\r\n\t */\n\t .then(editor.renderer.createBlockFromData)\n\t\n\t /**\r\n\t * now insert block to redactor\r\n\t */\n\t .then(function (blockData) {\n\t\n\t /**\r\n\t * blockData has 'block', 'type' and 'stretched' information\r\n\t */\n\t editor.content.insertBlock(blockData);\n\t\n\t /** Pass created block to next step */\n\t return blockData.block;\n\t })\n\t\n\t /** Log if something wrong with node */\n\t .catch(function (error) {\n\t\n\t editor.core.log('Node skipped while parsing because %o', 'error', error);\n\t });\n\t };\n\t\n\t /**\r\n\t * Asynchronously returns block data from blocksList by index\r\n\t * @return Promise to node\r\n\t */\n\t renderer.getNodeAsync = function (blocksList, index) {\n\t\n\t return Promise.resolve().then(function () {\n\t\n\t return {\n\t tool: blocksList[index],\n\t position: index\n\t };\n\t });\n\t };\n\t\n\t /**\r\n\t * Creates editor block by JSON-data\r\n\t *\r\n\t * @uses render method of each plugin\r\n\t *\r\n\t * @param {Object} toolData.tool\r\n\t * { header : {\r\n\t * text: '',\r\n\t * type: 'H3', ...\r\n\t * }\r\n\t * }\r\n\t * @param {Number} toolData.position - index in input-blocks array\r\n\t * @return {Object} with type and Element\r\n\t */\n\t renderer.createBlockFromData = function (toolData) {\n\t\n\t /** New parser */\n\t var block,\n\t tool = toolData.tool,\n\t pluginName = tool.type;\n\t\n\t /** Get first key of object that stores plugin name */\n\t // for (var pluginName in blockData) break;\n\t\n\t /** Check for plugin existance */\n\t if (!editor.tools[pluginName]) {\n\t\n\t throw Error('Plugin \\xAB' + pluginName + '\\xBB not found');\n\t }\n\t\n\t /** Check for plugin having render method */\n\t if (typeof editor.tools[pluginName].render != 'function') {\n\t\n\t throw Error('Plugin \\xAB' + pluginName + '\\xBB must have \\xABrender\\xBB method');\n\t }\n\t\n\t if (editor.tools[pluginName].available === false) {\n\t\n\t block = editor.draw.unavailableBlock();\n\t\n\t block.innerHTML = editor.tools[pluginName].loadingMessage;\n\t\n\t /**\r\n\t * Saver will extract data from initial block data by position in array\r\n\t */\n\t block.dataset.inputPosition = toolData.position;\n\t } else {\n\t\n\t /** New Parser */\n\t block = editor.tools[pluginName].render(tool.data);\n\t }\n\t\n\t /** is first-level block stretched */\n\t var stretched = editor.tools[pluginName].isStretched || false;\n\t\n\t /** Retrun type and block */\n\t return {\n\t type: pluginName,\n\t block: block,\n\t stretched: stretched\n\t };\n\t };\n\t\n\t return renderer;\n\t}({});\n\n/***/ },\n/* 6 */\n/***/ function(module, exports) {\n\n\t'use strict';\n\t\n\t/**\r\n\t * Codex Editor Saver\r\n\t *\r\n\t * @author Codex Team\r\n\t * @version 1.0.2\r\n\t */\n\t\n\tmodule.exports = function (saver) {\n\t\n\t var editor = codex.editor;\n\t\n\t /**\r\n\t * Saves blocks\r\n\t * @private\r\n\t */\n\t saver.saveBlocks = function () {\n\t\n\t /** Save html content of redactor to memory */\n\t editor.state.html = editor.nodes.redactor.innerHTML;\n\t\n\t /** Empty jsonOutput state */\n\t editor.state.jsonOutput = [];\n\t\n\t Promise.resolve().then(function () {\n\t\n\t return editor.nodes.redactor.childNodes;\n\t })\n\t /** Making a sequence from separate blocks */\n\t .then(editor.saver.makeQueue).then(function () {\n\t // editor.nodes.textarea.innerHTML = editor.state.html;\n\t }).catch(function (error) {\n\t\n\t editor.core.log(error);\n\t });\n\t };\n\t\n\t saver.makeQueue = function (blocks) {\n\t\n\t var queue = Promise.resolve();\n\t\n\t for (var index = 0; index < blocks.length; index++) {\n\t\n\t /** Add node to sequence at specified index */\n\t editor.saver.getBlockData(queue, blocks, index);\n\t }\n\t };\n\t\n\t /** Gets every block and makes From Data */\n\t saver.getBlockData = function (queue, blocks, index) {\n\t\n\t queue.then(function () {\n\t\n\t return editor.saver.getNodeAsync(blocks, index);\n\t }).then(editor.saver.makeFormDataFromBlocks);\n\t };\n\t\n\t /**\r\n\t * Asynchronously returns block data from blocksList by index\r\n\t * @return Promise to node\r\n\t */\n\t saver.getNodeAsync = function (blocksList, index) {\n\t\n\t return Promise.resolve().then(function () {\n\t\n\t return blocksList[index];\n\t });\n\t };\n\t\n\t saver.makeFormDataFromBlocks = function (block) {\n\t\n\t var pluginName = block.dataset.tool;\n\t\n\t /** Check for plugin existance */\n\t if (!editor.tools[pluginName]) {\n\t\n\t throw Error('Plugin \\xAB' + pluginName + '\\xBB not found');\n\t }\n\t\n\t /** Check for plugin having render method */\n\t if (typeof editor.tools[pluginName].save != 'function') {\n\t\n\t throw Error('Plugin \\xAB' + pluginName + '\\xBB must have save method');\n\t }\n\t\n\t /** Result saver */\n\t var blockContent = block.childNodes[0],\n\t pluginsContent = blockContent.childNodes[0],\n\t savedData,\n\t position,\n\t output;\n\t\n\t /** If plugin wasn't available then return data from cache */\n\t if (editor.tools[pluginName].available === false) {\n\t\n\t position = pluginsContent.dataset.inputPosition;\n\t savedData = codex.editor.state.blocks.items[position].data;\n\t } else {\n\t\n\t savedData = editor.tools[pluginName].save(pluginsContent);\n\t\n\t if (editor.tools[pluginName].validate) {\n\t\n\t var result = editor.tools[pluginName].validate(savedData);\n\t\n\t /**\r\n\t * Do not allow invalid data\r\n\t */\n\t if (!result) return;\n\t }\n\t }\n\t\n\t output = {\n\t type: pluginName,\n\t data: savedData\n\t };\n\t\n\t editor.state.jsonOutput.push(output);\n\t };\n\t\n\t return saver;\n\t}({});\n\n/***/ },\n/* 7 */\n/***/ function(module, exports) {\n\n\t'use strict';\n\t\n\t/**\r\n\t * Codex Editor Content Module\r\n\t * Works with DOM\r\n\t *\r\n\t * @module Codex Editor content module\r\n\t *\r\n\t * @author Codex Team\r\n\t * @version 1.3.13\r\n\t *\r\n\t * @description Module works with Elements that have been appended to the main DOM\r\n\t */\n\t\n\tmodule.exports = function (content) {\n\t\n\t var editor = codex.editor;\n\t\n\t /**\r\n\t * Links to current active block\r\n\t * @type {null | Element}\r\n\t */\n\t content.currentNode = null;\n\t\n\t /**\r\n\t * clicked in redactor area\r\n\t * @type {null | Boolean}\r\n\t */\n\t content.editorAreaHightlighted = null;\n\t\n\t /**\r\n\t * @deprecated\r\n\t * Synchronizes redactor with original textarea\r\n\t */\n\t content.sync = function () {\n\t\n\t editor.core.log('syncing...');\n\t\n\t /**\r\n\t * Save redactor content to editor.state\r\n\t */\n\t editor.state.html = editor.nodes.redactor.innerHTML;\n\t };\n\t\n\t /**\r\n\t * Appends background to the block\r\n\t *\r\n\t * @description add CSS class to highlight visually first-level block area\r\n\t */\n\t content.markBlock = function () {\n\t\n\t editor.content.currentNode.classList.add(editor.ui.className.BLOCK_HIGHLIGHTED);\n\t };\n\t\n\t /**\r\n\t * Clear background\r\n\t *\r\n\t * @description clears styles that highlights block\r\n\t */\n\t content.clearMark = function () {\n\t\n\t if (editor.content.currentNode) {\n\t\n\t editor.content.currentNode.classList.remove(editor.ui.className.BLOCK_HIGHLIGHTED);\n\t }\n\t };\n\t\n\t /**\r\n\t * Finds first-level block\r\n\t *\r\n\t * @param {Element} node - selected or clicked in redactors area node\r\n\t * @protected\r\n\t *\r\n\t * @description looks for first-level block.\r\n\t * gets parent while node is not first-level\r\n\t */\n\t content.getFirstLevelBlock = function (node) {\n\t\n\t if (!editor.core.isDomNode(node)) {\n\t\n\t node = node.parentNode;\n\t }\n\t\n\t if (node === editor.nodes.redactor || node === document.body) {\n\t\n\t return null;\n\t } else {\n\t\n\t while (!node.classList.contains(editor.ui.className.BLOCK_CLASSNAME)) {\n\t\n\t node = node.parentNode;\n\t }\n\t\n\t return node;\n\t }\n\t };\n\t\n\t /**\r\n\t * Trigger this event when working node changed\r\n\t * @param {Element} targetNode - first-level of this node will be current\r\n\t * @protected\r\n\t *\r\n\t * @description If targetNode is first-level then we set it as current else we look for parents to find first-level\r\n\t */\n\t content.workingNodeChanged = function (targetNode) {\n\t\n\t /** Clear background from previous marked block before we change */\n\t editor.content.clearMark();\n\t\n\t if (!targetNode) {\n\t\n\t return;\n\t }\n\t\n\t content.currentNode = content.getFirstLevelBlock(targetNode);\n\t };\n\t\n\t /**\r\n\t * Replaces one redactor block with another\r\n\t * @protected\r\n\t * @param {Element} targetBlock - block to replace. Mostly currentNode.\r\n\t * @param {Element} newBlock\r\n\t * @param {string} newBlockType - type of new block; we need to store it to data-attribute\r\n\t *\r\n\t * [!] Function does not saves old block content.\r\n\t * You can get it manually and pass with newBlock.innerHTML\r\n\t */\n\t content.replaceBlock = function (targetBlock, newBlock) {\n\t\n\t if (!targetBlock || !newBlock) {\n\t\n\t editor.core.log('replaceBlock: missed params');\n\t return;\n\t }\n\t\n\t /** If target-block is not a frist-level block, then we iterate parents to find it */\n\t while (!targetBlock.classList.contains(editor.ui.className.BLOCK_CLASSNAME)) {\n\t\n\t targetBlock = targetBlock.parentNode;\n\t }\n\t\n\t /** Replacing */\n\t editor.nodes.redactor.replaceChild(newBlock, targetBlock);\n\t\n\t /**\r\n\t * Set new node as current\r\n\t */\n\t editor.content.workingNodeChanged(newBlock);\n\t\n\t /**\r\n\t * Add block handlers\r\n\t */\n\t editor.ui.addBlockHandlers(newBlock);\n\t\n\t /**\r\n\t * Save changes\r\n\t */\n\t editor.ui.saveInputs();\n\t };\n\t\n\t /**\r\n\t * @protected\r\n\t *\r\n\t * Inserts new block to redactor\r\n\t * Wrapps block into a DIV with BLOCK_CLASSNAME class\r\n\t *\r\n\t * @param blockData {object}\r\n\t * @param blockData.block {Element} element with block content\r\n\t * @param blockData.type {string} block plugin\r\n\t * @param needPlaceCaret {bool} pass true to set caret in new block\r\n\t *\r\n\t */\n\t content.insertBlock = function (blockData, needPlaceCaret) {\n\t\n\t var workingBlock = editor.content.currentNode,\n\t newBlockContent = blockData.block,\n\t blockType = blockData.type,\n\t isStretched = blockData.stretched;\n\t\n\t var newBlock = composeNewBlock_(newBlockContent, blockType, isStretched);\n\t\n\t if (workingBlock) {\n\t\n\t editor.core.insertAfter(workingBlock, newBlock);\n\t } else {\n\t\n\t /**\r\n\t * If redactor is empty, append as first child\r\n\t */\n\t editor.nodes.redactor.appendChild(newBlock);\n\t }\n\t\n\t /**\r\n\t * Block handler\r\n\t */\n\t editor.ui.addBlockHandlers(newBlock);\n\t\n\t /**\r\n\t * Set new node as current\r\n\t */\n\t editor.content.workingNodeChanged(newBlock);\n\t\n\t /**\r\n\t * Save changes\r\n\t */\n\t editor.ui.saveInputs();\n\t\n\t if (needPlaceCaret) {\n\t\n\t /**\r\n\t * If we don't know input index then we set default value -1\r\n\t */\n\t var currentInputIndex = editor.caret.getCurrentInputIndex() || -1;\n\t\n\t if (currentInputIndex == -1) {\n\t\n\t var editableElement = newBlock.querySelector('[contenteditable]'),\n\t emptyText = document.createTextNode('');\n\t\n\t editableElement.appendChild(emptyText);\n\t editor.caret.set(editableElement, 0, 0);\n\t\n\t editor.toolbar.move();\n\t editor.toolbar.showPlusButton();\n\t } else {\n\t\n\t if (currentInputIndex === editor.state.inputs.length - 1) return;\n\t\n\t /** Timeout for browsers execution */\n\t window.setTimeout(function () {\n\t\n\t /** Setting to the new input */\n\t editor.caret.setToNextBlock(currentInputIndex);\n\t editor.toolbar.move();\n\t editor.toolbar.open();\n\t }, 10);\n\t }\n\t }\n\t\n\t /**\r\n\t * Block is inserted, wait for new click that defined focusing on editors area\r\n\t * @type {boolean}\r\n\t */\n\t content.editorAreaHightlighted = false;\n\t };\n\t\n\t /**\r\n\t * Replaces blocks with saving content\r\n\t * @protected\r\n\t * @param {Element} noteToReplace\r\n\t * @param {Element} newNode\r\n\t * @param {Element} blockType\r\n\t */\n\t content.switchBlock = function (blockToReplace, newBlock, tool) {\n\t\n\t tool = tool || editor.content.currentNode.dataset.tool;\n\t var newBlockComposed = composeNewBlock_(newBlock, tool);\n\t\n\t /** Replacing */\n\t editor.content.replaceBlock(blockToReplace, newBlockComposed);\n\t\n\t /** Save new Inputs when block is changed */\n\t editor.ui.saveInputs();\n\t };\n\t\n\t /**\r\n\t * Iterates between child noted and looking for #text node on deepest level\r\n\t * @protected\r\n\t *\r\n\t * @param {Element} block - node where find\r\n\t * @param {int} postiton - starting postion\r\n\t * Example: childNodex.length to find from the end\r\n\t * or 0 to find from the start\r\n\t * @return {Text} block\r\n\t * @uses DFS\r\n\t */\n\t content.getDeepestTextNodeFromPosition = function (block, position) {\n\t\n\t /**\r\n\t * Clear Block from empty and useless spaces with trim.\r\n\t * Such nodes we should remove\r\n\t */\n\t var blockChilds = block.childNodes,\n\t index,\n\t node,\n\t text;\n\t\n\t for (index = 0; index < blockChilds.length; index++) {\n\t\n\t node = blockChilds[index];\n\t\n\t if (node.nodeType == editor.core.nodeTypes.TEXT) {\n\t\n\t text = node.textContent.trim();\n\t\n\t /** Text is empty. We should remove this child from node before we start DFS\r\n\t * decrease the quantity of childs.\r\n\t */\n\t if (text === '') {\n\t\n\t block.removeChild(node);\n\t position--;\n\t }\n\t }\n\t }\n\t\n\t if (block.childNodes.length === 0) {\n\t\n\t return document.createTextNode('');\n\t }\n\t\n\t /** Setting default position when we deleted all empty nodes */\n\t if (position < 0) position = 1;\n\t\n\t var lookingFromStart = false;\n\t\n\t /** For looking from START */\n\t if (position === 0) {\n\t\n\t lookingFromStart = true;\n\t position = 1;\n\t }\n\t\n\t while (position) {\n\t\n\t /** initial verticle of node. */\n\t if (lookingFromStart) {\n\t\n\t block = block.childNodes[0];\n\t } else {\n\t\n\t block = block.childNodes[position - 1];\n\t }\n\t\n\t if (block.nodeType == editor.core.nodeTypes.TAG) {\n\t\n\t position = block.childNodes.length;\n\t } else if (block.nodeType == editor.core.nodeTypes.TEXT) {\n\t\n\t position = 0;\n\t }\n\t }\n\t\n\t return block;\n\t };\n\t\n\t /**\r\n\t * @private\r\n\t * @param {Element} block - current plugins render\r\n\t * @param {String} tool - plugins name\r\n\t * @param {Boolean} isStretched - make stretched block or not\r\n\t *\r\n\t * @description adds necessary information to wrap new created block by first-level holder\r\n\t */\n\t var composeNewBlock_ = function composeNewBlock_(block, tool, isStretched) {\n\t\n\t var newBlock = editor.draw.node('DIV', editor.ui.className.BLOCK_CLASSNAME, {}),\n\t blockContent = editor.draw.node('DIV', editor.ui.className.BLOCK_CONTENT, {});\n\t\n\t blockContent.appendChild(block);\n\t newBlock.appendChild(blockContent);\n\t\n\t if (isStretched) {\n\t\n\t blockContent.classList.add(editor.ui.className.BLOCK_STRETCHED);\n\t }\n\t\n\t newBlock.dataset.tool = tool;\n\t return newBlock;\n\t };\n\t\n\t /**\r\n\t * Returns Range object of current selection\r\n\t * @protected\r\n\t */\n\t content.getRange = function () {\n\t\n\t var selection = window.getSelection().getRangeAt(0);\n\t\n\t return selection;\n\t };\n\t\n\t /**\r\n\t * Divides block in two blocks (after and before caret)\r\n\t *\r\n\t * @protected\r\n\t * @param {int} inputIndex - target input index\r\n\t *\r\n\t * @description splits current input content to the separate blocks\r\n\t * When enter is pressed among the words, that text will be splited.\r\n\t */\n\t content.splitBlock = function (inputIndex) {\n\t\n\t var selection = window.getSelection(),\n\t anchorNode = selection.anchorNode,\n\t anchorNodeText = anchorNode.textContent,\n\t caretOffset = selection.anchorOffset,\n\t textBeforeCaret,\n\t textNodeBeforeCaret,\n\t textAfterCaret,\n\t textNodeAfterCaret;\n\t\n\t var currentBlock = editor.content.currentNode.querySelector('[contentEditable]');\n\t\n\t textBeforeCaret = anchorNodeText.substring(0, caretOffset);\n\t textAfterCaret = anchorNodeText.substring(caretOffset);\n\t\n\t textNodeBeforeCaret = document.createTextNode(textBeforeCaret);\n\t\n\t if (textAfterCaret) {\n\t\n\t textNodeAfterCaret = document.createTextNode(textAfterCaret);\n\t }\n\t\n\t var previousChilds = [],\n\t nextChilds = [],\n\t reachedCurrent = false;\n\t\n\t if (textNodeAfterCaret) {\n\t\n\t nextChilds.push(textNodeAfterCaret);\n\t }\n\t\n\t for (var i = 0, child; !!(child = currentBlock.childNodes[i]); i++) {\n\t\n\t if (child != anchorNode) {\n\t\n\t if (!reachedCurrent) {\n\t\n\t previousChilds.push(child);\n\t } else {\n\t\n\t nextChilds.push(child);\n\t }\n\t } else {\n\t\n\t reachedCurrent = true;\n\t }\n\t }\n\t\n\t /** Clear current input */\n\t editor.state.inputs[inputIndex].innerHTML = '';\n\t\n\t /**\r\n\t * Append all childs founded before anchorNode\r\n\t */\n\t var previousChildsLength = previousChilds.length;\n\t\n\t for (i = 0; i < previousChildsLength; i++) {\n\t\n\t editor.state.inputs[inputIndex].appendChild(previousChilds[i]);\n\t }\n\t\n\t editor.state.inputs[inputIndex].appendChild(textNodeBeforeCaret);\n\t\n\t /**\r\n\t * Append text node which is after caret\r\n\t */\n\t var nextChildsLength = nextChilds.length,\n\t newNode = document.createElement('div');\n\t\n\t for (i = 0; i < nextChildsLength; i++) {\n\t\n\t newNode.appendChild(nextChilds[i]);\n\t }\n\t\n\t newNode = newNode.innerHTML;\n\t\n\t /** This type of block creates when enter is pressed */\n\t var NEW_BLOCK_TYPE = editor.settings.initialBlockPlugin;\n\t\n\t /**\r\n\t * Make new paragraph with text after caret\r\n\t */\n\t editor.content.insertBlock({\n\t type: NEW_BLOCK_TYPE,\n\t block: editor.tools[NEW_BLOCK_TYPE].render({\n\t text: newNode\n\t })\n\t }, true);\n\t };\n\t\n\t /**\r\n\t * Merges two blocks — current and target\r\n\t * If target index is not exist, then previous will be as target\r\n\t *\r\n\t * @protected\r\n\t * @param {int} currentInputIndex\r\n\t * @param {int} targetInputIndex\r\n\t *\r\n\t * @description gets two inputs indexes and merges into one\r\n\t */\n\t content.mergeBlocks = function (currentInputIndex, targetInputIndex) {\n\t\n\t /** If current input index is zero, then prevent method execution */\n\t if (currentInputIndex === 0) {\n\t\n\t return;\n\t }\n\t\n\t var targetInput,\n\t currentInputContent = editor.state.inputs[currentInputIndex].innerHTML;\n\t\n\t if (!targetInputIndex) {\n\t\n\t targetInput = editor.state.inputs[currentInputIndex - 1];\n\t } else {\n\t\n\t targetInput = editor.state.inputs[targetInputIndex];\n\t }\n\t\n\t targetInput.innerHTML += currentInputContent;\n\t };\n\t\n\t /**\r\n\t * Iterates all right siblings and parents, which has right siblings\r\n\t * while it does not reached the first-level block\r\n\t *\r\n\t * @param {Element} node\r\n\t * @return {boolean}\r\n\t */\n\t content.isLastNode = function (node) {\n\t\n\t // console.log('погнали перебор родителей');\n\t\n\t var allChecked = false;\n\t\n\t while (!allChecked) {\n\t\n\t // console.log('Смотрим на %o', node);\n\t // console.log('Проверим, пустые ли соседи справа');\n\t\n\t if (!allSiblingsEmpty_(node)) {\n\t\n\t // console.log('Есть непустые соседи. Узел не последний. Выходим.');\n\t return false;\n\t }\n\t\n\t node = node.parentNode;\n\t\n\t /**\r\n\t * Проверяем родителей до тех пор, пока не найдем блок первого уровня\r\n\t */\n\t if (node.classList.contains(editor.ui.className.BLOCK_CONTENT)) {\n\t\n\t allChecked = true;\n\t }\n\t }\n\t\n\t return true;\n\t };\n\t\n\t /**\r\n\t * Checks if all element right siblings is empty\r\n\t * @param node\r\n\t */\n\t var allSiblingsEmpty_ = function allSiblingsEmpty_(node) {\n\t\n\t /**\r\n\t * Нужно убедиться, что после пустого соседа ничего нет\r\n\t */\n\t var sibling = node.nextSibling;\n\t\n\t while (sibling) {\n\t\n\t if (sibling.textContent.length) {\n\t\n\t return false;\n\t }\n\t\n\t sibling = sibling.nextSibling;\n\t }\n\t\n\t return true;\n\t };\n\t\n\t /**\r\n\t * @public\r\n\t *\r\n\t * @param {string} htmlData - html content as string\r\n\t * @param {string} plainData - plain text\r\n\t * @return {string} - html content as string\r\n\t */\n\t content.wrapTextWithParagraphs = function (htmlData, plainData) {\n\t\n\t if (!htmlData) {\n\t\n\t return '

    ' + plainData.split('\\n\\n').join('

    ') + '

    ';\n\t }\n\t\n\t var wrapper = document.createElement('DIV'),\n\t newWrapper = document.createElement('DIV'),\n\t i,\n\t paragraph,\n\t firstLevelBlocks = ['DIV', 'P'],\n\t blockTyped,\n\t node;\n\t\n\t /**\r\n\t * Make HTML Element to Wrap Text\r\n\t * It allows us to work with input data as HTML content\r\n\t */\n\t wrapper.innerHTML = htmlData;\n\t paragraph = document.createElement('P');\n\t\n\t for (i = 0; i < wrapper.childNodes.length; i++) {\n\t\n\t node = wrapper.childNodes[i];\n\t\n\t blockTyped = firstLevelBlocks.indexOf(node.tagName) != -1;\n\t\n\t /**\r\n\t * If node is first-levet\r\n\t * we add this node to our new wrapper\r\n\t */\n\t if (blockTyped) {\n\t\n\t /**\r\n\t * If we had splitted inline nodes to paragraph before\r\n\t */\n\t if (paragraph.childNodes.length) {\n\t\n\t newWrapper.appendChild(paragraph.cloneNode(true));\n\t\n\t /** empty paragraph */\n\t paragraph = null;\n\t paragraph = document.createElement('P');\n\t }\n\t\n\t newWrapper.appendChild(node.cloneNode(true));\n\t } else {\n\t\n\t /** Collect all inline nodes to one as paragraph */\n\t paragraph.appendChild(node.cloneNode(true));\n\t\n\t /** if node is last we should append this node to paragraph and paragraph to new wrapper */\n\t if (i == wrapper.childNodes.length - 1) {\n\t\n\t newWrapper.appendChild(paragraph.cloneNode(true));\n\t }\n\t }\n\t }\n\t\n\t return newWrapper.innerHTML;\n\t };\n\t\n\t /**\r\n\t * Finds closest Contenteditable parent from Element\r\n\t * @param {Element} node element looking from\r\n\t * @return {Element} node contenteditable\r\n\t */\n\t content.getEditableParent = function (node) {\n\t\n\t while (node && node.contentEditable != 'true') {\n\t\n\t node = node.parentNode;\n\t }\n\t\n\t return node;\n\t };\n\t\n\t return content;\n\t}({});\n\n/***/ },\n/* 8 */\n/***/ function(module, exports, __webpack_require__) {\n\n\t'use strict';\n\t\n\t/**\r\n\t * Codex Editor toolbar module\r\n\t *\r\n\t * Contains:\r\n\t * - Inline toolbox\r\n\t * - Toolbox within plus button\r\n\t * - Settings section\r\n\t *\r\n\t * @author Codex Team\r\n\t * @version 1.0\r\n\t */\n\t\n\tmodule.exports = function (toolbar) {\n\t\n\t var editor = codex.editor;\n\t\n\t toolbar.settings = __webpack_require__(9);\n\t toolbar.inline = __webpack_require__(10);\n\t toolbar.toolbox = __webpack_require__(11);\n\t\n\t /**\r\n\t * Margin between focused node and toolbar\r\n\t */\n\t toolbar.defaultToolbarHeight = 49;\n\t\n\t toolbar.defaultOffset = 34;\n\t\n\t toolbar.opened = false;\n\t\n\t toolbar.current = null;\n\t\n\t /**\r\n\t * @protected\r\n\t */\n\t toolbar.open = function () {\n\t\n\t if (editor.hideToolbar) {\n\t\n\t return;\n\t }\n\t\n\t editor.nodes.toolbar.classList.add('opened');\n\t this.opened = true;\n\t };\n\t\n\t /**\r\n\t * @protected\r\n\t */\n\t toolbar.close = function () {\n\t\n\t editor.nodes.toolbar.classList.remove('opened');\n\t\n\t toolbar.opened = false;\n\t toolbar.current = null;\n\t\n\t for (var button in editor.nodes.toolbarButtons) {\n\t\n\t editor.nodes.toolbarButtons[button].classList.remove('selected');\n\t }\n\t\n\t /** Close toolbox when toolbar is not displayed */\n\t editor.toolbar.toolbox.close();\n\t editor.toolbar.settings.close();\n\t };\n\t\n\t toolbar.toggle = function () {\n\t\n\t if (!this.opened) {\n\t\n\t this.open();\n\t } else {\n\t\n\t this.close();\n\t }\n\t };\n\t\n\t toolbar.hidePlusButton = function () {\n\t\n\t editor.nodes.plusButton.classList.add('hide');\n\t };\n\t\n\t toolbar.showPlusButton = function () {\n\t\n\t editor.nodes.plusButton.classList.remove('hide');\n\t };\n\t\n\t /**\r\n\t * Moving toolbar to the specified node\r\n\t */\n\t toolbar.move = function () {\n\t\n\t /** Close Toolbox when we move toolbar */\n\t editor.toolbar.toolbox.close();\n\t\n\t if (!editor.content.currentNode) {\n\t\n\t return;\n\t }\n\t\n\t var newYCoordinate = editor.content.currentNode.offsetTop - editor.toolbar.defaultToolbarHeight / 2 + editor.toolbar.defaultOffset;\n\t\n\t editor.nodes.toolbar.style.transform = 'translate3D(0, ' + Math.floor(newYCoordinate) + 'px, 0)';\n\t\n\t /** Close trash actions */\n\t editor.toolbar.settings.hideRemoveActions();\n\t };\n\t\n\t return toolbar;\n\t}({});\n\n/***/ },\n/* 9 */\n/***/ function(module, exports) {\n\n\t'use strict';\n\t\n\t/**\r\n\t * Toolbar settings\r\n\t *\r\n\t * @version 1.0.4\r\n\t */\n\t\n\tmodule.exports = function (settings) {\n\t\n\t var editor = codex.editor;\n\t\n\t settings.opened = false;\n\t\n\t settings.setting = null;\n\t settings.actions = null;\n\t\n\t /**\r\n\t * Append and open settings\r\n\t */\n\t settings.open = function (toolType) {\n\t\n\t /**\r\n\t * Append settings content\r\n\t * It's stored in tool.settings\r\n\t */\n\t if (!editor.tools[toolType] || !editor.tools[toolType].makeSettings) {\n\t\n\t editor.core.log('Plugin \\xAB' + toolType + '\\xBB has no settings', 'warn');\n\t editor.nodes.pluginSettings.innerHTML = '\\u041F\\u043B\\u0430\\u0433\\u0438\\u043D \\xAB' + toolType + '\\xBB \\u043D\\u0435 \\u0438\\u043C\\u0435\\u0435\\u0442 \\u043D\\u0430\\u0441\\u0442\\u0440\\u043E\\u0435\\u043A';\n\t } else {\n\t\n\t /**\r\n\t * Draw settings block\r\n\t */\n\t var settingsBlock = editor.tools[toolType].makeSettings();\n\t\n\t editor.nodes.pluginSettings.appendChild(settingsBlock);\n\t }\n\t\n\t /** Open settings block */\n\t editor.nodes.blockSettings.classList.add('opened');\n\t this.opened = true;\n\t };\n\t\n\t /**\r\n\t * Close and clear settings\r\n\t */\n\t settings.close = function () {\n\t\n\t editor.nodes.blockSettings.classList.remove('opened');\n\t editor.nodes.pluginSettings.innerHTML = '';\n\t\n\t this.opened = false;\n\t };\n\t\n\t /**\r\n\t * @param {string} toolType - plugin type\r\n\t */\n\t settings.toggle = function (toolType) {\n\t\n\t if (!this.opened) {\n\t\n\t this.open(toolType);\n\t } else {\n\t\n\t this.close();\n\t }\n\t };\n\t\n\t /**\r\n\t * Here we will draw buttons and add listeners to components\r\n\t */\n\t settings.makeRemoveBlockButton = function () {\n\t\n\t var removeBlockWrapper = editor.draw.node('SPAN', 'ce-toolbar__remove-btn', {}),\n\t settingButton = editor.draw.node('SPAN', 'ce-toolbar__remove-setting', { innerHTML: '' }),\n\t actionWrapper = editor.draw.node('DIV', 'ce-toolbar__remove-confirmation', {}),\n\t confirmAction = editor.draw.node('DIV', 'ce-toolbar__remove-confirm', { textContent: 'Удалить блок' }),\n\t cancelAction = editor.draw.node('DIV', 'ce-toolbar__remove-cancel', { textContent: 'Отмена' });\n\t\n\t editor.listeners.add(settingButton, 'click', editor.toolbar.settings.removeButtonClicked, false);\n\t\n\t editor.listeners.add(confirmAction, 'click', editor.toolbar.settings.confirmRemovingRequest, false);\n\t\n\t editor.listeners.add(cancelAction, 'click', editor.toolbar.settings.cancelRemovingRequest, false);\n\t\n\t actionWrapper.appendChild(confirmAction);\n\t actionWrapper.appendChild(cancelAction);\n\t\n\t removeBlockWrapper.appendChild(settingButton);\n\t removeBlockWrapper.appendChild(actionWrapper);\n\t\n\t /** Save setting */\n\t editor.toolbar.settings.setting = settingButton;\n\t editor.toolbar.settings.actions = actionWrapper;\n\t\n\t return removeBlockWrapper;\n\t };\n\t\n\t settings.removeButtonClicked = function () {\n\t\n\t var action = editor.toolbar.settings.actions;\n\t\n\t if (action.classList.contains('opened')) {\n\t\n\t editor.toolbar.settings.hideRemoveActions();\n\t } else {\n\t\n\t editor.toolbar.settings.showRemoveActions();\n\t }\n\t\n\t editor.toolbar.toolbox.close();\n\t editor.toolbar.settings.close();\n\t };\n\t\n\t settings.cancelRemovingRequest = function () {\n\t\n\t editor.toolbar.settings.actions.classList.remove('opened');\n\t };\n\t\n\t settings.confirmRemovingRequest = function () {\n\t\n\t var currentBlock = editor.content.currentNode,\n\t firstLevelBlocksCount;\n\t\n\t currentBlock.remove();\n\t\n\t firstLevelBlocksCount = editor.nodes.redactor.childNodes.length;\n\t\n\t /**\r\n\t * If all blocks are removed\r\n\t */\n\t if (firstLevelBlocksCount === 0) {\n\t\n\t /** update currentNode variable */\n\t editor.content.currentNode = null;\n\t\n\t /** Inserting new empty initial block */\n\t editor.ui.addInitialBlock();\n\t }\n\t\n\t editor.ui.saveInputs();\n\t\n\t editor.toolbar.close();\n\t };\n\t\n\t settings.showRemoveActions = function () {\n\t\n\t editor.toolbar.settings.actions.classList.add('opened');\n\t };\n\t\n\t settings.hideRemoveActions = function () {\n\t\n\t editor.toolbar.settings.actions.classList.remove('opened');\n\t };\n\t\n\t return settings;\n\t}({});\n\n/***/ },\n/* 10 */\n/***/ function(module, exports) {\n\n\t'use strict';\n\t\n\t/**\r\n\t * Inline toolbar\r\n\t *\r\n\t * Contains from tools:\r\n\t * Bold, Italic, Underline and Anchor\r\n\t *\r\n\t * @author Codex Team\r\n\t * @version 1.0\r\n\t */\n\t\n\tmodule.exports = function (inline) {\n\t\n\t var editor = codex.editor;\n\t\n\t inline.buttonsOpened = null;\n\t inline.actionsOpened = null;\n\t inline.wrappersOffset = null;\n\t\n\t /**\r\n\t * saving selection that need for execCommand for styling\r\n\t *\r\n\t */\n\t inline.storedSelection = null;\n\t\n\t /**\r\n\t * @protected\r\n\t *\r\n\t * Open inline toobar\r\n\t */\n\t inline.show = function () {\n\t\n\t var currentNode = editor.content.currentNode,\n\t tool = currentNode.dataset.tool,\n\t plugin;\n\t\n\t /**\r\n\t * tool allowed to open inline toolbar\r\n\t */\n\t plugin = editor.tools[tool];\n\t\n\t if (!plugin.showInlineToolbar) return;\n\t\n\t var selectedText = inline.getSelectionText(),\n\t toolbar = editor.nodes.inlineToolbar.wrapper;\n\t\n\t if (selectedText.length > 0) {\n\t\n\t /** Move toolbar and open */\n\t editor.toolbar.inline.move();\n\t\n\t /** Open inline toolbar */\n\t toolbar.classList.add('opened');\n\t\n\t /** show buttons of inline toolbar */\n\t editor.toolbar.inline.showButtons();\n\t }\n\t };\n\t\n\t /**\r\n\t * @protected\r\n\t *\r\n\t * Closes inline toolbar\r\n\t */\n\t inline.close = function () {\n\t\n\t var toolbar = editor.nodes.inlineToolbar.wrapper;\n\t\n\t toolbar.classList.remove('opened');\n\t };\n\t\n\t /**\r\n\t * @private\r\n\t *\r\n\t * Moving toolbar\r\n\t */\n\t inline.move = function () {\n\t\n\t if (!this.wrappersOffset) {\n\t\n\t this.wrappersOffset = this.getWrappersOffset();\n\t }\n\t\n\t var coords = this.getSelectionCoords(),\n\t defaultOffset = 0,\n\t toolbar = editor.nodes.inlineToolbar.wrapper,\n\t newCoordinateX,\n\t newCoordinateY;\n\t\n\t if (toolbar.offsetHeight === 0) {\n\t\n\t defaultOffset = 40;\n\t }\n\t\n\t newCoordinateX = coords.x - this.wrappersOffset.left;\n\t newCoordinateY = coords.y + window.scrollY - this.wrappersOffset.top - defaultOffset - toolbar.offsetHeight;\n\t\n\t toolbar.style.transform = 'translate3D(' + Math.floor(newCoordinateX) + 'px, ' + Math.floor(newCoordinateY) + 'px, 0)';\n\t\n\t /** Close everything */\n\t editor.toolbar.inline.closeButtons();\n\t editor.toolbar.inline.closeAction();\n\t };\n\t\n\t /**\r\n\t * @private\r\n\t *\r\n\t * Tool Clicked\r\n\t */\n\t\n\t inline.toolClicked = function (event, type) {\n\t\n\t /**\r\n\t * For simple tools we use default browser function\r\n\t * For more complicated tools, we should write our own behavior\r\n\t */\n\t switch (type) {\n\t case 'createLink':\n\t editor.toolbar.inline.createLinkAction(event, type);break;\n\t default:\n\t editor.toolbar.inline.defaultToolAction(type);break;\n\t }\n\t\n\t /**\r\n\t * highlight buttons\r\n\t * after making some action\r\n\t */\n\t editor.nodes.inlineToolbar.buttons.childNodes.forEach(editor.toolbar.inline.hightlight);\n\t };\n\t\n\t /**\r\n\t * @private\r\n\t *\r\n\t * Saving wrappers offset in DOM\r\n\t */\n\t inline.getWrappersOffset = function () {\n\t\n\t var wrapper = editor.nodes.wrapper,\n\t offset = this.getOffset(wrapper);\n\t\n\t this.wrappersOffset = offset;\n\t return offset;\n\t };\n\t\n\t /**\r\n\t * @private\r\n\t *\r\n\t * Calculates offset of DOM element\r\n\t *\r\n\t * @param el\r\n\t * @returns {{top: number, left: number}}\r\n\t */\n\t inline.getOffset = function (el) {\n\t\n\t var _x = 0;\n\t var _y = 0;\n\t\n\t while (el && !isNaN(el.offsetLeft) && !isNaN(el.offsetTop)) {\n\t\n\t _x += el.offsetLeft + el.clientLeft;\n\t _y += el.offsetTop + el.clientTop;\n\t el = el.offsetParent;\n\t }\n\t return { top: _y, left: _x };\n\t };\n\t\n\t /**\r\n\t * @private\r\n\t *\r\n\t * Calculates position of selected text\r\n\t * @returns {{x: number, y: number}}\r\n\t */\n\t inline.getSelectionCoords = function () {\n\t\n\t var sel = document.selection,\n\t range;\n\t var x = 0,\n\t y = 0;\n\t\n\t if (sel) {\n\t\n\t if (sel.type != 'Control') {\n\t\n\t range = sel.createRange();\n\t range.collapse(true);\n\t x = range.boundingLeft;\n\t y = range.boundingTop;\n\t }\n\t } else if (window.getSelection) {\n\t\n\t sel = window.getSelection();\n\t\n\t if (sel.rangeCount) {\n\t\n\t range = sel.getRangeAt(0).cloneRange();\n\t if (range.getClientRects) {\n\t\n\t range.collapse(true);\n\t var rect = range.getClientRects()[0];\n\t\n\t if (!rect) {\n\t\n\t return;\n\t }\n\t\n\t x = rect.left;\n\t y = rect.top;\n\t }\n\t }\n\t }\n\t return { x: x, y: y };\n\t };\n\t\n\t /**\r\n\t * @private\r\n\t *\r\n\t * Returns selected text as String\r\n\t * @returns {string}\r\n\t */\n\t inline.getSelectionText = function () {\n\t\n\t var selectedText = '';\n\t\n\t // all modern browsers and IE9+\n\t if (window.getSelection) {\n\t\n\t selectedText = window.getSelection().toString();\n\t }\n\t\n\t return selectedText;\n\t };\n\t\n\t /** Opens buttons block */\n\t inline.showButtons = function () {\n\t\n\t var buttons = editor.nodes.inlineToolbar.buttons;\n\t\n\t buttons.classList.add('opened');\n\t\n\t editor.toolbar.inline.buttonsOpened = true;\n\t\n\t /** highlight buttons */\n\t editor.nodes.inlineToolbar.buttons.childNodes.forEach(editor.toolbar.inline.hightlight);\n\t };\n\t\n\t /** Makes buttons disappear */\n\t inline.closeButtons = function () {\n\t\n\t var buttons = editor.nodes.inlineToolbar.buttons;\n\t\n\t buttons.classList.remove('opened');\n\t\n\t editor.toolbar.inline.buttonsOpened = false;\n\t };\n\t\n\t /** Open buttons defined action if exist */\n\t inline.showActions = function () {\n\t\n\t var action = editor.nodes.inlineToolbar.actions;\n\t\n\t action.classList.add('opened');\n\t\n\t editor.toolbar.inline.actionsOpened = true;\n\t };\n\t\n\t /** Close actions block */\n\t inline.closeAction = function () {\n\t\n\t var action = editor.nodes.inlineToolbar.actions;\n\t\n\t action.innerHTML = '';\n\t action.classList.remove('opened');\n\t editor.toolbar.inline.actionsOpened = false;\n\t };\n\t\n\t /**\r\n\t * Callback for keydowns in inline toolbar \"Insert link...\" input\r\n\t */\n\t var inlineToolbarAnchorInputKeydown_ = function inlineToolbarAnchorInputKeydown_(event) {\n\t\n\t if (event.keyCode != editor.core.keys.ENTER) {\n\t\n\t return;\n\t }\n\t\n\t var editable = editor.content.currentNode,\n\t storedSelection = editor.toolbar.inline.storedSelection;\n\t\n\t editor.toolbar.inline.restoreSelection(editable, storedSelection);\n\t editor.toolbar.inline.setAnchor(this.value);\n\t\n\t /**\r\n\t * Preventing events that will be able to happen\r\n\t */\n\t event.preventDefault();\n\t event.stopImmediatePropagation();\n\t\n\t editor.toolbar.inline.clearRange();\n\t };\n\t\n\t /** Action for link creation or for setting anchor */\n\t inline.createLinkAction = function (event) {\n\t\n\t var isActive = this.isLinkActive();\n\t\n\t var editable = editor.content.currentNode,\n\t storedSelection = editor.toolbar.inline.saveSelection(editable);\n\t\n\t /** Save globally selection */\n\t editor.toolbar.inline.storedSelection = storedSelection;\n\t\n\t if (isActive) {\n\t\n\t /**\r\n\t * Changing stored selection. if we want to remove anchor from word\r\n\t * we should remove anchor from whole word, not only selected part.\r\n\t * The solution is than we get the length of current link\r\n\t * Change start position to - end of selection minus length of anchor\r\n\t */\n\t editor.toolbar.inline.restoreSelection(editable, storedSelection);\n\t\n\t editor.toolbar.inline.defaultToolAction('unlink');\n\t } else {\n\t\n\t /** Create input and close buttons */\n\t var action = editor.draw.inputForLink();\n\t\n\t editor.nodes.inlineToolbar.actions.appendChild(action);\n\t\n\t editor.toolbar.inline.closeButtons();\n\t editor.toolbar.inline.showActions();\n\t\n\t /**\r\n\t * focus to input\r\n\t * Solution: https://developer.mozilla.org/ru/docs/Web/API/HTMLElement/focus\r\n\t * Prevents event after showing input and when we need to focus an input which is in unexisted form\r\n\t */\n\t action.focus();\n\t event.preventDefault();\n\t\n\t /** Callback to link action */\n\t editor.listeners.add(action, 'keydown', inlineToolbarAnchorInputKeydown_, false);\n\t }\n\t };\n\t\n\t inline.isLinkActive = function () {\n\t\n\t var isActive = false;\n\t\n\t editor.nodes.inlineToolbar.buttons.childNodes.forEach(function (tool) {\n\t\n\t var dataType = tool.dataset.type;\n\t\n\t if (dataType == 'link' && tool.classList.contains('hightlighted')) {\n\t\n\t isActive = true;\n\t }\n\t });\n\t\n\t return isActive;\n\t };\n\t\n\t /** default action behavior of tool */\n\t inline.defaultToolAction = function (type) {\n\t\n\t document.execCommand(type, false, null);\n\t };\n\t\n\t /**\r\n\t * @private\r\n\t *\r\n\t * Sets URL\r\n\t *\r\n\t * @param {String} url - URL\r\n\t */\n\t inline.setAnchor = function (url) {\n\t\n\t document.execCommand('createLink', false, url);\n\t\n\t /** Close after URL inserting */\n\t editor.toolbar.inline.closeAction();\n\t };\n\t\n\t /**\r\n\t * @private\r\n\t *\r\n\t * Saves selection\r\n\t */\n\t inline.saveSelection = function (containerEl) {\n\t\n\t var range = window.getSelection().getRangeAt(0),\n\t preSelectionRange = range.cloneRange(),\n\t start;\n\t\n\t preSelectionRange.selectNodeContents(containerEl);\n\t preSelectionRange.setEnd(range.startContainer, range.startOffset);\n\t\n\t start = preSelectionRange.toString().length;\n\t\n\t return {\n\t start: start,\n\t end: start + range.toString().length\n\t };\n\t };\n\t\n\t /**\r\n\t * @private\r\n\t *\r\n\t * Sets to previous selection (Range)\r\n\t *\r\n\t * @param {Element} containerEl - editable element where we restore range\r\n\t * @param {Object} savedSel - range basic information to restore\r\n\t */\n\t inline.restoreSelection = function (containerEl, savedSel) {\n\t\n\t var range = document.createRange(),\n\t charIndex = 0;\n\t\n\t range.setStart(containerEl, 0);\n\t range.collapse(true);\n\t\n\t var nodeStack = [containerEl],\n\t node,\n\t foundStart = false,\n\t stop = false,\n\t nextCharIndex;\n\t\n\t while (!stop && (node = nodeStack.pop())) {\n\t\n\t if (node.nodeType == 3) {\n\t\n\t nextCharIndex = charIndex + node.length;\n\t\n\t if (!foundStart && savedSel.start >= charIndex && savedSel.start <= nextCharIndex) {\n\t\n\t range.setStart(node, savedSel.start - charIndex);\n\t foundStart = true;\n\t }\n\t if (foundStart && savedSel.end >= charIndex && savedSel.end <= nextCharIndex) {\n\t\n\t range.setEnd(node, savedSel.end - charIndex);\n\t stop = true;\n\t }\n\t charIndex = nextCharIndex;\n\t } else {\n\t\n\t var i = node.childNodes.length;\n\t\n\t while (i--) {\n\t\n\t nodeStack.push(node.childNodes[i]);\n\t }\n\t }\n\t }\n\t\n\t var sel = window.getSelection();\n\t\n\t sel.removeAllRanges();\n\t sel.addRange(range);\n\t };\n\t\n\t /**\r\n\t * @private\r\n\t *\r\n\t * Removes all ranges from window selection\r\n\t */\n\t inline.clearRange = function () {\n\t\n\t var selection = window.getSelection();\n\t\n\t selection.removeAllRanges();\n\t };\n\t\n\t /**\r\n\t * @private\r\n\t *\r\n\t * sets or removes hightlight\r\n\t */\n\t inline.hightlight = function (tool) {\n\t\n\t var dataType = tool.dataset.type;\n\t\n\t if (document.queryCommandState(dataType)) {\n\t\n\t editor.toolbar.inline.setButtonHighlighted(tool);\n\t } else {\n\t\n\t editor.toolbar.inline.removeButtonsHighLight(tool);\n\t }\n\t\n\t /**\r\n\t *\r\n\t * hightlight for anchors\r\n\t */\n\t var selection = window.getSelection(),\n\t tag = selection.anchorNode.parentNode;\n\t\n\t if (tag.tagName == 'A' && dataType == 'link') {\n\t\n\t editor.toolbar.inline.setButtonHighlighted(tool);\n\t }\n\t };\n\t\n\t /**\r\n\t * @private\r\n\t *\r\n\t * Mark button if text is already executed\r\n\t */\n\t inline.setButtonHighlighted = function (button) {\n\t\n\t button.classList.add('hightlighted');\n\t\n\t /** At link tool we also change icon */\n\t if (button.dataset.type == 'link') {\n\t\n\t var icon = button.childNodes[0];\n\t\n\t icon.classList.remove('ce-icon-link');\n\t icon.classList.add('ce-icon-unlink');\n\t }\n\t };\n\t\n\t /**\r\n\t * @private\r\n\t *\r\n\t * Removes hightlight\r\n\t */\n\t inline.removeButtonsHighLight = function (button) {\n\t\n\t button.classList.remove('hightlighted');\n\t\n\t /** At link tool we also change icon */\n\t if (button.dataset.type == 'link') {\n\t\n\t var icon = button.childNodes[0];\n\t\n\t icon.classList.remove('ce-icon-unlink');\n\t icon.classList.add('ce-icon-link');\n\t }\n\t };\n\t\n\t return inline;\n\t}({});\n\n/***/ },\n/* 11 */\n/***/ function(module, exports) {\n\n\t'use strict';\n\t\n\t/**\r\n\t * Codex Editor toolbox\r\n\t *\r\n\t * All tools be able to appended here\r\n\t *\r\n\t * @author Codex Team\r\n\t * @version 1.0\r\n\t */\n\t\n\tmodule.exports = function (toolbox) {\n\t\n\t var editor = codex.editor;\n\t\n\t toolbox.opened = false;\n\t\n\t /** Shows toolbox */\n\t toolbox.open = function () {\n\t\n\t /** Close setting if toolbox is opened */\n\t if (editor.toolbar.settings.opened) {\n\t\n\t editor.toolbar.settings.close();\n\t }\n\t\n\t /** display toolbox */\n\t editor.nodes.toolbox.classList.add('opened');\n\t\n\t /** Animate plus button */\n\t editor.nodes.plusButton.classList.add('clicked');\n\t\n\t /** toolbox state */\n\t editor.toolbar.toolbox.opened = true;\n\t };\n\t\n\t /** Closes toolbox */\n\t toolbox.close = function () {\n\t\n\t /** Makes toolbox disapear */\n\t editor.nodes.toolbox.classList.remove('opened');\n\t\n\t /** Rotate plus button */\n\t editor.nodes.plusButton.classList.remove('clicked');\n\t\n\t /** toolbox state */\n\t editor.toolbar.toolbox.opened = false;\n\t\n\t editor.toolbar.current = null;\n\t };\n\t\n\t toolbox.leaf = function () {\n\t\n\t var currentTool = editor.toolbar.current,\n\t tools = Object.keys(editor.tools),\n\t barButtons = editor.nodes.toolbarButtons,\n\t nextToolIndex = 0,\n\t toolToSelect = void 0,\n\t visibleTool = void 0,\n\t tool = void 0;\n\t\n\t if (!currentTool) {\n\t\n\t /** Get first tool from object*/\n\t for (tool in editor.tools) {\n\t\n\t if (editor.tools[tool].displayInToolbox) {\n\t\n\t break;\n\t }\n\t\n\t nextToolIndex++;\n\t }\n\t } else {\n\t\n\t nextToolIndex = (tools.indexOf(currentTool) + 1) % tools.length;\n\t visibleTool = tools[nextToolIndex];\n\t\n\t while (!editor.tools[visibleTool].displayInToolbox) {\n\t\n\t nextToolIndex = (nextToolIndex + 1) % tools.length;\n\t visibleTool = tools[nextToolIndex];\n\t }\n\t }\n\t\n\t toolToSelect = tools[nextToolIndex];\n\t\n\t for (var button in barButtons) {\n\t\n\t barButtons[button].classList.remove('selected');\n\t }\n\t\n\t barButtons[toolToSelect].classList.add('selected');\n\t editor.toolbar.current = toolToSelect;\n\t };\n\t\n\t /**\r\n\t * Transforming selected node type into selected toolbar element type\r\n\t * @param {event} event\r\n\t */\n\t toolbox.toolClicked = function (event) {\n\t\n\t /**\r\n\t * UNREPLACEBLE_TOOLS this types of tools are forbidden to replace even they are empty\r\n\t */\n\t var UNREPLACEBLE_TOOLS = ['image', 'link', 'list', 'instagram', 'twitter', 'embed'],\n\t tool = editor.tools[editor.toolbar.current],\n\t workingNode = editor.content.currentNode,\n\t currentInputIndex = editor.caret.inputIndex,\n\t newBlockContent,\n\t appendCallback,\n\t blockData;\n\t\n\t /** Make block from plugin */\n\t newBlockContent = tool.render();\n\t\n\t /** information about block */\n\t blockData = {\n\t block: newBlockContent,\n\t type: tool.type,\n\t stretched: false\n\t };\n\t\n\t if (workingNode && UNREPLACEBLE_TOOLS.indexOf(workingNode.dataset.tool) === -1 && workingNode.textContent.trim() === '') {\n\t\n\t /** Replace current block */\n\t editor.content.switchBlock(workingNode, newBlockContent, tool.type);\n\t } else {\n\t\n\t /** Insert new Block from plugin */\n\t editor.content.insertBlock(blockData);\n\t\n\t /** increase input index */\n\t currentInputIndex++;\n\t }\n\t\n\t /** Fire tool append callback */\n\t appendCallback = tool.appendCallback;\n\t\n\t if (appendCallback && typeof appendCallback == 'function') {\n\t\n\t appendCallback.call(event);\n\t }\n\t\n\t window.setTimeout(function () {\n\t\n\t /** Set caret to current block */\n\t editor.caret.setToBlock(currentInputIndex);\n\t }, 10);\n\t\n\t /**\r\n\t * Changing current Node\r\n\t */\n\t editor.content.workingNodeChanged();\n\t\n\t /**\r\n\t * Move toolbar when node is changed\r\n\t */\n\t editor.toolbar.move();\n\t };\n\t\n\t return toolbox;\n\t}({});\n\n/***/ },\n/* 12 */\n/***/ function(module, exports) {\n\n\t'use strict';\n\t\n\t/**\r\n\t * @module Codex Editor Callbacks module\r\n\t * @description Module works with editor added Elements\r\n\t *\r\n\t * @author Codex Team\r\n\t * @version 1.4.0\r\n\t */\n\t\n\tmodule.exports = function (callbacks) {\n\t\n\t var editor = codex.editor;\n\t\n\t /**\r\n\t * used by UI module\r\n\t * @description Routes all keydowns on document\r\n\t * @param {Object} event\r\n\t */\n\t callbacks.globalKeydown = function (event) {\n\t\n\t switch (event.keyCode) {\n\t case editor.core.keys.ENTER:\n\t enterKeyPressed_(event);break;\n\t }\n\t };\n\t\n\t /**\r\n\t * used by UI module\r\n\t * @description Routes all keydowns on redactors area\r\n\t * @param {Object} event\r\n\t */\n\t callbacks.redactorKeyDown = function (event) {\n\t\n\t switch (event.keyCode) {\n\t case editor.core.keys.TAB:\n\t tabKeyPressedOnRedactorsZone_(event);break;\n\t case editor.core.keys.ENTER:\n\t enterKeyPressedOnRedactorsZone_(event);break;\n\t case editor.core.keys.ESC:\n\t escapeKeyPressedOnRedactorsZone_(event);break;\n\t default:\n\t defaultKeyPressedOnRedactorsZone_(event);break;\n\t }\n\t };\n\t\n\t /**\r\n\t * used by UI module\r\n\t * @description Routes all keyup events\r\n\t * @param {Object} event\r\n\t */\n\t callbacks.globalKeyup = function (event) {\n\t\n\t switch (event.keyCode) {\n\t case editor.core.keys.UP:\n\t case editor.core.keys.LEFT:\n\t case editor.core.keys.RIGHT:\n\t case editor.core.keys.DOWN:\n\t arrowKeyPressed_(event);break;\n\t }\n\t };\n\t\n\t /**\r\n\t * @param {Object} event\r\n\t * @private\r\n\t *\r\n\t * Handles behaviour when tab pressed\r\n\t * @description if Content is empty show toolbox (if it is closed) or leaf tools\r\n\t * uses Toolbars toolbox module to handle the situation\r\n\t */\n\t var tabKeyPressedOnRedactorsZone_ = function tabKeyPressedOnRedactorsZone_(event) {\n\t\n\t /**\r\n\t * Wait for solution. Would like to know the behaviour\r\n\t * @todo Add spaces\r\n\t */\n\t event.preventDefault();\n\t\n\t var nativeInputs = editor.content.currentNode.querySelectorAll('textarea, input'),\n\t nativeInputsAreEmpty = true,\n\t textContentIsEmpty = !editor.content.currentNode.textContent.trim();\n\t\n\t Array.prototype.map.call(nativeInputs, function (input) {\n\t\n\t if (input.type == 'textarea' || input.type == 'text') {\n\t\n\t nativeInputsAreEmpty = nativeInputsAreEmpty && !input.value.trim();\n\t }\n\t });\n\t\n\t var blockIsEmpty = textContentIsEmpty && nativeInputsAreEmpty;\n\t\n\t if (!blockIsEmpty) {\n\t\n\t return;\n\t }\n\t\n\t if (!editor.toolbar.opened) {\n\t\n\t editor.toolbar.open();\n\t }\n\t\n\t if (editor.toolbar.opened && !editor.toolbar.toolbox.opened) {\n\t\n\t editor.toolbar.toolbox.open();\n\t } else {\n\t\n\t editor.toolbar.toolbox.leaf();\n\t }\n\t };\n\t\n\t /**\r\n\t * Handles global EnterKey Press\r\n\t * @see enterPressedOnBlock_\r\n\t * @param {Object} event\r\n\t */\n\t var enterKeyPressed_ = function enterKeyPressed_() {\n\t\n\t if (editor.content.editorAreaHightlighted) {\n\t\n\t /**\r\n\t * it means that we lose input index, saved index before is not correct\r\n\t * therefore we need to set caret when we insert new block\r\n\t */\n\t editor.caret.inputIndex = -1;\n\t\n\t enterPressedOnBlock_();\n\t }\n\t };\n\t\n\t /**\r\n\t * Callback for enter key pressing in first-level block area\r\n\t *\r\n\t * @param {Event} event\r\n\t * @private\r\n\t *\r\n\t * @description Inserts new block with initial type from settings\r\n\t */\n\t var enterPressedOnBlock_ = function enterPressedOnBlock_() {\n\t\n\t var NEW_BLOCK_TYPE = editor.settings.initialBlockPlugin;\n\t\n\t editor.content.insertBlock({\n\t type: NEW_BLOCK_TYPE,\n\t block: editor.tools[NEW_BLOCK_TYPE].render()\n\t }, true);\n\t\n\t editor.toolbar.move();\n\t editor.toolbar.open();\n\t };\n\t\n\t /**\r\n\t * ENTER key handler\r\n\t *\r\n\t * @param {Object} event\r\n\t * @private\r\n\t *\r\n\t * @description Makes new block with initial type from settings\r\n\t */\n\t var enterKeyPressedOnRedactorsZone_ = function enterKeyPressedOnRedactorsZone_(event) {\n\t\n\t if (event.target.contentEditable == 'true') {\n\t\n\t /** Update input index */\n\t editor.caret.saveCurrentInputIndex();\n\t }\n\t\n\t var currentInputIndex = editor.caret.getCurrentInputIndex() || 0,\n\t workingNode = editor.content.currentNode,\n\t tool = workingNode.dataset.tool,\n\t isEnterPressedOnToolbar = editor.toolbar.opened && editor.toolbar.current && event.target == editor.state.inputs[currentInputIndex];\n\t\n\t /** The list of tools which needs the default browser behaviour */\n\t var enableLineBreaks = editor.tools[tool].enableLineBreaks;\n\t\n\t /** This type of block creates when enter is pressed */\n\t var NEW_BLOCK_TYPE = editor.settings.initialBlockPlugin;\n\t\n\t /**\r\n\t * When toolbar is opened, select tool instead of making new paragraph\r\n\t */\n\t if (isEnterPressedOnToolbar) {\n\t\n\t event.preventDefault();\n\t\n\t editor.toolbar.toolbox.toolClicked(event);\n\t\n\t editor.toolbar.close();\n\t\n\t /**\r\n\t * Stop other listeners callback executions\r\n\t */\n\t event.stopPropagation();\n\t event.stopImmediatePropagation();\n\t\n\t return;\n\t }\n\t\n\t /**\r\n\t * Allow paragraph lineBreaks with shift enter\r\n\t * Or if shiftkey pressed and enter and enabledLineBreaks, the let new block creation\r\n\t */\n\t if (event.shiftKey || enableLineBreaks) {\n\t\n\t event.stopPropagation();\n\t event.stopImmediatePropagation();\n\t return;\n\t }\n\t\n\t var currentSelection = window.getSelection(),\n\t currentSelectedNode = currentSelection.anchorNode,\n\t caretAtTheEndOfText = editor.caret.position.atTheEnd(),\n\t isTextNodeHasParentBetweenContenteditable = false;\n\t\n\t /**\r\n\t * Allow making new

    in same block by SHIFT+ENTER and forbids to prevent default browser behaviour\r\n\t */\n\t if (event.shiftKey && !enableLineBreaks) {\n\t\n\t editor.callback.enterPressedOnBlock(editor.content.currentBlock, event);\n\t event.preventDefault();\n\t return;\n\t }\n\t\n\t /**\r\n\t * Workaround situation when caret at the Text node that has some wrapper Elements\r\n\t * Split block cant handle this.\r\n\t * We need to save default behavior\r\n\t */\n\t isTextNodeHasParentBetweenContenteditable = currentSelectedNode && currentSelectedNode.parentNode.contentEditable != 'true';\n\t\n\t /**\r\n\t * Split blocks when input has several nodes and caret placed in textNode\r\n\t */\n\t if (currentSelectedNode.nodeType == editor.core.nodeTypes.TEXT && !isTextNodeHasParentBetweenContenteditable && !caretAtTheEndOfText) {\n\t\n\t event.preventDefault();\n\t\n\t editor.core.log('Splitting Text node...');\n\t\n\t editor.content.splitBlock(currentInputIndex);\n\t\n\t /** Show plus button when next input after split is empty*/\n\t if (!editor.state.inputs[currentInputIndex + 1].textContent.trim()) {\n\t\n\t editor.toolbar.showPlusButton();\n\t }\n\t } else {\n\t\n\t var islastNode = editor.content.isLastNode(currentSelectedNode);\n\t\n\t if (islastNode && caretAtTheEndOfText) {\n\t\n\t event.preventDefault();\n\t event.stopPropagation();\n\t event.stopImmediatePropagation();\n\t\n\t editor.core.log('ENTER clicked in last textNode. Create new BLOCK');\n\t\n\t editor.content.insertBlock({\n\t type: NEW_BLOCK_TYPE,\n\t block: editor.tools[NEW_BLOCK_TYPE].render()\n\t }, true);\n\t\n\t editor.toolbar.move();\n\t editor.toolbar.open();\n\t\n\t /** Show plus button with empty block */\n\t editor.toolbar.showPlusButton();\n\t }\n\t }\n\t\n\t /** get all inputs after new appending block */\n\t editor.ui.saveInputs();\n\t };\n\t\n\t /**\r\n\t * Escape behaviour\r\n\t * @param event\r\n\t * @private\r\n\t *\r\n\t * @description Closes toolbox and toolbar. Prevents default behaviour\r\n\t */\n\t var escapeKeyPressedOnRedactorsZone_ = function escapeKeyPressedOnRedactorsZone_(event) {\n\t\n\t /** Close all toolbar */\n\t editor.toolbar.close();\n\t\n\t /** Close toolbox */\n\t editor.toolbar.toolbox.close();\n\t\n\t event.preventDefault();\n\t };\n\t\n\t /**\r\n\t * @param {Event} event\r\n\t * @private\r\n\t *\r\n\t * closes and moves toolbar\r\n\t */\n\t var arrowKeyPressed_ = function arrowKeyPressed_(event) {\n\t\n\t editor.content.workingNodeChanged();\n\t\n\t /* Closing toolbar */\n\t editor.toolbar.close();\n\t editor.toolbar.move();\n\t };\n\t\n\t /**\r\n\t * @private\r\n\t * @param {Event} event\r\n\t *\r\n\t * @description Closes all opened bars from toolbar.\r\n\t * If block is mark, clears highlightning\r\n\t */\n\t var defaultKeyPressedOnRedactorsZone_ = function defaultKeyPressedOnRedactorsZone_() {\n\t\n\t editor.toolbar.close();\n\t\n\t if (!editor.toolbar.inline.actionsOpened) {\n\t\n\t editor.toolbar.inline.close();\n\t editor.content.clearMark();\n\t }\n\t };\n\t\n\t /**\r\n\t * Handler when clicked on redactors area\r\n\t *\r\n\t * @protected\r\n\t * @param event\r\n\t *\r\n\t * @description Detects clicked area. If it is first-level block area, marks as detected and\r\n\t * on next enter press will be inserted new block\r\n\t * Otherwise, save carets position (input index) and put caret to the editable zone.\r\n\t *\r\n\t * @see detectWhenClickedOnFirstLevelBlockArea_\r\n\t *\r\n\t */\n\t callbacks.redactorClicked = function (event) {\n\t\n\t detectWhenClickedOnFirstLevelBlockArea_();\n\t\n\t editor.content.workingNodeChanged(event.target);\n\t editor.ui.saveInputs();\n\t\n\t var selectedText = editor.toolbar.inline.getSelectionText(),\n\t firstLevelBlock;\n\t\n\t /** If selection range took off, then we hide inline toolbar */\n\t if (selectedText.length === 0) {\n\t\n\t editor.toolbar.inline.close();\n\t }\n\t\n\t /** Update current input index in memory when caret focused into existed input */\n\t if (event.target.contentEditable == 'true') {\n\t\n\t editor.caret.saveCurrentInputIndex();\n\t }\n\t\n\t if (editor.content.currentNode === null) {\n\t\n\t /**\r\n\t * If inputs in redactor does not exits, then we put input index 0 not -1\r\n\t */\n\t var indexOfLastInput = editor.state.inputs.length > 0 ? editor.state.inputs.length - 1 : 0;\n\t\n\t /** If we have any inputs */\n\t if (editor.state.inputs.length) {\n\t\n\t /** getting firstlevel parent of input */\n\t firstLevelBlock = editor.content.getFirstLevelBlock(editor.state.inputs[indexOfLastInput]);\n\t }\n\t\n\t /** If input is empty, then we set caret to the last input */\n\t if (editor.state.inputs.length && editor.state.inputs[indexOfLastInput].textContent === '' && firstLevelBlock.dataset.tool == editor.settings.initialBlockPlugin) {\n\t\n\t editor.caret.setToBlock(indexOfLastInput);\n\t } else {\n\t\n\t /** Create new input when caret clicked in redactors area */\n\t var NEW_BLOCK_TYPE = editor.settings.initialBlockPlugin;\n\t\n\t editor.content.insertBlock({\n\t type: NEW_BLOCK_TYPE,\n\t block: editor.tools[NEW_BLOCK_TYPE].render()\n\t });\n\t\n\t /** If there is no inputs except inserted */\n\t if (editor.state.inputs.length === 1) {\n\t\n\t editor.caret.setToBlock(indexOfLastInput);\n\t } else {\n\t\n\t /** Set caret to this appended input */\n\t editor.caret.setToNextBlock(indexOfLastInput);\n\t }\n\t }\n\t } else {\n\t\n\t /** Close all panels */\n\t editor.toolbar.settings.close();\n\t editor.toolbar.toolbox.close();\n\t }\n\t\n\t /**\r\n\t * Move toolbar and open\r\n\t */\n\t editor.toolbar.move();\n\t editor.toolbar.open();\n\t\n\t var inputIsEmpty = !editor.content.currentNode.textContent.trim(),\n\t currentNodeType = editor.content.currentNode.dataset.tool,\n\t isInitialType = currentNodeType == editor.settings.initialBlockPlugin;\n\t\n\t /** Hide plus buttons */\n\t editor.toolbar.hidePlusButton();\n\t\n\t if (!inputIsEmpty) {\n\t\n\t /** Mark current block */\n\t editor.content.markBlock();\n\t }\n\t\n\t if (isInitialType && inputIsEmpty) {\n\t\n\t /** Show plus button */\n\t editor.toolbar.showPlusButton();\n\t }\n\t };\n\t\n\t /**\r\n\t * This method allows to define, is caret in contenteditable element or not.\r\n\t *\r\n\t * @private\r\n\t *\r\n\t * @description Otherwise, if we get TEXT node from range container, that will means we have input index.\r\n\t * In this case we use default browsers behaviour (if plugin allows that) or overwritten action.\r\n\t * Therefore, to be sure that we've clicked first-level block area, we should have currentNode, which always\r\n\t * specifies to the first-level block. Other cases we just ignore.\r\n\t */\n\t var detectWhenClickedOnFirstLevelBlockArea_ = function detectWhenClickedOnFirstLevelBlockArea_() {\n\t\n\t var selection = window.getSelection(),\n\t anchorNode = selection.anchorNode,\n\t flag = false;\n\t\n\t if (selection.rangeCount === 0) {\n\t\n\t editor.content.editorAreaHightlighted = true;\n\t } else {\n\t\n\t if (!editor.core.isDomNode(anchorNode)) {\n\t\n\t anchorNode = anchorNode.parentNode;\n\t }\n\t\n\t /** Already founded, without loop */\n\t if (anchorNode.contentEditable == 'true') {\n\t\n\t flag = true;\n\t }\n\t\n\t while (anchorNode.contentEditable != 'true') {\n\t\n\t anchorNode = anchorNode.parentNode;\n\t\n\t if (anchorNode.contentEditable == 'true') {\n\t\n\t flag = true;\n\t }\n\t\n\t if (anchorNode == document.body) {\n\t\n\t break;\n\t }\n\t }\n\t\n\t /** If editable element founded, flag is \"TRUE\", Therefore we return \"FALSE\" */\n\t editor.content.editorAreaHightlighted = !flag;\n\t }\n\t };\n\t\n\t /**\r\n\t * Toolbar button click handler\r\n\t *\r\n\t * @param {Object} event - cursor to the button\r\n\t * @protected\r\n\t *\r\n\t * @description gets current tool and calls render method\r\n\t */\n\t callbacks.toolbarButtonClicked = function (event) {\n\t\n\t var button = this;\n\t\n\t editor.toolbar.current = button.dataset.type;\n\t\n\t editor.toolbar.toolbox.toolClicked(event);\n\t editor.toolbar.close();\n\t };\n\t\n\t /**\r\n\t * Show or Hide toolbox when plus button is clicked\r\n\t */\n\t callbacks.plusButtonClicked = function () {\n\t\n\t if (!editor.nodes.toolbox.classList.contains('opened')) {\n\t\n\t editor.toolbar.toolbox.open();\n\t } else {\n\t\n\t editor.toolbar.toolbox.close();\n\t }\n\t };\n\t\n\t /**\r\n\t * Block handlers for KeyDown events\r\n\t *\r\n\t * @protected\r\n\t * @param {Object} event\r\n\t *\r\n\t * Handles keydowns on block\r\n\t * @see blockRightOrDownArrowPressed_\r\n\t * @see backspacePressed_\r\n\t * @see blockLeftOrUpArrowPressed_\r\n\t */\n\t callbacks.blockKeydown = function (event) {\n\t\n\t var block = event.target; // event.target is input\n\t\n\t switch (event.keyCode) {\n\t\n\t case editor.core.keys.DOWN:\n\t case editor.core.keys.RIGHT:\n\t blockRightOrDownArrowPressed_(event);\n\t break;\n\t\n\t case editor.core.keys.BACKSPACE:\n\t backspacePressed_(block, event);\n\t break;\n\t\n\t case editor.core.keys.UP:\n\t case editor.core.keys.LEFT:\n\t blockLeftOrUpArrowPressed_(event);\n\t break;\n\t\n\t }\n\t };\n\t\n\t /**\r\n\t * RIGHT or DOWN keydowns on block\r\n\t *\r\n\t * @param {Object} event\r\n\t * @private\r\n\t *\r\n\t * @description watches the selection and gets closest editable element.\r\n\t * Uses method getDeepestTextNodeFromPosition to get the last node of next block\r\n\t * Sets caret if it is contenteditable\r\n\t */\n\t var blockRightOrDownArrowPressed_ = function blockRightOrDownArrowPressed_(event) {\n\t\n\t var selection = window.getSelection(),\n\t inputs = editor.state.inputs,\n\t focusedNode = selection.anchorNode,\n\t focusedNodeHolder;\n\t\n\t /** Check for caret existance */\n\t if (!focusedNode) {\n\t\n\t return false;\n\t }\n\t\n\t /** Looking for closest (parent) contentEditable element of focused node */\n\t while (focusedNode.contentEditable != 'true') {\n\t\n\t focusedNodeHolder = focusedNode.parentNode;\n\t focusedNode = focusedNodeHolder;\n\t }\n\t\n\t /** Input index in DOM level */\n\t var editableElementIndex = 0;\n\t\n\t while (focusedNode != inputs[editableElementIndex]) {\n\t\n\t editableElementIndex++;\n\t }\n\t\n\t /**\r\n\t * Founded contentEditable element doesn't have childs\r\n\t * Or maybe New created block\r\n\t */\n\t if (!focusedNode.textContent) {\n\t\n\t editor.caret.setToNextBlock(editableElementIndex);\n\t return;\n\t }\n\t\n\t /**\r\n\t * Do nothing when caret doesn not reaches the end of last child\r\n\t */\n\t var caretInLastChild = false,\n\t caretAtTheEndOfText = false;\n\t\n\t var lastChild, deepestTextnode;\n\t\n\t lastChild = focusedNode.childNodes[focusedNode.childNodes.length - 1];\n\t\n\t if (editor.core.isDomNode(lastChild)) {\n\t\n\t deepestTextnode = editor.content.getDeepestTextNodeFromPosition(lastChild, lastChild.childNodes.length);\n\t } else {\n\t\n\t deepestTextnode = lastChild;\n\t }\n\t\n\t caretInLastChild = selection.anchorNode == deepestTextnode;\n\t caretAtTheEndOfText = deepestTextnode.length == selection.anchorOffset;\n\t\n\t if (!caretInLastChild || !caretAtTheEndOfText) {\n\t\n\t editor.core.log('arrow [down|right] : caret does not reached the end');\n\t return false;\n\t }\n\t\n\t editor.caret.setToNextBlock(editableElementIndex);\n\t };\n\t\n\t /**\r\n\t * LEFT or UP keydowns on block\r\n\t *\r\n\t * @param {Object} event\r\n\t * @private\r\n\t *\r\n\t * watches the selection and gets closest editable element.\r\n\t * Uses method getDeepestTextNodeFromPosition to get the last node of previous block\r\n\t * Sets caret if it is contenteditable\r\n\t *\r\n\t */\n\t var blockLeftOrUpArrowPressed_ = function blockLeftOrUpArrowPressed_(event) {\n\t\n\t var selection = window.getSelection(),\n\t inputs = editor.state.inputs,\n\t focusedNode = selection.anchorNode,\n\t focusedNodeHolder;\n\t\n\t /** Check for caret existance */\n\t if (!focusedNode) {\n\t\n\t return false;\n\t }\n\t\n\t /**\r\n\t * LEFT or UP not at the beginning\r\n\t */\n\t if (selection.anchorOffset !== 0) {\n\t\n\t return false;\n\t }\n\t\n\t /** Looking for parent contentEditable block */\n\t while (focusedNode.contentEditable != 'true') {\n\t\n\t focusedNodeHolder = focusedNode.parentNode;\n\t focusedNode = focusedNodeHolder;\n\t }\n\t\n\t /** Input index in DOM level */\n\t var editableElementIndex = 0;\n\t\n\t while (focusedNode != inputs[editableElementIndex]) {\n\t\n\t editableElementIndex++;\n\t }\n\t\n\t /**\r\n\t * Do nothing if caret is not at the beginning of first child\r\n\t */\n\t var caretInFirstChild = false,\n\t caretAtTheBeginning = false;\n\t\n\t var firstChild, deepestTextnode;\n\t\n\t /**\r\n\t * Founded contentEditable element doesn't have childs\r\n\t * Or maybe New created block\r\n\t */\n\t if (!focusedNode.textContent) {\n\t\n\t editor.caret.setToPreviousBlock(editableElementIndex);\n\t return;\n\t }\n\t\n\t firstChild = focusedNode.childNodes[0];\n\t\n\t if (editor.core.isDomNode(firstChild)) {\n\t\n\t deepestTextnode = editor.content.getDeepestTextNodeFromPosition(firstChild, 0);\n\t } else {\n\t\n\t deepestTextnode = firstChild;\n\t }\n\t\n\t caretInFirstChild = selection.anchorNode == deepestTextnode;\n\t caretAtTheBeginning = selection.anchorOffset === 0;\n\t\n\t if (caretInFirstChild && caretAtTheBeginning) {\n\t\n\t editor.caret.setToPreviousBlock(editableElementIndex);\n\t }\n\t };\n\t\n\t /**\r\n\t * Handles backspace keydown\r\n\t *\r\n\t * @param {Element} block\r\n\t * @param {Object} event\r\n\t * @private\r\n\t *\r\n\t * @description if block is empty, delete the block and set caret to the previous block\r\n\t * If block is not empty, try to merge two blocks - current and previous\r\n\t * But it we try'n to remove first block, then we should set caret to the next block, not previous.\r\n\t * If we removed the last block, create new one\r\n\t */\n\t var backspacePressed_ = function backspacePressed_(block, event) {\n\t\n\t var currentInputIndex = editor.caret.getCurrentInputIndex(),\n\t range,\n\t selectionLength,\n\t firstLevelBlocksCount;\n\t\n\t if (isNativeInput_(event.target)) {\n\t\n\t /** If input value is empty - remove block */\n\t if (event.target.value.trim() == '') {\n\t\n\t block.remove();\n\t } else {\n\t\n\t return;\n\t }\n\t }\n\t\n\t if (block.textContent.trim()) {\n\t\n\t range = editor.content.getRange();\n\t selectionLength = range.endOffset - range.startOffset;\n\t\n\t if (editor.caret.position.atStart() && !selectionLength && editor.state.inputs[currentInputIndex - 1]) {\n\t\n\t editor.content.mergeBlocks(currentInputIndex);\n\t } else {\n\t\n\t return;\n\t }\n\t }\n\t\n\t if (!selectionLength) {\n\t\n\t block.remove();\n\t }\n\t\n\t firstLevelBlocksCount = editor.nodes.redactor.childNodes.length;\n\t\n\t /**\r\n\t * If all blocks are removed\r\n\t */\n\t if (firstLevelBlocksCount === 0) {\n\t\n\t /** update currentNode variable */\n\t editor.content.currentNode = null;\n\t\n\t /** Inserting new empty initial block */\n\t editor.ui.addInitialBlock();\n\t\n\t /** Updating inputs state after deleting last block */\n\t editor.ui.saveInputs();\n\t\n\t /** Set to current appended block */\n\t window.setTimeout(function () {\n\t\n\t editor.caret.setToPreviousBlock(1);\n\t }, 10);\n\t } else {\n\t\n\t if (editor.caret.inputIndex !== 0) {\n\t\n\t /** Target block is not first */\n\t editor.caret.setToPreviousBlock(editor.caret.inputIndex);\n\t } else {\n\t\n\t /** If we try to delete first block */\n\t editor.caret.setToNextBlock(editor.caret.inputIndex);\n\t }\n\t }\n\t\n\t editor.toolbar.move();\n\t\n\t if (!editor.toolbar.opened) {\n\t\n\t editor.toolbar.open();\n\t }\n\t\n\t /** Updating inputs state */\n\t editor.ui.saveInputs();\n\t\n\t /** Prevent default browser behaviour */\n\t event.preventDefault();\n\t };\n\t\n\t /**\r\n\t * This method prevents default behaviour.\r\n\t *\r\n\t * @param {Object} event\r\n\t * @protected\r\n\t *\r\n\t * @description We get from clipboard pasted data, sanitize, make a fragment that contains of this sanitized nodes.\r\n\t * Firstly, we need to memorize the caret position. We can do that by getting the range of selection.\r\n\t * After all, we insert clear fragment into caret placed position. Then, we should move the caret to the last node\r\n\t */\n\t callbacks.blockPasteCallback = function (event) {\n\t\n\t /** If area is input or textarea then allow default behaviour */\n\t if (isNativeInput_(event.target)) {\n\t\n\t return;\n\t }\n\t\n\t /** Prevent default behaviour */\n\t event.preventDefault();\n\t\n\t var editableParent = editor.content.getEditableParent(event.target);\n\t\n\t /** Allow paste when event target placed in Editable element */\n\t if (!editableParent) {\n\t\n\t return;\n\t }\n\t\n\t /** get html pasted data - dirty data */\n\t var htmlData = event.clipboardData.getData('text/html'),\n\t plainData = event.clipboardData.getData('text/plain');\n\t\n\t /** Temporary DIV that is used to work with childs as arrays item */\n\t var div = editor.draw.node('DIV', '', {}),\n\t cleanData,\n\t wrappedData;\n\t\n\t /** Create fragment, that we paste to range after proccesing */\n\t cleanData = editor.sanitizer.clean(htmlData);\n\t\n\t /**\r\n\t * We wrap pasted text with

    tags to split it logically\r\n\t *\r\n\t * @type {string}\r\n\t */\n\t wrappedData = editor.content.wrapTextWithParagraphs(cleanData, plainData);\n\t div.innerHTML = wrappedData;\n\t\n\t var NEW_BLOCK_TYPE = editor.settings.initialBlockPlugin,\n\t currentBlockContent = editor.content.currentNode.firstChild.firstChild;\n\t\n\t /**\r\n\t * If there only one paragraph, just insert it\r\n\t */\n\t if (div.childNodes.length == 1) {\n\t\n\t editor.caret.insertNode(document.createTextNode(div.firstChild.innerHTML));\n\t return;\n\t }\n\t\n\t div.childNodes.forEach(function (paragraph, index) {\n\t\n\t /**\r\n\t * If there was no data in working node, replace it with first paragraph of pasted text\r\n\t */\n\t if (index == 0 && currentBlockContent.innerHTML.trim() === '') {\n\t\n\t editor.content.switchBlock(editor.content.currentNode, editor.tools[NEW_BLOCK_TYPE].render({\n\t text: paragraph.innerHTML\n\t }), NEW_BLOCK_TYPE);\n\t\n\t return;\n\t }\n\t\n\t editor.content.insertBlock({\n\t type: NEW_BLOCK_TYPE,\n\t block: editor.tools[NEW_BLOCK_TYPE].render({\n\t text: paragraph.innerHTML\n\t })\n\t });\n\t\n\t editor.caret.inputIndex++;\n\t });\n\t\n\t editor.caret.setToPreviousBlock(editor.caret.getCurrentInputIndex() + 1);\n\t };\n\t\n\t /**\r\n\t * used by UI module\r\n\t * Clicks on block settings button\r\n\t *\r\n\t * @param {Object} event\r\n\t * @protected\r\n\t * @description Opens toolbar settings\r\n\t */\n\t callbacks.showSettingsButtonClicked = function (event) {\n\t\n\t /**\r\n\t * Get type of current block\r\n\t * It uses to append settings from tool.settings property.\r\n\t * ...\r\n\t * Type is stored in data-type attribute on block\r\n\t */\n\t var currentToolType = editor.content.currentNode.dataset.tool;\n\t\n\t editor.toolbar.settings.toggle(currentToolType);\n\t\n\t /** Close toolbox when settings button is active */\n\t editor.toolbar.toolbox.close();\n\t editor.toolbar.settings.hideRemoveActions();\n\t };\n\t\n\t /**\r\n\t * Check block\r\n\t * @param target\r\n\t * @private\r\n\t *\r\n\t * @description Checks target is it native input\r\n\t */\n\t var isNativeInput_ = function isNativeInput_(target) {\n\t\n\t var nativeInputAreas = ['INPUT', 'TEXTAREA'];\n\t\n\t return nativeInputAreas.indexOf(target.tagName) != -1;\n\t };\n\t\n\t return callbacks;\n\t}({});\n\n/***/ },\n/* 13 */\n/***/ function(module, exports) {\n\n\t'use strict';\n\t\n\t/**\r\n\t * Codex Editor Draw module\r\n\t *\r\n\t * @author Codex Team\r\n\t * @version 1.0.\r\n\t */\n\t\n\tmodule.exports = function (draw) {\n\t\n\t /**\r\n\t * Base editor wrapper\r\n\t */\n\t draw.wrapper = function () {\n\t\n\t var wrapper = document.createElement('div');\n\t\n\t wrapper.className += 'codex-editor';\n\t\n\t return wrapper;\n\t };\n\t\n\t /**\r\n\t * Content-editable holder\r\n\t */\n\t draw.redactor = function () {\n\t\n\t var redactor = document.createElement('div');\n\t\n\t redactor.className += 'ce-redactor';\n\t\n\t return redactor;\n\t };\n\t\n\t draw.ceBlock = function () {\n\t\n\t var block = document.createElement('DIV');\n\t\n\t block.className += 'ce_block';\n\t\n\t return block;\n\t };\n\t\n\t /**\r\n\t * Empty toolbar with toggler\r\n\t */\n\t draw.toolbar = function () {\n\t\n\t var bar = document.createElement('div');\n\t\n\t bar.className += 'ce-toolbar';\n\t\n\t return bar;\n\t };\n\t\n\t draw.toolbarContent = function () {\n\t\n\t var wrapper = document.createElement('DIV');\n\t\n\t wrapper.classList.add('ce-toolbar__content');\n\t\n\t return wrapper;\n\t };\n\t\n\t /**\r\n\t * Inline toolbar\r\n\t */\n\t draw.inlineToolbar = function () {\n\t\n\t var bar = document.createElement('DIV');\n\t\n\t bar.className += 'ce-toolbar-inline';\n\t\n\t return bar;\n\t };\n\t\n\t /**\r\n\t * Wrapper for inline toobar buttons\r\n\t */\n\t draw.inlineToolbarButtons = function () {\n\t\n\t var wrapper = document.createElement('DIV');\n\t\n\t wrapper.className += 'ce-toolbar-inline__buttons';\n\t\n\t return wrapper;\n\t };\n\t\n\t /**\r\n\t * For some actions\r\n\t */\n\t draw.inlineToolbarActions = function () {\n\t\n\t var wrapper = document.createElement('DIV');\n\t\n\t wrapper.className += 'ce-toolbar-inline__actions';\n\t\n\t return wrapper;\n\t };\n\t\n\t draw.inputForLink = function () {\n\t\n\t var input = document.createElement('INPUT');\n\t\n\t input.type = 'input';\n\t input.className += 'inputForLink';\n\t input.placeholder = 'Вставьте ссылку ...';\n\t input.setAttribute('form', 'defaultForm');\n\t\n\t input.setAttribute('autofocus', 'autofocus');\n\t\n\t return input;\n\t };\n\t\n\t /**\r\n\t * @todo Desc\r\n\t */\n\t draw.blockButtons = function () {\n\t\n\t var block = document.createElement('div');\n\t\n\t block.className += 'ce-toolbar__actions';\n\t\n\t return block;\n\t };\n\t\n\t /**\r\n\t * Block settings panel\r\n\t */\n\t draw.blockSettings = function () {\n\t\n\t var settings = document.createElement('div');\n\t\n\t settings.className += 'ce-settings';\n\t\n\t return settings;\n\t };\n\t\n\t draw.defaultSettings = function () {\n\t\n\t var div = document.createElement('div');\n\t\n\t div.classList.add('ce-settings_default');\n\t\n\t return div;\n\t };\n\t\n\t draw.pluginsSettings = function () {\n\t\n\t var div = document.createElement('div');\n\t\n\t div.classList.add('ce-settings_plugin');\n\t\n\t return div;\n\t };\n\t\n\t draw.plusButton = function () {\n\t\n\t var button = document.createElement('span');\n\t\n\t button.className = 'ce-toolbar__plus';\n\t // button.innerHTML = '';\n\t\n\t return button;\n\t };\n\t\n\t /**\r\n\t * Settings button in toolbar\r\n\t */\n\t draw.settingsButton = function () {\n\t\n\t var toggler = document.createElement('span');\n\t\n\t toggler.className = 'ce-toolbar__settings-btn';\n\t\n\t /** Toggler button*/\n\t toggler.innerHTML = '';\n\t\n\t return toggler;\n\t };\n\t\n\t /**\r\n\t * Redactor tools wrapper\r\n\t */\n\t\n\t draw.toolbox = function () {\n\t\n\t var wrapper = document.createElement('div');\n\t\n\t wrapper.className = 'ce-toolbar__tools';\n\t\n\t return wrapper;\n\t };\n\t\n\t /**\r\n\t * @protected\r\n\t *\r\n\t * Draws tool buttons for toolbox\r\n\t *\r\n\t * @param {String} type\r\n\t * @param {String} classname\r\n\t * @returns {Element}\r\n\t */\n\t draw.toolbarButton = function (type, classname) {\n\t\n\t var button = document.createElement('li'),\n\t toolIcon = document.createElement('i'),\n\t toolTitle = document.createElement('span');\n\t\n\t button.dataset.type = type;\n\t button.setAttribute('title', type);\n\t\n\t toolIcon.classList.add(classname);\n\t toolTitle.classList.add('ce_toolbar_tools--title');\n\t\n\t button.appendChild(toolIcon);\n\t button.appendChild(toolTitle);\n\t\n\t return button;\n\t };\n\t\n\t /**\r\n\t * @protected\r\n\t *\r\n\t * Draws tools for inline toolbar\r\n\t *\r\n\t * @param {String} type\r\n\t * @param {String} classname\r\n\t */\n\t draw.toolbarButtonInline = function (type, classname) {\n\t\n\t var button = document.createElement('BUTTON'),\n\t toolIcon = document.createElement('I');\n\t\n\t button.type = 'button';\n\t button.dataset.type = type;\n\t toolIcon.classList.add(classname);\n\t\n\t button.appendChild(toolIcon);\n\t\n\t return button;\n\t };\n\t\n\t /**\r\n\t * Redactor block\r\n\t */\n\t draw.block = function (tagName, content) {\n\t\n\t var node = document.createElement(tagName);\n\t\n\t node.innerHTML = content || '';\n\t\n\t return node;\n\t };\n\t\n\t /**\r\n\t * Creates Node with passed tagName and className\r\n\t * @param {string} tagName\r\n\t * @param {string} className\r\n\t * @param {object} properties - allow to assign properties\r\n\t */\n\t draw.node = function (tagName, className, properties) {\n\t\n\t var el = document.createElement(tagName);\n\t\n\t if (className) el.className = className;\n\t\n\t if (properties) {\n\t\n\t for (var name in properties) {\n\t\n\t el[name] = properties[name];\n\t }\n\t }\n\t\n\t return el;\n\t };\n\t\n\t /**\r\n\t * Unavailable plugin block\r\n\t */\n\t draw.unavailableBlock = function () {\n\t\n\t var wrapper = document.createElement('DIV');\n\t\n\t wrapper.classList.add('cdx-unavailable-block');\n\t\n\t return wrapper;\n\t };\n\t\n\t return draw;\n\t}({});\n\n/***/ },\n/* 14 */\n/***/ function(module, exports) {\n\n\t'use strict';\n\t\n\t/**\r\n\t * Codex Editor Caret Module\r\n\t *\r\n\t * @author Codex Team\r\n\t * @version 1.0\r\n\t */\n\t\n\tmodule.exports = function (caret) {\n\t\n\t var editor = codex.editor;\n\t\n\t /**\r\n\t * @var {int} InputIndex - editable element in DOM\r\n\t */\n\t caret.inputIndex = null;\n\t\n\t /**\r\n\t * @var {int} offset - caret position in a text node.\r\n\t */\n\t caret.offset = null;\n\t\n\t /**\r\n\t * @var {int} focusedNodeIndex - we get index of child node from first-level block\r\n\t */\n\t caret.focusedNodeIndex = null;\n\t\n\t /**\r\n\t * Creates Document Range and sets caret to the element.\r\n\t * @protected\r\n\t * @uses caret.save — if you need to save caret position\r\n\t * @param {Element} el - Changed Node.\r\n\t */\n\t caret.set = function (el, index, offset) {\n\t\n\t offset = offset || caret.offset || 0;\n\t index = index || caret.focusedNodeIndex || 0;\n\t\n\t var childs = el.childNodes,\n\t nodeToSet;\n\t\n\t if (childs.length === 0) {\n\t\n\t nodeToSet = el;\n\t } else {\n\t\n\t nodeToSet = childs[index];\n\t }\n\t\n\t /** If Element is INPUT */\n\t if (el.contentEditable != 'true') {\n\t\n\t el.focus();\n\t return;\n\t }\n\t\n\t if (editor.core.isDomNode(nodeToSet)) {\n\t\n\t nodeToSet = editor.content.getDeepestTextNodeFromPosition(nodeToSet, nodeToSet.childNodes.length);\n\t }\n\t\n\t var range = document.createRange(),\n\t selection = window.getSelection();\n\t\n\t window.setTimeout(function () {\n\t\n\t range.setStart(nodeToSet, offset);\n\t range.setEnd(nodeToSet, offset);\n\t\n\t selection.removeAllRanges();\n\t selection.addRange(range);\n\t\n\t editor.caret.saveCurrentInputIndex();\n\t }, 20);\n\t };\n\t\n\t /**\r\n\t * @protected\r\n\t * Updates index of input and saves it in caret object\r\n\t */\n\t caret.saveCurrentInputIndex = function () {\n\t\n\t /** Index of Input that we paste sanitized content */\n\t var selection = window.getSelection(),\n\t inputs = editor.state.inputs,\n\t focusedNode = selection.anchorNode,\n\t focusedNodeHolder;\n\t\n\t if (!focusedNode) {\n\t\n\t return;\n\t }\n\t\n\t /** Looking for parent contentEditable block */\n\t while (focusedNode.contentEditable != 'true') {\n\t\n\t focusedNodeHolder = focusedNode.parentNode;\n\t focusedNode = focusedNodeHolder;\n\t }\n\t\n\t /** Input index in DOM level */\n\t var editableElementIndex = 0;\n\t\n\t while (focusedNode != inputs[editableElementIndex]) {\n\t\n\t editableElementIndex++;\n\t }\n\t\n\t caret.inputIndex = editableElementIndex;\n\t };\n\t\n\t /**\r\n\t * Returns current input index (caret object)\r\n\t */\n\t caret.getCurrentInputIndex = function () {\n\t\n\t return caret.inputIndex;\n\t };\n\t\n\t /**\r\n\t * @param {int} index - index of first-level block after that we set caret into next input\r\n\t */\n\t caret.setToNextBlock = function (index) {\n\t\n\t var inputs = editor.state.inputs,\n\t nextInput = inputs[index + 1];\n\t\n\t if (!nextInput) {\n\t\n\t editor.core.log('We are reached the end');\n\t return;\n\t }\n\t\n\t /**\r\n\t * When new Block created or deleted content of input\r\n\t * We should add some text node to set caret\r\n\t */\n\t if (!nextInput.childNodes.length) {\n\t\n\t var emptyTextElement = document.createTextNode('');\n\t\n\t nextInput.appendChild(emptyTextElement);\n\t }\n\t\n\t editor.caret.inputIndex = index + 1;\n\t editor.caret.set(nextInput, 0, 0);\n\t editor.content.workingNodeChanged(nextInput);\n\t };\n\t\n\t /**\r\n\t * @param {int} index - index of target input.\r\n\t * Sets caret to input with this index\r\n\t */\n\t caret.setToBlock = function (index) {\n\t\n\t var inputs = editor.state.inputs,\n\t targetInput = inputs[index];\n\t\n\t if (!targetInput) {\n\t\n\t return;\n\t }\n\t\n\t /**\r\n\t * When new Block created or deleted content of input\r\n\t * We should add some text node to set caret\r\n\t */\n\t if (!targetInput.childNodes.length) {\n\t\n\t var emptyTextElement = document.createTextNode('');\n\t\n\t targetInput.appendChild(emptyTextElement);\n\t }\n\t\n\t editor.caret.inputIndex = index;\n\t editor.caret.set(targetInput, 0, 0);\n\t editor.content.workingNodeChanged(targetInput);\n\t };\n\t\n\t /**\r\n\t * @param {int} index - index of input\r\n\t */\n\t caret.setToPreviousBlock = function (index) {\n\t\n\t index = index || 0;\n\t\n\t var inputs = editor.state.inputs,\n\t previousInput = inputs[index - 1],\n\t lastChildNode,\n\t lengthOfLastChildNode,\n\t emptyTextElement;\n\t\n\t if (!previousInput) {\n\t\n\t editor.core.log('We are reached first node');\n\t return;\n\t }\n\t\n\t lastChildNode = editor.content.getDeepestTextNodeFromPosition(previousInput, previousInput.childNodes.length);\n\t lengthOfLastChildNode = lastChildNode.length;\n\t\n\t /**\r\n\t * When new Block created or deleted content of input\r\n\t * We should add some text node to set caret\r\n\t */\n\t if (!previousInput.childNodes.length) {\n\t\n\t emptyTextElement = document.createTextNode('');\n\t previousInput.appendChild(emptyTextElement);\n\t }\n\t editor.caret.inputIndex = index - 1;\n\t editor.caret.set(previousInput, previousInput.childNodes.length - 1, lengthOfLastChildNode);\n\t editor.content.workingNodeChanged(inputs[index - 1]);\n\t };\n\t\n\t caret.position = {\n\t\n\t atStart: function atStart() {\n\t\n\t var selection = window.getSelection(),\n\t anchorOffset = selection.anchorOffset,\n\t anchorNode = selection.anchorNode,\n\t firstLevelBlock = editor.content.getFirstLevelBlock(anchorNode),\n\t pluginsRender = firstLevelBlock.childNodes[0];\n\t\n\t if (!editor.core.isDomNode(anchorNode)) {\n\t\n\t anchorNode = anchorNode.parentNode;\n\t }\n\t\n\t var isFirstNode = anchorNode === pluginsRender.childNodes[0],\n\t isOffsetZero = anchorOffset === 0;\n\t\n\t return isFirstNode && isOffsetZero;\n\t },\n\t\n\t atTheEnd: function atTheEnd() {\n\t\n\t var selection = window.getSelection(),\n\t anchorOffset = selection.anchorOffset,\n\t anchorNode = selection.anchorNode;\n\t\n\t /** Caret is at the end of input */\n\t return !anchorNode || !anchorNode.length || anchorOffset === anchorNode.length;\n\t }\n\t };\n\t\n\t /**\r\n\t * Inserts node at the caret location\r\n\t * @param node\r\n\t */\n\t caret.insertNode = function (node) {\n\t\n\t var selection, range;\n\t\n\t selection = window.getSelection();\n\t\n\t range = selection.getRangeAt(0);\n\t range.deleteContents();\n\t\n\t range.insertNode(node);\n\t\n\t range.setStartAfter(node);\n\t range.collapse(true);\n\t\n\t selection.removeAllRanges();\n\t selection.addRange(range);\n\t };\n\t\n\t return caret;\n\t}({});\n\n/***/ },\n/* 15 */\n/***/ function(module, exports) {\n\n\t'use strict';\n\t\n\t/**\r\n\t * Codex Editor Notification Module\r\n\t *\r\n\t * @author Codex Team\r\n\t * @version 1.0\r\n\t */\n\t\n\tmodule.exports = function (notifications) {\n\t\n\t var editor = codex.editor;\n\t\n\t var queue = [];\n\t\n\t var addToQueue = function addToQueue(settings) {\n\t\n\t queue.push(settings);\n\t\n\t var index = 0;\n\t\n\t while (index < queue.length && queue.length > 5) {\n\t\n\t if (queue[index].type == 'confirm' || queue[index].type == 'prompt') {\n\t\n\t index++;\n\t continue;\n\t }\n\t\n\t queue[index].close();\n\t queue.splice(index, 1);\n\t }\n\t };\n\t\n\t notifications.createHolder = function () {\n\t\n\t var holder = editor.draw.node('DIV', 'cdx-notifications-block');\n\t\n\t editor.nodes.notifications = document.body.appendChild(holder);\n\t\n\t return holder;\n\t };\n\t\n\t /**\r\n\t * Error notificator. Shows block with message\r\n\t * @protected\r\n\t */\n\t notifications.errorThrown = function (errorMsg, event) {\n\t\n\t editor.notifications.notification({ message: 'This action is not available currently', type: event.type });\n\t };\n\t\n\t /**\r\n\t *\r\n\t * Appends notification\r\n\t *\r\n\t * settings = {\r\n\t * type - notification type (reserved types: alert, confirm, prompt). Just add class 'cdx-notification-'+type\r\n\t * message - notification message\r\n\t * okMsg - confirm button text (default - 'Ok')\r\n\t * cancelBtn - cancel button text (default - 'Cancel'). Only for confirm and prompt types\r\n\t * confirm - function-handler for ok button click\r\n\t * cancel - function-handler for cancel button click. Only for confirm and prompt types\r\n\t * time - time (in seconds) after which notification will close (default - 10s)\r\n\t * }\r\n\t *\r\n\t * @param settings\r\n\t */\n\t notifications.notification = function (constructorSettings) {\n\t\n\t /** Private vars and methods */\n\t var notification = null,\n\t cancel = null,\n\t type = null,\n\t confirm = null,\n\t inputField = null;\n\t\n\t var confirmHandler = function confirmHandler() {\n\t\n\t close();\n\t\n\t if (typeof confirm !== 'function') {\n\t\n\t return;\n\t }\n\t\n\t if (type == 'prompt') {\n\t\n\t confirm(inputField.value);\n\t return;\n\t }\n\t\n\t confirm();\n\t };\n\t\n\t var cancelHandler = function cancelHandler() {\n\t\n\t close();\n\t\n\t if (typeof cancel !== 'function') {\n\t\n\t return;\n\t }\n\t\n\t cancel();\n\t };\n\t\n\t /** Public methods */\n\t function create(settings) {\n\t\n\t if (!(settings && settings.message)) {\n\t\n\t editor.core.log('Can\\'t create notification. Message is missed');\n\t return;\n\t }\n\t\n\t settings.type = settings.type || 'alert';\n\t settings.time = settings.time * 1000 || 10000;\n\t\n\t var wrapper = editor.draw.node('DIV', 'cdx-notification'),\n\t message = editor.draw.node('DIV', 'cdx-notification__message'),\n\t input = editor.draw.node('INPUT', 'cdx-notification__input'),\n\t okBtn = editor.draw.node('SPAN', 'cdx-notification__ok-btn'),\n\t cancelBtn = editor.draw.node('SPAN', 'cdx-notification__cancel-btn');\n\t\n\t message.textContent = settings.message;\n\t okBtn.textContent = settings.okMsg || 'ОК';\n\t cancelBtn.textContent = settings.cancelMsg || 'Отмена';\n\t\n\t editor.listeners.add(okBtn, 'click', confirmHandler);\n\t editor.listeners.add(cancelBtn, 'click', cancelHandler);\n\t\n\t wrapper.appendChild(message);\n\t\n\t if (settings.type == 'prompt') {\n\t\n\t wrapper.appendChild(input);\n\t }\n\t\n\t wrapper.appendChild(okBtn);\n\t\n\t if (settings.type == 'prompt' || settings.type == 'confirm') {\n\t\n\t wrapper.appendChild(cancelBtn);\n\t }\n\t\n\t wrapper.classList.add('cdx-notification-' + settings.type);\n\t wrapper.dataset.type = settings.type;\n\t\n\t notification = wrapper;\n\t type = settings.type;\n\t confirm = settings.confirm;\n\t cancel = settings.cancel;\n\t inputField = input;\n\t\n\t if (settings.type != 'prompt' && settings.type != 'confirm') {\n\t\n\t window.setTimeout(close, settings.time);\n\t }\n\t };\n\t\n\t function send() {\n\t\n\t editor.nodes.notifications.appendChild(notification);\n\t inputField.focus();\n\t\n\t editor.nodes.notifications.classList.add('cdx-notification__notification-appending');\n\t\n\t window.setTimeout(function () {\n\t\n\t editor.nodes.notifications.classList.remove('cdx-notification__notification-appending');\n\t }, 100);\n\t\n\t addToQueue({ type: type, close: close });\n\t };\n\t\n\t function close() {\n\t\n\t notification.remove();\n\t };\n\t\n\t if (constructorSettings) {\n\t\n\t create(constructorSettings);\n\t send();\n\t }\n\t\n\t return {\n\t create: create,\n\t send: send,\n\t close: close\n\t };\n\t };\n\t\n\t notifications.clear = function () {\n\t\n\t editor.nodes.notifications.innerHTML = '';\n\t queue = [];\n\t };\n\t\n\t return notifications;\n\t}({});\n\n/***/ },\n/* 16 */\n/***/ function(module, exports) {\n\n\t\"use strict\";\n\t\n\t/**\r\n\t * Codex Editor Parser Module\r\n\t *\r\n\t * @author Codex Team\r\n\t * @version 1.1\r\n\t */\n\t\n\tmodule.exports = function (parser) {\n\t\n\t var editor = codex.editor;\n\t\n\t /** inserting text */\n\t parser.insertPastedContent = function (blockType, tag) {\n\t\n\t editor.content.insertBlock({\n\t type: blockType.type,\n\t block: blockType.render({\n\t text: tag.innerHTML\n\t })\n\t });\n\t };\n\t\n\t /**\r\n\t * Check DOM node for display style: separated block or child-view\r\n\t */\n\t parser.isFirstLevelBlock = function (node) {\n\t\n\t return node.nodeType == editor.core.nodeTypes.TAG && node.classList.contains(editor.ui.className.BLOCK_CLASSNAME);\n\t };\n\t\n\t return parser;\n\t}({});\n\n/***/ },\n/* 17 */\n/***/ function(module, exports, __webpack_require__) {\n\n\t'use strict';\n\t\n\t/**\r\n\t * Codex Sanitizer\r\n\t */\n\t\n\tmodule.exports = function (sanitizer) {\n\t\n\t /** HTML Janitor library */\n\t var janitor = __webpack_require__(18);\n\t\n\t /** Codex Editor */\n\t var editor = codex.editor;\n\t\n\t sanitizer.prepare = function () {\n\t\n\t if (editor.settings.sanitizer && !editor.core.isEmpty(editor.settings.sanitizer)) {\n\t\n\t Config.CUSTOM = editor.settings.sanitizer;\n\t }\n\t };\n\t\n\t /**\r\n\t * Basic config\r\n\t */\n\t var Config = {\n\t\n\t /** User configuration */\n\t CUSTOM: null,\n\t\n\t BASIC: {\n\t\n\t tags: {\n\t p: {},\n\t a: {\n\t href: true,\n\t target: '_blank',\n\t rel: 'nofollow'\n\t },\n\t i: {},\n\t b: {},\n\t strong: {},\n\t em: {},\n\t span: {}\n\t }\n\t }\n\t };\n\t\n\t sanitizer.Config = Config;\n\t\n\t /**\r\n\t *\r\n\t * @param userCustomConfig\r\n\t * @returns {*}\r\n\t * @private\r\n\t *\r\n\t * @description If developer uses editor's API, then he can customize sane restrictions.\r\n\t * Or, sane config can be defined globally in editors initialization. That config will be used everywhere\r\n\t * At least, if there is no config overrides, that API uses BASIC Default configation\r\n\t */\n\t var init_ = function init_(userCustomConfig) {\n\t\n\t var configuration = userCustomConfig || Config.CUSTOM || Config.BASIC;\n\t\n\t return new janitor(configuration);\n\t };\n\t\n\t /**\r\n\t * Cleans string from unwanted tags\r\n\t * @protected\r\n\t * @param {String} dirtyString - taint string\r\n\t * @param {Object} customConfig - allowed tags\r\n\t */\n\t sanitizer.clean = function (dirtyString, customConfig) {\n\t\n\t var janitorInstance = init_(customConfig);\n\t\n\t return janitorInstance.clean(dirtyString);\n\t };\n\t\n\t return sanitizer;\n\t}({});\n\n/***/ },\n/* 18 */\n/***/ function(module, exports, __webpack_require__) {\n\n\tvar __WEBPACK_AMD_DEFINE_FACTORY__, __WEBPACK_AMD_DEFINE_RESULT__;(function (root, factory) {\n\t if (true) {\n\t !(__WEBPACK_AMD_DEFINE_FACTORY__ = (factory), __WEBPACK_AMD_DEFINE_RESULT__ = (typeof __WEBPACK_AMD_DEFINE_FACTORY__ === 'function' ? (__WEBPACK_AMD_DEFINE_FACTORY__.call(exports, __webpack_require__, exports, module)) : __WEBPACK_AMD_DEFINE_FACTORY__), __WEBPACK_AMD_DEFINE_RESULT__ !== undefined && (module.exports = __WEBPACK_AMD_DEFINE_RESULT__));\n\t } else if (typeof exports === 'object') {\n\t module.exports = factory();\n\t } else {\n\t root.HTMLJanitor = factory();\n\t }\n\t}(this, function () {\n\t\n\t /**\n\t * @param {Object} config.tags Dictionary of allowed tags.\n\t * @param {boolean} config.keepNestedBlockElements Default false.\n\t */\n\t function HTMLJanitor(config) {\n\t\n\t var tagDefinitions = config['tags'];\n\t var tags = Object.keys(tagDefinitions);\n\t\n\t var validConfigValues = tags\n\t .map(function(k) { return typeof tagDefinitions[k]; })\n\t .every(function(type) { return type === 'object' || type === 'boolean' || type === 'function'; });\n\t\n\t if(!validConfigValues) {\n\t throw new Error(\"The configuration was invalid\");\n\t }\n\t\n\t this.config = config;\n\t }\n\t\n\t // TODO: not exhaustive?\n\t var blockElementNames = ['P', 'LI', 'TD', 'TH', 'DIV', 'H1', 'H2', 'H3', 'H4', 'H5', 'H6', 'PRE'];\n\t function isBlockElement(node) {\n\t return blockElementNames.indexOf(node.nodeName) !== -1;\n\t }\n\t\n\t var inlineElementNames = ['A', 'B', 'STRONG', 'I', 'EM', 'SUB', 'SUP', 'U', 'STRIKE'];\n\t function isInlineElement(node) {\n\t return inlineElementNames.indexOf(node.nodeName) !== -1;\n\t }\n\t\n\t HTMLJanitor.prototype.clean = function (html) {\n\t var sandbox = document.createElement('div');\n\t sandbox.innerHTML = html;\n\t\n\t this._sanitize(sandbox);\n\t\n\t return sandbox.innerHTML;\n\t };\n\t\n\t HTMLJanitor.prototype._sanitize = function (parentNode) {\n\t var treeWalker = createTreeWalker(parentNode);\n\t var node = treeWalker.firstChild();\n\t if (!node) { return; }\n\t\n\t do {\n\t // Ignore nodes that have already been sanitized\n\t if (node._sanitized) {\n\t continue;\n\t }\n\t\n\t if (node.nodeType === Node.TEXT_NODE) {\n\t // If this text node is just whitespace and the previous or next element\n\t // sibling is a block element, remove it\n\t // N.B.: This heuristic could change. Very specific to a bug with\n\t // `contenteditable` in Firefox: http://jsbin.com/EyuKase/1/edit?js,output\n\t // FIXME: make this an option?\n\t if (node.data.trim() === ''\n\t && ((node.previousElementSibling && isBlockElement(node.previousElementSibling))\n\t || (node.nextElementSibling && isBlockElement(node.nextElementSibling)))) {\n\t parentNode.removeChild(node);\n\t this._sanitize(parentNode);\n\t break;\n\t } else {\n\t continue;\n\t }\n\t }\n\t\n\t // Remove all comments\n\t if (node.nodeType === Node.COMMENT_NODE) {\n\t parentNode.removeChild(node);\n\t this._sanitize(parentNode);\n\t break;\n\t }\n\t\n\t var isInline = isInlineElement(node);\n\t var containsBlockElement;\n\t if (isInline) {\n\t containsBlockElement = Array.prototype.some.call(node.childNodes, isBlockElement);\n\t }\n\t\n\t // Block elements should not be nested (e.g.

  • ...); if\n\t // they are, we want to unwrap the inner block element.\n\t var isNotTopContainer = !! parentNode.parentNode;\n\t var isNestedBlockElement =\n\t isBlockElement(parentNode) &&\n\t isBlockElement(node) &&\n\t isNotTopContainer;\n\t\n\t var nodeName = node.nodeName.toLowerCase();\n\t\n\t var allowedAttrs = getAllowedAttrs(this.config, nodeName, node);\n\t\n\t var isInvalid = isInline && containsBlockElement;\n\t\n\t // Drop tag entirely according to the whitelist *and* if the markup\n\t // is invalid.\n\t if (isInvalid || shouldRejectNode(node, allowedAttrs)\n\t || (!this.config.keepNestedBlockElements && isNestedBlockElement)) {\n\t // Do not keep the inner text of SCRIPT/STYLE elements.\n\t if (! (node.nodeName === 'SCRIPT' || node.nodeName === 'STYLE')) {\n\t while (node.childNodes.length > 0) {\n\t parentNode.insertBefore(node.childNodes[0], node);\n\t }\n\t }\n\t parentNode.removeChild(node);\n\t\n\t this._sanitize(parentNode);\n\t break;\n\t }\n\t\n\t // Sanitize attributes\n\t for (var a = 0; a < node.attributes.length; a += 1) {\n\t var attr = node.attributes[a];\n\t\n\t if (shouldRejectAttr(attr, allowedAttrs, node)) {\n\t node.removeAttribute(attr.name);\n\t // Shift the array to continue looping.\n\t a = a - 1;\n\t }\n\t }\n\t\n\t // Sanitize children\n\t this._sanitize(node);\n\t\n\t // Mark node as sanitized so it's ignored in future runs\n\t node._sanitized = true;\n\t } while ((node = treeWalker.nextSibling()));\n\t };\n\t\n\t function createTreeWalker(node) {\n\t return document.createTreeWalker(node,\n\t NodeFilter.SHOW_TEXT | NodeFilter.SHOW_ELEMENT | NodeFilter.SHOW_COMMENT,\n\t null, false);\n\t }\n\t\n\t function getAllowedAttrs(config, nodeName, node){\n\t if (typeof config.tags[nodeName] === 'function') {\n\t return config.tags[nodeName](node);\n\t } else {\n\t return config.tags[nodeName];\n\t }\n\t }\n\t\n\t function shouldRejectNode(node, allowedAttrs){\n\t if (typeof allowedAttrs === 'undefined') {\n\t return true;\n\t } else if (typeof allowedAttrs === 'boolean') {\n\t return !allowedAttrs;\n\t }\n\t\n\t return false;\n\t }\n\t\n\t function shouldRejectAttr(attr, allowedAttrs, node){\n\t var attrName = attr.name.toLowerCase();\n\t\n\t if (allowedAttrs === true){\n\t return false;\n\t } else if (typeof allowedAttrs[attrName] === 'function'){\n\t return !allowedAttrs[attrName](attr.value, node);\n\t } else if (typeof allowedAttrs[attrName] === 'undefined'){\n\t return true;\n\t } else if (allowedAttrs[attrName] === false) {\n\t return true;\n\t } else if (typeof allowedAttrs[attrName] === 'string') {\n\t return (allowedAttrs[attrName] !== attr.value);\n\t }\n\t\n\t return false;\n\t }\n\t\n\t return HTMLJanitor;\n\t\n\t}));\n\n\n/***/ },\n/* 19 */\n/***/ function(module, exports) {\n\n\t\"use strict\";\n\t\n\t/**\r\n\t * Codex Editor Listeners module\r\n\t *\r\n\t * @author Codex Team\r\n\t * @version 1.0\r\n\t */\n\t\n\t/**\r\n\t * Module-decorator for event listeners assignment\r\n\t */\n\tmodule.exports = function (listeners) {\n\t\n\t var allListeners = [];\n\t\n\t /**\r\n\t * Search methods\r\n\t *\r\n\t * byElement, byType and byHandler returns array of suitable listeners\r\n\t * one and all takes element, eventType, and handler and returns first (all) suitable listener\r\n\t *\r\n\t */\n\t listeners.search = function () {\n\t\n\t var byElement = function byElement(element, context) {\n\t\n\t var listenersOnElement = [];\n\t\n\t context = context || allListeners;\n\t\n\t for (var i = 0; i < context.length; i++) {\n\t\n\t var listener = context[i];\n\t\n\t if (listener.element === element) {\n\t\n\t listenersOnElement.push(listener);\n\t }\n\t }\n\t\n\t return listenersOnElement;\n\t };\n\t\n\t var byType = function byType(eventType, context) {\n\t\n\t var listenersWithType = [];\n\t\n\t context = context || allListeners;\n\t\n\t for (var i = 0; i < context.length; i++) {\n\t\n\t var listener = context[i];\n\t\n\t if (listener.type === eventType) {\n\t\n\t listenersWithType.push(listener);\n\t }\n\t }\n\t\n\t return listenersWithType;\n\t };\n\t\n\t var byHandler = function byHandler(handler, context) {\n\t\n\t var listenersWithHandler = [];\n\t\n\t context = context || allListeners;\n\t\n\t for (var i = 0; i < context.length; i++) {\n\t\n\t var listener = context[i];\n\t\n\t if (listener.handler === handler) {\n\t\n\t listenersWithHandler.push(listener);\n\t }\n\t }\n\t\n\t return listenersWithHandler;\n\t };\n\t\n\t var one = function one(element, eventType, handler) {\n\t\n\t var result = allListeners;\n\t\n\t if (element) result = byElement(element, result);\n\t\n\t if (eventType) result = byType(eventType, result);\n\t\n\t if (handler) result = byHandler(handler, result);\n\t\n\t return result[0];\n\t };\n\t\n\t var all = function all(element, eventType, handler) {\n\t\n\t var result = allListeners;\n\t\n\t if (element) result = byElement(element, result);\n\t\n\t if (eventType) result = byType(eventType, result);\n\t\n\t if (handler) result = byHandler(handler, result);\n\t\n\t return result;\n\t };\n\t\n\t return {\n\t byElement: byElement,\n\t byType: byType,\n\t byHandler: byHandler,\n\t one: one,\n\t all: all\n\t };\n\t }();\n\t\n\t listeners.add = function (element, eventType, handler, isCapture) {\n\t\n\t element.addEventListener(eventType, handler, isCapture);\n\t\n\t var data = {\n\t element: element,\n\t type: eventType,\n\t handler: handler\n\t };\n\t\n\t var alreadyAddedListener = listeners.search.one(element, eventType, handler);\n\t\n\t if (!alreadyAddedListener) {\n\t\n\t allListeners.push(data);\n\t }\n\t };\n\t\n\t listeners.remove = function (element, eventType, handler) {\n\t\n\t element.removeEventListener(eventType, handler);\n\t\n\t var existingListeners = listeners.search.all(element, eventType, handler);\n\t\n\t for (var i = 0; i < existingListeners.length; i++) {\n\t\n\t var index = allListeners.indexOf(existingListeners[i]);\n\t\n\t if (index > 0) {\n\t\n\t allListeners.splice(index, 1);\n\t }\n\t }\n\t };\n\t\n\t listeners.removeAll = function () {\n\t\n\t allListeners.map(function (current) {\n\t\n\t listeners.remove(current.element, current.type, current.handler);\n\t });\n\t };\n\t\n\t listeners.get = function (element, eventType, handler) {\n\t\n\t return listeners.search.all(element, eventType, handler);\n\t };\n\t\n\t return listeners;\n\t}({});\n\n/***/ },\n/* 20 */\n/***/ function(module, exports) {\n\n\t'use strict';\n\t\n\tvar _typeof = typeof Symbol === \"function\" && typeof Symbol.iterator === \"symbol\" ? function (obj) { return typeof obj; } : function (obj) { return obj && typeof Symbol === \"function\" && obj.constructor === Symbol && obj !== Symbol.prototype ? \"symbol\" : typeof obj; };\n\t\n\t/**\r\n\t * Codex Editor Destroyer module\r\n\t *\r\n\t * @auhor Codex Team\r\n\t * @version 1.0\r\n\t */\n\t\n\tmodule.exports = function (destroyer) {\n\t\n\t var editor = codex.editor;\n\t\n\t destroyer.removeNodes = function () {\n\t\n\t editor.nodes.wrapper.remove();\n\t editor.nodes.notifications.remove();\n\t };\n\t\n\t destroyer.destroyPlugins = function () {\n\t\n\t for (var tool in editor.tools) {\n\t\n\t if (typeof editor.tools[tool].destroy === 'function') {\n\t\n\t editor.tools[tool].destroy();\n\t }\n\t }\n\t };\n\t\n\t destroyer.destroyScripts = function () {\n\t\n\t var scripts = document.getElementsByTagName('SCRIPT');\n\t\n\t for (var i = 0; i < scripts.length; i++) {\n\t\n\t if (scripts[i].id.indexOf(editor.scriptPrefix) + 1) {\n\t\n\t scripts[i].remove();\n\t i--;\n\t }\n\t }\n\t };\n\t\n\t /**\r\n\t * Delete editor data from webpage.\r\n\t * You should send settings argument with boolean flags:\r\n\t * @param settings.ui- remove redactor event listeners and DOM nodes\r\n\t * @param settings.scripts - remove redactor scripts from DOM\r\n\t * @param settings.plugins - remove plugin's objects\r\n\t * @param settings.core - remove editor core. You can remove core only if UI and scripts flags is true\r\n\t * }\r\n\t *\r\n\t */\n\t destroyer.destroy = function (settings) {\n\t\n\t if (!settings || (typeof settings === 'undefined' ? 'undefined' : _typeof(settings)) !== 'object') {\n\t\n\t return;\n\t }\n\t\n\t if (settings.ui) {\n\t\n\t destroyer.removeNodes();\n\t editor.listeners.removeAll();\n\t }\n\t\n\t if (settings.scripts) {\n\t\n\t destroyer.destroyScripts();\n\t }\n\t\n\t if (settings.plugins) {\n\t\n\t destroyer.destroyPlugins();\n\t }\n\t\n\t if (settings.ui && settings.scripts && settings.core) {\n\t\n\t delete codex.editor;\n\t }\n\t };\n\t\n\t return destroyer;\n\t}({});\n\n/***/ },\n/* 21 */\n/***/ function(module, exports) {\n\n\t'use strict';\n\t\n\t/**\r\n\t * Codex Editor Paste module\r\n\t *\r\n\t * @author Codex Team\r\n\t * @version 1.1.1\r\n\t */\n\t\n\tmodule.exports = function (paste) {\n\t\n\t var editor = codex.editor;\n\t\n\t var patterns = [];\n\t\n\t paste.prepare = function () {\n\t\n\t var tools = editor.tools;\n\t\n\t for (var tool in tools) {\n\t\n\t if (!tools[tool].renderOnPastePatterns || !Array.isArray(tools[tool].renderOnPastePatterns)) {\n\t\n\t continue;\n\t }\n\t\n\t tools[tool].renderOnPastePatterns.map(function (pattern) {\n\t\n\t patterns.push(pattern);\n\t });\n\t }\n\t\n\t return Promise.resolve();\n\t };\n\t\n\t /**\r\n\t * Saves data\r\n\t * @param event\r\n\t */\n\t paste.pasted = function (event) {\n\t\n\t var clipBoardData = event.clipboardData || window.clipboardData,\n\t content = clipBoardData.getData('Text');\n\t\n\t var result = analize(content);\n\t\n\t if (result) {\n\t\n\t event.preventDefault();\n\t event.stopImmediatePropagation();\n\t }\n\t\n\t return result;\n\t };\n\t\n\t /**\r\n\t * Analizes pated string and calls necessary method\r\n\t */\n\t\n\t var analize = function analize(string) {\n\t\n\t var result = false,\n\t content = editor.content.currentNode,\n\t plugin = content.dataset.tool;\n\t\n\t patterns.map(function (pattern) {\n\t\n\t var execArray = pattern.regex.exec(string),\n\t match = execArray && execArray[0];\n\t\n\t if (match && match === string.trim()) {\n\t\n\t /** current block is not empty */\n\t if (content.textContent.trim() && plugin == editor.settings.initialBlockPlugin) {\n\t\n\t pasteToNewBlock_();\n\t }\n\t\n\t pattern.callback(string, pattern);\n\t result = true;\n\t }\n\t });\n\t\n\t return result;\n\t };\n\t\n\t var pasteToNewBlock_ = function pasteToNewBlock_() {\n\t\n\t /** Create new initial block */\n\t editor.content.insertBlock({\n\t\n\t type: editor.settings.initialBlockPlugin,\n\t block: editor.tools[editor.settings.initialBlockPlugin].render({\n\t text: ''\n\t })\n\t\n\t }, false);\n\t };\n\t\n\t return paste;\n\t}({});\n\n/***/ }\n/******/ ]);\n\n\n// WEBPACK FOOTER //\n// codex-editor.js"," \t// The module cache\n \tvar installedModules = {};\n\n \t// The require function\n \tfunction __webpack_require__(moduleId) {\n\n \t\t// Check if module is in cache\n \t\tif(installedModules[moduleId])\n \t\t\treturn installedModules[moduleId].exports;\n\n \t\t// Create a new module (and put it into the cache)\n \t\tvar module = installedModules[moduleId] = {\n \t\t\texports: {},\n \t\t\tid: moduleId,\n \t\t\tloaded: false\n \t\t};\n\n \t\t// Execute the module function\n \t\tmodules[moduleId].call(module.exports, module, module.exports, __webpack_require__);\n\n \t\t// Flag the module as loaded\n \t\tmodule.loaded = true;\n\n \t\t// Return the exports of the module\n \t\treturn module.exports;\n \t}\n\n\n \t// expose the modules object (__webpack_modules__)\n \t__webpack_require__.m = modules;\n\n \t// expose the module cache\n \t__webpack_require__.c = installedModules;\n\n \t// __webpack_public_path__\n \t__webpack_require__.p = \"\";\n\n \t// Load entry module and return exports\n \treturn __webpack_require__(0);\n\n\n\n// WEBPACK FOOTER //\n// webpack/bootstrap b4a1ef56ed45877046f1","/**\r\n *\r\n * Codex Editor\r\n *\r\n * @author Codex Team\r\n */\r\n\r\nmodule.exports = (function (editor) {\r\n\r\n 'use strict';\r\n\r\n editor.version = VERSION;\r\n editor.scriptPrefix = 'cdx-script-';\r\n\r\n var init = function () {\r\n\r\n editor.core = require('./modules/core');\r\n editor.tools = require('./modules/tools');\r\n editor.ui = require('./modules/ui');\r\n editor.transport = require('./modules/transport');\r\n editor.renderer = require('./modules/renderer');\r\n editor.saver = require('./modules/saver');\r\n editor.content = require('./modules/content');\r\n editor.toolbar = require('./modules/toolbar/toolbar');\r\n editor.callback = require('./modules/callbacks');\r\n editor.draw = require('./modules/draw');\r\n editor.caret = require('./modules/caret');\r\n editor.notifications = require('./modules/notifications');\r\n editor.parser = require('./modules/parser');\r\n editor.sanitizer = require('./modules/sanitizer');\r\n editor.listeners = require('./modules/listeners');\r\n editor.destroyer = require('./modules/destroyer');\r\n editor.paste = require('./modules/paste');\r\n\r\n };\r\n\r\n /**\r\n * @public\r\n * holds initial settings\r\n */\r\n editor.settings = {\r\n tools : ['paragraph', 'header', 'picture', 'list', 'quote', 'code', 'twitter', 'instagram', 'smile'],\r\n holderId : 'codex-editor',\r\n\r\n // Type of block showing on empty editor\r\n initialBlockPlugin: 'paragraph'\r\n };\r\n\r\n /**\r\n * public\r\n *\r\n * Static nodes\r\n */\r\n editor.nodes = {\r\n holder : null,\r\n wrapper : null,\r\n toolbar : null,\r\n inlineToolbar : {\r\n wrapper : null,\r\n buttons : null,\r\n actions : null\r\n },\r\n toolbox : null,\r\n notifications : null,\r\n plusButton : null,\r\n showSettingsButton: null,\r\n showTrashButton : null,\r\n blockSettings : null,\r\n pluginSettings : null,\r\n defaultSettings : null,\r\n toolbarButtons : {}, // { type : DomEl, ... }\r\n redactor : null\r\n };\r\n\r\n /**\r\n * @public\r\n *\r\n * Output state\r\n */\r\n editor.state = {\r\n jsonOutput : [],\r\n blocks : [],\r\n inputs : []\r\n };\r\n\r\n /**\r\n * @public\r\n * Editor plugins\r\n */\r\n editor.tools = {};\r\n\r\n /**\r\n * Initialization\r\n * @uses Promise cEditor.core.prepare\r\n * @param {Object} userSettings\r\n * @param {Array} userSettings.tools list of plugins\r\n * @param {String} userSettings.holderId Element's id to append editor\r\n *\r\n * Load user defined tools\r\n * Tools must contain this important objects :\r\n * @param {String} type - this is a type of plugin. It can be used as plugin name\r\n * @param {String} iconClassname - this a icon in toolbar\r\n * @param {Object} make - what should plugin do, when it is clicked\r\n * @param {Object} appendCallback - callback after clicking\r\n * @param {Element} settings - what settings does it have\r\n * @param {Object} render - plugin get JSON, and should return HTML\r\n * @param {Object} save - plugin gets HTML content, returns JSON\r\n * @param {Boolean} displayInToolbox - will be displayed in toolbox. Default value is TRUE\r\n * @param {Boolean} enableLineBreaks - inserts new block or break lines. Default value is FALSE\r\n *\r\n * @example\r\n * - type : 'header',\r\n * - iconClassname : 'ce-icon-header',\r\n * - make : headerTool.make,\r\n * - appendCallback : headerTool.appendCallback,\r\n * - settings : headerTool.makeSettings(),\r\n * - render : headerTool.render,\r\n * - save : headerTool.save,\r\n * - displayInToolbox : true,\r\n * - enableLineBreaks : false\r\n */\r\n editor.start = function (userSettings) {\r\n\r\n init();\r\n\r\n editor.core.prepare(userSettings)\r\n\r\n // If all ok, make UI, bind events and parse initial-content\r\n .then(editor.ui.prepare)\r\n .then(editor.tools.prepare)\r\n .then(editor.sanitizer.prepare)\r\n .then(editor.paste.prepare)\r\n .then(editor.transport.prepare)\r\n .then(editor.renderer.makeBlocksFromData)\r\n .then(editor.ui.saveInputs)\r\n .catch(function (error) {\r\n\r\n editor.core.log('Initialization failed with error: %o', 'warn', error);\r\n\r\n });\r\n\r\n };\r\n\r\n return editor;\r\n\r\n})({});\r\n\n\n\n// WEBPACK FOOTER //\n// ./codex.js","/**\r\n * Codex Editor Core\r\n *\r\n * @author Codex Team\r\n * @version 1.1.3\r\n */\r\n\r\nmodule.exports = (function (core) {\r\n\r\n let editor = codex.editor;\r\n\r\n /**\r\n * @public\r\n *\r\n * Editor preparing method\r\n * @return Promise\r\n */\r\n core.prepare = function (userSettings) {\r\n\r\n return new Promise(function (resolve, reject) {\r\n\r\n if ( userSettings ) {\r\n\r\n editor.settings.tools = userSettings.tools || editor.settings.tools;\r\n\r\n }\r\n\r\n if (userSettings.data) {\r\n\r\n editor.state.blocks = userSettings.data;\r\n\r\n }\r\n\r\n if (userSettings.initialBlockPlugin) {\r\n\r\n editor.settings.initialBlockPlugin = userSettings.initialBlockPlugin;\r\n\r\n }\r\n\r\n if (userSettings.sanitizer) {\r\n\r\n editor.settings.sanitizer = userSettings.sanitizer;\r\n\r\n }\r\n\r\n editor.hideToolbar = userSettings.hideToolbar;\r\n\r\n editor.nodes.holder = document.getElementById(userSettings.holderId || editor.settings.holderId);\r\n\r\n if (typeof editor.nodes.holder === undefined || editor.nodes.holder === null) {\r\n\r\n reject(Error(\"Holder wasn't found by ID: #\" + userSettings.holderId));\r\n\r\n } else {\r\n\r\n resolve();\r\n\r\n }\r\n\r\n });\r\n\r\n };\r\n\r\n /**\r\n * Logging method\r\n * @param type = ['log', 'info', 'warn']\r\n */\r\n core.log = function (msg, type, arg) {\r\n\r\n type = type || 'log';\r\n\r\n if (!arg) {\r\n\r\n arg = msg || 'undefined';\r\n msg = '[codex-editor]: %o';\r\n\r\n } else {\r\n\r\n msg = '[codex-editor]: ' + msg;\r\n\r\n }\r\n\r\n try{\r\n\r\n if ( 'console' in window && window.console[ type ] ) {\r\n\r\n if ( arg ) window.console[ type ]( msg, arg );\r\n else window.console[ type ]( msg );\r\n\r\n }\r\n\r\n }catch(e) {}\r\n\r\n };\r\n\r\n /**\r\n * @protected\r\n *\r\n * Helper for insert one element after another\r\n */\r\n core.insertAfter = function (target, element) {\r\n\r\n target.parentNode.insertBefore(element, target.nextSibling);\r\n\r\n };\r\n\r\n /**\r\n * @const\r\n *\r\n * Readable DOM-node types map\r\n */\r\n core.nodeTypes = {\r\n TAG : 1,\r\n TEXT : 3,\r\n COMMENT : 8\r\n };\r\n\r\n /**\r\n * @const\r\n * Readable keys map\r\n */\r\n core.keys = { BACKSPACE: 8, TAB: 9, ENTER: 13, SHIFT: 16, CTRL: 17, ALT: 18, ESC: 27, SPACE: 32, LEFT: 37, UP: 38, DOWN: 40, RIGHT: 39, DELETE: 46, META: 91 };\r\n\r\n /**\r\n * @protected\r\n *\r\n * Check object for DOM node\r\n */\r\n core.isDomNode = function (el) {\r\n\r\n return el && typeof el === 'object' && el.nodeType && el.nodeType == this.nodeTypes.TAG;\r\n\r\n };\r\n\r\n /**\r\n * Checks passed object for emptiness\r\n * @require ES5 - Object.keys\r\n * @param {object}\r\n */\r\n core.isEmpty = function ( obj ) {\r\n\r\n return Object.keys(obj).length === 0;\r\n\r\n };\r\n\r\n /**\r\n * Native Ajax\r\n * @param {String} settings.url - request URL\r\n * @param {function} settings.beforeSend - returned value will be passed as context to the Success, Error and Progress callbacks\r\n * @param {function} settings.success\r\n * @param {function} settings.progress\r\n */\r\n core.ajax = function (settings) {\r\n\r\n if (!settings || !settings.url) {\r\n\r\n return;\r\n\r\n }\r\n\r\n var XMLHTTP = window.XMLHttpRequest ? new XMLHttpRequest() : new ActiveXObject('Microsoft.XMLHTTP'),\r\n encodedString,\r\n isFormData,\r\n prop;\r\n\r\n\r\n settings.async = true;\r\n settings.type = settings.type || 'GET';\r\n settings.data = settings.data || '';\r\n settings['content-type'] = settings['content-type'] || 'application/json; charset=utf-8';\r\n\r\n if (settings.type == 'GET' && settings.data) {\r\n\r\n settings.url = /\\?/.test(settings.url) ? settings.url + '&' + settings.data : settings.url + '?' + settings.data;\r\n\r\n } else {\r\n\r\n encodedString = '';\r\n for(prop in settings.data) {\r\n\r\n encodedString += (prop + '=' + encodeURIComponent(settings.data[prop]) + '&');\r\n\r\n }\r\n\r\n }\r\n\r\n if (settings.withCredentials) {\r\n\r\n XMLHTTP.withCredentials = true;\r\n\r\n }\r\n\r\n /**\r\n * Value returned in beforeSend funtion will be passed as context to the other response callbacks\r\n * If beforeSend returns false, AJAX will be blocked\r\n */\r\n let responseContext,\r\n beforeSendResult;\r\n\r\n if (typeof settings.beforeSend === 'function') {\r\n\r\n beforeSendResult = settings.beforeSend.call();\r\n\r\n if (beforeSendResult === false) {\r\n\r\n return;\r\n\r\n }\r\n\r\n }\r\n\r\n XMLHTTP.open( settings.type, settings.url, settings.async );\r\n\r\n /**\r\n * If we send FormData, we need no content-type header\r\n */\r\n isFormData = isFormData_(settings.data);\r\n\r\n if (!isFormData) {\r\n\r\n if (settings.type !== 'POST') {\r\n\r\n XMLHTTP.setRequestHeader('Content-type', settings['content-type']);\r\n\r\n } else {\r\n\r\n XMLHTTP.setRequestHeader('Content-type', 'application/x-www-form-urlencoded');\r\n\r\n }\r\n\r\n }\r\n\r\n XMLHTTP.setRequestHeader('X-Requested-With', 'XMLHttpRequest');\r\n\r\n responseContext = beforeSendResult || XMLHTTP;\r\n\r\n if (typeof settings.progress === 'function') {\r\n\r\n XMLHTTP.upload.onprogress = settings.progress.bind(responseContext);\r\n\r\n }\r\n\r\n XMLHTTP.onreadystatechange = function () {\r\n\r\n if (XMLHTTP.readyState === 4) {\r\n\r\n if (XMLHTTP.status === 200) {\r\n\r\n if (typeof settings.success === 'function') {\r\n\r\n settings.success.call(responseContext, XMLHTTP.responseText);\r\n\r\n }\r\n\r\n } else {\r\n\r\n if (typeof settings.error === 'function') {\r\n\r\n settings.error.call(responseContext, XMLHTTP.responseText, XMLHTTP.status);\r\n\r\n }\r\n\r\n }\r\n\r\n }\r\n\r\n };\r\n\r\n if (isFormData) {\r\n\r\n // Sending FormData\r\n XMLHTTP.send(settings.data);\r\n\r\n } else {\r\n\r\n // POST requests\r\n XMLHTTP.send(encodedString);\r\n\r\n }\r\n\r\n return XMLHTTP;\r\n\r\n };\r\n\r\n /**\r\n * Appends script to head of document\r\n * @return Promise\r\n */\r\n core.importScript = function (scriptPath, instanceName) {\r\n\r\n return new Promise(function (resolve, reject) {\r\n\r\n let script;\r\n\r\n /** Script is already loaded */\r\n if ( !instanceName ) {\r\n\r\n reject('Instance name is missed');\r\n\r\n } else if ( document.getElementById(editor.scriptPrefix + instanceName) ) {\r\n\r\n resolve(scriptPath);\r\n\r\n }\r\n\r\n script = document.createElement('SCRIPT');\r\n script.async = true;\r\n script.defer = true;\r\n script.id = editor.scriptPrefix + instanceName;\r\n\r\n script.onload = function () {\r\n\r\n resolve(scriptPath);\r\n\r\n };\r\n\r\n script.onerror = function () {\r\n\r\n reject(scriptPath);\r\n\r\n };\r\n\r\n script.src = scriptPath;\r\n document.head.appendChild(script);\r\n\r\n });\r\n\r\n };\r\n\r\n /**\r\n * Function for checking is it FormData object to send.\r\n * @param {Object} object to check\r\n * @return boolean\r\n */\r\n var isFormData_ = function (object) {\r\n\r\n return object instanceof FormData;\r\n\r\n };\r\n\r\n return core;\r\n\r\n})({});\r\n\n\n\n// WEBPACK FOOTER //\n// ./modules/core.js","/**\r\n* Module working with plugins\r\n*/\r\nmodule.exports = (function () {\r\n\r\n let editor = codex.editor;\r\n\r\n /**\r\n * Initialize plugins before using\r\n * Ex. Load scripts or call some internal methods\r\n * @return Promise\r\n */\r\n function prepare() {\r\n\r\n return new Promise(function (resolve_, reject_) {\r\n\r\n Promise.resolve()\r\n\r\n /**\r\n * Compose a sequence of plugins that requires preparation\r\n */\r\n .then(function () {\r\n\r\n let pluginsRequiresPreparation = [],\r\n allPlugins = editor.tools;\r\n\r\n for ( let pluginName in allPlugins ) {\r\n\r\n let plugin = allPlugins[pluginName];\r\n\r\n if (plugin.prepare && typeof plugin.prepare != 'function' || !plugin.prepare) {\r\n\r\n continue;\r\n\r\n }\r\n\r\n pluginsRequiresPreparation.push(plugin);\r\n\r\n }\r\n\r\n /**\r\n * If no one passed plugins requires preparation, finish prepare() and go ahead\r\n */\r\n if (!pluginsRequiresPreparation.length) {\r\n\r\n resolve_();\r\n\r\n }\r\n\r\n return pluginsRequiresPreparation;\r\n\r\n })\r\n\r\n /** Wait plugins while they prepares */\r\n .then(waitAllPluginsPreparation_)\r\n\r\n .then(function () {\r\n\r\n editor.core.log('Plugins loaded', 'info');\r\n resolve_();\r\n\r\n }).catch(function (error) {\r\n\r\n reject_(error);\r\n\r\n });\r\n\r\n });\r\n\r\n }\r\n\r\n /**\r\n * @param {array} plugins - list of tools that requires preparation\r\n * @return {Promise} resolved while all plugins will be ready or failed\r\n */\r\n function waitAllPluginsPreparation_(plugins) {\r\n\r\n /**\r\n * @calls allPluginsProcessed__ when all plugins prepared or failed\r\n */\r\n return new Promise (function (allPluginsProcessed__) {\r\n\r\n /**\r\n * pluck each element from queue\r\n * First, send resolved Promise as previous value\r\n * Each plugins \"prepare\" method returns a Promise, that's why\r\n * reduce current element will not be able to continue while can't get\r\n * a resolved Promise\r\n *\r\n * If last plugin is \"prepared\" then go to the next stage of initialization\r\n */\r\n plugins.reduce(function (previousValue, plugin, iteration) {\r\n\r\n return previousValue.then(function () {\r\n\r\n /**\r\n * Wait till plugins prepared\r\n * @calls pluginIsReady__ when plugin is ready or failed\r\n */\r\n return new Promise ( function (pluginIsReady__) {\r\n\r\n callPluginsPrepareMethod_( plugin )\r\n\r\n .then( pluginIsReady__ )\r\n .then( function () {\r\n\r\n plugin.available = true;\r\n\r\n })\r\n\r\n .catch(function (error) {\r\n\r\n editor.core.log(`Plugin «${plugin.type}» was not loaded. Preparation failed because %o`, 'warn', error);\r\n plugin.available = false;\r\n plugin.loadingMessage = error;\r\n\r\n /** Go ahead even some plugin has problems */\r\n pluginIsReady__();\r\n\r\n })\r\n\r\n .then(function () {\r\n\r\n /** If last plugin has problems then just ignore and continue */\r\n if (iteration == plugins.length - 1) {\r\n\r\n allPluginsProcessed__();\r\n\r\n }\r\n\r\n });\r\n\r\n });\r\n\r\n });\r\n\r\n }, Promise.resolve() );\r\n\r\n });\r\n\r\n }\r\n\r\n var callPluginsPrepareMethod_ = function (plugin) {\r\n\r\n return plugin.prepare( plugin.config || {} );\r\n\r\n };\r\n\r\n return {\r\n prepare: prepare\r\n };\r\n\r\n}());\n\n\n// WEBPACK FOOTER //\n// ./modules/tools.js","/**\r\n * Codex Editor UI module\r\n *\r\n * @author Codex Team\r\n * @version 1.2.0\r\n */\r\n\r\nmodule.exports = (function (ui) {\r\n\r\n let editor = codex.editor;\r\n\r\n /**\r\n * Basic editor classnames\r\n */\r\n ui.className = {\r\n\r\n /**\r\n * @const {string} BLOCK_CLASSNAME - redactor blocks name\r\n */\r\n BLOCK_CLASSNAME : 'ce-block',\r\n\r\n /**\r\n * @const {String} wrapper for plugins content\r\n */\r\n BLOCK_CONTENT : 'ce-block__content',\r\n\r\n /**\r\n * @const {String} BLOCK_STRETCHED - makes block stretched\r\n */\r\n BLOCK_STRETCHED : 'ce-block--stretched',\r\n\r\n /**\r\n * @const {String} BLOCK_HIGHLIGHTED - adds background\r\n */\r\n BLOCK_HIGHLIGHTED : 'ce-block--focused',\r\n\r\n /**\r\n * @const {String} - for all default settings\r\n */\r\n SETTINGS_ITEM : 'ce-settings__item'\r\n\r\n };\r\n\r\n /**\r\n * @protected\r\n *\r\n * Making main interface\r\n */\r\n ui.prepare = function () {\r\n\r\n return new Promise(function (resolve) {\r\n\r\n let wrapper = editor.draw.wrapper(),\r\n redactor = editor.draw.redactor(),\r\n toolbar = makeToolBar_();\r\n\r\n wrapper.appendChild(toolbar);\r\n wrapper.appendChild(redactor);\r\n\r\n /** Save created ui-elements to static nodes state */\r\n editor.nodes.wrapper = wrapper;\r\n editor.nodes.redactor = redactor;\r\n\r\n /** Append editor wrapper with redactor zone into holder */\r\n editor.nodes.holder.appendChild(wrapper);\r\n\r\n resolve();\r\n\r\n })\r\n\r\n /** Add toolbox tools */\r\n .then(addTools_)\r\n\r\n /** Make container for inline toolbar */\r\n .then(makeInlineToolbar_)\r\n\r\n /** Add inline toolbar tools */\r\n .then(addInlineToolbarTools_)\r\n\r\n /** Draw wrapper for notifications */\r\n .then(makeNotificationHolder_)\r\n\r\n /** Add eventlisteners to redactor elements */\r\n .then(bindEvents_)\r\n\r\n .catch( function () {\r\n\r\n editor.core.log(\"Can't draw editor interface\");\r\n\r\n });\r\n\r\n };\r\n\r\n /**\r\n * @private\r\n * Draws inline toolbar zone\r\n */\r\n var makeInlineToolbar_ = function () {\r\n\r\n var container = editor.draw.inlineToolbar();\r\n\r\n /** Append to redactor new inline block */\r\n editor.nodes.inlineToolbar.wrapper = container;\r\n\r\n /** Draw toolbar buttons */\r\n editor.nodes.inlineToolbar.buttons = editor.draw.inlineToolbarButtons();\r\n\r\n /** Buttons action or settings */\r\n editor.nodes.inlineToolbar.actions = editor.draw.inlineToolbarActions();\r\n\r\n /** Append to inline toolbar buttons as part of it */\r\n editor.nodes.inlineToolbar.wrapper.appendChild(editor.nodes.inlineToolbar.buttons);\r\n editor.nodes.inlineToolbar.wrapper.appendChild(editor.nodes.inlineToolbar.actions);\r\n\r\n editor.nodes.wrapper.appendChild(editor.nodes.inlineToolbar.wrapper);\r\n\r\n };\r\n\r\n var makeToolBar_ = function () {\r\n\r\n let toolbar = editor.draw.toolbar(),\r\n blockButtons = makeToolbarSettings_(),\r\n toolbarContent = makeToolbarContent_();\r\n\r\n /** Appending first-level block buttons */\r\n toolbar.appendChild(blockButtons);\r\n\r\n /** Append toolbarContent to toolbar */\r\n toolbar.appendChild(toolbarContent);\r\n\r\n /** Make toolbar global */\r\n editor.nodes.toolbar = toolbar;\r\n\r\n return toolbar;\r\n\r\n };\r\n\r\n var makeToolbarContent_ = function () {\r\n\r\n let toolbarContent = editor.draw.toolbarContent(),\r\n toolbox = editor.draw.toolbox(),\r\n plusButton = editor.draw.plusButton();\r\n\r\n /** Append plus button */\r\n toolbarContent.appendChild(plusButton);\r\n\r\n /** Appending toolbar tools */\r\n toolbarContent.appendChild(toolbox);\r\n\r\n /** Make Toolbox and plusButton global */\r\n editor.nodes.toolbox = toolbox;\r\n editor.nodes.plusButton = plusButton;\r\n\r\n return toolbarContent;\r\n\r\n };\r\n\r\n var makeToolbarSettings_ = function () {\r\n\r\n let blockSettings = editor.draw.blockSettings(),\r\n blockButtons = editor.draw.blockButtons(),\r\n defaultSettings = editor.draw.defaultSettings(),\r\n showSettingsButton = editor.draw.settingsButton(),\r\n showTrashButton = editor.toolbar.settings.makeRemoveBlockButton(),\r\n pluginSettings = editor.draw.pluginsSettings();\r\n\r\n /** Add default and plugins settings */\r\n blockSettings.appendChild(pluginSettings);\r\n blockSettings.appendChild(defaultSettings);\r\n\r\n /**\r\n * Make blocks buttons\r\n * This block contains settings button and remove block button\r\n */\r\n blockButtons.appendChild(showSettingsButton);\r\n blockButtons.appendChild(showTrashButton);\r\n blockButtons.appendChild(blockSettings);\r\n\r\n /** Make BlockSettings, PluginSettings, DefaultSettings global */\r\n editor.nodes.blockSettings = blockSettings;\r\n editor.nodes.pluginSettings = pluginSettings;\r\n editor.nodes.defaultSettings = defaultSettings;\r\n editor.nodes.showSettingsButton = showSettingsButton;\r\n editor.nodes.showTrashButton = showTrashButton;\r\n\r\n return blockButtons;\r\n\r\n };\r\n\r\n /** Draw notifications holder */\r\n var makeNotificationHolder_ = function () {\r\n\r\n /** Append block with notifications to the document */\r\n editor.nodes.notifications = editor.notifications.createHolder();\r\n\r\n };\r\n\r\n /**\r\n * @private\r\n * Append tools passed in editor.tools\r\n */\r\n var addTools_ = function () {\r\n\r\n var tool,\r\n toolName,\r\n toolButton;\r\n\r\n for ( toolName in editor.settings.tools ) {\r\n\r\n tool = editor.settings.tools[toolName];\r\n\r\n editor.tools[toolName] = tool;\r\n\r\n if (!tool.iconClassname && tool.displayInToolbox) {\r\n\r\n editor.core.log('Toolbar icon classname missed. Tool %o skipped', 'warn', toolName);\r\n continue;\r\n\r\n }\r\n\r\n if (typeof tool.render != 'function') {\r\n\r\n editor.core.log('render method missed. Tool %o skipped', 'warn', toolName);\r\n continue;\r\n\r\n }\r\n\r\n if (!tool.displayInToolbox) {\r\n\r\n continue;\r\n\r\n } else {\r\n\r\n /** if tools is for toolbox */\r\n toolButton = editor.draw.toolbarButton(toolName, tool.iconClassname);\r\n\r\n editor.nodes.toolbox.appendChild(toolButton);\r\n\r\n editor.nodes.toolbarButtons[toolName] = toolButton;\r\n\r\n }\r\n\r\n }\r\n\r\n };\r\n\r\n var addInlineToolbarTools_ = function () {\r\n\r\n var tools = {\r\n\r\n bold: {\r\n icon : 'ce-icon-bold',\r\n command : 'bold'\r\n },\r\n\r\n italic: {\r\n icon : 'ce-icon-italic',\r\n command : 'italic'\r\n },\r\n\r\n underline: {\r\n icon : 'ce-icon-underline',\r\n command : 'underline'\r\n },\r\n\r\n link: {\r\n icon : 'ce-icon-link',\r\n command : 'createLink'\r\n }\r\n };\r\n\r\n var toolButton,\r\n tool;\r\n\r\n for(var name in tools) {\r\n\r\n tool = tools[name];\r\n\r\n toolButton = editor.draw.toolbarButtonInline(name, tool.icon);\r\n\r\n editor.nodes.inlineToolbar.buttons.appendChild(toolButton);\r\n /**\r\n * Add callbacks to this buttons\r\n */\r\n editor.ui.setInlineToolbarButtonBehaviour(toolButton, tool.command);\r\n\r\n }\r\n\r\n };\r\n\r\n /**\r\n * @private\r\n * Bind editor UI events\r\n */\r\n var bindEvents_ = function () {\r\n\r\n editor.core.log('ui.bindEvents fired', 'info');\r\n\r\n // window.addEventListener('error', function (errorMsg, url, lineNumber) {\r\n // editor.notifications.errorThrown(errorMsg, event);\r\n // }, false );\r\n\r\n /** All keydowns on Document */\r\n editor.listeners.add(document, 'keydown', editor.callback.globalKeydown, false);\r\n\r\n /** All keydowns on Redactor zone */\r\n editor.listeners.add(editor.nodes.redactor, 'keydown', editor.callback.redactorKeyDown, false);\r\n\r\n /** All keydowns on Document */\r\n editor.listeners.add(document, 'keyup', editor.callback.globalKeyup, false );\r\n\r\n /**\r\n * Mouse click to radactor\r\n */\r\n editor.listeners.add(editor.nodes.redactor, 'click', editor.callback.redactorClicked, false );\r\n\r\n /**\r\n * Clicks to the Plus button\r\n */\r\n editor.listeners.add(editor.nodes.plusButton, 'click', editor.callback.plusButtonClicked, false);\r\n\r\n /**\r\n * Clicks to SETTINGS button in toolbar\r\n */\r\n editor.listeners.add(editor.nodes.showSettingsButton, 'click', editor.callback.showSettingsButtonClicked, false );\r\n\r\n /** Bind click listeners on toolbar buttons */\r\n for (var button in editor.nodes.toolbarButtons) {\r\n\r\n editor.listeners.add(editor.nodes.toolbarButtons[button], 'click', editor.callback.toolbarButtonClicked, false);\r\n\r\n }\r\n\r\n };\r\n\r\n ui.addBlockHandlers = function (block) {\r\n\r\n if (!block) return;\r\n\r\n /**\r\n * Block keydowns\r\n */\r\n editor.listeners.add(block, 'keydown', editor.callback.blockKeydown, false);\r\n\r\n /**\r\n * Pasting content from another source\r\n * We have two type of sanitization\r\n * First - uses deep-first search algorithm to get sub nodes,\r\n * sanitizes whole Block_content and replaces cleared nodes\r\n * This method is deprecated\r\n * Method is used in editor.callback.blockPaste(event)\r\n *\r\n * Secont - uses Mutation observer.\r\n * Observer \"observe\" DOM changes and send changings to callback.\r\n * Callback gets changed node, not whole Block_content.\r\n * Inserted or changed node, which we've gotten have been cleared and replaced with diry node\r\n *\r\n * Method is used in editor.callback.blockPasteViaSanitize(event)\r\n *\r\n * @uses html-janitor\r\n * @example editor.callback.blockPasteViaSanitize(event), the second method.\r\n *\r\n */\r\n editor.listeners.add(block, 'paste', editor.callback.blockPasteCallback, false);\r\n\r\n editor.listeners.add(block, 'mouseup', editor.toolbar.inline.show, false);\r\n\r\n };\r\n\r\n /** getting all contenteditable elements */\r\n ui.saveInputs = function () {\r\n\r\n var redactor = editor.nodes.redactor;\r\n\r\n editor.state.inputs = [];\r\n\r\n /** Save all inputs in global variable state */\r\n var inputs = redactor.querySelectorAll('[contenteditable], input, textarea');\r\n\r\n Array.prototype.map.call(inputs, function (current) {\r\n\r\n if (!current.type || current.type == 'text' || current.type == 'textarea') {\r\n\r\n editor.state.inputs.push(current);\r\n\r\n }\r\n\r\n });\r\n\r\n };\r\n\r\n /**\r\n * Adds first initial block on empty redactor\r\n */\r\n ui.addInitialBlock = function () {\r\n\r\n var initialBlockType = editor.settings.initialBlockPlugin,\r\n initialBlock;\r\n\r\n if ( !editor.tools[initialBlockType] ) {\r\n\r\n editor.core.log('Plugin %o was not implemented and can\\'t be used as initial block', 'warn', initialBlockType);\r\n return;\r\n\r\n }\r\n\r\n initialBlock = editor.tools[initialBlockType].render();\r\n\r\n initialBlock.setAttribute('data-placeholder', 'Расскажите свою историю...');\r\n\r\n editor.content.insertBlock({\r\n type : initialBlockType,\r\n block : initialBlock\r\n });\r\n\r\n editor.content.workingNodeChanged(initialBlock);\r\n\r\n };\r\n\r\n ui.setInlineToolbarButtonBehaviour = function (button, type) {\r\n\r\n editor.listeners.add(button, 'mousedown', function (event) {\r\n\r\n editor.toolbar.inline.toolClicked(event, type);\r\n\r\n }, false);\r\n\r\n };\r\n\r\n return ui;\r\n\r\n})({});\r\n\n\n\n// WEBPACK FOOTER //\n// ./modules/ui.js","/**\r\n *\r\n * Codex.Editor Transport Module\r\n *\r\n * @copyright 2017 Codex-Team\r\n * @version 1.2.0\r\n */\r\n\r\nmodule.exports = (function (transport) {\r\n\r\n let editor = codex.editor;\r\n\r\n\r\n /**\r\n * @private {Object} current XmlHttpRequest instance\r\n */\r\n var currentRequest = null;\r\n\r\n\r\n /**\r\n * @type {null} | {DOMElement} input - keeps input element in memory\r\n */\r\n transport.input = null;\r\n\r\n /**\r\n * @property {Object} arguments - keep plugin settings and defined callbacks\r\n */\r\n transport.arguments = null;\r\n\r\n /**\r\n * Prepares input element where will be files\r\n */\r\n transport.prepare = function () {\r\n\r\n let input = editor.draw.node( 'INPUT', '', { type : 'file' } );\r\n\r\n editor.listeners.add(input, 'change', editor.transport.fileSelected);\r\n editor.transport.input = input;\r\n\r\n };\r\n\r\n /** Clear input when files is uploaded */\r\n transport.clearInput = function () {\r\n\r\n /** Remove old input */\r\n transport.input = null;\r\n\r\n /** Prepare new one */\r\n transport.prepare();\r\n\r\n };\r\n\r\n /**\r\n * Callback for file selection\r\n * @param {Event} event\r\n */\r\n transport.fileSelected = function () {\r\n\r\n var input = this,\r\n i,\r\n files = input.files,\r\n formData = new FormData();\r\n\r\n if (editor.transport.arguments.multiple === true) {\r\n\r\n for ( i = 0; i < files.length; i++) {\r\n\r\n formData.append('files[]', files[i], files[i].name);\r\n\r\n }\r\n\r\n } else {\r\n\r\n formData.append('files', files[0], files[0].name);\r\n\r\n }\r\n\r\n currentRequest = editor.core.ajax({\r\n type : 'POST',\r\n data : formData,\r\n url : editor.transport.arguments.url,\r\n beforeSend : editor.transport.arguments.beforeSend,\r\n success : editor.transport.arguments.success,\r\n error : editor.transport.arguments.error,\r\n progress : editor.transport.arguments.progress\r\n });\r\n\r\n /** Clear input */\r\n transport.clearInput();\r\n\r\n };\r\n\r\n /**\r\n * Use plugin callbacks\r\n * @protected\r\n *\r\n * @param {Object} args - can have :\r\n * @param {String} args.url - fetch URL\r\n * @param {Function} args.beforeSend - function calls before sending ajax\r\n * @param {Function} args.success - success callback\r\n * @param {Function} args.error - on error handler\r\n * @param {Function} args.progress - xhr onprogress handler\r\n * @param {Boolean} args.multiple - allow select several files\r\n * @param {String} args.accept - adds accept attribute\r\n */\r\n transport.selectAndUpload = function (args) {\r\n\r\n transport.arguments = args;\r\n\r\n if ( args.multiple === true) {\r\n\r\n transport.input.setAttribute('multiple', 'multiple');\r\n\r\n }\r\n\r\n if ( args.accept ) {\r\n\r\n transport.input.setAttribute('accept', args.accept);\r\n\r\n }\r\n\r\n transport.input.click();\r\n\r\n };\r\n\r\n transport.abort = function () {\r\n\r\n currentRequest.abort();\r\n\r\n currentRequest = null;\r\n\r\n };\r\n\r\n return transport;\r\n\r\n})({});\n\n\n// WEBPACK FOOTER //\n// ./modules/transport.js","/**\r\n * Codex Editor Renderer Module\r\n *\r\n * @author Codex Team\r\n * @version 1.0\r\n */\r\n\r\nmodule.exports = (function (renderer) {\r\n\r\n let editor = codex.editor;\r\n\r\n /**\r\n * Asyncronously parses input JSON to redactor blocks\r\n */\r\n renderer.makeBlocksFromData = function () {\r\n\r\n /**\r\n * If redactor is empty, add first paragraph to start writing\r\n */\r\n if (editor.core.isEmpty(editor.state.blocks) || !editor.state.blocks.items.length) {\r\n\r\n editor.ui.addInitialBlock();\r\n return;\r\n\r\n }\r\n\r\n Promise.resolve()\r\n\r\n /** First, get JSON from state */\r\n .then(function () {\r\n\r\n return editor.state.blocks;\r\n\r\n })\r\n\r\n /** Then, start to iterate they */\r\n .then(editor.renderer.appendBlocks)\r\n\r\n /** Write log if something goes wrong */\r\n .catch(function (error) {\r\n\r\n editor.core.log('Error while parsing JSON: %o', 'error', error);\r\n\r\n });\r\n\r\n };\r\n\r\n /**\r\n * Parses JSON to blocks\r\n * @param {object} data\r\n * @return Primise -> nodeList\r\n */\r\n renderer.appendBlocks = function (data) {\r\n\r\n var blocks = data.items;\r\n\r\n /**\r\n * Sequence of one-by-one blocks appending\r\n * Uses to save blocks order after async-handler\r\n */\r\n var nodeSequence = Promise.resolve();\r\n\r\n for (var index = 0; index < blocks.length ; index++ ) {\r\n\r\n /** Add node to sequence at specified index */\r\n editor.renderer.appendNodeAtIndex(nodeSequence, blocks, index);\r\n\r\n }\r\n\r\n };\r\n\r\n /**\r\n * Append node at specified index\r\n */\r\n renderer.appendNodeAtIndex = function (nodeSequence, blocks, index) {\r\n\r\n /** We need to append node to sequence */\r\n nodeSequence\r\n\r\n /** first, get node async-aware */\r\n .then(function () {\r\n\r\n return editor.renderer.getNodeAsync(blocks, index);\r\n\r\n })\r\n\r\n /**\r\n * second, compose editor-block from JSON object\r\n */\r\n .then(editor.renderer.createBlockFromData)\r\n\r\n /**\r\n * now insert block to redactor\r\n */\r\n .then(function (blockData) {\r\n\r\n /**\r\n * blockData has 'block', 'type' and 'stretched' information\r\n */\r\n editor.content.insertBlock(blockData);\r\n\r\n /** Pass created block to next step */\r\n return blockData.block;\r\n\r\n })\r\n\r\n /** Log if something wrong with node */\r\n .catch(function (error) {\r\n\r\n editor.core.log('Node skipped while parsing because %o', 'error', error);\r\n\r\n });\r\n\r\n };\r\n\r\n /**\r\n * Asynchronously returns block data from blocksList by index\r\n * @return Promise to node\r\n */\r\n renderer.getNodeAsync = function (blocksList, index) {\r\n\r\n return Promise.resolve().then(function () {\r\n\r\n return {\r\n tool : blocksList[index],\r\n position : index\r\n };\r\n\r\n });\r\n\r\n };\r\n\r\n /**\r\n * Creates editor block by JSON-data\r\n *\r\n * @uses render method of each plugin\r\n *\r\n * @param {Object} toolData.tool\r\n * { header : {\r\n * text: '',\r\n * type: 'H3', ...\r\n * }\r\n * }\r\n * @param {Number} toolData.position - index in input-blocks array\r\n * @return {Object} with type and Element\r\n */\r\n renderer.createBlockFromData = function ( toolData ) {\r\n\r\n /** New parser */\r\n var block,\r\n tool = toolData.tool,\r\n pluginName = tool.type;\r\n\r\n /** Get first key of object that stores plugin name */\r\n // for (var pluginName in blockData) break;\r\n\r\n /** Check for plugin existance */\r\n if (!editor.tools[pluginName]) {\r\n\r\n throw Error(`Plugin «${pluginName}» not found`);\r\n\r\n }\r\n\r\n /** Check for plugin having render method */\r\n if (typeof editor.tools[pluginName].render != 'function') {\r\n\r\n throw Error(`Plugin «${pluginName}» must have «render» method`);\r\n\r\n }\r\n\r\n if ( editor.tools[pluginName].available === false ) {\r\n\r\n block = editor.draw.unavailableBlock();\r\n\r\n block.innerHTML = editor.tools[pluginName].loadingMessage;\r\n\r\n /**\r\n * Saver will extract data from initial block data by position in array\r\n */\r\n block.dataset.inputPosition = toolData.position;\r\n\r\n } else {\r\n\r\n /** New Parser */\r\n block = editor.tools[pluginName].render(tool.data);\r\n\r\n }\r\n\r\n /** is first-level block stretched */\r\n var stretched = editor.tools[pluginName].isStretched || false;\r\n\r\n /** Retrun type and block */\r\n return {\r\n type : pluginName,\r\n block : block,\r\n stretched : stretched\r\n };\r\n\r\n };\r\n\r\n return renderer;\r\n\r\n})({});\n\n\n// WEBPACK FOOTER //\n// ./modules/renderer.js","/**\r\n * Codex Editor Saver\r\n *\r\n * @author Codex Team\r\n * @version 1.0.2\r\n */\r\n\r\nmodule.exports = (function (saver) {\r\n\r\n let editor = codex.editor;\r\n\r\n /**\r\n * Saves blocks\r\n * @private\r\n */\r\n saver.saveBlocks = function () {\r\n\r\n /** Save html content of redactor to memory */\r\n editor.state.html = editor.nodes.redactor.innerHTML;\r\n\r\n /** Empty jsonOutput state */\r\n editor.state.jsonOutput = [];\r\n\r\n Promise.resolve()\r\n\r\n .then(function () {\r\n\r\n return editor.nodes.redactor.childNodes;\r\n\r\n })\r\n /** Making a sequence from separate blocks */\r\n .then(editor.saver.makeQueue)\r\n\r\n .then(function () {\r\n // editor.nodes.textarea.innerHTML = editor.state.html;\r\n })\r\n\r\n .catch( function (error) {\r\n\r\n editor.core.log(error);\r\n\r\n });\r\n\r\n };\r\n\r\n saver.makeQueue = function (blocks) {\r\n\r\n var queue = Promise.resolve();\r\n\r\n for(var index = 0; index < blocks.length; index++) {\r\n\r\n /** Add node to sequence at specified index */\r\n editor.saver.getBlockData(queue, blocks, index);\r\n\r\n }\r\n\r\n };\r\n\r\n /** Gets every block and makes From Data */\r\n saver.getBlockData = function (queue, blocks, index) {\r\n\r\n queue.then(function () {\r\n\r\n return editor.saver.getNodeAsync(blocks, index);\r\n\r\n })\r\n\r\n .then(editor.saver.makeFormDataFromBlocks);\r\n\r\n };\r\n\r\n\r\n /**\r\n * Asynchronously returns block data from blocksList by index\r\n * @return Promise to node\r\n */\r\n saver.getNodeAsync = function (blocksList, index) {\r\n\r\n return Promise.resolve().then(function () {\r\n\r\n return blocksList[index];\r\n\r\n });\r\n\r\n };\r\n\r\n saver.makeFormDataFromBlocks = function (block) {\r\n\r\n var pluginName = block.dataset.tool;\r\n\r\n /** Check for plugin existance */\r\n if (!editor.tools[pluginName]) {\r\n\r\n throw Error(`Plugin «${pluginName}» not found`);\r\n\r\n }\r\n\r\n /** Check for plugin having render method */\r\n if (typeof editor.tools[pluginName].save != 'function') {\r\n\r\n throw Error(`Plugin «${pluginName}» must have save method`);\r\n\r\n }\r\n\r\n /** Result saver */\r\n var blockContent = block.childNodes[0],\r\n pluginsContent = blockContent.childNodes[0],\r\n savedData,\r\n position,\r\n output;\r\n\r\n /** If plugin wasn't available then return data from cache */\r\n if ( editor.tools[pluginName].available === false ) {\r\n\r\n position = pluginsContent.dataset.inputPosition;\r\n savedData = codex.editor.state.blocks.items[position].data;\r\n\r\n } else {\r\n\r\n savedData = editor.tools[pluginName].save(pluginsContent);\r\n\r\n if (editor.tools[pluginName].validate) {\r\n\r\n var result = editor.tools[pluginName].validate(savedData);\r\n\r\n /**\r\n * Do not allow invalid data\r\n */\r\n if (!result)\r\n return;\r\n\r\n }\r\n\r\n }\r\n\r\n output = {\r\n type : pluginName,\r\n data : savedData\r\n };\r\n\r\n editor.state.jsonOutput.push(output);\r\n\r\n };\r\n\r\n return saver;\r\n\r\n})({});\n\n\n// WEBPACK FOOTER //\n// ./modules/saver.js","/**\r\n * Codex Editor Content Module\r\n * Works with DOM\r\n *\r\n * @module Codex Editor content module\r\n *\r\n * @author Codex Team\r\n * @version 1.3.13\r\n *\r\n * @description Module works with Elements that have been appended to the main DOM\r\n */\r\n\r\nmodule.exports = (function (content) {\r\n\r\n let editor = codex.editor;\r\n\r\n /**\r\n * Links to current active block\r\n * @type {null | Element}\r\n */\r\n content.currentNode = null;\r\n\r\n /**\r\n * clicked in redactor area\r\n * @type {null | Boolean}\r\n */\r\n content.editorAreaHightlighted = null;\r\n\r\n /**\r\n * @deprecated\r\n * Synchronizes redactor with original textarea\r\n */\r\n content.sync = function () {\r\n\r\n editor.core.log('syncing...');\r\n\r\n /**\r\n * Save redactor content to editor.state\r\n */\r\n editor.state.html = editor.nodes.redactor.innerHTML;\r\n\r\n };\r\n\r\n /**\r\n * Appends background to the block\r\n *\r\n * @description add CSS class to highlight visually first-level block area\r\n */\r\n content.markBlock = function () {\r\n\r\n editor.content.currentNode.classList.add(editor.ui.className.BLOCK_HIGHLIGHTED);\r\n\r\n };\r\n\r\n /**\r\n * Clear background\r\n *\r\n * @description clears styles that highlights block\r\n */\r\n content.clearMark = function () {\r\n\r\n if (editor.content.currentNode) {\r\n\r\n editor.content.currentNode.classList.remove(editor.ui.className.BLOCK_HIGHLIGHTED);\r\n\r\n }\r\n\r\n };\r\n\r\n /**\r\n * Finds first-level block\r\n *\r\n * @param {Element} node - selected or clicked in redactors area node\r\n * @protected\r\n *\r\n * @description looks for first-level block.\r\n * gets parent while node is not first-level\r\n */\r\n content.getFirstLevelBlock = function (node) {\r\n\r\n if (!editor.core.isDomNode(node)) {\r\n\r\n node = node.parentNode;\r\n\r\n }\r\n\r\n if (node === editor.nodes.redactor || node === document.body) {\r\n\r\n return null;\r\n\r\n } else {\r\n\r\n while(!node.classList.contains(editor.ui.className.BLOCK_CLASSNAME)) {\r\n\r\n node = node.parentNode;\r\n\r\n }\r\n\r\n return node;\r\n\r\n }\r\n\r\n };\r\n\r\n /**\r\n * Trigger this event when working node changed\r\n * @param {Element} targetNode - first-level of this node will be current\r\n * @protected\r\n *\r\n * @description If targetNode is first-level then we set it as current else we look for parents to find first-level\r\n */\r\n content.workingNodeChanged = function (targetNode) {\r\n\r\n /** Clear background from previous marked block before we change */\r\n editor.content.clearMark();\r\n\r\n if (!targetNode) {\r\n\r\n return;\r\n\r\n }\r\n\r\n content.currentNode = content.getFirstLevelBlock(targetNode);\r\n\r\n };\r\n\r\n /**\r\n * Replaces one redactor block with another\r\n * @protected\r\n * @param {Element} targetBlock - block to replace. Mostly currentNode.\r\n * @param {Element} newBlock\r\n * @param {string} newBlockType - type of new block; we need to store it to data-attribute\r\n *\r\n * [!] Function does not saves old block content.\r\n * You can get it manually and pass with newBlock.innerHTML\r\n */\r\n content.replaceBlock = function (targetBlock, newBlock) {\r\n\r\n if (!targetBlock || !newBlock) {\r\n\r\n editor.core.log('replaceBlock: missed params');\r\n return;\r\n\r\n }\r\n\r\n /** If target-block is not a frist-level block, then we iterate parents to find it */\r\n while(!targetBlock.classList.contains(editor.ui.className.BLOCK_CLASSNAME)) {\r\n\r\n targetBlock = targetBlock.parentNode;\r\n\r\n }\r\n\r\n /** Replacing */\r\n editor.nodes.redactor.replaceChild(newBlock, targetBlock);\r\n\r\n /**\r\n * Set new node as current\r\n */\r\n editor.content.workingNodeChanged(newBlock);\r\n\r\n /**\r\n * Add block handlers\r\n */\r\n editor.ui.addBlockHandlers(newBlock);\r\n\r\n /**\r\n * Save changes\r\n */\r\n editor.ui.saveInputs();\r\n\r\n };\r\n\r\n /**\r\n * @protected\r\n *\r\n * Inserts new block to redactor\r\n * Wrapps block into a DIV with BLOCK_CLASSNAME class\r\n *\r\n * @param blockData {object}\r\n * @param blockData.block {Element} element with block content\r\n * @param blockData.type {string} block plugin\r\n * @param needPlaceCaret {bool} pass true to set caret in new block\r\n *\r\n */\r\n content.insertBlock = function ( blockData, needPlaceCaret ) {\r\n\r\n var workingBlock = editor.content.currentNode,\r\n newBlockContent = blockData.block,\r\n blockType = blockData.type,\r\n isStretched = blockData.stretched;\r\n\r\n var newBlock = composeNewBlock_(newBlockContent, blockType, isStretched);\r\n\r\n if (workingBlock) {\r\n\r\n editor.core.insertAfter(workingBlock, newBlock);\r\n\r\n } else {\r\n\r\n /**\r\n * If redactor is empty, append as first child\r\n */\r\n editor.nodes.redactor.appendChild(newBlock);\r\n\r\n }\r\n\r\n /**\r\n * Block handler\r\n */\r\n editor.ui.addBlockHandlers(newBlock);\r\n\r\n /**\r\n * Set new node as current\r\n */\r\n editor.content.workingNodeChanged(newBlock);\r\n\r\n /**\r\n * Save changes\r\n */\r\n editor.ui.saveInputs();\r\n\r\n\r\n if ( needPlaceCaret ) {\r\n\r\n /**\r\n * If we don't know input index then we set default value -1\r\n */\r\n var currentInputIndex = editor.caret.getCurrentInputIndex() || -1;\r\n\r\n\r\n if (currentInputIndex == -1) {\r\n\r\n\r\n var editableElement = newBlock.querySelector('[contenteditable]'),\r\n emptyText = document.createTextNode('');\r\n\r\n editableElement.appendChild(emptyText);\r\n editor.caret.set(editableElement, 0, 0);\r\n\r\n editor.toolbar.move();\r\n editor.toolbar.showPlusButton();\r\n\r\n\r\n } else {\r\n\r\n if (currentInputIndex === editor.state.inputs.length - 1)\r\n return;\r\n\r\n /** Timeout for browsers execution */\r\n window.setTimeout(function () {\r\n\r\n /** Setting to the new input */\r\n editor.caret.setToNextBlock(currentInputIndex);\r\n editor.toolbar.move();\r\n editor.toolbar.open();\r\n\r\n }, 10);\r\n\r\n }\r\n\r\n }\r\n\r\n /**\r\n * Block is inserted, wait for new click that defined focusing on editors area\r\n * @type {boolean}\r\n */\r\n content.editorAreaHightlighted = false;\r\n\r\n };\r\n\r\n /**\r\n * Replaces blocks with saving content\r\n * @protected\r\n * @param {Element} noteToReplace\r\n * @param {Element} newNode\r\n * @param {Element} blockType\r\n */\r\n content.switchBlock = function (blockToReplace, newBlock, tool) {\r\n\r\n tool = tool || editor.content.currentNode.dataset.tool;\r\n var newBlockComposed = composeNewBlock_(newBlock, tool);\r\n\r\n /** Replacing */\r\n editor.content.replaceBlock(blockToReplace, newBlockComposed);\r\n\r\n /** Save new Inputs when block is changed */\r\n editor.ui.saveInputs();\r\n\r\n };\r\n\r\n /**\r\n * Iterates between child noted and looking for #text node on deepest level\r\n * @protected\r\n *\r\n * @param {Element} block - node where find\r\n * @param {int} postiton - starting postion\r\n * Example: childNodex.length to find from the end\r\n * or 0 to find from the start\r\n * @return {Text} block\r\n * @uses DFS\r\n */\r\n content.getDeepestTextNodeFromPosition = function (block, position) {\r\n\r\n /**\r\n * Clear Block from empty and useless spaces with trim.\r\n * Such nodes we should remove\r\n */\r\n var blockChilds = block.childNodes,\r\n index,\r\n node,\r\n text;\r\n\r\n for(index = 0; index < blockChilds.length; index++) {\r\n\r\n node = blockChilds[index];\r\n\r\n if (node.nodeType == editor.core.nodeTypes.TEXT) {\r\n\r\n text = node.textContent.trim();\r\n\r\n /** Text is empty. We should remove this child from node before we start DFS\r\n * decrease the quantity of childs.\r\n */\r\n if (text === '') {\r\n\r\n block.removeChild(node);\r\n position--;\r\n\r\n }\r\n\r\n }\r\n\r\n }\r\n\r\n if (block.childNodes.length === 0) {\r\n\r\n return document.createTextNode('');\r\n\r\n }\r\n\r\n /** Setting default position when we deleted all empty nodes */\r\n if ( position < 0 )\r\n position = 1;\r\n\r\n var lookingFromStart = false;\r\n\r\n /** For looking from START */\r\n if (position === 0) {\r\n\r\n lookingFromStart = true;\r\n position = 1;\r\n\r\n }\r\n\r\n while ( position ) {\r\n\r\n /** initial verticle of node. */\r\n if ( lookingFromStart ) {\r\n\r\n block = block.childNodes[0];\r\n\r\n } else {\r\n\r\n block = block.childNodes[position - 1];\r\n\r\n }\r\n\r\n if ( block.nodeType == editor.core.nodeTypes.TAG ) {\r\n\r\n position = block.childNodes.length;\r\n\r\n } else if (block.nodeType == editor.core.nodeTypes.TEXT ) {\r\n\r\n position = 0;\r\n\r\n }\r\n\r\n }\r\n\r\n return block;\r\n\r\n };\r\n\r\n /**\r\n * @private\r\n * @param {Element} block - current plugins render\r\n * @param {String} tool - plugins name\r\n * @param {Boolean} isStretched - make stretched block or not\r\n *\r\n * @description adds necessary information to wrap new created block by first-level holder\r\n */\r\n var composeNewBlock_ = function (block, tool, isStretched) {\r\n\r\n var newBlock = editor.draw.node('DIV', editor.ui.className.BLOCK_CLASSNAME, {}),\r\n blockContent = editor.draw.node('DIV', editor.ui.className.BLOCK_CONTENT, {});\r\n\r\n blockContent.appendChild(block);\r\n newBlock.appendChild(blockContent);\r\n\r\n if (isStretched) {\r\n\r\n blockContent.classList.add(editor.ui.className.BLOCK_STRETCHED);\r\n\r\n }\r\n\r\n newBlock.dataset.tool = tool;\r\n return newBlock;\r\n\r\n };\r\n\r\n /**\r\n * Returns Range object of current selection\r\n * @protected\r\n */\r\n content.getRange = function () {\r\n\r\n var selection = window.getSelection().getRangeAt(0);\r\n\r\n return selection;\r\n\r\n };\r\n\r\n /**\r\n * Divides block in two blocks (after and before caret)\r\n *\r\n * @protected\r\n * @param {int} inputIndex - target input index\r\n *\r\n * @description splits current input content to the separate blocks\r\n * When enter is pressed among the words, that text will be splited.\r\n */\r\n content.splitBlock = function (inputIndex) {\r\n\r\n var selection = window.getSelection(),\r\n anchorNode = selection.anchorNode,\r\n anchorNodeText = anchorNode.textContent,\r\n caretOffset = selection.anchorOffset,\r\n textBeforeCaret,\r\n textNodeBeforeCaret,\r\n textAfterCaret,\r\n textNodeAfterCaret;\r\n\r\n var currentBlock = editor.content.currentNode.querySelector('[contentEditable]');\r\n\r\n\r\n textBeforeCaret = anchorNodeText.substring(0, caretOffset);\r\n textAfterCaret = anchorNodeText.substring(caretOffset);\r\n\r\n textNodeBeforeCaret = document.createTextNode(textBeforeCaret);\r\n\r\n if (textAfterCaret) {\r\n\r\n textNodeAfterCaret = document.createTextNode(textAfterCaret);\r\n\r\n }\r\n\r\n var previousChilds = [],\r\n nextChilds = [],\r\n reachedCurrent = false;\r\n\r\n if (textNodeAfterCaret) {\r\n\r\n nextChilds.push(textNodeAfterCaret);\r\n\r\n }\r\n\r\n for ( var i = 0, child; !!(child = currentBlock.childNodes[i]); i++) {\r\n\r\n if ( child != anchorNode ) {\r\n\r\n if ( !reachedCurrent ) {\r\n\r\n previousChilds.push(child);\r\n\r\n } else {\r\n\r\n nextChilds.push(child);\r\n\r\n }\r\n\r\n } else {\r\n\r\n reachedCurrent = true;\r\n\r\n }\r\n\r\n }\r\n\r\n /** Clear current input */\r\n editor.state.inputs[inputIndex].innerHTML = '';\r\n\r\n /**\r\n * Append all childs founded before anchorNode\r\n */\r\n var previousChildsLength = previousChilds.length;\r\n\r\n for(i = 0; i < previousChildsLength; i++) {\r\n\r\n editor.state.inputs[inputIndex].appendChild(previousChilds[i]);\r\n\r\n }\r\n\r\n editor.state.inputs[inputIndex].appendChild(textNodeBeforeCaret);\r\n\r\n /**\r\n * Append text node which is after caret\r\n */\r\n var nextChildsLength = nextChilds.length,\r\n newNode = document.createElement('div');\r\n\r\n for(i = 0; i < nextChildsLength; i++) {\r\n\r\n newNode.appendChild(nextChilds[i]);\r\n\r\n }\r\n\r\n newNode = newNode.innerHTML;\r\n\r\n /** This type of block creates when enter is pressed */\r\n var NEW_BLOCK_TYPE = editor.settings.initialBlockPlugin;\r\n\r\n /**\r\n * Make new paragraph with text after caret\r\n */\r\n editor.content.insertBlock({\r\n type : NEW_BLOCK_TYPE,\r\n block : editor.tools[NEW_BLOCK_TYPE].render({\r\n text : newNode\r\n })\r\n }, true );\r\n\r\n };\r\n\r\n /**\r\n * Merges two blocks — current and target\r\n * If target index is not exist, then previous will be as target\r\n *\r\n * @protected\r\n * @param {int} currentInputIndex\r\n * @param {int} targetInputIndex\r\n *\r\n * @description gets two inputs indexes and merges into one\r\n */\r\n content.mergeBlocks = function (currentInputIndex, targetInputIndex) {\r\n\r\n /** If current input index is zero, then prevent method execution */\r\n if (currentInputIndex === 0) {\r\n\r\n return;\r\n\r\n }\r\n\r\n var targetInput,\r\n currentInputContent = editor.state.inputs[currentInputIndex].innerHTML;\r\n\r\n if (!targetInputIndex) {\r\n\r\n targetInput = editor.state.inputs[currentInputIndex - 1];\r\n\r\n } else {\r\n\r\n targetInput = editor.state.inputs[targetInputIndex];\r\n\r\n }\r\n\r\n targetInput.innerHTML += currentInputContent;\r\n\r\n };\r\n\r\n /**\r\n * Iterates all right siblings and parents, which has right siblings\r\n * while it does not reached the first-level block\r\n *\r\n * @param {Element} node\r\n * @return {boolean}\r\n */\r\n content.isLastNode = function (node) {\r\n\r\n // console.log('погнали перебор родителей');\r\n\r\n var allChecked = false;\r\n\r\n while ( !allChecked ) {\r\n\r\n // console.log('Смотрим на %o', node);\r\n // console.log('Проверим, пустые ли соседи справа');\r\n\r\n if ( !allSiblingsEmpty_(node) ) {\r\n\r\n // console.log('Есть непустые соседи. Узел не последний. Выходим.');\r\n return false;\r\n\r\n }\r\n\r\n node = node.parentNode;\r\n\r\n /**\r\n * Проверяем родителей до тех пор, пока не найдем блок первого уровня\r\n */\r\n if ( node.classList.contains(editor.ui.className.BLOCK_CONTENT) ) {\r\n\r\n allChecked = true;\r\n\r\n }\r\n\r\n }\r\n\r\n return true;\r\n\r\n };\r\n\r\n /**\r\n * Checks if all element right siblings is empty\r\n * @param node\r\n */\r\n var allSiblingsEmpty_ = function (node) {\r\n\r\n /**\r\n * Нужно убедиться, что после пустого соседа ничего нет\r\n */\r\n var sibling = node.nextSibling;\r\n\r\n while ( sibling ) {\r\n\r\n if (sibling.textContent.length) {\r\n\r\n return false;\r\n\r\n }\r\n\r\n sibling = sibling.nextSibling;\r\n\r\n }\r\n\r\n return true;\r\n\r\n };\r\n\r\n /**\r\n * @public\r\n *\r\n * @param {string} htmlData - html content as string\r\n * @param {string} plainData - plain text\r\n * @return {string} - html content as string\r\n */\r\n content.wrapTextWithParagraphs = function (htmlData, plainData) {\r\n\r\n if (!htmlData) {\r\n\r\n return '

    ' + plainData.split('\\n\\n').join('

    ') + '

    ';\r\n\r\n }\r\n\r\n var wrapper = document.createElement('DIV'),\r\n newWrapper = document.createElement('DIV'),\r\n i,\r\n paragraph,\r\n firstLevelBlocks = ['DIV', 'P'],\r\n blockTyped,\r\n node;\r\n\r\n /**\r\n * Make HTML Element to Wrap Text\r\n * It allows us to work with input data as HTML content\r\n */\r\n wrapper.innerHTML = htmlData;\r\n paragraph = document.createElement('P');\r\n\r\n for (i = 0; i < wrapper.childNodes.length; i++) {\r\n\r\n node = wrapper.childNodes[i];\r\n\r\n blockTyped = firstLevelBlocks.indexOf(node.tagName) != -1;\r\n\r\n /**\r\n * If node is first-levet\r\n * we add this node to our new wrapper\r\n */\r\n if ( blockTyped ) {\r\n\r\n /**\r\n * If we had splitted inline nodes to paragraph before\r\n */\r\n if ( paragraph.childNodes.length ) {\r\n\r\n newWrapper.appendChild(paragraph.cloneNode(true));\r\n\r\n /** empty paragraph */\r\n paragraph = null;\r\n paragraph = document.createElement('P');\r\n\r\n }\r\n\r\n newWrapper.appendChild(node.cloneNode(true));\r\n\r\n } else {\r\n\r\n /** Collect all inline nodes to one as paragraph */\r\n paragraph.appendChild(node.cloneNode(true));\r\n\r\n /** if node is last we should append this node to paragraph and paragraph to new wrapper */\r\n if ( i == wrapper.childNodes.length - 1 ) {\r\n\r\n newWrapper.appendChild(paragraph.cloneNode(true));\r\n\r\n }\r\n\r\n }\r\n\r\n }\r\n\r\n return newWrapper.innerHTML;\r\n\r\n };\r\n\r\n /**\r\n * Finds closest Contenteditable parent from Element\r\n * @param {Element} node element looking from\r\n * @return {Element} node contenteditable\r\n */\r\n content.getEditableParent = function (node) {\r\n\r\n while (node && node.contentEditable != 'true') {\r\n\r\n node = node.parentNode;\r\n\r\n }\r\n\r\n return node;\r\n\r\n };\r\n\r\n return content;\r\n\r\n})({});\n\n\n// WEBPACK FOOTER //\n// ./modules/content.js","/**\r\n * Codex Editor toolbar module\r\n *\r\n * Contains:\r\n * - Inline toolbox\r\n * - Toolbox within plus button\r\n * - Settings section\r\n *\r\n * @author Codex Team\r\n * @version 1.0\r\n */\r\n\r\nmodule.exports = (function (toolbar) {\r\n\r\n let editor = codex.editor;\r\n\r\n toolbar.settings = require('./settings');\r\n toolbar.inline = require('./inline');\r\n toolbar.toolbox = require('./toolbox');\r\n\r\n /**\r\n * Margin between focused node and toolbar\r\n */\r\n toolbar.defaultToolbarHeight = 49;\r\n\r\n toolbar.defaultOffset = 34;\r\n\r\n toolbar.opened = false;\r\n\r\n toolbar.current = null;\r\n\r\n /**\r\n * @protected\r\n */\r\n toolbar.open = function () {\r\n\r\n if (editor.hideToolbar) {\r\n\r\n return;\r\n\r\n }\r\n\r\n editor.nodes.toolbar.classList.add('opened');\r\n this.opened = true;\r\n\r\n };\r\n\r\n /**\r\n * @protected\r\n */\r\n toolbar.close = function () {\r\n\r\n editor.nodes.toolbar.classList.remove('opened');\r\n\r\n toolbar.opened = false;\r\n toolbar.current = null;\r\n\r\n for (var button in editor.nodes.toolbarButtons) {\r\n\r\n editor.nodes.toolbarButtons[button].classList.remove('selected');\r\n\r\n }\r\n\r\n /** Close toolbox when toolbar is not displayed */\r\n editor.toolbar.toolbox.close();\r\n editor.toolbar.settings.close();\r\n\r\n };\r\n\r\n toolbar.toggle = function () {\r\n\r\n if ( !this.opened ) {\r\n\r\n this.open();\r\n\r\n } else {\r\n\r\n this.close();\r\n\r\n }\r\n\r\n };\r\n\r\n toolbar.hidePlusButton = function () {\r\n\r\n editor.nodes.plusButton.classList.add('hide');\r\n\r\n };\r\n\r\n toolbar.showPlusButton = function () {\r\n\r\n editor.nodes.plusButton.classList.remove('hide');\r\n\r\n };\r\n\r\n /**\r\n * Moving toolbar to the specified node\r\n */\r\n toolbar.move = function () {\r\n\r\n /** Close Toolbox when we move toolbar */\r\n editor.toolbar.toolbox.close();\r\n\r\n if (!editor.content.currentNode) {\r\n\r\n return;\r\n\r\n }\r\n\r\n var newYCoordinate = editor.content.currentNode.offsetTop - (editor.toolbar.defaultToolbarHeight / 2) + editor.toolbar.defaultOffset;\r\n\r\n editor.nodes.toolbar.style.transform = `translate3D(0, ${Math.floor(newYCoordinate)}px, 0)`;\r\n\r\n /** Close trash actions */\r\n editor.toolbar.settings.hideRemoveActions();\r\n\r\n };\r\n\r\n return toolbar;\r\n\r\n})({});\n\n\n// WEBPACK FOOTER //\n// ./modules/toolbar/toolbar.js","/**\r\n * Toolbar settings\r\n *\r\n * @version 1.0.4\r\n */\r\n\r\nmodule.exports = (function (settings) {\r\n\r\n let editor = codex.editor;\r\n\r\n settings.opened = false;\r\n\r\n settings.setting = null;\r\n settings.actions = null;\r\n\r\n /**\r\n * Append and open settings\r\n */\r\n settings.open = function (toolType) {\r\n\r\n /**\r\n * Append settings content\r\n * It's stored in tool.settings\r\n */\r\n if (!editor.tools[toolType] || !editor.tools[toolType].makeSettings ) {\r\n\r\n editor.core.log(`Plugin «${toolType}» has no settings`, 'warn');\r\n editor.nodes.pluginSettings.innerHTML = `Плагин «${toolType}» не имеет настроек`;\r\n\r\n } else {\r\n\r\n /**\r\n * Draw settings block\r\n */\r\n var settingsBlock = editor.tools[toolType].makeSettings();\r\n\r\n editor.nodes.pluginSettings.appendChild(settingsBlock);\r\n\r\n }\r\n\r\n /** Open settings block */\r\n editor.nodes.blockSettings.classList.add('opened');\r\n this.opened = true;\r\n\r\n };\r\n\r\n /**\r\n * Close and clear settings\r\n */\r\n settings.close = function () {\r\n\r\n editor.nodes.blockSettings.classList.remove('opened');\r\n editor.nodes.pluginSettings.innerHTML = '';\r\n\r\n this.opened = false;\r\n\r\n };\r\n\r\n /**\r\n * @param {string} toolType - plugin type\r\n */\r\n settings.toggle = function ( toolType ) {\r\n\r\n if ( !this.opened ) {\r\n\r\n this.open(toolType);\r\n\r\n } else {\r\n\r\n this.close();\r\n\r\n }\r\n\r\n };\r\n\r\n /**\r\n * Here we will draw buttons and add listeners to components\r\n */\r\n settings.makeRemoveBlockButton = function () {\r\n\r\n var removeBlockWrapper = editor.draw.node('SPAN', 'ce-toolbar__remove-btn', {}),\r\n settingButton = editor.draw.node('SPAN', 'ce-toolbar__remove-setting', { innerHTML : '' }),\r\n actionWrapper = editor.draw.node('DIV', 'ce-toolbar__remove-confirmation', {}),\r\n confirmAction = editor.draw.node('DIV', 'ce-toolbar__remove-confirm', { textContent : 'Удалить блок' }),\r\n cancelAction = editor.draw.node('DIV', 'ce-toolbar__remove-cancel', { textContent : 'Отмена' });\r\n\r\n editor.listeners.add(settingButton, 'click', editor.toolbar.settings.removeButtonClicked, false);\r\n\r\n editor.listeners.add(confirmAction, 'click', editor.toolbar.settings.confirmRemovingRequest, false);\r\n\r\n editor.listeners.add(cancelAction, 'click', editor.toolbar.settings.cancelRemovingRequest, false);\r\n\r\n actionWrapper.appendChild(confirmAction);\r\n actionWrapper.appendChild(cancelAction);\r\n\r\n removeBlockWrapper.appendChild(settingButton);\r\n removeBlockWrapper.appendChild(actionWrapper);\r\n\r\n /** Save setting */\r\n editor.toolbar.settings.setting = settingButton;\r\n editor.toolbar.settings.actions = actionWrapper;\r\n\r\n return removeBlockWrapper;\r\n\r\n };\r\n\r\n settings.removeButtonClicked = function () {\r\n\r\n var action = editor.toolbar.settings.actions;\r\n\r\n if (action.classList.contains('opened')) {\r\n\r\n editor.toolbar.settings.hideRemoveActions();\r\n\r\n } else {\r\n\r\n editor.toolbar.settings.showRemoveActions();\r\n\r\n }\r\n\r\n editor.toolbar.toolbox.close();\r\n editor.toolbar.settings.close();\r\n\r\n };\r\n\r\n settings.cancelRemovingRequest = function () {\r\n\r\n editor.toolbar.settings.actions.classList.remove('opened');\r\n\r\n };\r\n\r\n settings.confirmRemovingRequest = function () {\r\n\r\n var currentBlock = editor.content.currentNode,\r\n firstLevelBlocksCount;\r\n\r\n currentBlock.remove();\r\n\r\n firstLevelBlocksCount = editor.nodes.redactor.childNodes.length;\r\n\r\n /**\r\n * If all blocks are removed\r\n */\r\n if (firstLevelBlocksCount === 0) {\r\n\r\n /** update currentNode variable */\r\n editor.content.currentNode = null;\r\n\r\n /** Inserting new empty initial block */\r\n editor.ui.addInitialBlock();\r\n\r\n }\r\n\r\n editor.ui.saveInputs();\r\n\r\n editor.toolbar.close();\r\n\r\n };\r\n\r\n settings.showRemoveActions = function () {\r\n\r\n editor.toolbar.settings.actions.classList.add('opened');\r\n\r\n };\r\n\r\n settings.hideRemoveActions = function () {\r\n\r\n editor.toolbar.settings.actions.classList.remove('opened');\r\n\r\n };\r\n\r\n return settings;\r\n\r\n})({});\n\n\n// WEBPACK FOOTER //\n// ./modules/toolbar/settings.js","/**\r\n * Inline toolbar\r\n *\r\n * Contains from tools:\r\n * Bold, Italic, Underline and Anchor\r\n *\r\n * @author Codex Team\r\n * @version 1.0\r\n */\r\n\r\nmodule.exports = (function (inline) {\r\n\r\n let editor = codex.editor;\r\n\r\n inline.buttonsOpened = null;\r\n inline.actionsOpened = null;\r\n inline.wrappersOffset = null;\r\n\r\n /**\r\n * saving selection that need for execCommand for styling\r\n *\r\n */\r\n inline.storedSelection = null;\r\n\r\n /**\r\n * @protected\r\n *\r\n * Open inline toobar\r\n */\r\n inline.show = function () {\r\n\r\n var currentNode = editor.content.currentNode,\r\n tool = currentNode.dataset.tool,\r\n plugin;\r\n\r\n /**\r\n * tool allowed to open inline toolbar\r\n */\r\n plugin = editor.tools[tool];\r\n\r\n if (!plugin.showInlineToolbar)\r\n return;\r\n\r\n var selectedText = inline.getSelectionText(),\r\n toolbar = editor.nodes.inlineToolbar.wrapper;\r\n\r\n if (selectedText.length > 0) {\r\n\r\n /** Move toolbar and open */\r\n editor.toolbar.inline.move();\r\n\r\n /** Open inline toolbar */\r\n toolbar.classList.add('opened');\r\n\r\n /** show buttons of inline toolbar */\r\n editor.toolbar.inline.showButtons();\r\n\r\n }\r\n\r\n };\r\n\r\n /**\r\n * @protected\r\n *\r\n * Closes inline toolbar\r\n */\r\n inline.close = function () {\r\n\r\n var toolbar = editor.nodes.inlineToolbar.wrapper;\r\n\r\n toolbar.classList.remove('opened');\r\n\r\n };\r\n\r\n /**\r\n * @private\r\n *\r\n * Moving toolbar\r\n */\r\n inline.move = function () {\r\n\r\n if (!this.wrappersOffset) {\r\n\r\n this.wrappersOffset = this.getWrappersOffset();\r\n\r\n }\r\n\r\n var coords = this.getSelectionCoords(),\r\n defaultOffset = 0,\r\n toolbar = editor.nodes.inlineToolbar.wrapper,\r\n newCoordinateX,\r\n newCoordinateY;\r\n\r\n if (toolbar.offsetHeight === 0) {\r\n\r\n defaultOffset = 40;\r\n\r\n }\r\n\r\n newCoordinateX = coords.x - this.wrappersOffset.left;\r\n newCoordinateY = coords.y + window.scrollY - this.wrappersOffset.top - defaultOffset - toolbar.offsetHeight;\r\n\r\n toolbar.style.transform = `translate3D(${Math.floor(newCoordinateX)}px, ${Math.floor(newCoordinateY)}px, 0)`;\r\n\r\n /** Close everything */\r\n editor.toolbar.inline.closeButtons();\r\n editor.toolbar.inline.closeAction();\r\n\r\n };\r\n\r\n /**\r\n * @private\r\n *\r\n * Tool Clicked\r\n */\r\n\r\n inline.toolClicked = function (event, type) {\r\n\r\n /**\r\n * For simple tools we use default browser function\r\n * For more complicated tools, we should write our own behavior\r\n */\r\n switch (type) {\r\n case 'createLink' : editor.toolbar.inline.createLinkAction(event, type); break;\r\n default : editor.toolbar.inline.defaultToolAction(type); break;\r\n }\r\n\r\n /**\r\n * highlight buttons\r\n * after making some action\r\n */\r\n editor.nodes.inlineToolbar.buttons.childNodes.forEach(editor.toolbar.inline.hightlight);\r\n\r\n };\r\n\r\n /**\r\n * @private\r\n *\r\n * Saving wrappers offset in DOM\r\n */\r\n inline.getWrappersOffset = function () {\r\n\r\n var wrapper = editor.nodes.wrapper,\r\n offset = this.getOffset(wrapper);\r\n\r\n this.wrappersOffset = offset;\r\n return offset;\r\n\r\n };\r\n\r\n /**\r\n * @private\r\n *\r\n * Calculates offset of DOM element\r\n *\r\n * @param el\r\n * @returns {{top: number, left: number}}\r\n */\r\n inline.getOffset = function ( el ) {\r\n\r\n var _x = 0;\r\n var _y = 0;\r\n\r\n while( el && !isNaN( el.offsetLeft ) && !isNaN( el.offsetTop ) ) {\r\n\r\n _x += (el.offsetLeft + el.clientLeft);\r\n _y += (el.offsetTop + el.clientTop);\r\n el = el.offsetParent;\r\n\r\n }\r\n return { top: _y, left: _x };\r\n\r\n };\r\n\r\n /**\r\n * @private\r\n *\r\n * Calculates position of selected text\r\n * @returns {{x: number, y: number}}\r\n */\r\n inline.getSelectionCoords = function () {\r\n\r\n var sel = document.selection, range;\r\n var x = 0, y = 0;\r\n\r\n if (sel) {\r\n\r\n if (sel.type != 'Control') {\r\n\r\n range = sel.createRange();\r\n range.collapse(true);\r\n x = range.boundingLeft;\r\n y = range.boundingTop;\r\n\r\n }\r\n\r\n } else if (window.getSelection) {\r\n\r\n sel = window.getSelection();\r\n\r\n if (sel.rangeCount) {\r\n\r\n range = sel.getRangeAt(0).cloneRange();\r\n if (range.getClientRects) {\r\n\r\n range.collapse(true);\r\n var rect = range.getClientRects()[0];\r\n\r\n if (!rect) {\r\n\r\n return;\r\n\r\n }\r\n\r\n x = rect.left;\r\n y = rect.top;\r\n\r\n }\r\n\r\n }\r\n\r\n }\r\n return { x: x, y: y };\r\n\r\n };\r\n\r\n /**\r\n * @private\r\n *\r\n * Returns selected text as String\r\n * @returns {string}\r\n */\r\n inline.getSelectionText = function () {\r\n\r\n var selectedText = '';\r\n\r\n // all modern browsers and IE9+\r\n if (window.getSelection) {\r\n\r\n selectedText = window.getSelection().toString();\r\n\r\n }\r\n\r\n return selectedText;\r\n\r\n };\r\n\r\n /** Opens buttons block */\r\n inline.showButtons = function () {\r\n\r\n var buttons = editor.nodes.inlineToolbar.buttons;\r\n\r\n buttons.classList.add('opened');\r\n\r\n editor.toolbar.inline.buttonsOpened = true;\r\n\r\n /** highlight buttons */\r\n editor.nodes.inlineToolbar.buttons.childNodes.forEach(editor.toolbar.inline.hightlight);\r\n\r\n };\r\n\r\n /** Makes buttons disappear */\r\n inline.closeButtons = function () {\r\n\r\n var buttons = editor.nodes.inlineToolbar.buttons;\r\n\r\n buttons.classList.remove('opened');\r\n\r\n editor.toolbar.inline.buttonsOpened = false;\r\n\r\n };\r\n\r\n /** Open buttons defined action if exist */\r\n inline.showActions = function () {\r\n\r\n var action = editor.nodes.inlineToolbar.actions;\r\n\r\n action.classList.add('opened');\r\n\r\n editor.toolbar.inline.actionsOpened = true;\r\n\r\n };\r\n\r\n /** Close actions block */\r\n inline.closeAction = function () {\r\n\r\n var action = editor.nodes.inlineToolbar.actions;\r\n\r\n action.innerHTML = '';\r\n action.classList.remove('opened');\r\n editor.toolbar.inline.actionsOpened = false;\r\n\r\n };\r\n\r\n\r\n /**\r\n * Callback for keydowns in inline toolbar \"Insert link...\" input\r\n */\r\n let inlineToolbarAnchorInputKeydown_ = function (event) {\r\n\r\n if (event.keyCode != editor.core.keys.ENTER) {\r\n\r\n return;\r\n\r\n }\r\n\r\n let editable = editor.content.currentNode,\r\n storedSelection = editor.toolbar.inline.storedSelection;\r\n\r\n editor.toolbar.inline.restoreSelection(editable, storedSelection);\r\n editor.toolbar.inline.setAnchor(this.value);\r\n\r\n /**\r\n * Preventing events that will be able to happen\r\n */\r\n event.preventDefault();\r\n event.stopImmediatePropagation();\r\n\r\n editor.toolbar.inline.clearRange();\r\n\r\n };\r\n\r\n /** Action for link creation or for setting anchor */\r\n inline.createLinkAction = function (event) {\r\n\r\n var isActive = this.isLinkActive();\r\n\r\n var editable = editor.content.currentNode,\r\n storedSelection = editor.toolbar.inline.saveSelection(editable);\r\n\r\n /** Save globally selection */\r\n editor.toolbar.inline.storedSelection = storedSelection;\r\n\r\n if (isActive) {\r\n\r\n\r\n /**\r\n * Changing stored selection. if we want to remove anchor from word\r\n * we should remove anchor from whole word, not only selected part.\r\n * The solution is than we get the length of current link\r\n * Change start position to - end of selection minus length of anchor\r\n */\r\n editor.toolbar.inline.restoreSelection(editable, storedSelection);\r\n\r\n editor.toolbar.inline.defaultToolAction('unlink');\r\n\r\n } else {\r\n\r\n /** Create input and close buttons */\r\n var action = editor.draw.inputForLink();\r\n\r\n editor.nodes.inlineToolbar.actions.appendChild(action);\r\n\r\n editor.toolbar.inline.closeButtons();\r\n editor.toolbar.inline.showActions();\r\n\r\n /**\r\n * focus to input\r\n * Solution: https://developer.mozilla.org/ru/docs/Web/API/HTMLElement/focus\r\n * Prevents event after showing input and when we need to focus an input which is in unexisted form\r\n */\r\n action.focus();\r\n event.preventDefault();\r\n\r\n /** Callback to link action */\r\n editor.listeners.add(action, 'keydown', inlineToolbarAnchorInputKeydown_, false);\r\n\r\n }\r\n\r\n };\r\n\r\n inline.isLinkActive = function () {\r\n\r\n var isActive = false;\r\n\r\n editor.nodes.inlineToolbar.buttons.childNodes.forEach(function (tool) {\r\n\r\n var dataType = tool.dataset.type;\r\n\r\n if (dataType == 'link' && tool.classList.contains('hightlighted')) {\r\n\r\n isActive = true;\r\n\r\n }\r\n\r\n });\r\n\r\n return isActive;\r\n\r\n };\r\n\r\n /** default action behavior of tool */\r\n inline.defaultToolAction = function (type) {\r\n\r\n document.execCommand(type, false, null);\r\n\r\n };\r\n\r\n /**\r\n * @private\r\n *\r\n * Sets URL\r\n *\r\n * @param {String} url - URL\r\n */\r\n inline.setAnchor = function (url) {\r\n\r\n document.execCommand('createLink', false, url);\r\n\r\n /** Close after URL inserting */\r\n editor.toolbar.inline.closeAction();\r\n\r\n };\r\n\r\n /**\r\n * @private\r\n *\r\n * Saves selection\r\n */\r\n inline.saveSelection = function (containerEl) {\r\n\r\n var range = window.getSelection().getRangeAt(0),\r\n preSelectionRange = range.cloneRange(),\r\n start;\r\n\r\n preSelectionRange.selectNodeContents(containerEl);\r\n preSelectionRange.setEnd(range.startContainer, range.startOffset);\r\n\r\n start = preSelectionRange.toString().length;\r\n\r\n return {\r\n start: start,\r\n end: start + range.toString().length\r\n };\r\n\r\n };\r\n\r\n /**\r\n * @private\r\n *\r\n * Sets to previous selection (Range)\r\n *\r\n * @param {Element} containerEl - editable element where we restore range\r\n * @param {Object} savedSel - range basic information to restore\r\n */\r\n inline.restoreSelection = function (containerEl, savedSel) {\r\n\r\n var range = document.createRange(),\r\n charIndex = 0;\r\n\r\n range.setStart(containerEl, 0);\r\n range.collapse(true);\r\n\r\n var nodeStack = [ containerEl ],\r\n node,\r\n foundStart = false,\r\n stop = false,\r\n nextCharIndex;\r\n\r\n while (!stop && (node = nodeStack.pop())) {\r\n\r\n if (node.nodeType == 3) {\r\n\r\n nextCharIndex = charIndex + node.length;\r\n\r\n if (!foundStart && savedSel.start >= charIndex && savedSel.start <= nextCharIndex) {\r\n\r\n range.setStart(node, savedSel.start - charIndex);\r\n foundStart = true;\r\n\r\n }\r\n if (foundStart && savedSel.end >= charIndex && savedSel.end <= nextCharIndex) {\r\n\r\n range.setEnd(node, savedSel.end - charIndex);\r\n stop = true;\r\n\r\n }\r\n charIndex = nextCharIndex;\r\n\r\n } else {\r\n\r\n var i = node.childNodes.length;\r\n\r\n while (i--) {\r\n\r\n nodeStack.push(node.childNodes[i]);\r\n\r\n }\r\n\r\n }\r\n\r\n }\r\n\r\n var sel = window.getSelection();\r\n\r\n sel.removeAllRanges();\r\n sel.addRange(range);\r\n\r\n };\r\n\r\n /**\r\n * @private\r\n *\r\n * Removes all ranges from window selection\r\n */\r\n inline.clearRange = function () {\r\n\r\n var selection = window.getSelection();\r\n\r\n selection.removeAllRanges();\r\n\r\n };\r\n\r\n /**\r\n * @private\r\n *\r\n * sets or removes hightlight\r\n */\r\n inline.hightlight = function (tool) {\r\n\r\n var dataType = tool.dataset.type;\r\n\r\n if (document.queryCommandState(dataType)) {\r\n\r\n editor.toolbar.inline.setButtonHighlighted(tool);\r\n\r\n } else {\r\n\r\n editor.toolbar.inline.removeButtonsHighLight(tool);\r\n\r\n }\r\n\r\n /**\r\n *\r\n * hightlight for anchors\r\n */\r\n var selection = window.getSelection(),\r\n tag = selection.anchorNode.parentNode;\r\n\r\n if (tag.tagName == 'A' && dataType == 'link') {\r\n\r\n editor.toolbar.inline.setButtonHighlighted(tool);\r\n\r\n }\r\n\r\n };\r\n\r\n /**\r\n * @private\r\n *\r\n * Mark button if text is already executed\r\n */\r\n inline.setButtonHighlighted = function (button) {\r\n\r\n button.classList.add('hightlighted');\r\n\r\n /** At link tool we also change icon */\r\n if (button.dataset.type == 'link') {\r\n\r\n var icon = button.childNodes[0];\r\n\r\n icon.classList.remove('ce-icon-link');\r\n icon.classList.add('ce-icon-unlink');\r\n\r\n }\r\n\r\n };\r\n\r\n /**\r\n * @private\r\n *\r\n * Removes hightlight\r\n */\r\n inline.removeButtonsHighLight = function (button) {\r\n\r\n button.classList.remove('hightlighted');\r\n\r\n /** At link tool we also change icon */\r\n if (button.dataset.type == 'link') {\r\n\r\n var icon = button.childNodes[0];\r\n\r\n icon.classList.remove('ce-icon-unlink');\r\n icon.classList.add('ce-icon-link');\r\n\r\n }\r\n\r\n };\r\n\r\n\r\n return inline;\r\n\r\n})({});\n\n\n// WEBPACK FOOTER //\n// ./modules/toolbar/inline.js","/**\r\n * Codex Editor toolbox\r\n *\r\n * All tools be able to appended here\r\n *\r\n * @author Codex Team\r\n * @version 1.0\r\n */\r\n\r\nmodule.exports = (function (toolbox) {\r\n\r\n let editor = codex.editor;\r\n\r\n toolbox.opened = false;\r\n\r\n /** Shows toolbox */\r\n toolbox.open = function () {\r\n\r\n /** Close setting if toolbox is opened */\r\n if (editor.toolbar.settings.opened) {\r\n\r\n editor.toolbar.settings.close();\r\n\r\n }\r\n\r\n /** display toolbox */\r\n editor.nodes.toolbox.classList.add('opened');\r\n\r\n /** Animate plus button */\r\n editor.nodes.plusButton.classList.add('clicked');\r\n\r\n /** toolbox state */\r\n editor.toolbar.toolbox.opened = true;\r\n\r\n };\r\n\r\n /** Closes toolbox */\r\n toolbox.close = function () {\r\n\r\n /** Makes toolbox disapear */\r\n editor.nodes.toolbox.classList.remove('opened');\r\n\r\n /** Rotate plus button */\r\n editor.nodes.plusButton.classList.remove('clicked');\r\n\r\n /** toolbox state */\r\n editor.toolbar.toolbox.opened = false;\r\n\r\n editor.toolbar.current = null;\r\n\r\n };\r\n\r\n toolbox.leaf = function () {\r\n\r\n let currentTool = editor.toolbar.current,\r\n tools = Object.keys(editor.tools),\r\n barButtons = editor.nodes.toolbarButtons,\r\n nextToolIndex = 0,\r\n toolToSelect,\r\n visibleTool,\r\n tool;\r\n\r\n if ( !currentTool ) {\r\n\r\n /** Get first tool from object*/\r\n for(tool in editor.tools) {\r\n\r\n if (editor.tools[tool].displayInToolbox) {\r\n\r\n break;\r\n\r\n }\r\n\r\n nextToolIndex ++;\r\n\r\n }\r\n\r\n } else {\r\n\r\n nextToolIndex = (tools.indexOf(currentTool) + 1) % tools.length;\r\n visibleTool = tools[nextToolIndex];\r\n\r\n while (!editor.tools[visibleTool].displayInToolbox) {\r\n\r\n nextToolIndex = (nextToolIndex + 1) % tools.length;\r\n visibleTool = tools[nextToolIndex];\r\n\r\n }\r\n\r\n }\r\n\r\n toolToSelect = tools[nextToolIndex];\r\n\r\n for ( var button in barButtons ) {\r\n\r\n barButtons[button].classList.remove('selected');\r\n\r\n }\r\n\r\n barButtons[toolToSelect].classList.add('selected');\r\n editor.toolbar.current = toolToSelect;\r\n\r\n };\r\n\r\n /**\r\n * Transforming selected node type into selected toolbar element type\r\n * @param {event} event\r\n */\r\n toolbox.toolClicked = function (event) {\r\n\r\n /**\r\n * UNREPLACEBLE_TOOLS this types of tools are forbidden to replace even they are empty\r\n */\r\n var UNREPLACEBLE_TOOLS = ['image', 'link', 'list', 'instagram', 'twitter', 'embed'],\r\n tool = editor.tools[editor.toolbar.current],\r\n workingNode = editor.content.currentNode,\r\n currentInputIndex = editor.caret.inputIndex,\r\n newBlockContent,\r\n appendCallback,\r\n blockData;\r\n\r\n /** Make block from plugin */\r\n newBlockContent = tool.render();\r\n\r\n /** information about block */\r\n blockData = {\r\n block : newBlockContent,\r\n type : tool.type,\r\n stretched : false\r\n };\r\n\r\n if (\r\n workingNode &&\r\n UNREPLACEBLE_TOOLS.indexOf(workingNode.dataset.tool) === -1 &&\r\n workingNode.textContent.trim() === ''\r\n ) {\r\n\r\n /** Replace current block */\r\n editor.content.switchBlock(workingNode, newBlockContent, tool.type);\r\n\r\n } else {\r\n\r\n /** Insert new Block from plugin */\r\n editor.content.insertBlock(blockData);\r\n\r\n /** increase input index */\r\n currentInputIndex++;\r\n\r\n }\r\n\r\n /** Fire tool append callback */\r\n appendCallback = tool.appendCallback;\r\n\r\n if (appendCallback && typeof appendCallback == 'function') {\r\n\r\n appendCallback.call(event);\r\n\r\n }\r\n\r\n window.setTimeout(function () {\r\n\r\n /** Set caret to current block */\r\n editor.caret.setToBlock(currentInputIndex);\r\n\r\n }, 10);\r\n\r\n\r\n /**\r\n * Changing current Node\r\n */\r\n editor.content.workingNodeChanged();\r\n\r\n /**\r\n * Move toolbar when node is changed\r\n */\r\n editor.toolbar.move();\r\n\r\n };\r\n\r\n return toolbox;\r\n\r\n})({});\n\n\n// WEBPACK FOOTER //\n// ./modules/toolbar/toolbox.js","/**\r\n * @module Codex Editor Callbacks module\r\n * @description Module works with editor added Elements\r\n *\r\n * @author Codex Team\r\n * @version 1.4.0\r\n */\r\n\r\nmodule.exports = (function (callbacks) {\r\n\r\n let editor = codex.editor;\r\n\r\n /**\r\n * used by UI module\r\n * @description Routes all keydowns on document\r\n * @param {Object} event\r\n */\r\n callbacks.globalKeydown = function (event) {\r\n\r\n switch (event.keyCode) {\r\n case editor.core.keys.ENTER : enterKeyPressed_(event); break;\r\n }\r\n\r\n };\r\n\r\n /**\r\n * used by UI module\r\n * @description Routes all keydowns on redactors area\r\n * @param {Object} event\r\n */\r\n callbacks.redactorKeyDown = function (event) {\r\n\r\n switch (event.keyCode) {\r\n case editor.core.keys.TAB : tabKeyPressedOnRedactorsZone_(event); break;\r\n case editor.core.keys.ENTER : enterKeyPressedOnRedactorsZone_(event); break;\r\n case editor.core.keys.ESC : escapeKeyPressedOnRedactorsZone_(event); break;\r\n default : defaultKeyPressedOnRedactorsZone_(event); break;\r\n }\r\n\r\n };\r\n\r\n /**\r\n * used by UI module\r\n * @description Routes all keyup events\r\n * @param {Object} event\r\n */\r\n callbacks.globalKeyup = function (event) {\r\n\r\n switch (event.keyCode) {\r\n case editor.core.keys.UP :\r\n case editor.core.keys.LEFT :\r\n case editor.core.keys.RIGHT :\r\n case editor.core.keys.DOWN : arrowKeyPressed_(event); break;\r\n }\r\n\r\n };\r\n\r\n /**\r\n * @param {Object} event\r\n * @private\r\n *\r\n * Handles behaviour when tab pressed\r\n * @description if Content is empty show toolbox (if it is closed) or leaf tools\r\n * uses Toolbars toolbox module to handle the situation\r\n */\r\n var tabKeyPressedOnRedactorsZone_ = function (event) {\r\n\r\n /**\r\n * Wait for solution. Would like to know the behaviour\r\n * @todo Add spaces\r\n */\r\n event.preventDefault();\r\n\r\n var nativeInputs = editor.content.currentNode.querySelectorAll('textarea, input'),\r\n nativeInputsAreEmpty = true,\r\n textContentIsEmpty = !editor.content.currentNode.textContent.trim();\r\n\r\n Array.prototype.map.call(nativeInputs, function (input) {\r\n\r\n if (input.type == 'textarea' || input.type == 'text') {\r\n\r\n nativeInputsAreEmpty = nativeInputsAreEmpty && !input.value.trim();\r\n\r\n }\r\n\r\n });\r\n\r\n var blockIsEmpty = textContentIsEmpty && nativeInputsAreEmpty;\r\n\r\n if (!blockIsEmpty) {\r\n\r\n return;\r\n\r\n }\r\n\r\n if ( !editor.toolbar.opened ) {\r\n\r\n editor.toolbar.open();\r\n\r\n }\r\n\r\n if (editor.toolbar.opened && !editor.toolbar.toolbox.opened) {\r\n\r\n editor.toolbar.toolbox.open();\r\n\r\n } else {\r\n\r\n editor.toolbar.toolbox.leaf();\r\n\r\n }\r\n\r\n };\r\n\r\n /**\r\n * Handles global EnterKey Press\r\n * @see enterPressedOnBlock_\r\n * @param {Object} event\r\n */\r\n var enterKeyPressed_ = function () {\r\n\r\n if (editor.content.editorAreaHightlighted) {\r\n\r\n /**\r\n * it means that we lose input index, saved index before is not correct\r\n * therefore we need to set caret when we insert new block\r\n */\r\n editor.caret.inputIndex = -1;\r\n\r\n enterPressedOnBlock_();\r\n\r\n }\r\n\r\n };\r\n\r\n /**\r\n * Callback for enter key pressing in first-level block area\r\n *\r\n * @param {Event} event\r\n * @private\r\n *\r\n * @description Inserts new block with initial type from settings\r\n */\r\n var enterPressedOnBlock_ = function () {\r\n\r\n var NEW_BLOCK_TYPE = editor.settings.initialBlockPlugin;\r\n\r\n editor.content.insertBlock({\r\n type : NEW_BLOCK_TYPE,\r\n block : editor.tools[NEW_BLOCK_TYPE].render()\r\n }, true );\r\n\r\n editor.toolbar.move();\r\n editor.toolbar.open();\r\n\r\n };\r\n\r\n\r\n /**\r\n * ENTER key handler\r\n *\r\n * @param {Object} event\r\n * @private\r\n *\r\n * @description Makes new block with initial type from settings\r\n */\r\n var enterKeyPressedOnRedactorsZone_ = function (event) {\r\n\r\n if (event.target.contentEditable == 'true') {\r\n\r\n /** Update input index */\r\n editor.caret.saveCurrentInputIndex();\r\n\r\n }\r\n\r\n var currentInputIndex = editor.caret.getCurrentInputIndex() || 0,\r\n workingNode = editor.content.currentNode,\r\n tool = workingNode.dataset.tool,\r\n isEnterPressedOnToolbar = editor.toolbar.opened &&\r\n editor.toolbar.current &&\r\n event.target == editor.state.inputs[currentInputIndex];\r\n\r\n /** The list of tools which needs the default browser behaviour */\r\n var enableLineBreaks = editor.tools[tool].enableLineBreaks;\r\n\r\n /** This type of block creates when enter is pressed */\r\n var NEW_BLOCK_TYPE = editor.settings.initialBlockPlugin;\r\n\r\n /**\r\n * When toolbar is opened, select tool instead of making new paragraph\r\n */\r\n if ( isEnterPressedOnToolbar ) {\r\n\r\n event.preventDefault();\r\n\r\n editor.toolbar.toolbox.toolClicked(event);\r\n\r\n editor.toolbar.close();\r\n\r\n /**\r\n * Stop other listeners callback executions\r\n */\r\n event.stopPropagation();\r\n event.stopImmediatePropagation();\r\n\r\n return;\r\n\r\n }\r\n\r\n /**\r\n * Allow paragraph lineBreaks with shift enter\r\n * Or if shiftkey pressed and enter and enabledLineBreaks, the let new block creation\r\n */\r\n if ( event.shiftKey || enableLineBreaks ) {\r\n\r\n event.stopPropagation();\r\n event.stopImmediatePropagation();\r\n return;\r\n\r\n }\r\n\r\n var currentSelection = window.getSelection(),\r\n currentSelectedNode = currentSelection.anchorNode,\r\n caretAtTheEndOfText = editor.caret.position.atTheEnd(),\r\n isTextNodeHasParentBetweenContenteditable = false;\r\n\r\n /**\r\n * Allow making new

    in same block by SHIFT+ENTER and forbids to prevent default browser behaviour\r\n */\r\n if ( event.shiftKey && !enableLineBreaks ) {\r\n\r\n editor.callback.enterPressedOnBlock(editor.content.currentBlock, event);\r\n event.preventDefault();\r\n return;\r\n\r\n }\r\n\r\n /**\r\n * Workaround situation when caret at the Text node that has some wrapper Elements\r\n * Split block cant handle this.\r\n * We need to save default behavior\r\n */\r\n isTextNodeHasParentBetweenContenteditable = currentSelectedNode && currentSelectedNode.parentNode.contentEditable != 'true';\r\n\r\n /**\r\n * Split blocks when input has several nodes and caret placed in textNode\r\n */\r\n if (\r\n currentSelectedNode.nodeType == editor.core.nodeTypes.TEXT &&\r\n !isTextNodeHasParentBetweenContenteditable &&\r\n !caretAtTheEndOfText\r\n ) {\r\n\r\n event.preventDefault();\r\n\r\n editor.core.log('Splitting Text node...');\r\n\r\n editor.content.splitBlock(currentInputIndex);\r\n\r\n /** Show plus button when next input after split is empty*/\r\n if (!editor.state.inputs[currentInputIndex + 1].textContent.trim()) {\r\n\r\n editor.toolbar.showPlusButton();\r\n\r\n }\r\n\r\n } else {\r\n\r\n var islastNode = editor.content.isLastNode(currentSelectedNode);\r\n\r\n if ( islastNode && caretAtTheEndOfText ) {\r\n\r\n event.preventDefault();\r\n event.stopPropagation();\r\n event.stopImmediatePropagation();\r\n\r\n editor.core.log('ENTER clicked in last textNode. Create new BLOCK');\r\n\r\n editor.content.insertBlock({\r\n type: NEW_BLOCK_TYPE,\r\n block: editor.tools[NEW_BLOCK_TYPE].render()\r\n }, true);\r\n\r\n editor.toolbar.move();\r\n editor.toolbar.open();\r\n\r\n /** Show plus button with empty block */\r\n editor.toolbar.showPlusButton();\r\n\r\n }\r\n\r\n }\r\n\r\n /** get all inputs after new appending block */\r\n editor.ui.saveInputs();\r\n\r\n };\r\n\r\n /**\r\n * Escape behaviour\r\n * @param event\r\n * @private\r\n *\r\n * @description Closes toolbox and toolbar. Prevents default behaviour\r\n */\r\n var escapeKeyPressedOnRedactorsZone_ = function (event) {\r\n\r\n /** Close all toolbar */\r\n editor.toolbar.close();\r\n\r\n /** Close toolbox */\r\n editor.toolbar.toolbox.close();\r\n\r\n event.preventDefault();\r\n\r\n };\r\n\r\n /**\r\n * @param {Event} event\r\n * @private\r\n *\r\n * closes and moves toolbar\r\n */\r\n var arrowKeyPressed_ = function (event) {\r\n\r\n editor.content.workingNodeChanged();\r\n\r\n /* Closing toolbar */\r\n editor.toolbar.close();\r\n editor.toolbar.move();\r\n\r\n };\r\n\r\n /**\r\n * @private\r\n * @param {Event} event\r\n *\r\n * @description Closes all opened bars from toolbar.\r\n * If block is mark, clears highlightning\r\n */\r\n var defaultKeyPressedOnRedactorsZone_ = function () {\r\n\r\n editor.toolbar.close();\r\n\r\n if (!editor.toolbar.inline.actionsOpened) {\r\n\r\n editor.toolbar.inline.close();\r\n editor.content.clearMark();\r\n\r\n }\r\n\r\n };\r\n\r\n /**\r\n * Handler when clicked on redactors area\r\n *\r\n * @protected\r\n * @param event\r\n *\r\n * @description Detects clicked area. If it is first-level block area, marks as detected and\r\n * on next enter press will be inserted new block\r\n * Otherwise, save carets position (input index) and put caret to the editable zone.\r\n *\r\n * @see detectWhenClickedOnFirstLevelBlockArea_\r\n *\r\n */\r\n callbacks.redactorClicked = function (event) {\r\n\r\n detectWhenClickedOnFirstLevelBlockArea_();\r\n\r\n editor.content.workingNodeChanged(event.target);\r\n editor.ui.saveInputs();\r\n\r\n var selectedText = editor.toolbar.inline.getSelectionText(),\r\n firstLevelBlock;\r\n\r\n /** If selection range took off, then we hide inline toolbar */\r\n if (selectedText.length === 0) {\r\n\r\n editor.toolbar.inline.close();\r\n\r\n }\r\n\r\n /** Update current input index in memory when caret focused into existed input */\r\n if (event.target.contentEditable == 'true') {\r\n\r\n editor.caret.saveCurrentInputIndex();\r\n\r\n }\r\n\r\n if (editor.content.currentNode === null) {\r\n\r\n /**\r\n * If inputs in redactor does not exits, then we put input index 0 not -1\r\n */\r\n var indexOfLastInput = editor.state.inputs.length > 0 ? editor.state.inputs.length - 1 : 0;\r\n\r\n /** If we have any inputs */\r\n if (editor.state.inputs.length) {\r\n\r\n /** getting firstlevel parent of input */\r\n firstLevelBlock = editor.content.getFirstLevelBlock(editor.state.inputs[indexOfLastInput]);\r\n\r\n }\r\n\r\n /** If input is empty, then we set caret to the last input */\r\n if (editor.state.inputs.length && editor.state.inputs[indexOfLastInput].textContent === '' && firstLevelBlock.dataset.tool == editor.settings.initialBlockPlugin) {\r\n\r\n editor.caret.setToBlock(indexOfLastInput);\r\n\r\n } else {\r\n\r\n /** Create new input when caret clicked in redactors area */\r\n var NEW_BLOCK_TYPE = editor.settings.initialBlockPlugin;\r\n\r\n editor.content.insertBlock({\r\n type : NEW_BLOCK_TYPE,\r\n block : editor.tools[NEW_BLOCK_TYPE].render()\r\n });\r\n\r\n /** If there is no inputs except inserted */\r\n if (editor.state.inputs.length === 1) {\r\n\r\n editor.caret.setToBlock(indexOfLastInput);\r\n\r\n } else {\r\n\r\n /** Set caret to this appended input */\r\n editor.caret.setToNextBlock(indexOfLastInput);\r\n\r\n }\r\n\r\n }\r\n\r\n } else {\r\n\r\n /** Close all panels */\r\n editor.toolbar.settings.close();\r\n editor.toolbar.toolbox.close();\r\n\r\n }\r\n\r\n /**\r\n * Move toolbar and open\r\n */\r\n editor.toolbar.move();\r\n editor.toolbar.open();\r\n\r\n var inputIsEmpty = !editor.content.currentNode.textContent.trim(),\r\n currentNodeType = editor.content.currentNode.dataset.tool,\r\n isInitialType = currentNodeType == editor.settings.initialBlockPlugin;\r\n\r\n\r\n /** Hide plus buttons */\r\n editor.toolbar.hidePlusButton();\r\n\r\n if (!inputIsEmpty) {\r\n\r\n /** Mark current block */\r\n editor.content.markBlock();\r\n\r\n }\r\n\r\n if ( isInitialType && inputIsEmpty ) {\r\n\r\n /** Show plus button */\r\n editor.toolbar.showPlusButton();\r\n\r\n }\r\n\r\n\r\n };\r\n\r\n /**\r\n * This method allows to define, is caret in contenteditable element or not.\r\n *\r\n * @private\r\n *\r\n * @description Otherwise, if we get TEXT node from range container, that will means we have input index.\r\n * In this case we use default browsers behaviour (if plugin allows that) or overwritten action.\r\n * Therefore, to be sure that we've clicked first-level block area, we should have currentNode, which always\r\n * specifies to the first-level block. Other cases we just ignore.\r\n */\r\n var detectWhenClickedOnFirstLevelBlockArea_ = function () {\r\n\r\n var selection = window.getSelection(),\r\n anchorNode = selection.anchorNode,\r\n flag = false;\r\n\r\n if (selection.rangeCount === 0) {\r\n\r\n editor.content.editorAreaHightlighted = true;\r\n\r\n } else {\r\n\r\n if (!editor.core.isDomNode(anchorNode)) {\r\n\r\n anchorNode = anchorNode.parentNode;\r\n\r\n }\r\n\r\n /** Already founded, without loop */\r\n if (anchorNode.contentEditable == 'true') {\r\n\r\n flag = true;\r\n\r\n }\r\n\r\n while (anchorNode.contentEditable != 'true') {\r\n\r\n anchorNode = anchorNode.parentNode;\r\n\r\n if (anchorNode.contentEditable == 'true') {\r\n\r\n flag = true;\r\n\r\n }\r\n\r\n if (anchorNode == document.body) {\r\n\r\n break;\r\n\r\n }\r\n\r\n }\r\n\r\n /** If editable element founded, flag is \"TRUE\", Therefore we return \"FALSE\" */\r\n editor.content.editorAreaHightlighted = !flag;\r\n\r\n }\r\n\r\n };\r\n\r\n /**\r\n * Toolbar button click handler\r\n *\r\n * @param {Object} event - cursor to the button\r\n * @protected\r\n *\r\n * @description gets current tool and calls render method\r\n */\r\n callbacks.toolbarButtonClicked = function (event) {\r\n\r\n var button = this;\r\n\r\n editor.toolbar.current = button.dataset.type;\r\n\r\n editor.toolbar.toolbox.toolClicked(event);\r\n editor.toolbar.close();\r\n\r\n };\r\n\r\n /**\r\n * Show or Hide toolbox when plus button is clicked\r\n */\r\n callbacks.plusButtonClicked = function () {\r\n\r\n if (!editor.nodes.toolbox.classList.contains('opened')) {\r\n\r\n editor.toolbar.toolbox.open();\r\n\r\n } else {\r\n\r\n editor.toolbar.toolbox.close();\r\n\r\n }\r\n\r\n };\r\n\r\n /**\r\n * Block handlers for KeyDown events\r\n *\r\n * @protected\r\n * @param {Object} event\r\n *\r\n * Handles keydowns on block\r\n * @see blockRightOrDownArrowPressed_\r\n * @see backspacePressed_\r\n * @see blockLeftOrUpArrowPressed_\r\n */\r\n callbacks.blockKeydown = function (event) {\r\n\r\n let block = event.target; // event.target is input\r\n\r\n switch (event.keyCode) {\r\n\r\n case editor.core.keys.DOWN:\r\n case editor.core.keys.RIGHT:\r\n blockRightOrDownArrowPressed_(event);\r\n break;\r\n\r\n case editor.core.keys.BACKSPACE:\r\n backspacePressed_(block, event);\r\n break;\r\n\r\n case editor.core.keys.UP:\r\n case editor.core.keys.LEFT:\r\n blockLeftOrUpArrowPressed_(event);\r\n break;\r\n\r\n }\r\n\r\n };\r\n\r\n /**\r\n * RIGHT or DOWN keydowns on block\r\n *\r\n * @param {Object} event\r\n * @private\r\n *\r\n * @description watches the selection and gets closest editable element.\r\n * Uses method getDeepestTextNodeFromPosition to get the last node of next block\r\n * Sets caret if it is contenteditable\r\n */\r\n var blockRightOrDownArrowPressed_ = function (event) {\r\n\r\n var selection = window.getSelection(),\r\n inputs = editor.state.inputs,\r\n focusedNode = selection.anchorNode,\r\n focusedNodeHolder;\r\n\r\n /** Check for caret existance */\r\n if (!focusedNode) {\r\n\r\n return false;\r\n\r\n }\r\n\r\n /** Looking for closest (parent) contentEditable element of focused node */\r\n while (focusedNode.contentEditable != 'true') {\r\n\r\n focusedNodeHolder = focusedNode.parentNode;\r\n focusedNode = focusedNodeHolder;\r\n\r\n }\r\n\r\n /** Input index in DOM level */\r\n var editableElementIndex = 0;\r\n\r\n while (focusedNode != inputs[editableElementIndex]) {\r\n\r\n editableElementIndex ++;\r\n\r\n }\r\n\r\n /**\r\n * Founded contentEditable element doesn't have childs\r\n * Or maybe New created block\r\n */\r\n if (!focusedNode.textContent) {\r\n\r\n editor.caret.setToNextBlock(editableElementIndex);\r\n return;\r\n\r\n }\r\n\r\n /**\r\n * Do nothing when caret doesn not reaches the end of last child\r\n */\r\n var caretInLastChild = false,\r\n caretAtTheEndOfText = false;\r\n\r\n var lastChild,\r\n deepestTextnode;\r\n\r\n lastChild = focusedNode.childNodes[focusedNode.childNodes.length - 1 ];\r\n\r\n if (editor.core.isDomNode(lastChild)) {\r\n\r\n deepestTextnode = editor.content.getDeepestTextNodeFromPosition(lastChild, lastChild.childNodes.length);\r\n\r\n } else {\r\n\r\n deepestTextnode = lastChild;\r\n\r\n }\r\n\r\n caretInLastChild = selection.anchorNode == deepestTextnode;\r\n caretAtTheEndOfText = deepestTextnode.length == selection.anchorOffset;\r\n\r\n if ( !caretInLastChild || !caretAtTheEndOfText ) {\r\n\r\n editor.core.log('arrow [down|right] : caret does not reached the end');\r\n return false;\r\n\r\n }\r\n\r\n editor.caret.setToNextBlock(editableElementIndex);\r\n\r\n };\r\n\r\n /**\r\n * LEFT or UP keydowns on block\r\n *\r\n * @param {Object} event\r\n * @private\r\n *\r\n * watches the selection and gets closest editable element.\r\n * Uses method getDeepestTextNodeFromPosition to get the last node of previous block\r\n * Sets caret if it is contenteditable\r\n *\r\n */\r\n var blockLeftOrUpArrowPressed_ = function (event) {\r\n\r\n var selection = window.getSelection(),\r\n inputs = editor.state.inputs,\r\n focusedNode = selection.anchorNode,\r\n focusedNodeHolder;\r\n\r\n /** Check for caret existance */\r\n if (!focusedNode) {\r\n\r\n return false;\r\n\r\n }\r\n\r\n /**\r\n * LEFT or UP not at the beginning\r\n */\r\n if ( selection.anchorOffset !== 0) {\r\n\r\n return false;\r\n\r\n }\r\n\r\n /** Looking for parent contentEditable block */\r\n while (focusedNode.contentEditable != 'true') {\r\n\r\n focusedNodeHolder = focusedNode.parentNode;\r\n focusedNode = focusedNodeHolder;\r\n\r\n }\r\n\r\n /** Input index in DOM level */\r\n var editableElementIndex = 0;\r\n\r\n while (focusedNode != inputs[editableElementIndex]) {\r\n\r\n editableElementIndex ++;\r\n\r\n }\r\n\r\n /**\r\n * Do nothing if caret is not at the beginning of first child\r\n */\r\n var caretInFirstChild = false,\r\n caretAtTheBeginning = false;\r\n\r\n var firstChild,\r\n deepestTextnode;\r\n\r\n /**\r\n * Founded contentEditable element doesn't have childs\r\n * Or maybe New created block\r\n */\r\n if (!focusedNode.textContent) {\r\n\r\n editor.caret.setToPreviousBlock(editableElementIndex);\r\n return;\r\n\r\n }\r\n\r\n firstChild = focusedNode.childNodes[0];\r\n\r\n if (editor.core.isDomNode(firstChild)) {\r\n\r\n deepestTextnode = editor.content.getDeepestTextNodeFromPosition(firstChild, 0);\r\n\r\n } else {\r\n\r\n deepestTextnode = firstChild;\r\n\r\n }\r\n\r\n caretInFirstChild = selection.anchorNode == deepestTextnode;\r\n caretAtTheBeginning = selection.anchorOffset === 0;\r\n\r\n if ( caretInFirstChild && caretAtTheBeginning ) {\r\n\r\n editor.caret.setToPreviousBlock(editableElementIndex);\r\n\r\n }\r\n\r\n };\r\n\r\n /**\r\n * Handles backspace keydown\r\n *\r\n * @param {Element} block\r\n * @param {Object} event\r\n * @private\r\n *\r\n * @description if block is empty, delete the block and set caret to the previous block\r\n * If block is not empty, try to merge two blocks - current and previous\r\n * But it we try'n to remove first block, then we should set caret to the next block, not previous.\r\n * If we removed the last block, create new one\r\n */\r\n var backspacePressed_ = function (block, event) {\r\n\r\n var currentInputIndex = editor.caret.getCurrentInputIndex(),\r\n range,\r\n selectionLength,\r\n firstLevelBlocksCount;\r\n\r\n if (isNativeInput_(event.target)) {\r\n\r\n /** If input value is empty - remove block */\r\n if (event.target.value.trim() == '') {\r\n\r\n block.remove();\r\n\r\n } else {\r\n\r\n return;\r\n\r\n }\r\n\r\n }\r\n\r\n if (block.textContent.trim()) {\r\n\r\n range = editor.content.getRange();\r\n selectionLength = range.endOffset - range.startOffset;\r\n\r\n if (editor.caret.position.atStart() && !selectionLength && editor.state.inputs[currentInputIndex - 1]) {\r\n\r\n editor.content.mergeBlocks(currentInputIndex);\r\n\r\n } else {\r\n\r\n return;\r\n\r\n }\r\n\r\n }\r\n\r\n if (!selectionLength) {\r\n\r\n block.remove();\r\n\r\n }\r\n\r\n\r\n firstLevelBlocksCount = editor.nodes.redactor.childNodes.length;\r\n\r\n /**\r\n * If all blocks are removed\r\n */\r\n if (firstLevelBlocksCount === 0) {\r\n\r\n /** update currentNode variable */\r\n editor.content.currentNode = null;\r\n\r\n /** Inserting new empty initial block */\r\n editor.ui.addInitialBlock();\r\n\r\n /** Updating inputs state after deleting last block */\r\n editor.ui.saveInputs();\r\n\r\n /** Set to current appended block */\r\n window.setTimeout(function () {\r\n\r\n editor.caret.setToPreviousBlock(1);\r\n\r\n }, 10);\r\n\r\n } else {\r\n\r\n if (editor.caret.inputIndex !== 0) {\r\n\r\n /** Target block is not first */\r\n editor.caret.setToPreviousBlock(editor.caret.inputIndex);\r\n\r\n } else {\r\n\r\n /** If we try to delete first block */\r\n editor.caret.setToNextBlock(editor.caret.inputIndex);\r\n\r\n }\r\n\r\n }\r\n\r\n editor.toolbar.move();\r\n\r\n if (!editor.toolbar.opened) {\r\n\r\n editor.toolbar.open();\r\n\r\n }\r\n\r\n /** Updating inputs state */\r\n editor.ui.saveInputs();\r\n\r\n /** Prevent default browser behaviour */\r\n event.preventDefault();\r\n\r\n };\r\n\r\n /**\r\n * This method prevents default behaviour.\r\n *\r\n * @param {Object} event\r\n * @protected\r\n *\r\n * @description We get from clipboard pasted data, sanitize, make a fragment that contains of this sanitized nodes.\r\n * Firstly, we need to memorize the caret position. We can do that by getting the range of selection.\r\n * After all, we insert clear fragment into caret placed position. Then, we should move the caret to the last node\r\n */\r\n callbacks.blockPasteCallback = function (event) {\r\n\r\n /** If area is input or textarea then allow default behaviour */\r\n if ( isNativeInput_(event.target) ) {\r\n\r\n return;\r\n\r\n }\r\n\r\n /** Prevent default behaviour */\r\n event.preventDefault();\r\n\r\n var editableParent = editor.content.getEditableParent(event.target);\r\n\r\n /** Allow paste when event target placed in Editable element */\r\n if (!editableParent) {\r\n\r\n return;\r\n\r\n }\r\n\r\n /** get html pasted data - dirty data */\r\n var htmlData = event.clipboardData.getData('text/html'),\r\n plainData = event.clipboardData.getData('text/plain');\r\n\r\n\r\n /** Temporary DIV that is used to work with childs as arrays item */\r\n var div = editor.draw.node('DIV', '', {}),\r\n cleanData,\r\n wrappedData;\r\n\r\n /** Create fragment, that we paste to range after proccesing */\r\n cleanData = editor.sanitizer.clean(htmlData);\r\n\r\n\r\n /**\r\n * We wrap pasted text with

    tags to split it logically\r\n *\r\n * @type {string}\r\n */\r\n wrappedData = editor.content.wrapTextWithParagraphs(cleanData, plainData);\r\n div.innerHTML = wrappedData;\r\n\r\n var NEW_BLOCK_TYPE = editor.settings.initialBlockPlugin,\r\n currentBlockContent = editor.content.currentNode.firstChild.firstChild;\r\n\r\n /**\r\n * If there only one paragraph, just insert it\r\n */\r\n if (div.childNodes.length == 1) {\r\n\r\n editor.caret.insertNode(document.createTextNode(div.firstChild.innerHTML));\r\n return;\r\n\r\n }\r\n\r\n div.childNodes.forEach(function (paragraph, index) {\r\n\r\n\r\n /**\r\n * If there was no data in working node, replace it with first paragraph of pasted text\r\n */\r\n if (index == 0 && currentBlockContent.innerHTML.trim() === '') {\r\n\r\n editor.content.switchBlock(editor.content.currentNode, editor.tools[NEW_BLOCK_TYPE].render({\r\n text : paragraph.innerHTML\r\n }), NEW_BLOCK_TYPE);\r\n\r\n return;\r\n\r\n }\r\n\r\n editor.content.insertBlock({\r\n type : NEW_BLOCK_TYPE,\r\n block : editor.tools[NEW_BLOCK_TYPE].render({\r\n text : paragraph.innerHTML\r\n })\r\n });\r\n\r\n editor.caret.inputIndex++;\r\n\r\n });\r\n\r\n editor.caret.setToPreviousBlock(editor.caret.getCurrentInputIndex() + 1);\r\n\r\n\r\n };\r\n\r\n /**\r\n * used by UI module\r\n * Clicks on block settings button\r\n *\r\n * @param {Object} event\r\n * @protected\r\n * @description Opens toolbar settings\r\n */\r\n callbacks.showSettingsButtonClicked = function (event) {\r\n\r\n /**\r\n * Get type of current block\r\n * It uses to append settings from tool.settings property.\r\n * ...\r\n * Type is stored in data-type attribute on block\r\n */\r\n var currentToolType = editor.content.currentNode.dataset.tool;\r\n\r\n editor.toolbar.settings.toggle(currentToolType);\r\n\r\n /** Close toolbox when settings button is active */\r\n editor.toolbar.toolbox.close();\r\n editor.toolbar.settings.hideRemoveActions();\r\n\r\n };\r\n\r\n /**\r\n * Check block\r\n * @param target\r\n * @private\r\n *\r\n * @description Checks target is it native input\r\n */\r\n var isNativeInput_ = function (target) {\r\n\r\n var nativeInputAreas = ['INPUT', 'TEXTAREA'];\r\n\r\n return (nativeInputAreas.indexOf(target.tagName) != -1);\r\n\r\n };\r\n\r\n return callbacks;\r\n\r\n})({});\n\n\n// WEBPACK FOOTER //\n// ./modules/callbacks.js","/**\r\n * Codex Editor Draw module\r\n *\r\n * @author Codex Team\r\n * @version 1.0.\r\n */\r\n\r\nmodule.exports = (function (draw) {\r\n\r\n /**\r\n * Base editor wrapper\r\n */\r\n draw.wrapper = function () {\r\n\r\n var wrapper = document.createElement('div');\r\n\r\n wrapper.className += 'codex-editor';\r\n\r\n return wrapper;\r\n\r\n };\r\n\r\n /**\r\n * Content-editable holder\r\n */\r\n draw.redactor = function () {\r\n\r\n var redactor = document.createElement('div');\r\n\r\n redactor.className += 'ce-redactor';\r\n\r\n return redactor;\r\n\r\n };\r\n\r\n draw.ceBlock = function () {\r\n\r\n var block = document.createElement('DIV');\r\n\r\n block.className += 'ce_block';\r\n\r\n return block;\r\n\r\n };\r\n\r\n /**\r\n * Empty toolbar with toggler\r\n */\r\n draw.toolbar = function () {\r\n\r\n var bar = document.createElement('div');\r\n\r\n bar.className += 'ce-toolbar';\r\n\r\n return bar;\r\n\r\n };\r\n\r\n draw.toolbarContent = function () {\r\n\r\n var wrapper = document.createElement('DIV');\r\n\r\n wrapper.classList.add('ce-toolbar__content');\r\n\r\n return wrapper;\r\n\r\n };\r\n\r\n /**\r\n * Inline toolbar\r\n */\r\n draw.inlineToolbar = function () {\r\n\r\n var bar = document.createElement('DIV');\r\n\r\n bar.className += 'ce-toolbar-inline';\r\n\r\n return bar;\r\n\r\n };\r\n\r\n /**\r\n * Wrapper for inline toobar buttons\r\n */\r\n draw.inlineToolbarButtons = function () {\r\n\r\n var wrapper = document.createElement('DIV');\r\n\r\n wrapper.className += 'ce-toolbar-inline__buttons';\r\n\r\n return wrapper;\r\n\r\n };\r\n\r\n /**\r\n * For some actions\r\n */\r\n draw.inlineToolbarActions = function () {\r\n\r\n var wrapper = document.createElement('DIV');\r\n\r\n wrapper.className += 'ce-toolbar-inline__actions';\r\n\r\n return wrapper;\r\n\r\n };\r\n\r\n draw.inputForLink = function () {\r\n\r\n var input = document.createElement('INPUT');\r\n\r\n input.type = 'input';\r\n input.className += 'inputForLink';\r\n input.placeholder = 'Вставьте ссылку ...';\r\n input.setAttribute('form', 'defaultForm');\r\n\r\n input.setAttribute('autofocus', 'autofocus');\r\n\r\n return input;\r\n\r\n };\r\n\r\n /**\r\n * @todo Desc\r\n */\r\n draw.blockButtons = function () {\r\n\r\n var block = document.createElement('div');\r\n\r\n block.className += 'ce-toolbar__actions';\r\n\r\n return block;\r\n\r\n };\r\n\r\n /**\r\n * Block settings panel\r\n */\r\n draw.blockSettings = function () {\r\n\r\n var settings = document.createElement('div');\r\n\r\n settings.className += 'ce-settings';\r\n\r\n return settings;\r\n\r\n };\r\n\r\n draw.defaultSettings = function () {\r\n\r\n var div = document.createElement('div');\r\n\r\n div.classList.add('ce-settings_default');\r\n\r\n return div;\r\n\r\n };\r\n\r\n draw.pluginsSettings = function () {\r\n\r\n var div = document.createElement('div');\r\n\r\n div.classList.add('ce-settings_plugin');\r\n\r\n return div;\r\n\r\n };\r\n\r\n draw.plusButton = function () {\r\n\r\n var button = document.createElement('span');\r\n\r\n button.className = 'ce-toolbar__plus';\r\n // button.innerHTML = '';\r\n\r\n return button;\r\n\r\n };\r\n\r\n /**\r\n * Settings button in toolbar\r\n */\r\n draw.settingsButton = function () {\r\n\r\n var toggler = document.createElement('span');\r\n\r\n toggler.className = 'ce-toolbar__settings-btn';\r\n\r\n /** Toggler button*/\r\n toggler.innerHTML = '';\r\n\r\n return toggler;\r\n\r\n };\r\n\r\n /**\r\n * Redactor tools wrapper\r\n */\r\n\r\n draw.toolbox = function () {\r\n\r\n var wrapper = document.createElement('div');\r\n\r\n wrapper.className = 'ce-toolbar__tools';\r\n\r\n return wrapper;\r\n\r\n };\r\n\r\n /**\r\n * @protected\r\n *\r\n * Draws tool buttons for toolbox\r\n *\r\n * @param {String} type\r\n * @param {String} classname\r\n * @returns {Element}\r\n */\r\n draw.toolbarButton = function (type, classname) {\r\n\r\n var button = document.createElement('li'),\r\n toolIcon = document.createElement('i'),\r\n toolTitle = document.createElement('span');\r\n\r\n button.dataset.type = type;\r\n button.setAttribute('title', type);\r\n\r\n toolIcon.classList.add(classname);\r\n toolTitle.classList.add('ce_toolbar_tools--title');\r\n\r\n\r\n button.appendChild(toolIcon);\r\n button.appendChild(toolTitle);\r\n\r\n return button;\r\n\r\n };\r\n\r\n /**\r\n * @protected\r\n *\r\n * Draws tools for inline toolbar\r\n *\r\n * @param {String} type\r\n * @param {String} classname\r\n */\r\n draw.toolbarButtonInline = function (type, classname) {\r\n\r\n var button = document.createElement('BUTTON'),\r\n toolIcon = document.createElement('I');\r\n\r\n button.type = 'button';\r\n button.dataset.type = type;\r\n toolIcon.classList.add(classname);\r\n\r\n button.appendChild(toolIcon);\r\n\r\n return button;\r\n\r\n };\r\n\r\n /**\r\n * Redactor block\r\n */\r\n draw.block = function (tagName, content) {\r\n\r\n var node = document.createElement(tagName);\r\n\r\n node.innerHTML = content || '';\r\n\r\n return node;\r\n\r\n };\r\n\r\n /**\r\n * Creates Node with passed tagName and className\r\n * @param {string} tagName\r\n * @param {string} className\r\n * @param {object} properties - allow to assign properties\r\n */\r\n draw.node = function ( tagName, className, properties ) {\r\n\r\n var el = document.createElement( tagName );\r\n\r\n if ( className ) el.className = className;\r\n\r\n if ( properties ) {\r\n\r\n for (var name in properties) {\r\n\r\n el[name] = properties[name];\r\n\r\n }\r\n\r\n }\r\n\r\n return el;\r\n\r\n };\r\n\r\n /**\r\n * Unavailable plugin block\r\n */\r\n draw.unavailableBlock = function () {\r\n\r\n var wrapper = document.createElement('DIV');\r\n\r\n wrapper.classList.add('cdx-unavailable-block');\r\n\r\n return wrapper;\r\n\r\n };\r\n\r\n return draw;\r\n\r\n})({});\n\n\n// WEBPACK FOOTER //\n// ./modules/draw.js","/**\r\n * Codex Editor Caret Module\r\n *\r\n * @author Codex Team\r\n * @version 1.0\r\n */\r\n\r\nmodule.exports = (function (caret) {\r\n\r\n let editor = codex.editor;\r\n\r\n /**\r\n * @var {int} InputIndex - editable element in DOM\r\n */\r\n caret.inputIndex = null;\r\n\r\n /**\r\n * @var {int} offset - caret position in a text node.\r\n */\r\n caret.offset = null;\r\n\r\n /**\r\n * @var {int} focusedNodeIndex - we get index of child node from first-level block\r\n */\r\n caret.focusedNodeIndex = null;\r\n\r\n /**\r\n * Creates Document Range and sets caret to the element.\r\n * @protected\r\n * @uses caret.save — if you need to save caret position\r\n * @param {Element} el - Changed Node.\r\n */\r\n caret.set = function ( el, index, offset) {\r\n\r\n offset = offset || caret.offset || 0;\r\n index = index || caret.focusedNodeIndex || 0;\r\n\r\n var childs = el.childNodes,\r\n nodeToSet;\r\n\r\n if ( childs.length === 0 ) {\r\n\r\n nodeToSet = el;\r\n\r\n } else {\r\n\r\n nodeToSet = childs[index];\r\n\r\n }\r\n\r\n /** If Element is INPUT */\r\n if (el.contentEditable != 'true') {\r\n\r\n el.focus();\r\n return;\r\n\r\n }\r\n\r\n if (editor.core.isDomNode(nodeToSet)) {\r\n\r\n nodeToSet = editor.content.getDeepestTextNodeFromPosition(nodeToSet, nodeToSet.childNodes.length);\r\n\r\n }\r\n\r\n var range = document.createRange(),\r\n selection = window.getSelection();\r\n\r\n window.setTimeout(function () {\r\n\r\n range.setStart(nodeToSet, offset);\r\n range.setEnd(nodeToSet, offset);\r\n\r\n selection.removeAllRanges();\r\n selection.addRange(range);\r\n\r\n editor.caret.saveCurrentInputIndex();\r\n\r\n }, 20);\r\n\r\n };\r\n\r\n /**\r\n * @protected\r\n * Updates index of input and saves it in caret object\r\n */\r\n caret.saveCurrentInputIndex = function () {\r\n\r\n /** Index of Input that we paste sanitized content */\r\n var selection = window.getSelection(),\r\n inputs = editor.state.inputs,\r\n focusedNode = selection.anchorNode,\r\n focusedNodeHolder;\r\n\r\n if (!focusedNode) {\r\n\r\n return;\r\n\r\n }\r\n\r\n /** Looking for parent contentEditable block */\r\n while (focusedNode.contentEditable != 'true') {\r\n\r\n focusedNodeHolder = focusedNode.parentNode;\r\n focusedNode = focusedNodeHolder;\r\n\r\n }\r\n\r\n /** Input index in DOM level */\r\n var editableElementIndex = 0;\r\n\r\n while (focusedNode != inputs[editableElementIndex]) {\r\n\r\n editableElementIndex ++;\r\n\r\n }\r\n\r\n caret.inputIndex = editableElementIndex;\r\n\r\n };\r\n\r\n /**\r\n * Returns current input index (caret object)\r\n */\r\n caret.getCurrentInputIndex = function () {\r\n\r\n return caret.inputIndex;\r\n\r\n };\r\n\r\n /**\r\n * @param {int} index - index of first-level block after that we set caret into next input\r\n */\r\n caret.setToNextBlock = function (index) {\r\n\r\n var inputs = editor.state.inputs,\r\n nextInput = inputs[index + 1];\r\n\r\n if (!nextInput) {\r\n\r\n editor.core.log('We are reached the end');\r\n return;\r\n\r\n }\r\n\r\n /**\r\n * When new Block created or deleted content of input\r\n * We should add some text node to set caret\r\n */\r\n if (!nextInput.childNodes.length) {\r\n\r\n var emptyTextElement = document.createTextNode('');\r\n\r\n nextInput.appendChild(emptyTextElement);\r\n\r\n }\r\n\r\n editor.caret.inputIndex = index + 1;\r\n editor.caret.set(nextInput, 0, 0);\r\n editor.content.workingNodeChanged(nextInput);\r\n\r\n };\r\n\r\n /**\r\n * @param {int} index - index of target input.\r\n * Sets caret to input with this index\r\n */\r\n caret.setToBlock = function (index) {\r\n\r\n var inputs = editor.state.inputs,\r\n targetInput = inputs[index];\r\n\r\n if ( !targetInput ) {\r\n\r\n return;\r\n\r\n }\r\n\r\n /**\r\n * When new Block created or deleted content of input\r\n * We should add some text node to set caret\r\n */\r\n if (!targetInput.childNodes.length) {\r\n\r\n var emptyTextElement = document.createTextNode('');\r\n\r\n targetInput.appendChild(emptyTextElement);\r\n\r\n }\r\n\r\n editor.caret.inputIndex = index;\r\n editor.caret.set(targetInput, 0, 0);\r\n editor.content.workingNodeChanged(targetInput);\r\n\r\n };\r\n\r\n /**\r\n * @param {int} index - index of input\r\n */\r\n caret.setToPreviousBlock = function (index) {\r\n\r\n index = index || 0;\r\n\r\n var inputs = editor.state.inputs,\r\n previousInput = inputs[index - 1],\r\n lastChildNode,\r\n lengthOfLastChildNode,\r\n emptyTextElement;\r\n\r\n\r\n if (!previousInput) {\r\n\r\n editor.core.log('We are reached first node');\r\n return;\r\n\r\n }\r\n\r\n lastChildNode = editor.content.getDeepestTextNodeFromPosition(previousInput, previousInput.childNodes.length);\r\n lengthOfLastChildNode = lastChildNode.length;\r\n\r\n /**\r\n * When new Block created or deleted content of input\r\n * We should add some text node to set caret\r\n */\r\n if (!previousInput.childNodes.length) {\r\n\r\n emptyTextElement = document.createTextNode('');\r\n previousInput.appendChild(emptyTextElement);\r\n\r\n }\r\n editor.caret.inputIndex = index - 1;\r\n editor.caret.set(previousInput, previousInput.childNodes.length - 1, lengthOfLastChildNode);\r\n editor.content.workingNodeChanged(inputs[index - 1]);\r\n\r\n };\r\n\r\n caret.position = {\r\n\r\n atStart : function () {\r\n\r\n var selection = window.getSelection(),\r\n anchorOffset = selection.anchorOffset,\r\n anchorNode = selection.anchorNode,\r\n firstLevelBlock = editor.content.getFirstLevelBlock(anchorNode),\r\n pluginsRender = firstLevelBlock.childNodes[0];\r\n\r\n if (!editor.core.isDomNode(anchorNode)) {\r\n\r\n anchorNode = anchorNode.parentNode;\r\n\r\n }\r\n\r\n var isFirstNode = anchorNode === pluginsRender.childNodes[0],\r\n isOffsetZero = anchorOffset === 0;\r\n\r\n return isFirstNode && isOffsetZero;\r\n\r\n },\r\n\r\n atTheEnd : function () {\r\n\r\n var selection = window.getSelection(),\r\n anchorOffset = selection.anchorOffset,\r\n anchorNode = selection.anchorNode;\r\n\r\n /** Caret is at the end of input */\r\n return !anchorNode || !anchorNode.length || anchorOffset === anchorNode.length;\r\n\r\n }\r\n };\r\n\r\n\r\n /**\r\n * Inserts node at the caret location\r\n * @param node\r\n */\r\n caret.insertNode = function (node) {\r\n\r\n var selection, range;\r\n\r\n selection = window.getSelection();\r\n\r\n range = selection.getRangeAt(0);\r\n range.deleteContents();\r\n\r\n range.insertNode(node);\r\n\r\n range.setStartAfter(node);\r\n range.collapse(true);\r\n\r\n selection.removeAllRanges();\r\n selection.addRange(range);\r\n\r\n\r\n };\r\n\r\n return caret;\r\n\r\n})({});\n\n\n// WEBPACK FOOTER //\n// ./modules/caret.js","/**\r\n * Codex Editor Notification Module\r\n *\r\n * @author Codex Team\r\n * @version 1.0\r\n */\r\n\r\nmodule.exports = (function (notifications) {\r\n\r\n let editor = codex.editor;\r\n\r\n var queue = [];\r\n\r\n var addToQueue = function (settings) {\r\n\r\n queue.push(settings);\r\n\r\n var index = 0;\r\n\r\n while ( index < queue.length && queue.length > 5) {\r\n\r\n if (queue[index].type == 'confirm' || queue[index].type == 'prompt') {\r\n\r\n index++;\r\n continue;\r\n\r\n }\r\n\r\n queue[index].close();\r\n queue.splice(index, 1);\r\n\r\n }\r\n\r\n };\r\n\r\n notifications.createHolder = function () {\r\n\r\n var holder = editor.draw.node('DIV', 'cdx-notifications-block');\r\n\r\n editor.nodes.notifications = document.body.appendChild(holder);\r\n\r\n return holder;\r\n\r\n };\r\n\r\n\r\n /**\r\n * Error notificator. Shows block with message\r\n * @protected\r\n */\r\n notifications.errorThrown = function (errorMsg, event) {\r\n\r\n editor.notifications.notification({message: 'This action is not available currently', type: event.type});\r\n\r\n };\r\n\r\n /**\r\n *\r\n * Appends notification\r\n *\r\n * settings = {\r\n * type - notification type (reserved types: alert, confirm, prompt). Just add class 'cdx-notification-'+type\r\n * message - notification message\r\n * okMsg - confirm button text (default - 'Ok')\r\n * cancelBtn - cancel button text (default - 'Cancel'). Only for confirm and prompt types\r\n * confirm - function-handler for ok button click\r\n * cancel - function-handler for cancel button click. Only for confirm and prompt types\r\n * time - time (in seconds) after which notification will close (default - 10s)\r\n * }\r\n *\r\n * @param settings\r\n */\r\n notifications.notification = function (constructorSettings) {\r\n\r\n /** Private vars and methods */\r\n var notification = null,\r\n cancel = null,\r\n type = null,\r\n confirm = null,\r\n inputField = null;\r\n\r\n var confirmHandler = function () {\r\n\r\n close();\r\n\r\n if (typeof confirm !== 'function' ) {\r\n\r\n return;\r\n\r\n }\r\n\r\n if (type == 'prompt') {\r\n\r\n confirm(inputField.value);\r\n return;\r\n\r\n }\r\n\r\n confirm();\r\n\r\n };\r\n\r\n var cancelHandler = function () {\r\n\r\n close();\r\n\r\n if (typeof cancel !== 'function' ) {\r\n\r\n return;\r\n\r\n }\r\n\r\n cancel();\r\n\r\n };\r\n\r\n\r\n /** Public methods */\r\n function create(settings) {\r\n\r\n if (!(settings && settings.message)) {\r\n\r\n editor.core.log('Can\\'t create notification. Message is missed');\r\n return;\r\n\r\n }\r\n\r\n settings.type = settings.type || 'alert';\r\n settings.time = settings.time*1000 || 10000;\r\n\r\n var wrapper = editor.draw.node('DIV', 'cdx-notification'),\r\n message = editor.draw.node('DIV', 'cdx-notification__message'),\r\n input = editor.draw.node('INPUT', 'cdx-notification__input'),\r\n okBtn = editor.draw.node('SPAN', 'cdx-notification__ok-btn'),\r\n cancelBtn = editor.draw.node('SPAN', 'cdx-notification__cancel-btn');\r\n\r\n message.textContent = settings.message;\r\n okBtn.textContent = settings.okMsg || 'ОК';\r\n cancelBtn.textContent = settings.cancelMsg || 'Отмена';\r\n\r\n editor.listeners.add(okBtn, 'click', confirmHandler);\r\n editor.listeners.add(cancelBtn, 'click', cancelHandler);\r\n\r\n wrapper.appendChild(message);\r\n\r\n if (settings.type == 'prompt') {\r\n\r\n wrapper.appendChild(input);\r\n\r\n }\r\n\r\n wrapper.appendChild(okBtn);\r\n\r\n if (settings.type == 'prompt' || settings.type == 'confirm') {\r\n\r\n wrapper.appendChild(cancelBtn);\r\n\r\n }\r\n\r\n wrapper.classList.add('cdx-notification-' + settings.type);\r\n wrapper.dataset.type = settings.type;\r\n\r\n notification = wrapper;\r\n type = settings.type;\r\n confirm = settings.confirm;\r\n cancel = settings.cancel;\r\n inputField = input;\r\n\r\n if (settings.type != 'prompt' && settings.type != 'confirm') {\r\n\r\n window.setTimeout(close, settings.time);\r\n\r\n }\r\n\r\n };\r\n\r\n function send() {\r\n\r\n editor.nodes.notifications.appendChild(notification);\r\n inputField.focus();\r\n\r\n editor.nodes.notifications.classList.add('cdx-notification__notification-appending');\r\n\r\n window.setTimeout(function () {\r\n\r\n editor.nodes.notifications.classList.remove('cdx-notification__notification-appending');\r\n\r\n }, 100);\r\n\r\n addToQueue({type: type, close: close});\r\n\r\n };\r\n\r\n function close() {\r\n\r\n notification.remove();\r\n\r\n };\r\n\r\n\r\n if (constructorSettings) {\r\n\r\n create(constructorSettings);\r\n send();\r\n\r\n }\r\n\r\n return {\r\n create: create,\r\n send: send,\r\n close: close\r\n };\r\n\r\n };\r\n\r\n notifications.clear = function () {\r\n\r\n editor.nodes.notifications.innerHTML = '';\r\n queue = [];\r\n\r\n };\r\n\r\n return notifications;\r\n\r\n})({});\n\n\n// WEBPACK FOOTER //\n// ./modules/notifications.js","/**\r\n * Codex Editor Parser Module\r\n *\r\n * @author Codex Team\r\n * @version 1.1\r\n */\r\n\r\nmodule.exports = (function (parser) {\r\n\r\n let editor = codex.editor;\r\n\r\n /** inserting text */\r\n parser.insertPastedContent = function (blockType, tag) {\r\n\r\n editor.content.insertBlock({\r\n type : blockType.type,\r\n block : blockType.render({\r\n text : tag.innerHTML\r\n })\r\n });\r\n\r\n };\r\n\r\n /**\r\n * Check DOM node for display style: separated block or child-view\r\n */\r\n parser.isFirstLevelBlock = function (node) {\r\n\r\n return node.nodeType == editor.core.nodeTypes.TAG &&\r\n node.classList.contains(editor.ui.className.BLOCK_CLASSNAME);\r\n\r\n };\r\n\r\n return parser;\r\n\r\n})({});\r\n\n\n\n// WEBPACK FOOTER //\n// ./modules/parser.js","/**\r\n * Codex Sanitizer\r\n */\r\n\r\nmodule.exports = (function (sanitizer) {\r\n\r\n /** HTML Janitor library */\r\n let janitor = require('html-janitor');\r\n\r\n /** Codex Editor */\r\n let editor = codex.editor;\r\n\r\n sanitizer.prepare = function () {\r\n\r\n if (editor.settings.sanitizer && !editor.core.isEmpty(editor.settings.sanitizer)) {\r\n\r\n Config.CUSTOM = editor.settings.sanitizer;\r\n\r\n }\r\n\r\n };\r\n\r\n /**\r\n * Basic config\r\n */\r\n var Config = {\r\n\r\n /** User configuration */\r\n CUSTOM : null,\r\n\r\n BASIC : {\r\n\r\n tags: {\r\n p: {},\r\n a: {\r\n href: true,\r\n target: '_blank',\r\n rel: 'nofollow'\r\n },\r\n i: {},\r\n b: {},\r\n strong: {},\r\n em: {},\r\n span: {}\r\n }\r\n }\r\n };\r\n\r\n sanitizer.Config = Config;\r\n\r\n /**\r\n *\r\n * @param userCustomConfig\r\n * @returns {*}\r\n * @private\r\n *\r\n * @description If developer uses editor's API, then he can customize sane restrictions.\r\n * Or, sane config can be defined globally in editors initialization. That config will be used everywhere\r\n * At least, if there is no config overrides, that API uses BASIC Default configation\r\n */\r\n let init_ = function (userCustomConfig) {\r\n\r\n let configuration = userCustomConfig || Config.CUSTOM || Config.BASIC;\r\n\r\n return new janitor(configuration);\r\n\r\n };\r\n\r\n /**\r\n * Cleans string from unwanted tags\r\n * @protected\r\n * @param {String} dirtyString - taint string\r\n * @param {Object} customConfig - allowed tags\r\n */\r\n sanitizer.clean = function (dirtyString, customConfig) {\r\n\r\n let janitorInstance = init_(customConfig);\r\n\r\n return janitorInstance.clean(dirtyString);\r\n\r\n };\r\n\r\n return sanitizer;\r\n\r\n})({});\n\n\n// WEBPACK FOOTER //\n// ./modules/sanitizer.js","(function (root, factory) {\n if (typeof define === 'function' && define.amd) {\n define('html-janitor', factory);\n } else if (typeof exports === 'object') {\n module.exports = factory();\n } else {\n root.HTMLJanitor = factory();\n }\n}(this, function () {\n\n /**\n * @param {Object} config.tags Dictionary of allowed tags.\n * @param {boolean} config.keepNestedBlockElements Default false.\n */\n function HTMLJanitor(config) {\n\n var tagDefinitions = config['tags'];\n var tags = Object.keys(tagDefinitions);\n\n var validConfigValues = tags\n .map(function(k) { return typeof tagDefinitions[k]; })\n .every(function(type) { return type === 'object' || type === 'boolean' || type === 'function'; });\n\n if(!validConfigValues) {\n throw new Error(\"The configuration was invalid\");\n }\n\n this.config = config;\n }\n\n // TODO: not exhaustive?\n var blockElementNames = ['P', 'LI', 'TD', 'TH', 'DIV', 'H1', 'H2', 'H3', 'H4', 'H5', 'H6', 'PRE'];\n function isBlockElement(node) {\n return blockElementNames.indexOf(node.nodeName) !== -1;\n }\n\n var inlineElementNames = ['A', 'B', 'STRONG', 'I', 'EM', 'SUB', 'SUP', 'U', 'STRIKE'];\n function isInlineElement(node) {\n return inlineElementNames.indexOf(node.nodeName) !== -1;\n }\n\n HTMLJanitor.prototype.clean = function (html) {\n var sandbox = document.createElement('div');\n sandbox.innerHTML = html;\n\n this._sanitize(sandbox);\n\n return sandbox.innerHTML;\n };\n\n HTMLJanitor.prototype._sanitize = function (parentNode) {\n var treeWalker = createTreeWalker(parentNode);\n var node = treeWalker.firstChild();\n if (!node) { return; }\n\n do {\n // Ignore nodes that have already been sanitized\n if (node._sanitized) {\n continue;\n }\n\n if (node.nodeType === Node.TEXT_NODE) {\n // If this text node is just whitespace and the previous or next element\n // sibling is a block element, remove it\n // N.B.: This heuristic could change. Very specific to a bug with\n // `contenteditable` in Firefox: http://jsbin.com/EyuKase/1/edit?js,output\n // FIXME: make this an option?\n if (node.data.trim() === ''\n && ((node.previousElementSibling && isBlockElement(node.previousElementSibling))\n || (node.nextElementSibling && isBlockElement(node.nextElementSibling)))) {\n parentNode.removeChild(node);\n this._sanitize(parentNode);\n break;\n } else {\n continue;\n }\n }\n\n // Remove all comments\n if (node.nodeType === Node.COMMENT_NODE) {\n parentNode.removeChild(node);\n this._sanitize(parentNode);\n break;\n }\n\n var isInline = isInlineElement(node);\n var containsBlockElement;\n if (isInline) {\n containsBlockElement = Array.prototype.some.call(node.childNodes, isBlockElement);\n }\n\n // Block elements should not be nested (e.g.

  • ...); if\n // they are, we want to unwrap the inner block element.\n var isNotTopContainer = !! parentNode.parentNode;\n var isNestedBlockElement =\n isBlockElement(parentNode) &&\n isBlockElement(node) &&\n isNotTopContainer;\n\n var nodeName = node.nodeName.toLowerCase();\n\n var allowedAttrs = getAllowedAttrs(this.config, nodeName, node);\n\n var isInvalid = isInline && containsBlockElement;\n\n // Drop tag entirely according to the whitelist *and* if the markup\n // is invalid.\n if (isInvalid || shouldRejectNode(node, allowedAttrs)\n || (!this.config.keepNestedBlockElements && isNestedBlockElement)) {\n // Do not keep the inner text of SCRIPT/STYLE elements.\n if (! (node.nodeName === 'SCRIPT' || node.nodeName === 'STYLE')) {\n while (node.childNodes.length > 0) {\n parentNode.insertBefore(node.childNodes[0], node);\n }\n }\n parentNode.removeChild(node);\n\n this._sanitize(parentNode);\n break;\n }\n\n // Sanitize attributes\n for (var a = 0; a < node.attributes.length; a += 1) {\n var attr = node.attributes[a];\n\n if (shouldRejectAttr(attr, allowedAttrs, node)) {\n node.removeAttribute(attr.name);\n // Shift the array to continue looping.\n a = a - 1;\n }\n }\n\n // Sanitize children\n this._sanitize(node);\n\n // Mark node as sanitized so it's ignored in future runs\n node._sanitized = true;\n } while ((node = treeWalker.nextSibling()));\n };\n\n function createTreeWalker(node) {\n return document.createTreeWalker(node,\n NodeFilter.SHOW_TEXT | NodeFilter.SHOW_ELEMENT | NodeFilter.SHOW_COMMENT,\n null, false);\n }\n\n function getAllowedAttrs(config, nodeName, node){\n if (typeof config.tags[nodeName] === 'function') {\n return config.tags[nodeName](node);\n } else {\n return config.tags[nodeName];\n }\n }\n\n function shouldRejectNode(node, allowedAttrs){\n if (typeof allowedAttrs === 'undefined') {\n return true;\n } else if (typeof allowedAttrs === 'boolean') {\n return !allowedAttrs;\n }\n\n return false;\n }\n\n function shouldRejectAttr(attr, allowedAttrs, node){\n var attrName = attr.name.toLowerCase();\n\n if (allowedAttrs === true){\n return false;\n } else if (typeof allowedAttrs[attrName] === 'function'){\n return !allowedAttrs[attrName](attr.value, node);\n } else if (typeof allowedAttrs[attrName] === 'undefined'){\n return true;\n } else if (allowedAttrs[attrName] === false) {\n return true;\n } else if (typeof allowedAttrs[attrName] === 'string') {\n return (allowedAttrs[attrName] !== attr.value);\n }\n\n return false;\n }\n\n return HTMLJanitor;\n\n}));\n\n\n\n//////////////////\n// WEBPACK FOOTER\n// ./~/html-janitor/src/html-janitor.js\n// module id = 18\n// module chunks = 0","/**\r\n * Codex Editor Listeners module\r\n *\r\n * @author Codex Team\r\n * @version 1.0\r\n */\r\n\r\n/**\r\n * Module-decorator for event listeners assignment\r\n */\r\nmodule.exports = function (listeners) {\r\n\r\n var allListeners = [];\r\n\r\n /**\r\n * Search methods\r\n *\r\n * byElement, byType and byHandler returns array of suitable listeners\r\n * one and all takes element, eventType, and handler and returns first (all) suitable listener\r\n *\r\n */\r\n listeners.search = function () {\r\n\r\n var byElement = function (element, context) {\r\n\r\n var listenersOnElement = [];\r\n\r\n context = context || allListeners;\r\n\r\n for (var i = 0; i < context.length; i++) {\r\n\r\n var listener = context[i];\r\n\r\n if (listener.element === element) {\r\n\r\n listenersOnElement.push(listener);\r\n\r\n }\r\n\r\n }\r\n\r\n return listenersOnElement;\r\n\r\n };\r\n\r\n var byType = function (eventType, context) {\r\n\r\n var listenersWithType = [];\r\n\r\n context = context || allListeners;\r\n\r\n for (var i = 0; i < context.length; i++) {\r\n\r\n var listener = context[i];\r\n\r\n if (listener.type === eventType) {\r\n\r\n listenersWithType.push(listener);\r\n\r\n }\r\n\r\n }\r\n\r\n return listenersWithType;\r\n\r\n };\r\n\r\n var byHandler = function (handler, context) {\r\n\r\n var listenersWithHandler = [];\r\n\r\n context = context || allListeners;\r\n\r\n for (var i = 0; i < context.length; i++) {\r\n\r\n var listener = context[i];\r\n\r\n if (listener.handler === handler) {\r\n\r\n listenersWithHandler.push(listener);\r\n\r\n }\r\n\r\n }\r\n\r\n return listenersWithHandler;\r\n\r\n };\r\n\r\n var one = function (element, eventType, handler) {\r\n\r\n var result = allListeners;\r\n\r\n if (element)\r\n result = byElement(element, result);\r\n\r\n if (eventType)\r\n result = byType(eventType, result);\r\n\r\n if (handler)\r\n result = byHandler(handler, result);\r\n\r\n return result[0];\r\n\r\n };\r\n\r\n var all = function (element, eventType, handler) {\r\n\r\n var result = allListeners;\r\n\r\n if (element)\r\n result = byElement(element, result);\r\n\r\n if (eventType)\r\n result = byType(eventType, result);\r\n\r\n if (handler)\r\n result = byHandler(handler, result);\r\n\r\n return result;\r\n\r\n };\r\n\r\n return {\r\n byElement : byElement,\r\n byType : byType,\r\n byHandler : byHandler,\r\n one : one,\r\n all : all\r\n };\r\n\r\n }();\r\n\r\n listeners.add = function (element, eventType, handler, isCapture) {\r\n\r\n element.addEventListener(eventType, handler, isCapture);\r\n\r\n var data = {\r\n element: element,\r\n type: eventType,\r\n handler: handler\r\n };\r\n\r\n var alreadyAddedListener = listeners.search.one(element, eventType, handler);\r\n\r\n if (!alreadyAddedListener) {\r\n\r\n allListeners.push(data);\r\n\r\n }\r\n\r\n };\r\n\r\n listeners.remove = function (element, eventType, handler) {\r\n\r\n element.removeEventListener(eventType, handler);\r\n\r\n var existingListeners = listeners.search.all(element, eventType, handler);\r\n\r\n for (var i = 0; i < existingListeners.length; i++) {\r\n\r\n var index = allListeners.indexOf(existingListeners[i]);\r\n\r\n if (index > 0) {\r\n\r\n allListeners.splice(index, 1);\r\n\r\n }\r\n\r\n }\r\n\r\n };\r\n\r\n listeners.removeAll = function () {\r\n\r\n allListeners.map(function (current) {\r\n\r\n listeners.remove(current.element, current.type, current.handler);\r\n\r\n });\r\n\r\n };\r\n\r\n listeners.get = function (element, eventType, handler) {\r\n\r\n return listeners.search.all(element, eventType, handler);\r\n\r\n };\r\n\r\n return listeners;\r\n\r\n}({});\n\n\n// WEBPACK FOOTER //\n// ./modules/listeners.js","/**\r\n * Codex Editor Destroyer module\r\n *\r\n * @auhor Codex Team\r\n * @version 1.0\r\n */\r\n\r\nmodule.exports = function (destroyer) {\r\n\r\n let editor = codex.editor;\r\n\r\n destroyer.removeNodes = function () {\r\n\r\n editor.nodes.wrapper.remove();\r\n editor.nodes.notifications.remove();\r\n\r\n };\r\n\r\n destroyer.destroyPlugins = function () {\r\n\r\n for (var tool in editor.tools) {\r\n\r\n if (typeof editor.tools[tool].destroy === 'function') {\r\n\r\n editor.tools[tool].destroy();\r\n\r\n }\r\n\r\n }\r\n\r\n };\r\n\r\n destroyer.destroyScripts = function () {\r\n\r\n var scripts = document.getElementsByTagName('SCRIPT');\r\n\r\n for (var i = 0; i < scripts.length; i++) {\r\n\r\n if (scripts[i].id.indexOf(editor.scriptPrefix) + 1) {\r\n\r\n scripts[i].remove();\r\n i--;\r\n\r\n }\r\n\r\n }\r\n\r\n };\r\n\r\n\r\n /**\r\n * Delete editor data from webpage.\r\n * You should send settings argument with boolean flags:\r\n * @param settings.ui- remove redactor event listeners and DOM nodes\r\n * @param settings.scripts - remove redactor scripts from DOM\r\n * @param settings.plugins - remove plugin's objects\r\n * @param settings.core - remove editor core. You can remove core only if UI and scripts flags is true\r\n * }\r\n *\r\n */\r\n destroyer.destroy = function (settings) {\r\n\r\n if (!settings || typeof settings !== 'object') {\r\n\r\n return;\r\n\r\n }\r\n\r\n if (settings.ui) {\r\n\r\n destroyer.removeNodes();\r\n editor.listeners.removeAll();\r\n\r\n }\r\n\r\n if (settings.scripts) {\r\n\r\n destroyer.destroyScripts();\r\n\r\n }\r\n\r\n if (settings.plugins) {\r\n\r\n destroyer.destroyPlugins();\r\n\r\n }\r\n\r\n if (settings.ui && settings.scripts && settings.core) {\r\n\r\n delete codex.editor;\r\n\r\n }\r\n\r\n };\r\n\r\n return destroyer;\r\n\r\n}({});\n\n\n// WEBPACK FOOTER //\n// ./modules/destroyer.js","/**\r\n * Codex Editor Paste module\r\n *\r\n * @author Codex Team\r\n * @version 1.1.1\r\n */\r\n\r\nmodule.exports = function (paste) {\r\n\r\n let editor = codex.editor;\r\n\r\n var patterns = [];\r\n\r\n paste.prepare = function () {\r\n\r\n var tools = editor.tools;\r\n\r\n for (var tool in tools) {\r\n\r\n if (!tools[tool].renderOnPastePatterns || !Array.isArray(tools[tool].renderOnPastePatterns)) {\r\n\r\n continue;\r\n\r\n }\r\n\r\n tools[tool].renderOnPastePatterns.map(function (pattern) {\r\n\r\n\r\n patterns.push(pattern);\r\n\r\n });\r\n\r\n }\r\n\r\n return Promise.resolve();\r\n\r\n };\r\n\r\n /**\r\n * Saves data\r\n * @param event\r\n */\r\n paste.pasted = function (event) {\r\n\r\n var clipBoardData = event.clipboardData || window.clipboardData,\r\n content = clipBoardData.getData('Text');\r\n\r\n var result = analize(content);\r\n\r\n if (result) {\r\n\r\n event.preventDefault();\r\n event.stopImmediatePropagation();\r\n\r\n }\r\n\r\n return result;\r\n\r\n };\r\n\r\n /**\r\n * Analizes pated string and calls necessary method\r\n */\r\n\r\n var analize = function (string) {\r\n\r\n var result = false,\r\n content = editor.content.currentNode,\r\n plugin = content.dataset.tool;\r\n\r\n patterns.map( function (pattern) {\r\n\r\n var execArray = pattern.regex.exec(string),\r\n match = execArray && execArray[0];\r\n\r\n if ( match && match === string.trim()) {\r\n\r\n /** current block is not empty */\r\n if ( content.textContent.trim() && plugin == editor.settings.initialBlockPlugin ) {\r\n\r\n pasteToNewBlock_();\r\n\r\n }\r\n\r\n pattern.callback(string, pattern);\r\n result = true;\r\n\r\n }\r\n\r\n });\r\n\r\n return result;\r\n\r\n };\r\n\r\n var pasteToNewBlock_ = function () {\r\n\r\n /** Create new initial block */\r\n editor.content.insertBlock({\r\n\r\n type : editor.settings.initialBlockPlugin,\r\n block : editor.tools[editor.settings.initialBlockPlugin].render({\r\n text : ''\r\n })\r\n\r\n }, false);\r\n\r\n };\r\n\r\n return paste;\r\n\r\n}({});\n\n\n// WEBPACK FOOTER //\n// ./modules/paste.js"],"sourceRoot":""} \ No newline at end of file +{"version":3,"sources":["webpack:///codex-editor.js","webpack:///webpack/bootstrap 4b9c333b21a67902b259","webpack:///./codex.js","webpack:///./modules/core.js","webpack:///./modules/tools.js","webpack:///./modules/ui.js","webpack:///./modules/transport.js","webpack:///./modules/renderer.js","webpack:///./modules/saver.js","webpack:///./modules/content.js","webpack:///./modules/toolbar/toolbar.js","webpack:///./modules/toolbar/settings.js","webpack:///./modules/toolbar/inline.js","webpack:///./modules/toolbar/toolbox.js","webpack:///./modules/callbacks.js","webpack:///./modules/draw.js","webpack:///./modules/caret.js","webpack:///./modules/notifications.js","webpack:///./modules/parser.js","webpack:///./modules/sanitizer.js","webpack:///./~/html-janitor/src/html-janitor.js","webpack:///./modules/listeners.js","webpack:///./modules/destroyer.js","webpack:///./modules/paste.js"],"names":["codex","modules","__webpack_require__","moduleId","installedModules","exports","module","id","loaded","call","m","c","p","editor","version","scriptPrefix","init","core","tools","ui","transport","renderer","saver","content","toolbar","callback","draw","caret","notifications","parser","sanitizer","listeners","destroyer","paste","settings","holderId","initialBlockPlugin","nodes","holder","wrapper","inlineToolbar","buttons","actions","toolbox","plusButton","showSettingsButton","showTrashButton","blockSettings","pluginSettings","defaultSettings","toolbarButtons","redactor","state","jsonOutput","blocks","inputs","start","userSettings","prepare","then","makeBlocksFromData","saveInputs","catch","error","log","_typeof","Symbol","iterator","obj","constructor","prototype","Promise","resolve","reject","data","hideToolbar","document","getElementById","undefined","Error","msg","type","arg","window","console","e","insertAfter","target","element","parentNode","insertBefore","nextSibling","nodeTypes","TAG","TEXT","COMMENT","keys","BACKSPACE","TAB","ENTER","SHIFT","CTRL","ALT","ESC","SPACE","LEFT","UP","DOWN","RIGHT","DELETE","META","isDomNode","el","nodeType","this","isEmpty","Object","length","ajax","url","encodedString","isFormData","prop","XMLHTTP","XMLHttpRequest","ActiveXObject","async","test","encodeURIComponent","withCredentials","responseContext","beforeSendResult","beforeSend","open","isFormData_","setRequestHeader","progress","upload","onprogress","bind","onreadystatechange","readyState","status","success","responseText","send","importScript","scriptPath","instanceName","script","createElement","defer","onload","onerror","src","head","appendChild","object","FormData","resolve_","reject_","pluginsRequiresPreparation","allPlugins","pluginName","plugin","push","waitAllPluginsPreparation_","plugins","allPluginsProcessed__","reduce","previousValue","iteration","pluginIsReady__","callPluginsPrepareMethod_","available","loadingMessage","config","className","BLOCK_CLASSNAME","BLOCK_CONTENT","BLOCK_STRETCHED","BLOCK_HIGHLIGHTED","SETTINGS_ITEM","makeToolBar_","addTools_","makeInlineToolbar_","addInlineToolbarTools_","makeNotificationHolder_","bindEvents_","container","inlineToolbarButtons","inlineToolbarActions","blockButtons","makeToolbarSettings_","toolbarContent","makeToolbarContent_","settingsButton","makeRemoveBlockButton","pluginsSettings","createHolder","tool","toolName","toolButton","iconClassname","displayInToolbox","render","toolbarButton","bold","icon","command","italic","underline","link","name","toolbarButtonInline","setInlineToolbarButtonBehaviour","add","globalKeydown","redactorKeyDown","globalKeyup","redactorClicked","plusButtonClicked","showSettingsButtonClicked","button","toolbarButtonClicked","addBlockHandlers","block","blockKeydown","blockPasteCallback","inline","show","querySelectorAll","Array","map","current","addInitialBlock","initialBlock","initialBlockType","setAttribute","insertBlock","workingNodeChanged","event","toolClicked","currentRequest","input","arguments","node","fileSelected","clearInput","i","files","formData","multiple","append","selectAndUpload","args","accept","click","abort","items","appendBlocks","nodeSequence","index","appendNodeAtIndex","getNodeAsync","createBlockFromData","blockData","blocksList","position","toolData","unavailableBlock","innerHTML","dataset","inputPosition","stretched","isStretched","saveBlocks","html","childNodes","makeQueue","queue","getBlockData","makeFormDataFromBlocks","save","savedData","output","blockContent","pluginsContent","validate","result","currentNode","editorAreaHightlighted","sync","markBlock","classList","clearMark","remove","getFirstLevelBlock","body","contains","targetNode","replaceBlock","targetBlock","newBlock","replaceChild","needPlaceCaret","workingBlock","newBlockContent","blockType","composeNewBlock_","currentInputIndex","getCurrentInputIndex","editableElement","querySelector","emptyText","createTextNode","set","move","showPlusButton","setTimeout","setToNextBlock","switchBlock","blockToReplace","newBlockComposed","getDeepestTextNodeFromPosition","text","blockChilds","textContent","trim","removeChild","lookingFromStart","getRange","selection","getSelection","getRangeAt","splitBlock","inputIndex","textBeforeCaret","textNodeBeforeCaret","textAfterCaret","textNodeAfterCaret","anchorNode","anchorNodeText","caretOffset","anchorOffset","currentBlock","substring","previousChilds","nextChilds","reachedCurrent","child","previousChildsLength","nextChildsLength","newNode","NEW_BLOCK_TYPE","mergeBlocks","targetInputIndex","targetInput","currentInputContent","isLastNode","allChecked","allSiblingsEmpty_","sibling","wrapTextWithParagraphs","htmlData","plainData","split","join","paragraph","blockTyped","newWrapper","firstLevelBlocks","indexOf","tagName","cloneNode","getEditableParent","contentEditable","defaultToolbarHeight","defaultOffset","opened","close","toggle","hidePlusButton","newYCoordinate","offsetTop","style","transform","Math","floor","hideRemoveActions","setting","toolType","makeSettings","settingsBlock","removeBlockWrapper","settingButton","actionWrapper","confirmAction","cancelAction","removeButtonClicked","confirmRemovingRequest","cancelRemovingRequest","action","showRemoveActions","firstLevelBlocksCount","buttonsOpened","actionsOpened","wrappersOffset","storedSelection","showInlineToolbar","selectedText","getSelectionText","showButtons","getWrappersOffset","newCoordinateX","newCoordinateY","coords","getSelectionCoords","offsetHeight","x","left","y","scrollY","top","closeButtons","closeAction","createLinkAction","defaultToolAction","forEach","hightlight","offset","getOffset","_x","_y","isNaN","offsetLeft","clientLeft","clientTop","offsetParent","range","sel","createRange","collapse","boundingLeft","boundingTop","rangeCount","cloneRange","getClientRects","rect","toString","showActions","inlineToolbarAnchorInputKeydown_","keyCode","editable","restoreSelection","setAnchor","value","preventDefault","stopImmediatePropagation","clearRange","isActive","isLinkActive","saveSelection","inputForLink","focus","dataType","execCommand","containerEl","preSelectionRange","selectNodeContents","setEnd","startContainer","startOffset","end","savedSel","charIndex","setStart","nextCharIndex","nodeStack","foundStart","stop","pop","removeAllRanges","addRange","queryCommandState","setButtonHighlighted","removeButtonsHighLight","tag","leaf","currentTool","barButtons","nextToolIndex","toolToSelect","visibleTool","appendCallback","UNREPLACEBLE_TOOLS","workingNode","setToBlock","callbacks","enterKeyPressed_","tabKeyPressedOnRedactorsZone_","enterKeyPressedOnRedactorsZone_","escapeKeyPressedOnRedactorsZone_","defaultKeyPressedOnRedactorsZone_","arrowKeyPressed_","nativeInputs","nativeInputsAreEmpty","textContentIsEmpty","blockIsEmpty","enterPressedOnBlock_","saveCurrentInputIndex","isEnterPressedOnToolbar","enableLineBreaks","stopPropagation","shiftKey","currentSelection","currentSelectedNode","caretAtTheEndOfText","atTheEnd","isTextNodeHasParentBetweenContenteditable","enterPressedOnBlock","islastNode","detectWhenClickedOnFirstLevelBlockArea_","firstLevelBlock","indexOfLastInput","inputIsEmpty","currentNodeType","isInitialType","flag","blockRightOrDownArrowPressed_","backspacePressed_","blockLeftOrUpArrowPressed_","focusedNodeHolder","focusedNode","editableElementIndex","lastChild","deepestTextnode","caretInLastChild","firstChild","caretInFirstChild","caretAtTheBeginning","setToPreviousBlock","selectionLength","isNativeInput_","endOffset","atStart","editableParent","cleanData","wrappedData","clipboardData","getData","div","clean","currentBlockContent","insertNode","currentToolType","nativeInputAreas","ceBlock","bar","placeholder","toggler","classname","toolIcon","toolTitle","properties","focusedNodeIndex","nodeToSet","childs","nextInput","emptyTextElement","lastChildNode","lengthOfLastChildNode","previousInput","pluginsRender","isFirstNode","isOffsetZero","deleteContents","setStartAfter","addToQueue","splice","errorThrown","errorMsg","notification","message","constructorSettings","create","time","okBtn","cancelBtn","okMsg","cancelMsg","confirmHandler","cancelHandler","confirm","cancel","inputField","clear","insertPastedContent","isFirstLevelBlock","janitor","Config","CUSTOM","BASIC","tags","a","href","rel","b","strong","em","span","init_","userCustomConfig","configuration","dirtyString","customConfig","janitorInstance","__WEBPACK_AMD_DEFINE_FACTORY__","__WEBPACK_AMD_DEFINE_RESULT__","root","factory","HTMLJanitor","tagDefinitions","validConfigValues","k","every","isBlockElement","blockElementNames","nodeName","isInlineElement","inlineElementNames","createTreeWalker","NodeFilter","SHOW_TEXT","SHOW_ELEMENT","SHOW_COMMENT","getAllowedAttrs","shouldRejectNode","allowedAttrs","shouldRejectAttr","attr","attrName","toLowerCase","sandbox","_sanitize","treeWalker","_sanitized","Node","TEXT_NODE","COMMENT_NODE","containsBlockElement","isInline","some","isNotTopContainer","isNestedBlockElement","isInvalid","keepNestedBlockElements","attributes","removeAttribute","previousElementSibling","nextElementSibling","allListeners","search","byElement","context","listenersOnElement","listener","byType","eventType","listenersWithType","byHandler","handler","listenersWithHandler","one","all","isCapture","addEventListener","alreadyAddedListener","removeEventListener","existingListeners","removeAll","get","removeNodes","destroyPlugins","destroy","destroyScripts","scripts","getElementsByTagName","patterns","renderOnPastePatterns","isArray","pattern","pasted","clipBoardData","analize","string","execArray","regex","exec","match","pasteToNewBlock_"],"mappings":"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","file":"codex-editor.js","sourcesContent":["var codex = codex || {}; codex[\"editor\"] =\n/******/ (function(modules) { // webpackBootstrap\n/******/ \t// The module cache\n/******/ \tvar installedModules = {};\n/******/\n/******/ \t// The require function\n/******/ \tfunction __webpack_require__(moduleId) {\n/******/\n/******/ \t\t// Check if module is in cache\n/******/ \t\tif(installedModules[moduleId])\n/******/ \t\t\treturn installedModules[moduleId].exports;\n/******/\n/******/ \t\t// Create a new module (and put it into the cache)\n/******/ \t\tvar module = installedModules[moduleId] = {\n/******/ \t\t\texports: {},\n/******/ \t\t\tid: moduleId,\n/******/ \t\t\tloaded: false\n/******/ \t\t};\n/******/\n/******/ \t\t// Execute the module function\n/******/ \t\tmodules[moduleId].call(module.exports, module, module.exports, __webpack_require__);\n/******/\n/******/ \t\t// Flag the module as loaded\n/******/ \t\tmodule.loaded = true;\n/******/\n/******/ \t\t// Return the exports of the module\n/******/ \t\treturn module.exports;\n/******/ \t}\n/******/\n/******/\n/******/ \t// expose the modules object (__webpack_modules__)\n/******/ \t__webpack_require__.m = modules;\n/******/\n/******/ \t// expose the module cache\n/******/ \t__webpack_require__.c = installedModules;\n/******/\n/******/ \t// __webpack_public_path__\n/******/ \t__webpack_require__.p = \"\";\n/******/\n/******/ \t// Load entry module and return exports\n/******/ \treturn __webpack_require__(0);\n/******/ })\n/************************************************************************/\n/******/ ([\n/* 0 */\n/***/ function(module, exports, __webpack_require__) {\n\n\t'use strict';\n\t\n\t/**\r\n\t *\r\n\t * Codex Editor\r\n\t *\r\n\t * @author Codex Team\r\n\t */\n\t\n\tmodule.exports = function (editor) {\n\t\n\t 'use strict';\n\t\n\t editor.version = (\"1.6.4\");\n\t editor.scriptPrefix = 'cdx-script-';\n\t\n\t var init = function init() {\n\t\n\t editor.core = __webpack_require__(1);\n\t editor.tools = __webpack_require__(2);\n\t editor.ui = __webpack_require__(3);\n\t editor.transport = __webpack_require__(4);\n\t editor.renderer = __webpack_require__(5);\n\t editor.saver = __webpack_require__(6);\n\t editor.content = __webpack_require__(7);\n\t editor.toolbar = __webpack_require__(8);\n\t editor.callback = __webpack_require__(12);\n\t editor.draw = __webpack_require__(13);\n\t editor.caret = __webpack_require__(14);\n\t editor.notifications = __webpack_require__(15);\n\t editor.parser = __webpack_require__(16);\n\t editor.sanitizer = __webpack_require__(17);\n\t editor.listeners = __webpack_require__(19);\n\t editor.destroyer = __webpack_require__(20);\n\t editor.paste = __webpack_require__(21);\n\t };\n\t\n\t /**\r\n\t * @public\r\n\t * holds initial settings\r\n\t */\n\t editor.settings = {\n\t tools: ['paragraph', 'header', 'picture', 'list', 'quote', 'code', 'twitter', 'instagram', 'smile'],\n\t holderId: 'codex-editor',\n\t\n\t // Type of block showing on empty editor\n\t initialBlockPlugin: 'paragraph'\n\t };\n\t\n\t /**\r\n\t * public\r\n\t *\r\n\t * Static nodes\r\n\t */\n\t editor.nodes = {\n\t holder: null,\n\t wrapper: null,\n\t toolbar: null,\n\t inlineToolbar: {\n\t wrapper: null,\n\t buttons: null,\n\t actions: null\n\t },\n\t toolbox: null,\n\t notifications: null,\n\t plusButton: null,\n\t showSettingsButton: null,\n\t showTrashButton: null,\n\t blockSettings: null,\n\t pluginSettings: null,\n\t defaultSettings: null,\n\t toolbarButtons: {}, // { type : DomEl, ... }\n\t redactor: null\n\t };\n\t\n\t /**\r\n\t * @public\r\n\t *\r\n\t * Output state\r\n\t */\n\t editor.state = {\n\t jsonOutput: [],\n\t blocks: [],\n\t inputs: []\n\t };\n\t\n\t /**\r\n\t * @public\r\n\t * Editor plugins\r\n\t */\n\t editor.tools = {};\n\t\n\t /**\r\n\t * Initialization\r\n\t * @uses Promise cEditor.core.prepare\r\n\t * @param {Object} userSettings\r\n\t * @param {Array} userSettings.tools list of plugins\r\n\t * @param {String} userSettings.holderId Element's id to append editor\r\n\t *\r\n\t * Load user defined tools\r\n\t * Tools must contain this important objects :\r\n\t * @param {String} type - this is a type of plugin. It can be used as plugin name\r\n\t * @param {String} iconClassname - this a icon in toolbar\r\n\t * @param {Object} make - what should plugin do, when it is clicked\r\n\t * @param {Object} appendCallback - callback after clicking\r\n\t * @param {Element} settings - what settings does it have\r\n\t * @param {Object} render - plugin get JSON, and should return HTML\r\n\t * @param {Object} save - plugin gets HTML content, returns JSON\r\n\t * @param {Boolean} displayInToolbox - will be displayed in toolbox. Default value is TRUE\r\n\t * @param {Boolean} enableLineBreaks - inserts new block or break lines. Default value is FALSE\r\n\t *\r\n\t * @example\r\n\t * - type : 'header',\r\n\t * - iconClassname : 'ce-icon-header',\r\n\t * - make : headerTool.make,\r\n\t * - appendCallback : headerTool.appendCallback,\r\n\t * - settings : headerTool.makeSettings(),\r\n\t * - render : headerTool.render,\r\n\t * - save : headerTool.save,\r\n\t * - displayInToolbox : true,\r\n\t * - enableLineBreaks : false\r\n\t */\n\t editor.start = function (userSettings) {\n\t\n\t init();\n\t\n\t editor.core.prepare(userSettings)\n\t\n\t // If all ok, make UI, bind events and parse initial-content\n\t .then(editor.ui.prepare).then(editor.tools.prepare).then(editor.sanitizer.prepare).then(editor.paste.prepare).then(editor.transport.prepare).then(editor.renderer.makeBlocksFromData).then(editor.ui.saveInputs).catch(function (error) {\n\t\n\t editor.core.log('Initialization failed with error: %o', 'warn', error);\n\t });\n\t };\n\t\n\t return editor;\n\t}({});\n\n/***/ },\n/* 1 */\n/***/ function(module, exports) {\n\n\t'use strict';\n\t\n\tvar _typeof = typeof Symbol === \"function\" && typeof Symbol.iterator === \"symbol\" ? function (obj) { return typeof obj; } : function (obj) { return obj && typeof Symbol === \"function\" && obj.constructor === Symbol && obj !== Symbol.prototype ? \"symbol\" : typeof obj; };\n\t\n\t/**\r\n\t * Codex Editor Core\r\n\t *\r\n\t * @author Codex Team\r\n\t * @version 1.1.3\r\n\t */\n\t\n\tmodule.exports = function (core) {\n\t\n\t var editor = codex.editor;\n\t\n\t /**\r\n\t * @public\r\n\t *\r\n\t * Editor preparing method\r\n\t * @return Promise\r\n\t */\n\t core.prepare = function (userSettings) {\n\t\n\t return new Promise(function (resolve, reject) {\n\t\n\t if (userSettings) {\n\t\n\t editor.settings.tools = userSettings.tools || editor.settings.tools;\n\t }\n\t\n\t if (userSettings.data) {\n\t\n\t editor.state.blocks = userSettings.data;\n\t }\n\t\n\t if (userSettings.initialBlockPlugin) {\n\t\n\t editor.settings.initialBlockPlugin = userSettings.initialBlockPlugin;\n\t }\n\t\n\t if (userSettings.sanitizer) {\n\t\n\t editor.settings.sanitizer = userSettings.sanitizer;\n\t }\n\t\n\t editor.hideToolbar = userSettings.hideToolbar;\n\t\n\t editor.nodes.holder = document.getElementById(userSettings.holderId || editor.settings.holderId);\n\t\n\t if (_typeof(editor.nodes.holder) === undefined || editor.nodes.holder === null) {\n\t\n\t reject(Error(\"Holder wasn't found by ID: #\" + userSettings.holderId));\n\t } else {\n\t\n\t resolve();\n\t }\n\t });\n\t };\n\t\n\t /**\r\n\t * Logging method\r\n\t * @param type = ['log', 'info', 'warn']\r\n\t */\n\t core.log = function (msg, type, arg) {\n\t\n\t type = type || 'log';\n\t\n\t if (!arg) {\n\t\n\t arg = msg || 'undefined';\n\t msg = '[codex-editor]: %o';\n\t } else {\n\t\n\t msg = '[codex-editor]: ' + msg;\n\t }\n\t\n\t try {\n\t\n\t if ('console' in window && window.console[type]) {\n\t\n\t if (arg) window.console[type](msg, arg);else window.console[type](msg);\n\t }\n\t } catch (e) {}\n\t };\n\t\n\t /**\r\n\t * @protected\r\n\t *\r\n\t * Helper for insert one element after another\r\n\t */\n\t core.insertAfter = function (target, element) {\n\t\n\t target.parentNode.insertBefore(element, target.nextSibling);\n\t };\n\t\n\t /**\r\n\t * @const\r\n\t *\r\n\t * Readable DOM-node types map\r\n\t */\n\t core.nodeTypes = {\n\t TAG: 1,\n\t TEXT: 3,\n\t COMMENT: 8\n\t };\n\t\n\t /**\r\n\t * @const\r\n\t * Readable keys map\r\n\t */\n\t core.keys = { BACKSPACE: 8, TAB: 9, ENTER: 13, SHIFT: 16, CTRL: 17, ALT: 18, ESC: 27, SPACE: 32, LEFT: 37, UP: 38, DOWN: 40, RIGHT: 39, DELETE: 46, META: 91 };\n\t\n\t /**\r\n\t * @protected\r\n\t *\r\n\t * Check object for DOM node\r\n\t */\n\t core.isDomNode = function (el) {\n\t\n\t return el && (typeof el === 'undefined' ? 'undefined' : _typeof(el)) === 'object' && el.nodeType && el.nodeType == this.nodeTypes.TAG;\n\t };\n\t\n\t /**\r\n\t * Checks passed object for emptiness\r\n\t * @require ES5 - Object.keys\r\n\t * @param {object}\r\n\t */\n\t core.isEmpty = function (obj) {\n\t\n\t return Object.keys(obj).length === 0;\n\t };\n\t\n\t /**\r\n\t * Native Ajax\r\n\t * @param {String} settings.url - request URL\r\n\t * @param {function} settings.beforeSend - returned value will be passed as context to the Success, Error and Progress callbacks\r\n\t * @param {function} settings.success\r\n\t * @param {function} settings.progress\r\n\t */\n\t core.ajax = function (settings) {\n\t\n\t if (!settings || !settings.url) {\n\t\n\t return;\n\t }\n\t\n\t var XMLHTTP = window.XMLHttpRequest ? new XMLHttpRequest() : new ActiveXObject('Microsoft.XMLHTTP'),\n\t encodedString,\n\t isFormData,\n\t prop;\n\t\n\t settings.async = true;\n\t settings.type = settings.type || 'GET';\n\t settings.data = settings.data || '';\n\t settings['content-type'] = settings['content-type'] || 'application/json; charset=utf-8';\n\t\n\t if (settings.type == 'GET' && settings.data) {\n\t\n\t settings.url = /\\?/.test(settings.url) ? settings.url + '&' + settings.data : settings.url + '?' + settings.data;\n\t } else {\n\t\n\t encodedString = '';\n\t for (prop in settings.data) {\n\t\n\t encodedString += prop + '=' + encodeURIComponent(settings.data[prop]) + '&';\n\t }\n\t }\n\t\n\t if (settings.withCredentials) {\n\t\n\t XMLHTTP.withCredentials = true;\n\t }\n\t\n\t /**\r\n\t * Value returned in beforeSend funtion will be passed as context to the other response callbacks\r\n\t * If beforeSend returns false, AJAX will be blocked\r\n\t */\n\t var responseContext = void 0,\n\t beforeSendResult = void 0;\n\t\n\t if (typeof settings.beforeSend === 'function') {\n\t\n\t beforeSendResult = settings.beforeSend.call();\n\t\n\t if (beforeSendResult === false) {\n\t\n\t return;\n\t }\n\t }\n\t\n\t XMLHTTP.open(settings.type, settings.url, settings.async);\n\t\n\t /**\r\n\t * If we send FormData, we need no content-type header\r\n\t */\n\t isFormData = isFormData_(settings.data);\n\t\n\t if (!isFormData) {\n\t\n\t if (settings.type !== 'POST') {\n\t\n\t XMLHTTP.setRequestHeader('Content-type', settings['content-type']);\n\t } else {\n\t\n\t XMLHTTP.setRequestHeader('Content-type', 'application/x-www-form-urlencoded');\n\t }\n\t }\n\t\n\t XMLHTTP.setRequestHeader('X-Requested-With', 'XMLHttpRequest');\n\t\n\t responseContext = beforeSendResult || XMLHTTP;\n\t\n\t if (typeof settings.progress === 'function') {\n\t\n\t XMLHTTP.upload.onprogress = settings.progress.bind(responseContext);\n\t }\n\t\n\t XMLHTTP.onreadystatechange = function () {\n\t\n\t if (XMLHTTP.readyState === 4) {\n\t\n\t if (XMLHTTP.status === 200) {\n\t\n\t if (typeof settings.success === 'function') {\n\t\n\t settings.success.call(responseContext, XMLHTTP.responseText);\n\t }\n\t } else {\n\t\n\t if (typeof settings.error === 'function') {\n\t\n\t settings.error.call(responseContext, XMLHTTP.responseText, XMLHTTP.status);\n\t }\n\t }\n\t }\n\t };\n\t\n\t if (isFormData) {\n\t\n\t // Sending FormData\n\t XMLHTTP.send(settings.data);\n\t } else {\n\t\n\t // POST requests\n\t XMLHTTP.send(encodedString);\n\t }\n\t\n\t return XMLHTTP;\n\t };\n\t\n\t /**\r\n\t * Appends script to head of document\r\n\t * @return Promise\r\n\t */\n\t core.importScript = function (scriptPath, instanceName) {\n\t\n\t return new Promise(function (resolve, reject) {\n\t\n\t var script = void 0;\n\t\n\t /** Script is already loaded */\n\t if (!instanceName) {\n\t\n\t reject('Instance name is missed');\n\t } else if (document.getElementById(editor.scriptPrefix + instanceName)) {\n\t\n\t resolve(scriptPath);\n\t }\n\t\n\t script = document.createElement('SCRIPT');\n\t script.async = true;\n\t script.defer = true;\n\t script.id = editor.scriptPrefix + instanceName;\n\t\n\t script.onload = function () {\n\t\n\t resolve(scriptPath);\n\t };\n\t\n\t script.onerror = function () {\n\t\n\t reject(scriptPath);\n\t };\n\t\n\t script.src = scriptPath;\n\t document.head.appendChild(script);\n\t });\n\t };\n\t\n\t /**\r\n\t * Function for checking is it FormData object to send.\r\n\t * @param {Object} object to check\r\n\t * @return boolean\r\n\t */\n\t var isFormData_ = function isFormData_(object) {\n\t\n\t return object instanceof FormData;\n\t };\n\t\n\t return core;\n\t}({});\n\n/***/ },\n/* 2 */\n/***/ function(module, exports) {\n\n\t'use strict';\n\t\n\t/**\r\n\t* Module working with plugins\r\n\t*/\n\tmodule.exports = function () {\n\t\n\t var editor = codex.editor;\n\t\n\t /**\r\n\t * Initialize plugins before using\r\n\t * Ex. Load scripts or call some internal methods\r\n\t * @return Promise\r\n\t */\n\t function prepare() {\n\t\n\t return new Promise(function (resolve_, reject_) {\n\t\n\t Promise.resolve()\n\t\n\t /**\r\n\t * Compose a sequence of plugins that requires preparation\r\n\t */\n\t .then(function () {\n\t\n\t var pluginsRequiresPreparation = [],\n\t allPlugins = editor.tools;\n\t\n\t for (var pluginName in allPlugins) {\n\t\n\t var plugin = allPlugins[pluginName];\n\t\n\t if (plugin.prepare && typeof plugin.prepare != 'function' || !plugin.prepare) {\n\t\n\t continue;\n\t }\n\t\n\t pluginsRequiresPreparation.push(plugin);\n\t }\n\t\n\t /**\r\n\t * If no one passed plugins requires preparation, finish prepare() and go ahead\r\n\t */\n\t if (!pluginsRequiresPreparation.length) {\n\t\n\t resolve_();\n\t }\n\t\n\t return pluginsRequiresPreparation;\n\t })\n\t\n\t /** Wait plugins while they prepares */\n\t .then(waitAllPluginsPreparation_).then(function () {\n\t\n\t editor.core.log('Plugins loaded', 'info');\n\t resolve_();\n\t }).catch(function (error) {\n\t\n\t reject_(error);\n\t });\n\t });\n\t }\n\t\n\t /**\r\n\t * @param {array} plugins - list of tools that requires preparation\r\n\t * @return {Promise} resolved while all plugins will be ready or failed\r\n\t */\n\t function waitAllPluginsPreparation_(plugins) {\n\t\n\t /**\r\n\t * @calls allPluginsProcessed__ when all plugins prepared or failed\r\n\t */\n\t return new Promise(function (allPluginsProcessed__) {\n\t\n\t /**\r\n\t * pluck each element from queue\r\n\t * First, send resolved Promise as previous value\r\n\t * Each plugins \"prepare\" method returns a Promise, that's why\r\n\t * reduce current element will not be able to continue while can't get\r\n\t * a resolved Promise\r\n\t *\r\n\t * If last plugin is \"prepared\" then go to the next stage of initialization\r\n\t */\n\t plugins.reduce(function (previousValue, plugin, iteration) {\n\t\n\t return previousValue.then(function () {\n\t\n\t /**\r\n\t * Wait till plugins prepared\r\n\t * @calls pluginIsReady__ when plugin is ready or failed\r\n\t */\n\t return new Promise(function (pluginIsReady__) {\n\t\n\t callPluginsPrepareMethod_(plugin).then(pluginIsReady__).then(function () {\n\t\n\t plugin.available = true;\n\t }).catch(function (error) {\n\t\n\t editor.core.log('Plugin \\xAB' + plugin.type + '\\xBB was not loaded. Preparation failed because %o', 'warn', error);\n\t plugin.available = false;\n\t plugin.loadingMessage = error;\n\t\n\t /** Go ahead even some plugin has problems */\n\t pluginIsReady__();\n\t }).then(function () {\n\t\n\t /** If last plugin has problems then just ignore and continue */\n\t if (iteration == plugins.length - 1) {\n\t\n\t allPluginsProcessed__();\n\t }\n\t });\n\t });\n\t });\n\t }, Promise.resolve());\n\t });\n\t }\n\t\n\t var callPluginsPrepareMethod_ = function callPluginsPrepareMethod_(plugin) {\n\t\n\t return plugin.prepare(plugin.config || {});\n\t };\n\t\n\t return {\n\t prepare: prepare\n\t };\n\t}();\n\n/***/ },\n/* 3 */\n/***/ function(module, exports) {\n\n\t'use strict';\n\t\n\t/**\r\n\t * Codex Editor UI module\r\n\t *\r\n\t * @author Codex Team\r\n\t * @version 1.2.0\r\n\t */\n\t\n\tmodule.exports = function (ui) {\n\t\n\t var editor = codex.editor;\n\t\n\t /**\r\n\t * Basic editor classnames\r\n\t */\n\t ui.className = {\n\t\n\t /**\r\n\t * @const {string} BLOCK_CLASSNAME - redactor blocks name\r\n\t */\n\t BLOCK_CLASSNAME: 'ce-block',\n\t\n\t /**\r\n\t * @const {String} wrapper for plugins content\r\n\t */\n\t BLOCK_CONTENT: 'ce-block__content',\n\t\n\t /**\r\n\t * @const {String} BLOCK_STRETCHED - makes block stretched\r\n\t */\n\t BLOCK_STRETCHED: 'ce-block--stretched',\n\t\n\t /**\r\n\t * @const {String} BLOCK_HIGHLIGHTED - adds background\r\n\t */\n\t BLOCK_HIGHLIGHTED: 'ce-block--focused',\n\t\n\t /**\r\n\t * @const {String} - for all default settings\r\n\t */\n\t SETTINGS_ITEM: 'ce-settings__item'\n\t\n\t };\n\t\n\t /**\r\n\t * @protected\r\n\t *\r\n\t * Making main interface\r\n\t */\n\t ui.prepare = function () {\n\t\n\t return new Promise(function (resolve) {\n\t\n\t var wrapper = editor.draw.wrapper(),\n\t redactor = editor.draw.redactor(),\n\t toolbar = makeToolBar_();\n\t\n\t wrapper.appendChild(toolbar);\n\t wrapper.appendChild(redactor);\n\t\n\t /** Save created ui-elements to static nodes state */\n\t editor.nodes.wrapper = wrapper;\n\t editor.nodes.redactor = redactor;\n\t\n\t /** Append editor wrapper with redactor zone into holder */\n\t editor.nodes.holder.appendChild(wrapper);\n\t\n\t resolve();\n\t })\n\t\n\t /** Add toolbox tools */\n\t .then(addTools_)\n\t\n\t /** Make container for inline toolbar */\n\t .then(makeInlineToolbar_)\n\t\n\t /** Add inline toolbar tools */\n\t .then(addInlineToolbarTools_)\n\t\n\t /** Draw wrapper for notifications */\n\t .then(makeNotificationHolder_)\n\t\n\t /** Add eventlisteners to redactor elements */\n\t .then(bindEvents_).catch(function () {\n\t\n\t editor.core.log(\"Can't draw editor interface\");\n\t });\n\t };\n\t\n\t /**\r\n\t * @private\r\n\t * Draws inline toolbar zone\r\n\t */\n\t var makeInlineToolbar_ = function makeInlineToolbar_() {\n\t\n\t var container = editor.draw.inlineToolbar();\n\t\n\t /** Append to redactor new inline block */\n\t editor.nodes.inlineToolbar.wrapper = container;\n\t\n\t /** Draw toolbar buttons */\n\t editor.nodes.inlineToolbar.buttons = editor.draw.inlineToolbarButtons();\n\t\n\t /** Buttons action or settings */\n\t editor.nodes.inlineToolbar.actions = editor.draw.inlineToolbarActions();\n\t\n\t /** Append to inline toolbar buttons as part of it */\n\t editor.nodes.inlineToolbar.wrapper.appendChild(editor.nodes.inlineToolbar.buttons);\n\t editor.nodes.inlineToolbar.wrapper.appendChild(editor.nodes.inlineToolbar.actions);\n\t\n\t editor.nodes.wrapper.appendChild(editor.nodes.inlineToolbar.wrapper);\n\t };\n\t\n\t var makeToolBar_ = function makeToolBar_() {\n\t\n\t var toolbar = editor.draw.toolbar(),\n\t blockButtons = makeToolbarSettings_(),\n\t toolbarContent = makeToolbarContent_();\n\t\n\t /** Appending first-level block buttons */\n\t toolbar.appendChild(blockButtons);\n\t\n\t /** Append toolbarContent to toolbar */\n\t toolbar.appendChild(toolbarContent);\n\t\n\t /** Make toolbar global */\n\t editor.nodes.toolbar = toolbar;\n\t\n\t return toolbar;\n\t };\n\t\n\t var makeToolbarContent_ = function makeToolbarContent_() {\n\t\n\t var toolbarContent = editor.draw.toolbarContent(),\n\t toolbox = editor.draw.toolbox(),\n\t plusButton = editor.draw.plusButton();\n\t\n\t /** Append plus button */\n\t toolbarContent.appendChild(plusButton);\n\t\n\t /** Appending toolbar tools */\n\t toolbarContent.appendChild(toolbox);\n\t\n\t /** Make Toolbox and plusButton global */\n\t editor.nodes.toolbox = toolbox;\n\t editor.nodes.plusButton = plusButton;\n\t\n\t return toolbarContent;\n\t };\n\t\n\t var makeToolbarSettings_ = function makeToolbarSettings_() {\n\t\n\t var blockSettings = editor.draw.blockSettings(),\n\t blockButtons = editor.draw.blockButtons(),\n\t defaultSettings = editor.draw.defaultSettings(),\n\t showSettingsButton = editor.draw.settingsButton(),\n\t showTrashButton = editor.toolbar.settings.makeRemoveBlockButton(),\n\t pluginSettings = editor.draw.pluginsSettings();\n\t\n\t /** Add default and plugins settings */\n\t blockSettings.appendChild(pluginSettings);\n\t blockSettings.appendChild(defaultSettings);\n\t\n\t /**\r\n\t * Make blocks buttons\r\n\t * This block contains settings button and remove block button\r\n\t */\n\t blockButtons.appendChild(showSettingsButton);\n\t blockButtons.appendChild(showTrashButton);\n\t blockButtons.appendChild(blockSettings);\n\t\n\t /** Make BlockSettings, PluginSettings, DefaultSettings global */\n\t editor.nodes.blockSettings = blockSettings;\n\t editor.nodes.pluginSettings = pluginSettings;\n\t editor.nodes.defaultSettings = defaultSettings;\n\t editor.nodes.showSettingsButton = showSettingsButton;\n\t editor.nodes.showTrashButton = showTrashButton;\n\t\n\t return blockButtons;\n\t };\n\t\n\t /** Draw notifications holder */\n\t var makeNotificationHolder_ = function makeNotificationHolder_() {\n\t\n\t /** Append block with notifications to the document */\n\t editor.nodes.notifications = editor.notifications.createHolder();\n\t };\n\t\n\t /**\r\n\t * @private\r\n\t * Append tools passed in editor.tools\r\n\t */\n\t var addTools_ = function addTools_() {\n\t\n\t var tool, toolName, toolButton;\n\t\n\t for (toolName in editor.settings.tools) {\n\t\n\t tool = editor.settings.tools[toolName];\n\t\n\t editor.tools[toolName] = tool;\n\t\n\t if (!tool.iconClassname && tool.displayInToolbox) {\n\t\n\t editor.core.log('Toolbar icon classname missed. Tool %o skipped', 'warn', toolName);\n\t continue;\n\t }\n\t\n\t if (typeof tool.render != 'function') {\n\t\n\t editor.core.log('render method missed. Tool %o skipped', 'warn', toolName);\n\t continue;\n\t }\n\t\n\t if (!tool.displayInToolbox) {\n\t\n\t continue;\n\t } else {\n\t\n\t /** if tools is for toolbox */\n\t toolButton = editor.draw.toolbarButton(toolName, tool.iconClassname);\n\t\n\t editor.nodes.toolbox.appendChild(toolButton);\n\t\n\t editor.nodes.toolbarButtons[toolName] = toolButton;\n\t }\n\t }\n\t };\n\t\n\t var addInlineToolbarTools_ = function addInlineToolbarTools_() {\n\t\n\t var tools = {\n\t\n\t bold: {\n\t icon: 'ce-icon-bold',\n\t command: 'bold'\n\t },\n\t\n\t italic: {\n\t icon: 'ce-icon-italic',\n\t command: 'italic'\n\t },\n\t\n\t underline: {\n\t icon: 'ce-icon-underline',\n\t command: 'underline'\n\t },\n\t\n\t link: {\n\t icon: 'ce-icon-link',\n\t command: 'createLink'\n\t }\n\t };\n\t\n\t var toolButton, tool;\n\t\n\t for (var name in tools) {\n\t\n\t tool = tools[name];\n\t\n\t toolButton = editor.draw.toolbarButtonInline(name, tool.icon);\n\t\n\t editor.nodes.inlineToolbar.buttons.appendChild(toolButton);\n\t /**\r\n\t * Add callbacks to this buttons\r\n\t */\n\t editor.ui.setInlineToolbarButtonBehaviour(toolButton, tool.command);\n\t }\n\t };\n\t\n\t /**\r\n\t * @private\r\n\t * Bind editor UI events\r\n\t */\n\t var bindEvents_ = function bindEvents_() {\n\t\n\t editor.core.log('ui.bindEvents fired', 'info');\n\t\n\t // window.addEventListener('error', function (errorMsg, url, lineNumber) {\n\t // editor.notifications.errorThrown(errorMsg, event);\n\t // }, false );\n\t\n\t /** All keydowns on Document */\n\t editor.listeners.add(document, 'keydown', editor.callback.globalKeydown, false);\n\t\n\t /** All keydowns on Redactor zone */\n\t editor.listeners.add(editor.nodes.redactor, 'keydown', editor.callback.redactorKeyDown, false);\n\t\n\t /** All keydowns on Document */\n\t editor.listeners.add(document, 'keyup', editor.callback.globalKeyup, false);\n\t\n\t /**\r\n\t * Mouse click to radactor\r\n\t */\n\t editor.listeners.add(editor.nodes.redactor, 'click', editor.callback.redactorClicked, false);\n\t\n\t /**\r\n\t * Clicks to the Plus button\r\n\t */\n\t editor.listeners.add(editor.nodes.plusButton, 'click', editor.callback.plusButtonClicked, false);\n\t\n\t /**\r\n\t * Clicks to SETTINGS button in toolbar\r\n\t */\n\t editor.listeners.add(editor.nodes.showSettingsButton, 'click', editor.callback.showSettingsButtonClicked, false);\n\t\n\t /** Bind click listeners on toolbar buttons */\n\t for (var button in editor.nodes.toolbarButtons) {\n\t\n\t editor.listeners.add(editor.nodes.toolbarButtons[button], 'click', editor.callback.toolbarButtonClicked, false);\n\t }\n\t };\n\t\n\t ui.addBlockHandlers = function (block) {\n\t\n\t if (!block) return;\n\t\n\t /**\r\n\t * Block keydowns\r\n\t */\n\t editor.listeners.add(block, 'keydown', editor.callback.blockKeydown, false);\n\t\n\t /**\r\n\t * Pasting content from another source\r\n\t * We have two type of sanitization\r\n\t * First - uses deep-first search algorithm to get sub nodes,\r\n\t * sanitizes whole Block_content and replaces cleared nodes\r\n\t * This method is deprecated\r\n\t * Method is used in editor.callback.blockPaste(event)\r\n\t *\r\n\t * Secont - uses Mutation observer.\r\n\t * Observer \"observe\" DOM changes and send changings to callback.\r\n\t * Callback gets changed node, not whole Block_content.\r\n\t * Inserted or changed node, which we've gotten have been cleared and replaced with diry node\r\n\t *\r\n\t * Method is used in editor.callback.blockPasteViaSanitize(event)\r\n\t *\r\n\t * @uses html-janitor\r\n\t * @example editor.callback.blockPasteViaSanitize(event), the second method.\r\n\t *\r\n\t */\n\t editor.listeners.add(block, 'paste', editor.callback.blockPasteCallback, false);\n\t\n\t editor.listeners.add(block, 'mouseup', editor.toolbar.inline.show, false);\n\t };\n\t\n\t /** getting all contenteditable elements */\n\t ui.saveInputs = function () {\n\t\n\t var redactor = editor.nodes.redactor;\n\t\n\t editor.state.inputs = [];\n\t\n\t /** Save all inputs in global variable state */\n\t var inputs = redactor.querySelectorAll('[contenteditable], input, textarea');\n\t\n\t Array.prototype.map.call(inputs, function (current) {\n\t\n\t if (!current.type || current.type == 'text' || current.type == 'textarea') {\n\t\n\t editor.state.inputs.push(current);\n\t }\n\t });\n\t };\n\t\n\t /**\r\n\t * Adds first initial block on empty redactor\r\n\t */\n\t ui.addInitialBlock = function () {\n\t\n\t var initialBlockType = editor.settings.initialBlockPlugin,\n\t initialBlock;\n\t\n\t if (!editor.tools[initialBlockType]) {\n\t\n\t editor.core.log('Plugin %o was not implemented and can\\'t be used as initial block', 'warn', initialBlockType);\n\t return;\n\t }\n\t\n\t initialBlock = editor.tools[initialBlockType].render();\n\t\n\t initialBlock.setAttribute('data-placeholder', 'Расскажите свою историю...');\n\t\n\t editor.content.insertBlock({\n\t type: initialBlockType,\n\t block: initialBlock\n\t });\n\t\n\t editor.content.workingNodeChanged(initialBlock);\n\t };\n\t\n\t ui.setInlineToolbarButtonBehaviour = function (button, type) {\n\t\n\t editor.listeners.add(button, 'mousedown', function (event) {\n\t\n\t editor.toolbar.inline.toolClicked(event, type);\n\t }, false);\n\t };\n\t\n\t return ui;\n\t}({});\n\n/***/ },\n/* 4 */\n/***/ function(module, exports) {\n\n\t'use strict';\n\t\n\t/**\r\n\t *\r\n\t * Codex.Editor Transport Module\r\n\t *\r\n\t * @copyright 2017 Codex-Team\r\n\t * @version 1.2.0\r\n\t */\n\t\n\tmodule.exports = function (transport) {\n\t\n\t var editor = codex.editor;\n\t\n\t /**\r\n\t * @private {Object} current XmlHttpRequest instance\r\n\t */\n\t var currentRequest = null;\n\t\n\t /**\r\n\t * @type {null} | {DOMElement} input - keeps input element in memory\r\n\t */\n\t transport.input = null;\n\t\n\t /**\r\n\t * @property {Object} arguments - keep plugin settings and defined callbacks\r\n\t */\n\t transport.arguments = null;\n\t\n\t /**\r\n\t * Prepares input element where will be files\r\n\t */\n\t transport.prepare = function () {\n\t\n\t var input = editor.draw.node('INPUT', '', { type: 'file' });\n\t\n\t editor.listeners.add(input, 'change', editor.transport.fileSelected);\n\t editor.transport.input = input;\n\t };\n\t\n\t /** Clear input when files is uploaded */\n\t transport.clearInput = function () {\n\t\n\t /** Remove old input */\n\t transport.input = null;\n\t\n\t /** Prepare new one */\n\t transport.prepare();\n\t };\n\t\n\t /**\r\n\t * Callback for file selection\r\n\t * @param {Event} event\r\n\t */\n\t transport.fileSelected = function () {\n\t\n\t var input = this,\n\t i,\n\t files = input.files,\n\t formData = new FormData();\n\t\n\t if (editor.transport.arguments.multiple === true) {\n\t\n\t for (i = 0; i < files.length; i++) {\n\t\n\t formData.append('files[]', files[i], files[i].name);\n\t }\n\t } else {\n\t\n\t formData.append('files', files[0], files[0].name);\n\t }\n\t\n\t currentRequest = editor.core.ajax({\n\t type: 'POST',\n\t data: formData,\n\t url: editor.transport.arguments.url,\n\t beforeSend: editor.transport.arguments.beforeSend,\n\t success: editor.transport.arguments.success,\n\t error: editor.transport.arguments.error,\n\t progress: editor.transport.arguments.progress\n\t });\n\t\n\t /** Clear input */\n\t transport.clearInput();\n\t };\n\t\n\t /**\r\n\t * Use plugin callbacks\r\n\t * @protected\r\n\t *\r\n\t * @param {Object} args - can have :\r\n\t * @param {String} args.url - fetch URL\r\n\t * @param {Function} args.beforeSend - function calls before sending ajax\r\n\t * @param {Function} args.success - success callback\r\n\t * @param {Function} args.error - on error handler\r\n\t * @param {Function} args.progress - xhr onprogress handler\r\n\t * @param {Boolean} args.multiple - allow select several files\r\n\t * @param {String} args.accept - adds accept attribute\r\n\t */\n\t transport.selectAndUpload = function (args) {\n\t\n\t transport.arguments = args;\n\t\n\t if (args.multiple === true) {\n\t\n\t transport.input.setAttribute('multiple', 'multiple');\n\t }\n\t\n\t if (args.accept) {\n\t\n\t transport.input.setAttribute('accept', args.accept);\n\t }\n\t\n\t transport.input.click();\n\t };\n\t\n\t transport.abort = function () {\n\t\n\t currentRequest.abort();\n\t\n\t currentRequest = null;\n\t };\n\t\n\t return transport;\n\t}({});\n\n/***/ },\n/* 5 */\n/***/ function(module, exports) {\n\n\t'use strict';\n\t\n\t/**\r\n\t * Codex Editor Renderer Module\r\n\t *\r\n\t * @author Codex Team\r\n\t * @version 1.0\r\n\t */\n\t\n\tmodule.exports = function (renderer) {\n\t\n\t var editor = codex.editor;\n\t\n\t /**\r\n\t * Asyncronously parses input JSON to redactor blocks\r\n\t */\n\t renderer.makeBlocksFromData = function () {\n\t\n\t /**\r\n\t * If redactor is empty, add first paragraph to start writing\r\n\t */\n\t if (editor.core.isEmpty(editor.state.blocks) || !editor.state.blocks.items.length) {\n\t\n\t editor.ui.addInitialBlock();\n\t return;\n\t }\n\t\n\t Promise.resolve()\n\t\n\t /** First, get JSON from state */\n\t .then(function () {\n\t\n\t return editor.state.blocks;\n\t })\n\t\n\t /** Then, start to iterate they */\n\t .then(editor.renderer.appendBlocks)\n\t\n\t /** Write log if something goes wrong */\n\t .catch(function (error) {\n\t\n\t editor.core.log('Error while parsing JSON: %o', 'error', error);\n\t });\n\t };\n\t\n\t /**\r\n\t * Parses JSON to blocks\r\n\t * @param {object} data\r\n\t * @return Primise -> nodeList\r\n\t */\n\t renderer.appendBlocks = function (data) {\n\t\n\t var blocks = data.items;\n\t\n\t /**\r\n\t * Sequence of one-by-one blocks appending\r\n\t * Uses to save blocks order after async-handler\r\n\t */\n\t var nodeSequence = Promise.resolve();\n\t\n\t for (var index = 0; index < blocks.length; index++) {\n\t\n\t /** Add node to sequence at specified index */\n\t editor.renderer.appendNodeAtIndex(nodeSequence, blocks, index);\n\t }\n\t };\n\t\n\t /**\r\n\t * Append node at specified index\r\n\t */\n\t renderer.appendNodeAtIndex = function (nodeSequence, blocks, index) {\n\t\n\t /** We need to append node to sequence */\n\t nodeSequence\n\t\n\t /** first, get node async-aware */\n\t .then(function () {\n\t\n\t return editor.renderer.getNodeAsync(blocks, index);\n\t })\n\t\n\t /**\r\n\t * second, compose editor-block from JSON object\r\n\t */\n\t .then(editor.renderer.createBlockFromData)\n\t\n\t /**\r\n\t * now insert block to redactor\r\n\t */\n\t .then(function (blockData) {\n\t\n\t /**\r\n\t * blockData has 'block', 'type' and 'stretched' information\r\n\t */\n\t editor.content.insertBlock(blockData);\n\t\n\t /** Pass created block to next step */\n\t return blockData.block;\n\t })\n\t\n\t /** Log if something wrong with node */\n\t .catch(function (error) {\n\t\n\t editor.core.log('Node skipped while parsing because %o', 'error', error);\n\t });\n\t };\n\t\n\t /**\r\n\t * Asynchronously returns block data from blocksList by index\r\n\t * @return Promise to node\r\n\t */\n\t renderer.getNodeAsync = function (blocksList, index) {\n\t\n\t return Promise.resolve().then(function () {\n\t\n\t return {\n\t tool: blocksList[index],\n\t position: index\n\t };\n\t });\n\t };\n\t\n\t /**\r\n\t * Creates editor block by JSON-data\r\n\t *\r\n\t * @uses render method of each plugin\r\n\t *\r\n\t * @param {Object} toolData.tool\r\n\t * { header : {\r\n\t * text: '',\r\n\t * type: 'H3', ...\r\n\t * }\r\n\t * }\r\n\t * @param {Number} toolData.position - index in input-blocks array\r\n\t * @return {Object} with type and Element\r\n\t */\n\t renderer.createBlockFromData = function (toolData) {\n\t\n\t /** New parser */\n\t var block,\n\t tool = toolData.tool,\n\t pluginName = tool.type;\n\t\n\t /** Get first key of object that stores plugin name */\n\t // for (var pluginName in blockData) break;\n\t\n\t /** Check for plugin existance */\n\t if (!editor.tools[pluginName]) {\n\t\n\t throw Error('Plugin \\xAB' + pluginName + '\\xBB not found');\n\t }\n\t\n\t /** Check for plugin having render method */\n\t if (typeof editor.tools[pluginName].render != 'function') {\n\t\n\t throw Error('Plugin \\xAB' + pluginName + '\\xBB must have \\xABrender\\xBB method');\n\t }\n\t\n\t if (editor.tools[pluginName].available === false) {\n\t\n\t block = editor.draw.unavailableBlock();\n\t\n\t block.innerHTML = editor.tools[pluginName].loadingMessage;\n\t\n\t /**\r\n\t * Saver will extract data from initial block data by position in array\r\n\t */\n\t block.dataset.inputPosition = toolData.position;\n\t } else {\n\t\n\t /** New Parser */\n\t block = editor.tools[pluginName].render(tool.data);\n\t }\n\t\n\t /** is first-level block stretched */\n\t var stretched = editor.tools[pluginName].isStretched || false;\n\t\n\t /** Retrun type and block */\n\t return {\n\t type: pluginName,\n\t block: block,\n\t stretched: stretched\n\t };\n\t };\n\t\n\t return renderer;\n\t}({});\n\n/***/ },\n/* 6 */\n/***/ function(module, exports) {\n\n\t'use strict';\n\t\n\t/**\r\n\t * Codex Editor Saver\r\n\t *\r\n\t * @author Codex Team\r\n\t * @version 1.0.2\r\n\t */\n\t\n\tmodule.exports = function (saver) {\n\t\n\t var editor = codex.editor;\n\t\n\t /**\r\n\t * Saves blocks\r\n\t * @private\r\n\t */\n\t saver.saveBlocks = function () {\n\t\n\t /** Save html content of redactor to memory */\n\t editor.state.html = editor.nodes.redactor.innerHTML;\n\t\n\t /** Empty jsonOutput state */\n\t editor.state.jsonOutput = [];\n\t\n\t Promise.resolve().then(function () {\n\t\n\t return editor.nodes.redactor.childNodes;\n\t })\n\t /** Making a sequence from separate blocks */\n\t .then(editor.saver.makeQueue).then(function () {\n\t // editor.nodes.textarea.innerHTML = editor.state.html;\n\t }).catch(function (error) {\n\t\n\t editor.core.log(error);\n\t });\n\t };\n\t\n\t saver.makeQueue = function (blocks) {\n\t\n\t var queue = Promise.resolve();\n\t\n\t for (var index = 0; index < blocks.length; index++) {\n\t\n\t /** Add node to sequence at specified index */\n\t editor.saver.getBlockData(queue, blocks, index);\n\t }\n\t };\n\t\n\t /** Gets every block and makes From Data */\n\t saver.getBlockData = function (queue, blocks, index) {\n\t\n\t queue.then(function () {\n\t\n\t return editor.saver.getNodeAsync(blocks, index);\n\t }).then(editor.saver.makeFormDataFromBlocks);\n\t };\n\t\n\t /**\r\n\t * Asynchronously returns block data from blocksList by index\r\n\t * @return Promise to node\r\n\t */\n\t saver.getNodeAsync = function (blocksList, index) {\n\t\n\t return Promise.resolve().then(function () {\n\t\n\t return blocksList[index];\n\t });\n\t };\n\t\n\t saver.makeFormDataFromBlocks = function (block) {\n\t\n\t var pluginName = block.dataset.tool;\n\t\n\t /** Check for plugin existance */\n\t if (!editor.tools[pluginName]) {\n\t\n\t throw Error('Plugin \\xAB' + pluginName + '\\xBB not found');\n\t }\n\t\n\t /** Check for plugin having render method */\n\t if (typeof editor.tools[pluginName].save != 'function') {\n\t\n\t throw Error('Plugin \\xAB' + pluginName + '\\xBB must have save method');\n\t }\n\t\n\t /** Result saver */\n\t var blockContent = block.childNodes[0],\n\t pluginsContent = blockContent.childNodes[0],\n\t savedData,\n\t position,\n\t output;\n\t\n\t /** If plugin wasn't available then return data from cache */\n\t if (editor.tools[pluginName].available === false) {\n\t\n\t position = pluginsContent.dataset.inputPosition;\n\t savedData = codex.editor.state.blocks.items[position].data;\n\t } else {\n\t\n\t savedData = editor.tools[pluginName].save(pluginsContent);\n\t\n\t if (editor.tools[pluginName].validate) {\n\t\n\t var result = editor.tools[pluginName].validate(savedData);\n\t\n\t /**\r\n\t * Do not allow invalid data\r\n\t */\n\t if (!result) return;\n\t }\n\t }\n\t\n\t output = {\n\t type: pluginName,\n\t data: savedData\n\t };\n\t\n\t editor.state.jsonOutput.push(output);\n\t };\n\t\n\t return saver;\n\t}({});\n\n/***/ },\n/* 7 */\n/***/ function(module, exports) {\n\n\t'use strict';\n\t\n\t/**\r\n\t * Codex Editor Content Module\r\n\t * Works with DOM\r\n\t *\r\n\t * @module Codex Editor content module\r\n\t *\r\n\t * @author Codex Team\r\n\t * @version 1.3.13\r\n\t *\r\n\t * @description Module works with Elements that have been appended to the main DOM\r\n\t */\n\t\n\tmodule.exports = function (content) {\n\t\n\t var editor = codex.editor;\n\t\n\t /**\r\n\t * Links to current active block\r\n\t * @type {null | Element}\r\n\t */\n\t content.currentNode = null;\n\t\n\t /**\r\n\t * clicked in redactor area\r\n\t * @type {null | Boolean}\r\n\t */\n\t content.editorAreaHightlighted = null;\n\t\n\t /**\r\n\t * @deprecated\r\n\t * Synchronizes redactor with original textarea\r\n\t */\n\t content.sync = function () {\n\t\n\t editor.core.log('syncing...');\n\t\n\t /**\r\n\t * Save redactor content to editor.state\r\n\t */\n\t editor.state.html = editor.nodes.redactor.innerHTML;\n\t };\n\t\n\t /**\r\n\t * Appends background to the block\r\n\t *\r\n\t * @description add CSS class to highlight visually first-level block area\r\n\t */\n\t content.markBlock = function () {\n\t\n\t editor.content.currentNode.classList.add(editor.ui.className.BLOCK_HIGHLIGHTED);\n\t };\n\t\n\t /**\r\n\t * Clear background\r\n\t *\r\n\t * @description clears styles that highlights block\r\n\t */\n\t content.clearMark = function () {\n\t\n\t if (editor.content.currentNode) {\n\t\n\t editor.content.currentNode.classList.remove(editor.ui.className.BLOCK_HIGHLIGHTED);\n\t }\n\t };\n\t\n\t /**\r\n\t * Finds first-level block\r\n\t *\r\n\t * @param {Element} node - selected or clicked in redactors area node\r\n\t * @protected\r\n\t *\r\n\t * @description looks for first-level block.\r\n\t * gets parent while node is not first-level\r\n\t */\n\t content.getFirstLevelBlock = function (node) {\n\t\n\t if (!editor.core.isDomNode(node)) {\n\t\n\t node = node.parentNode;\n\t }\n\t\n\t if (node === editor.nodes.redactor || node === document.body) {\n\t\n\t return null;\n\t } else {\n\t\n\t while (!node.classList.contains(editor.ui.className.BLOCK_CLASSNAME)) {\n\t\n\t node = node.parentNode;\n\t }\n\t\n\t return node;\n\t }\n\t };\n\t\n\t /**\r\n\t * Trigger this event when working node changed\r\n\t * @param {Element} targetNode - first-level of this node will be current\r\n\t * @protected\r\n\t *\r\n\t * @description If targetNode is first-level then we set it as current else we look for parents to find first-level\r\n\t */\n\t content.workingNodeChanged = function (targetNode) {\n\t\n\t /** Clear background from previous marked block before we change */\n\t editor.content.clearMark();\n\t\n\t if (!targetNode) {\n\t\n\t return;\n\t }\n\t\n\t content.currentNode = content.getFirstLevelBlock(targetNode);\n\t };\n\t\n\t /**\r\n\t * Replaces one redactor block with another\r\n\t * @protected\r\n\t * @param {Element} targetBlock - block to replace. Mostly currentNode.\r\n\t * @param {Element} newBlock\r\n\t * @param {string} newBlockType - type of new block; we need to store it to data-attribute\r\n\t *\r\n\t * [!] Function does not saves old block content.\r\n\t * You can get it manually and pass with newBlock.innerHTML\r\n\t */\n\t content.replaceBlock = function (targetBlock, newBlock) {\n\t\n\t if (!targetBlock || !newBlock) {\n\t\n\t editor.core.log('replaceBlock: missed params');\n\t return;\n\t }\n\t\n\t /** If target-block is not a frist-level block, then we iterate parents to find it */\n\t while (!targetBlock.classList.contains(editor.ui.className.BLOCK_CLASSNAME)) {\n\t\n\t targetBlock = targetBlock.parentNode;\n\t }\n\t\n\t /** Replacing */\n\t editor.nodes.redactor.replaceChild(newBlock, targetBlock);\n\t\n\t /**\r\n\t * Set new node as current\r\n\t */\n\t editor.content.workingNodeChanged(newBlock);\n\t\n\t /**\r\n\t * Add block handlers\r\n\t */\n\t editor.ui.addBlockHandlers(newBlock);\n\t\n\t /**\r\n\t * Save changes\r\n\t */\n\t editor.ui.saveInputs();\n\t };\n\t\n\t /**\r\n\t * @protected\r\n\t *\r\n\t * Inserts new block to redactor\r\n\t * Wrapps block into a DIV with BLOCK_CLASSNAME class\r\n\t *\r\n\t * @param blockData {object}\r\n\t * @param blockData.block {Element} element with block content\r\n\t * @param blockData.type {string} block plugin\r\n\t * @param needPlaceCaret {bool} pass true to set caret in new block\r\n\t *\r\n\t */\n\t content.insertBlock = function (blockData, needPlaceCaret) {\n\t\n\t var workingBlock = editor.content.currentNode,\n\t newBlockContent = blockData.block,\n\t blockType = blockData.type,\n\t isStretched = blockData.stretched;\n\t\n\t var newBlock = composeNewBlock_(newBlockContent, blockType, isStretched);\n\t\n\t if (workingBlock) {\n\t\n\t editor.core.insertAfter(workingBlock, newBlock);\n\t } else {\n\t\n\t /**\r\n\t * If redactor is empty, append as first child\r\n\t */\n\t editor.nodes.redactor.appendChild(newBlock);\n\t }\n\t\n\t /**\r\n\t * Block handler\r\n\t */\n\t editor.ui.addBlockHandlers(newBlock);\n\t\n\t /**\r\n\t * Set new node as current\r\n\t */\n\t editor.content.workingNodeChanged(newBlock);\n\t\n\t /**\r\n\t * Save changes\r\n\t */\n\t editor.ui.saveInputs();\n\t\n\t if (needPlaceCaret) {\n\t\n\t /**\r\n\t * If we don't know input index then we set default value -1\r\n\t */\n\t var currentInputIndex = editor.caret.getCurrentInputIndex() || -1;\n\t\n\t if (currentInputIndex == -1) {\n\t\n\t var editableElement = newBlock.querySelector('[contenteditable]'),\n\t emptyText = document.createTextNode('');\n\t\n\t editableElement.appendChild(emptyText);\n\t editor.caret.set(editableElement, 0, 0);\n\t\n\t editor.toolbar.move();\n\t editor.toolbar.showPlusButton();\n\t } else {\n\t\n\t if (currentInputIndex === editor.state.inputs.length - 1) return;\n\t\n\t /** Timeout for browsers execution */\n\t window.setTimeout(function () {\n\t\n\t /** Setting to the new input */\n\t editor.caret.setToNextBlock(currentInputIndex);\n\t editor.toolbar.move();\n\t editor.toolbar.open();\n\t }, 10);\n\t }\n\t }\n\t\n\t /**\r\n\t * Block is inserted, wait for new click that defined focusing on editors area\r\n\t * @type {boolean}\r\n\t */\n\t content.editorAreaHightlighted = false;\n\t };\n\t\n\t /**\r\n\t * Replaces blocks with saving content\r\n\t * @protected\r\n\t * @param {Element} noteToReplace\r\n\t * @param {Element} newNode\r\n\t * @param {Element} blockType\r\n\t */\n\t content.switchBlock = function (blockToReplace, newBlock, tool) {\n\t\n\t tool = tool || editor.content.currentNode.dataset.tool;\n\t var newBlockComposed = composeNewBlock_(newBlock, tool);\n\t\n\t /** Replacing */\n\t editor.content.replaceBlock(blockToReplace, newBlockComposed);\n\t\n\t /** Save new Inputs when block is changed */\n\t editor.ui.saveInputs();\n\t };\n\t\n\t /**\r\n\t * Iterates between child noted and looking for #text node on deepest level\r\n\t * @protected\r\n\t *\r\n\t * @param {Element} block - node where find\r\n\t * @param {int} postiton - starting postion\r\n\t * Example: childNodex.length to find from the end\r\n\t * or 0 to find from the start\r\n\t * @return {Text} block\r\n\t * @uses DFS\r\n\t */\n\t content.getDeepestTextNodeFromPosition = function (block, position) {\n\t\n\t /**\r\n\t * Clear Block from empty and useless spaces with trim.\r\n\t * Such nodes we should remove\r\n\t */\n\t var blockChilds = block.childNodes,\n\t index,\n\t node,\n\t text;\n\t\n\t for (index = 0; index < blockChilds.length; index++) {\n\t\n\t node = blockChilds[index];\n\t\n\t if (node.nodeType == editor.core.nodeTypes.TEXT) {\n\t\n\t text = node.textContent.trim();\n\t\n\t /** Text is empty. We should remove this child from node before we start DFS\r\n\t * decrease the quantity of childs.\r\n\t */\n\t if (text === '') {\n\t\n\t block.removeChild(node);\n\t position--;\n\t }\n\t }\n\t }\n\t\n\t if (block.childNodes.length === 0) {\n\t\n\t return document.createTextNode('');\n\t }\n\t\n\t /** Setting default position when we deleted all empty nodes */\n\t if (position < 0) position = 1;\n\t\n\t var lookingFromStart = false;\n\t\n\t /** For looking from START */\n\t if (position === 0) {\n\t\n\t lookingFromStart = true;\n\t position = 1;\n\t }\n\t\n\t while (position) {\n\t\n\t /** initial verticle of node. */\n\t if (lookingFromStart) {\n\t\n\t block = block.childNodes[0];\n\t } else {\n\t\n\t block = block.childNodes[position - 1];\n\t }\n\t\n\t if (block.nodeType == editor.core.nodeTypes.TAG) {\n\t\n\t position = block.childNodes.length;\n\t } else if (block.nodeType == editor.core.nodeTypes.TEXT) {\n\t\n\t position = 0;\n\t }\n\t }\n\t\n\t return block;\n\t };\n\t\n\t /**\r\n\t * @private\r\n\t * @param {Element} block - current plugins render\r\n\t * @param {String} tool - plugins name\r\n\t * @param {Boolean} isStretched - make stretched block or not\r\n\t *\r\n\t * @description adds necessary information to wrap new created block by first-level holder\r\n\t */\n\t var composeNewBlock_ = function composeNewBlock_(block, tool, isStretched) {\n\t\n\t var newBlock = editor.draw.node('DIV', editor.ui.className.BLOCK_CLASSNAME, {}),\n\t blockContent = editor.draw.node('DIV', editor.ui.className.BLOCK_CONTENT, {});\n\t\n\t blockContent.appendChild(block);\n\t newBlock.appendChild(blockContent);\n\t\n\t if (isStretched) {\n\t\n\t blockContent.classList.add(editor.ui.className.BLOCK_STRETCHED);\n\t }\n\t\n\t newBlock.dataset.tool = tool;\n\t return newBlock;\n\t };\n\t\n\t /**\r\n\t * Returns Range object of current selection\r\n\t * @protected\r\n\t */\n\t content.getRange = function () {\n\t\n\t var selection = window.getSelection().getRangeAt(0);\n\t\n\t return selection;\n\t };\n\t\n\t /**\r\n\t * Divides block in two blocks (after and before caret)\r\n\t *\r\n\t * @protected\r\n\t * @param {int} inputIndex - target input index\r\n\t *\r\n\t * @description splits current input content to the separate blocks\r\n\t * When enter is pressed among the words, that text will be splited.\r\n\t */\n\t content.splitBlock = function (inputIndex) {\n\t\n\t var selection = window.getSelection(),\n\t anchorNode = selection.anchorNode,\n\t anchorNodeText = anchorNode.textContent,\n\t caretOffset = selection.anchorOffset,\n\t textBeforeCaret,\n\t textNodeBeforeCaret,\n\t textAfterCaret,\n\t textNodeAfterCaret;\n\t\n\t var currentBlock = editor.content.currentNode.querySelector('[contentEditable]');\n\t\n\t textBeforeCaret = anchorNodeText.substring(0, caretOffset);\n\t textAfterCaret = anchorNodeText.substring(caretOffset);\n\t\n\t textNodeBeforeCaret = document.createTextNode(textBeforeCaret);\n\t\n\t if (textAfterCaret) {\n\t\n\t textNodeAfterCaret = document.createTextNode(textAfterCaret);\n\t }\n\t\n\t var previousChilds = [],\n\t nextChilds = [],\n\t reachedCurrent = false;\n\t\n\t if (textNodeAfterCaret) {\n\t\n\t nextChilds.push(textNodeAfterCaret);\n\t }\n\t\n\t for (var i = 0, child; !!(child = currentBlock.childNodes[i]); i++) {\n\t\n\t if (child != anchorNode) {\n\t\n\t if (!reachedCurrent) {\n\t\n\t previousChilds.push(child);\n\t } else {\n\t\n\t nextChilds.push(child);\n\t }\n\t } else {\n\t\n\t reachedCurrent = true;\n\t }\n\t }\n\t\n\t /** Clear current input */\n\t editor.state.inputs[inputIndex].innerHTML = '';\n\t\n\t /**\r\n\t * Append all childs founded before anchorNode\r\n\t */\n\t var previousChildsLength = previousChilds.length;\n\t\n\t for (i = 0; i < previousChildsLength; i++) {\n\t\n\t editor.state.inputs[inputIndex].appendChild(previousChilds[i]);\n\t }\n\t\n\t editor.state.inputs[inputIndex].appendChild(textNodeBeforeCaret);\n\t\n\t /**\r\n\t * Append text node which is after caret\r\n\t */\n\t var nextChildsLength = nextChilds.length,\n\t newNode = document.createElement('div');\n\t\n\t for (i = 0; i < nextChildsLength; i++) {\n\t\n\t newNode.appendChild(nextChilds[i]);\n\t }\n\t\n\t newNode = newNode.innerHTML;\n\t\n\t /** This type of block creates when enter is pressed */\n\t var NEW_BLOCK_TYPE = editor.settings.initialBlockPlugin;\n\t\n\t /**\r\n\t * Make new paragraph with text after caret\r\n\t */\n\t editor.content.insertBlock({\n\t type: NEW_BLOCK_TYPE,\n\t block: editor.tools[NEW_BLOCK_TYPE].render({\n\t text: newNode\n\t })\n\t }, true);\n\t };\n\t\n\t /**\r\n\t * Merges two blocks — current and target\r\n\t * If target index is not exist, then previous will be as target\r\n\t *\r\n\t * @protected\r\n\t * @param {int} currentInputIndex\r\n\t * @param {int} targetInputIndex\r\n\t *\r\n\t * @description gets two inputs indexes and merges into one\r\n\t */\n\t content.mergeBlocks = function (currentInputIndex, targetInputIndex) {\n\t\n\t /** If current input index is zero, then prevent method execution */\n\t if (currentInputIndex === 0) {\n\t\n\t return;\n\t }\n\t\n\t var targetInput,\n\t currentInputContent = editor.state.inputs[currentInputIndex].innerHTML;\n\t\n\t if (!targetInputIndex) {\n\t\n\t targetInput = editor.state.inputs[currentInputIndex - 1];\n\t } else {\n\t\n\t targetInput = editor.state.inputs[targetInputIndex];\n\t }\n\t\n\t targetInput.innerHTML += currentInputContent;\n\t };\n\t\n\t /**\r\n\t * Iterates all right siblings and parents, which has right siblings\r\n\t * while it does not reached the first-level block\r\n\t *\r\n\t * @param {Element} node\r\n\t * @return {boolean}\r\n\t */\n\t content.isLastNode = function (node) {\n\t\n\t // console.log('погнали перебор родителей');\n\t\n\t var allChecked = false;\n\t\n\t while (!allChecked) {\n\t\n\t // console.log('Смотрим на %o', node);\n\t // console.log('Проверим, пустые ли соседи справа');\n\t\n\t if (!allSiblingsEmpty_(node)) {\n\t\n\t // console.log('Есть непустые соседи. Узел не последний. Выходим.');\n\t return false;\n\t }\n\t\n\t node = node.parentNode;\n\t\n\t /**\r\n\t * Проверяем родителей до тех пор, пока не найдем блок первого уровня\r\n\t */\n\t if (node.classList.contains(editor.ui.className.BLOCK_CONTENT)) {\n\t\n\t allChecked = true;\n\t }\n\t }\n\t\n\t return true;\n\t };\n\t\n\t /**\r\n\t * Checks if all element right siblings is empty\r\n\t * @param node\r\n\t */\n\t var allSiblingsEmpty_ = function allSiblingsEmpty_(node) {\n\t\n\t /**\r\n\t * Нужно убедиться, что после пустого соседа ничего нет\r\n\t */\n\t var sibling = node.nextSibling;\n\t\n\t while (sibling) {\n\t\n\t if (sibling.textContent.length) {\n\t\n\t return false;\n\t }\n\t\n\t sibling = sibling.nextSibling;\n\t }\n\t\n\t return true;\n\t };\n\t\n\t /**\r\n\t * @public\r\n\t *\r\n\t * @param {string} htmlData - html content as string\r\n\t * @param {string} plainData - plain text\r\n\t * @return {string} - html content as string\r\n\t */\n\t content.wrapTextWithParagraphs = function (htmlData, plainData) {\n\t\n\t if (!htmlData) {\n\t\n\t return '

    ' + plainData.split('\\n\\n').join('

    ') + '

    ';\n\t }\n\t\n\t var wrapper = document.createElement('DIV'),\n\t newWrapper = document.createElement('DIV'),\n\t i,\n\t paragraph,\n\t firstLevelBlocks = ['DIV', 'P'],\n\t blockTyped,\n\t node;\n\t\n\t /**\r\n\t * Make HTML Element to Wrap Text\r\n\t * It allows us to work with input data as HTML content\r\n\t */\n\t wrapper.innerHTML = htmlData;\n\t paragraph = document.createElement('P');\n\t\n\t for (i = 0; i < wrapper.childNodes.length; i++) {\n\t\n\t node = wrapper.childNodes[i];\n\t\n\t blockTyped = firstLevelBlocks.indexOf(node.tagName) != -1;\n\t\n\t /**\r\n\t * If node is first-levet\r\n\t * we add this node to our new wrapper\r\n\t */\n\t if (blockTyped) {\n\t\n\t /**\r\n\t * If we had splitted inline nodes to paragraph before\r\n\t */\n\t if (paragraph.childNodes.length) {\n\t\n\t newWrapper.appendChild(paragraph.cloneNode(true));\n\t\n\t /** empty paragraph */\n\t paragraph = null;\n\t paragraph = document.createElement('P');\n\t }\n\t\n\t newWrapper.appendChild(node.cloneNode(true));\n\t } else {\n\t\n\t /** Collect all inline nodes to one as paragraph */\n\t paragraph.appendChild(node.cloneNode(true));\n\t\n\t /** if node is last we should append this node to paragraph and paragraph to new wrapper */\n\t if (i == wrapper.childNodes.length - 1) {\n\t\n\t newWrapper.appendChild(paragraph.cloneNode(true));\n\t }\n\t }\n\t }\n\t\n\t return newWrapper.innerHTML;\n\t };\n\t\n\t /**\r\n\t * Finds closest Contenteditable parent from Element\r\n\t * @param {Element} node element looking from\r\n\t * @return {Element} node contenteditable\r\n\t */\n\t content.getEditableParent = function (node) {\n\t\n\t while (node && node.contentEditable != 'true') {\n\t\n\t node = node.parentNode;\n\t }\n\t\n\t return node;\n\t };\n\t\n\t return content;\n\t}({});\n\n/***/ },\n/* 8 */\n/***/ function(module, exports, __webpack_require__) {\n\n\t'use strict';\n\t\n\t/**\r\n\t * Codex Editor toolbar module\r\n\t *\r\n\t * Contains:\r\n\t * - Inline toolbox\r\n\t * - Toolbox within plus button\r\n\t * - Settings section\r\n\t *\r\n\t * @author Codex Team\r\n\t * @version 1.0\r\n\t */\n\t\n\tmodule.exports = function (toolbar) {\n\t\n\t var editor = codex.editor;\n\t\n\t toolbar.settings = __webpack_require__(9);\n\t toolbar.inline = __webpack_require__(10);\n\t toolbar.toolbox = __webpack_require__(11);\n\t\n\t /**\r\n\t * Margin between focused node and toolbar\r\n\t */\n\t toolbar.defaultToolbarHeight = 49;\n\t\n\t toolbar.defaultOffset = 34;\n\t\n\t toolbar.opened = false;\n\t\n\t toolbar.current = null;\n\t\n\t /**\r\n\t * @protected\r\n\t */\n\t toolbar.open = function () {\n\t\n\t if (editor.hideToolbar) {\n\t\n\t return;\n\t }\n\t\n\t editor.nodes.toolbar.classList.add('opened');\n\t this.opened = true;\n\t };\n\t\n\t /**\r\n\t * @protected\r\n\t */\n\t toolbar.close = function () {\n\t\n\t editor.nodes.toolbar.classList.remove('opened');\n\t\n\t toolbar.opened = false;\n\t toolbar.current = null;\n\t\n\t for (var button in editor.nodes.toolbarButtons) {\n\t\n\t editor.nodes.toolbarButtons[button].classList.remove('selected');\n\t }\n\t\n\t /** Close toolbox when toolbar is not displayed */\n\t editor.toolbar.toolbox.close();\n\t editor.toolbar.settings.close();\n\t };\n\t\n\t toolbar.toggle = function () {\n\t\n\t if (!this.opened) {\n\t\n\t this.open();\n\t } else {\n\t\n\t this.close();\n\t }\n\t };\n\t\n\t toolbar.hidePlusButton = function () {\n\t\n\t editor.nodes.plusButton.classList.add('hide');\n\t };\n\t\n\t toolbar.showPlusButton = function () {\n\t\n\t editor.nodes.plusButton.classList.remove('hide');\n\t };\n\t\n\t /**\r\n\t * Moving toolbar to the specified node\r\n\t */\n\t toolbar.move = function () {\n\t\n\t /** Close Toolbox when we move toolbar */\n\t editor.toolbar.toolbox.close();\n\t\n\t if (!editor.content.currentNode) {\n\t\n\t return;\n\t }\n\t\n\t var newYCoordinate = editor.content.currentNode.offsetTop - editor.toolbar.defaultToolbarHeight / 2 + editor.toolbar.defaultOffset;\n\t\n\t editor.nodes.toolbar.style.transform = 'translate3D(0, ' + Math.floor(newYCoordinate) + 'px, 0)';\n\t\n\t /** Close trash actions */\n\t editor.toolbar.settings.hideRemoveActions();\n\t };\n\t\n\t return toolbar;\n\t}({});\n\n/***/ },\n/* 9 */\n/***/ function(module, exports) {\n\n\t'use strict';\n\t\n\t/**\r\n\t * Toolbar settings\r\n\t *\r\n\t * @version 1.0.4\r\n\t */\n\t\n\tmodule.exports = function (settings) {\n\t\n\t var editor = codex.editor;\n\t\n\t settings.opened = false;\n\t\n\t settings.setting = null;\n\t settings.actions = null;\n\t\n\t /**\r\n\t * Append and open settings\r\n\t */\n\t settings.open = function (toolType) {\n\t\n\t /**\r\n\t * Append settings content\r\n\t * It's stored in tool.settings\r\n\t */\n\t if (!editor.tools[toolType] || !editor.tools[toolType].makeSettings) {\n\t\n\t editor.core.log('Plugin \\xAB' + toolType + '\\xBB has no settings', 'warn');\n\t editor.nodes.pluginSettings.innerHTML = '\\u041F\\u043B\\u0430\\u0433\\u0438\\u043D \\xAB' + toolType + '\\xBB \\u043D\\u0435 \\u0438\\u043C\\u0435\\u0435\\u0442 \\u043D\\u0430\\u0441\\u0442\\u0440\\u043E\\u0435\\u043A';\n\t } else {\n\t\n\t /**\r\n\t * Draw settings block\r\n\t */\n\t var settingsBlock = editor.tools[toolType].makeSettings();\n\t\n\t editor.nodes.pluginSettings.appendChild(settingsBlock);\n\t }\n\t\n\t /** Open settings block */\n\t editor.nodes.blockSettings.classList.add('opened');\n\t this.opened = true;\n\t };\n\t\n\t /**\r\n\t * Close and clear settings\r\n\t */\n\t settings.close = function () {\n\t\n\t editor.nodes.blockSettings.classList.remove('opened');\n\t editor.nodes.pluginSettings.innerHTML = '';\n\t\n\t this.opened = false;\n\t };\n\t\n\t /**\r\n\t * @param {string} toolType - plugin type\r\n\t */\n\t settings.toggle = function (toolType) {\n\t\n\t if (!this.opened) {\n\t\n\t this.open(toolType);\n\t } else {\n\t\n\t this.close();\n\t }\n\t };\n\t\n\t /**\r\n\t * Here we will draw buttons and add listeners to components\r\n\t */\n\t settings.makeRemoveBlockButton = function () {\n\t\n\t var removeBlockWrapper = editor.draw.node('SPAN', 'ce-toolbar__remove-btn', {}),\n\t settingButton = editor.draw.node('SPAN', 'ce-toolbar__remove-setting', { innerHTML: '' }),\n\t actionWrapper = editor.draw.node('DIV', 'ce-toolbar__remove-confirmation', {}),\n\t confirmAction = editor.draw.node('DIV', 'ce-toolbar__remove-confirm', { textContent: 'Удалить блок' }),\n\t cancelAction = editor.draw.node('DIV', 'ce-toolbar__remove-cancel', { textContent: 'Отмена' });\n\t\n\t editor.listeners.add(settingButton, 'click', editor.toolbar.settings.removeButtonClicked, false);\n\t\n\t editor.listeners.add(confirmAction, 'click', editor.toolbar.settings.confirmRemovingRequest, false);\n\t\n\t editor.listeners.add(cancelAction, 'click', editor.toolbar.settings.cancelRemovingRequest, false);\n\t\n\t actionWrapper.appendChild(confirmAction);\n\t actionWrapper.appendChild(cancelAction);\n\t\n\t removeBlockWrapper.appendChild(settingButton);\n\t removeBlockWrapper.appendChild(actionWrapper);\n\t\n\t /** Save setting */\n\t editor.toolbar.settings.setting = settingButton;\n\t editor.toolbar.settings.actions = actionWrapper;\n\t\n\t return removeBlockWrapper;\n\t };\n\t\n\t settings.removeButtonClicked = function () {\n\t\n\t var action = editor.toolbar.settings.actions;\n\t\n\t if (action.classList.contains('opened')) {\n\t\n\t editor.toolbar.settings.hideRemoveActions();\n\t } else {\n\t\n\t editor.toolbar.settings.showRemoveActions();\n\t }\n\t\n\t editor.toolbar.toolbox.close();\n\t editor.toolbar.settings.close();\n\t };\n\t\n\t settings.cancelRemovingRequest = function () {\n\t\n\t editor.toolbar.settings.actions.classList.remove('opened');\n\t };\n\t\n\t settings.confirmRemovingRequest = function () {\n\t\n\t var currentBlock = editor.content.currentNode,\n\t firstLevelBlocksCount;\n\t\n\t currentBlock.remove();\n\t\n\t firstLevelBlocksCount = editor.nodes.redactor.childNodes.length;\n\t\n\t /**\r\n\t * If all blocks are removed\r\n\t */\n\t if (firstLevelBlocksCount === 0) {\n\t\n\t /** update currentNode variable */\n\t editor.content.currentNode = null;\n\t\n\t /** Inserting new empty initial block */\n\t editor.ui.addInitialBlock();\n\t }\n\t\n\t editor.ui.saveInputs();\n\t\n\t editor.toolbar.close();\n\t };\n\t\n\t settings.showRemoveActions = function () {\n\t\n\t editor.toolbar.settings.actions.classList.add('opened');\n\t };\n\t\n\t settings.hideRemoveActions = function () {\n\t\n\t editor.toolbar.settings.actions.classList.remove('opened');\n\t };\n\t\n\t return settings;\n\t}({});\n\n/***/ },\n/* 10 */\n/***/ function(module, exports) {\n\n\t'use strict';\n\t\n\t/**\r\n\t * Inline toolbar\r\n\t *\r\n\t * Contains from tools:\r\n\t * Bold, Italic, Underline and Anchor\r\n\t *\r\n\t * @author Codex Team\r\n\t * @version 1.0\r\n\t */\n\t\n\tmodule.exports = function (inline) {\n\t\n\t var editor = codex.editor;\n\t\n\t inline.buttonsOpened = null;\n\t inline.actionsOpened = null;\n\t inline.wrappersOffset = null;\n\t\n\t /**\r\n\t * saving selection that need for execCommand for styling\r\n\t *\r\n\t */\n\t inline.storedSelection = null;\n\t\n\t /**\r\n\t * @protected\r\n\t *\r\n\t * Open inline toobar\r\n\t */\n\t inline.show = function () {\n\t\n\t var currentNode = editor.content.currentNode,\n\t tool = currentNode.dataset.tool,\n\t plugin;\n\t\n\t /**\r\n\t * tool allowed to open inline toolbar\r\n\t */\n\t plugin = editor.tools[tool];\n\t\n\t if (!plugin.showInlineToolbar) return;\n\t\n\t var selectedText = inline.getSelectionText(),\n\t toolbar = editor.nodes.inlineToolbar.wrapper;\n\t\n\t if (selectedText.length > 0) {\n\t\n\t /** Move toolbar and open */\n\t editor.toolbar.inline.move();\n\t\n\t /** Open inline toolbar */\n\t toolbar.classList.add('opened');\n\t\n\t /** show buttons of inline toolbar */\n\t editor.toolbar.inline.showButtons();\n\t }\n\t };\n\t\n\t /**\r\n\t * @protected\r\n\t *\r\n\t * Closes inline toolbar\r\n\t */\n\t inline.close = function () {\n\t\n\t var toolbar = editor.nodes.inlineToolbar.wrapper;\n\t\n\t toolbar.classList.remove('opened');\n\t };\n\t\n\t /**\r\n\t * @private\r\n\t *\r\n\t * Moving toolbar\r\n\t */\n\t inline.move = function () {\n\t\n\t if (!this.wrappersOffset) {\n\t\n\t this.wrappersOffset = this.getWrappersOffset();\n\t }\n\t\n\t var coords = this.getSelectionCoords(),\n\t defaultOffset = 0,\n\t toolbar = editor.nodes.inlineToolbar.wrapper,\n\t newCoordinateX,\n\t newCoordinateY;\n\t\n\t if (toolbar.offsetHeight === 0) {\n\t\n\t defaultOffset = 40;\n\t }\n\t\n\t newCoordinateX = coords.x - this.wrappersOffset.left;\n\t newCoordinateY = coords.y + window.scrollY - this.wrappersOffset.top - defaultOffset - toolbar.offsetHeight;\n\t\n\t toolbar.style.transform = 'translate3D(' + Math.floor(newCoordinateX) + 'px, ' + Math.floor(newCoordinateY) + 'px, 0)';\n\t\n\t /** Close everything */\n\t editor.toolbar.inline.closeButtons();\n\t editor.toolbar.inline.closeAction();\n\t };\n\t\n\t /**\r\n\t * @private\r\n\t *\r\n\t * Tool Clicked\r\n\t */\n\t\n\t inline.toolClicked = function (event, type) {\n\t\n\t /**\r\n\t * For simple tools we use default browser function\r\n\t * For more complicated tools, we should write our own behavior\r\n\t */\n\t switch (type) {\n\t case 'createLink':\n\t editor.toolbar.inline.createLinkAction(event, type);break;\n\t default:\n\t editor.toolbar.inline.defaultToolAction(type);break;\n\t }\n\t\n\t /**\r\n\t * highlight buttons\r\n\t * after making some action\r\n\t */\n\t editor.nodes.inlineToolbar.buttons.childNodes.forEach(editor.toolbar.inline.hightlight);\n\t };\n\t\n\t /**\r\n\t * @private\r\n\t *\r\n\t * Saving wrappers offset in DOM\r\n\t */\n\t inline.getWrappersOffset = function () {\n\t\n\t var wrapper = editor.nodes.wrapper,\n\t offset = this.getOffset(wrapper);\n\t\n\t this.wrappersOffset = offset;\n\t return offset;\n\t };\n\t\n\t /**\r\n\t * @private\r\n\t *\r\n\t * Calculates offset of DOM element\r\n\t *\r\n\t * @param el\r\n\t * @returns {{top: number, left: number}}\r\n\t */\n\t inline.getOffset = function (el) {\n\t\n\t var _x = 0;\n\t var _y = 0;\n\t\n\t while (el && !isNaN(el.offsetLeft) && !isNaN(el.offsetTop)) {\n\t\n\t _x += el.offsetLeft + el.clientLeft;\n\t _y += el.offsetTop + el.clientTop;\n\t el = el.offsetParent;\n\t }\n\t return { top: _y, left: _x };\n\t };\n\t\n\t /**\r\n\t * @private\r\n\t *\r\n\t * Calculates position of selected text\r\n\t * @returns {{x: number, y: number}}\r\n\t */\n\t inline.getSelectionCoords = function () {\n\t\n\t var sel = document.selection,\n\t range;\n\t var x = 0,\n\t y = 0;\n\t\n\t if (sel) {\n\t\n\t if (sel.type != 'Control') {\n\t\n\t range = sel.createRange();\n\t range.collapse(true);\n\t x = range.boundingLeft;\n\t y = range.boundingTop;\n\t }\n\t } else if (window.getSelection) {\n\t\n\t sel = window.getSelection();\n\t\n\t if (sel.rangeCount) {\n\t\n\t range = sel.getRangeAt(0).cloneRange();\n\t if (range.getClientRects) {\n\t\n\t range.collapse(true);\n\t var rect = range.getClientRects()[0];\n\t\n\t if (!rect) {\n\t\n\t return;\n\t }\n\t\n\t x = rect.left;\n\t y = rect.top;\n\t }\n\t }\n\t }\n\t return { x: x, y: y };\n\t };\n\t\n\t /**\r\n\t * @private\r\n\t *\r\n\t * Returns selected text as String\r\n\t * @returns {string}\r\n\t */\n\t inline.getSelectionText = function () {\n\t\n\t var selectedText = '';\n\t\n\t // all modern browsers and IE9+\n\t if (window.getSelection) {\n\t\n\t selectedText = window.getSelection().toString();\n\t }\n\t\n\t return selectedText;\n\t };\n\t\n\t /** Opens buttons block */\n\t inline.showButtons = function () {\n\t\n\t var buttons = editor.nodes.inlineToolbar.buttons;\n\t\n\t buttons.classList.add('opened');\n\t\n\t editor.toolbar.inline.buttonsOpened = true;\n\t\n\t /** highlight buttons */\n\t editor.nodes.inlineToolbar.buttons.childNodes.forEach(editor.toolbar.inline.hightlight);\n\t };\n\t\n\t /** Makes buttons disappear */\n\t inline.closeButtons = function () {\n\t\n\t var buttons = editor.nodes.inlineToolbar.buttons;\n\t\n\t buttons.classList.remove('opened');\n\t\n\t editor.toolbar.inline.buttonsOpened = false;\n\t };\n\t\n\t /** Open buttons defined action if exist */\n\t inline.showActions = function () {\n\t\n\t var action = editor.nodes.inlineToolbar.actions;\n\t\n\t action.classList.add('opened');\n\t\n\t editor.toolbar.inline.actionsOpened = true;\n\t };\n\t\n\t /** Close actions block */\n\t inline.closeAction = function () {\n\t\n\t var action = editor.nodes.inlineToolbar.actions;\n\t\n\t action.innerHTML = '';\n\t action.classList.remove('opened');\n\t editor.toolbar.inline.actionsOpened = false;\n\t };\n\t\n\t /**\r\n\t * Callback for keydowns in inline toolbar \"Insert link...\" input\r\n\t */\n\t var inlineToolbarAnchorInputKeydown_ = function inlineToolbarAnchorInputKeydown_(event) {\n\t\n\t if (event.keyCode != editor.core.keys.ENTER) {\n\t\n\t return;\n\t }\n\t\n\t var editable = editor.content.currentNode,\n\t storedSelection = editor.toolbar.inline.storedSelection;\n\t\n\t editor.toolbar.inline.restoreSelection(editable, storedSelection);\n\t editor.toolbar.inline.setAnchor(this.value);\n\t\n\t /**\r\n\t * Preventing events that will be able to happen\r\n\t */\n\t event.preventDefault();\n\t event.stopImmediatePropagation();\n\t\n\t editor.toolbar.inline.clearRange();\n\t };\n\t\n\t /** Action for link creation or for setting anchor */\n\t inline.createLinkAction = function (event) {\n\t\n\t var isActive = this.isLinkActive();\n\t\n\t var editable = editor.content.currentNode,\n\t storedSelection = editor.toolbar.inline.saveSelection(editable);\n\t\n\t /** Save globally selection */\n\t editor.toolbar.inline.storedSelection = storedSelection;\n\t\n\t if (isActive) {\n\t\n\t /**\r\n\t * Changing stored selection. if we want to remove anchor from word\r\n\t * we should remove anchor from whole word, not only selected part.\r\n\t * The solution is than we get the length of current link\r\n\t * Change start position to - end of selection minus length of anchor\r\n\t */\n\t editor.toolbar.inline.restoreSelection(editable, storedSelection);\n\t\n\t editor.toolbar.inline.defaultToolAction('unlink');\n\t } else {\n\t\n\t /** Create input and close buttons */\n\t var action = editor.draw.inputForLink();\n\t\n\t editor.nodes.inlineToolbar.actions.appendChild(action);\n\t\n\t editor.toolbar.inline.closeButtons();\n\t editor.toolbar.inline.showActions();\n\t\n\t /**\r\n\t * focus to input\r\n\t * Solution: https://developer.mozilla.org/ru/docs/Web/API/HTMLElement/focus\r\n\t * Prevents event after showing input and when we need to focus an input which is in unexisted form\r\n\t */\n\t action.focus();\n\t event.preventDefault();\n\t\n\t /** Callback to link action */\n\t editor.listeners.add(action, 'keydown', inlineToolbarAnchorInputKeydown_, false);\n\t }\n\t };\n\t\n\t inline.isLinkActive = function () {\n\t\n\t var isActive = false;\n\t\n\t editor.nodes.inlineToolbar.buttons.childNodes.forEach(function (tool) {\n\t\n\t var dataType = tool.dataset.type;\n\t\n\t if (dataType == 'link' && tool.classList.contains('hightlighted')) {\n\t\n\t isActive = true;\n\t }\n\t });\n\t\n\t return isActive;\n\t };\n\t\n\t /** default action behavior of tool */\n\t inline.defaultToolAction = function (type) {\n\t\n\t document.execCommand(type, false, null);\n\t };\n\t\n\t /**\r\n\t * @private\r\n\t *\r\n\t * Sets URL\r\n\t *\r\n\t * @param {String} url - URL\r\n\t */\n\t inline.setAnchor = function (url) {\n\t\n\t document.execCommand('createLink', false, url);\n\t\n\t /** Close after URL inserting */\n\t editor.toolbar.inline.closeAction();\n\t };\n\t\n\t /**\r\n\t * @private\r\n\t *\r\n\t * Saves selection\r\n\t */\n\t inline.saveSelection = function (containerEl) {\n\t\n\t var range = window.getSelection().getRangeAt(0),\n\t preSelectionRange = range.cloneRange(),\n\t start;\n\t\n\t preSelectionRange.selectNodeContents(containerEl);\n\t preSelectionRange.setEnd(range.startContainer, range.startOffset);\n\t\n\t start = preSelectionRange.toString().length;\n\t\n\t return {\n\t start: start,\n\t end: start + range.toString().length\n\t };\n\t };\n\t\n\t /**\r\n\t * @private\r\n\t *\r\n\t * Sets to previous selection (Range)\r\n\t *\r\n\t * @param {Element} containerEl - editable element where we restore range\r\n\t * @param {Object} savedSel - range basic information to restore\r\n\t */\n\t inline.restoreSelection = function (containerEl, savedSel) {\n\t\n\t var range = document.createRange(),\n\t charIndex = 0;\n\t\n\t range.setStart(containerEl, 0);\n\t range.collapse(true);\n\t\n\t var nodeStack = [containerEl],\n\t node,\n\t foundStart = false,\n\t stop = false,\n\t nextCharIndex;\n\t\n\t while (!stop && (node = nodeStack.pop())) {\n\t\n\t if (node.nodeType == 3) {\n\t\n\t nextCharIndex = charIndex + node.length;\n\t\n\t if (!foundStart && savedSel.start >= charIndex && savedSel.start <= nextCharIndex) {\n\t\n\t range.setStart(node, savedSel.start - charIndex);\n\t foundStart = true;\n\t }\n\t if (foundStart && savedSel.end >= charIndex && savedSel.end <= nextCharIndex) {\n\t\n\t range.setEnd(node, savedSel.end - charIndex);\n\t stop = true;\n\t }\n\t charIndex = nextCharIndex;\n\t } else {\n\t\n\t var i = node.childNodes.length;\n\t\n\t while (i--) {\n\t\n\t nodeStack.push(node.childNodes[i]);\n\t }\n\t }\n\t }\n\t\n\t var sel = window.getSelection();\n\t\n\t sel.removeAllRanges();\n\t sel.addRange(range);\n\t };\n\t\n\t /**\r\n\t * @private\r\n\t *\r\n\t * Removes all ranges from window selection\r\n\t */\n\t inline.clearRange = function () {\n\t\n\t var selection = window.getSelection();\n\t\n\t selection.removeAllRanges();\n\t };\n\t\n\t /**\r\n\t * @private\r\n\t *\r\n\t * sets or removes hightlight\r\n\t */\n\t inline.hightlight = function (tool) {\n\t\n\t var dataType = tool.dataset.type;\n\t\n\t if (document.queryCommandState(dataType)) {\n\t\n\t editor.toolbar.inline.setButtonHighlighted(tool);\n\t } else {\n\t\n\t editor.toolbar.inline.removeButtonsHighLight(tool);\n\t }\n\t\n\t /**\r\n\t *\r\n\t * hightlight for anchors\r\n\t */\n\t var selection = window.getSelection(),\n\t tag = selection.anchorNode.parentNode;\n\t\n\t if (tag.tagName == 'A' && dataType == 'link') {\n\t\n\t editor.toolbar.inline.setButtonHighlighted(tool);\n\t }\n\t };\n\t\n\t /**\r\n\t * @private\r\n\t *\r\n\t * Mark button if text is already executed\r\n\t */\n\t inline.setButtonHighlighted = function (button) {\n\t\n\t button.classList.add('hightlighted');\n\t\n\t /** At link tool we also change icon */\n\t if (button.dataset.type == 'link') {\n\t\n\t var icon = button.childNodes[0];\n\t\n\t icon.classList.remove('ce-icon-link');\n\t icon.classList.add('ce-icon-unlink');\n\t }\n\t };\n\t\n\t /**\r\n\t * @private\r\n\t *\r\n\t * Removes hightlight\r\n\t */\n\t inline.removeButtonsHighLight = function (button) {\n\t\n\t button.classList.remove('hightlighted');\n\t\n\t /** At link tool we also change icon */\n\t if (button.dataset.type == 'link') {\n\t\n\t var icon = button.childNodes[0];\n\t\n\t icon.classList.remove('ce-icon-unlink');\n\t icon.classList.add('ce-icon-link');\n\t }\n\t };\n\t\n\t return inline;\n\t}({});\n\n/***/ },\n/* 11 */\n/***/ function(module, exports) {\n\n\t'use strict';\n\t\n\t/**\r\n\t * Codex Editor toolbox\r\n\t *\r\n\t * All tools be able to appended here\r\n\t *\r\n\t * @author Codex Team\r\n\t * @version 1.0\r\n\t */\n\t\n\tmodule.exports = function (toolbox) {\n\t\n\t var editor = codex.editor;\n\t\n\t toolbox.opened = false;\n\t\n\t /** Shows toolbox */\n\t toolbox.open = function () {\n\t\n\t /** Close setting if toolbox is opened */\n\t if (editor.toolbar.settings.opened) {\n\t\n\t editor.toolbar.settings.close();\n\t }\n\t\n\t /** display toolbox */\n\t editor.nodes.toolbox.classList.add('opened');\n\t\n\t /** Animate plus button */\n\t editor.nodes.plusButton.classList.add('clicked');\n\t\n\t /** toolbox state */\n\t editor.toolbar.toolbox.opened = true;\n\t };\n\t\n\t /** Closes toolbox */\n\t toolbox.close = function () {\n\t\n\t /** Makes toolbox disapear */\n\t editor.nodes.toolbox.classList.remove('opened');\n\t\n\t /** Rotate plus button */\n\t editor.nodes.plusButton.classList.remove('clicked');\n\t\n\t /** toolbox state */\n\t editor.toolbar.toolbox.opened = false;\n\t\n\t editor.toolbar.current = null;\n\t };\n\t\n\t toolbox.leaf = function () {\n\t\n\t var currentTool = editor.toolbar.current,\n\t tools = Object.keys(editor.tools),\n\t barButtons = editor.nodes.toolbarButtons,\n\t nextToolIndex = 0,\n\t toolToSelect = void 0,\n\t visibleTool = void 0,\n\t tool = void 0;\n\t\n\t if (!currentTool) {\n\t\n\t /** Get first tool from object*/\n\t for (tool in editor.tools) {\n\t\n\t if (editor.tools[tool].displayInToolbox) {\n\t\n\t break;\n\t }\n\t\n\t nextToolIndex++;\n\t }\n\t } else {\n\t\n\t nextToolIndex = (tools.indexOf(currentTool) + 1) % tools.length;\n\t visibleTool = tools[nextToolIndex];\n\t\n\t while (!editor.tools[visibleTool].displayInToolbox) {\n\t\n\t nextToolIndex = (nextToolIndex + 1) % tools.length;\n\t visibleTool = tools[nextToolIndex];\n\t }\n\t }\n\t\n\t toolToSelect = tools[nextToolIndex];\n\t\n\t for (var button in barButtons) {\n\t\n\t barButtons[button].classList.remove('selected');\n\t }\n\t\n\t barButtons[toolToSelect].classList.add('selected');\n\t editor.toolbar.current = toolToSelect;\n\t };\n\t\n\t /**\r\n\t * Transforming selected node type into selected toolbar element type\r\n\t * @param {event} event\r\n\t */\n\t toolbox.toolClicked = function (event) {\n\t\n\t /**\r\n\t * UNREPLACEBLE_TOOLS this types of tools are forbidden to replace even they are empty\r\n\t */\n\t var UNREPLACEBLE_TOOLS = ['image', 'link', 'list', 'instagram', 'twitter', 'embed'],\n\t tool = editor.tools[editor.toolbar.current],\n\t workingNode = editor.content.currentNode,\n\t currentInputIndex = editor.caret.inputIndex,\n\t newBlockContent,\n\t appendCallback,\n\t blockData;\n\t\n\t /** Make block from plugin */\n\t newBlockContent = tool.render();\n\t\n\t /** information about block */\n\t blockData = {\n\t block: newBlockContent,\n\t type: tool.type,\n\t stretched: false\n\t };\n\t\n\t if (workingNode && UNREPLACEBLE_TOOLS.indexOf(workingNode.dataset.tool) === -1 && workingNode.textContent.trim() === '') {\n\t\n\t /** Replace current block */\n\t editor.content.switchBlock(workingNode, newBlockContent, tool.type);\n\t } else {\n\t\n\t /** Insert new Block from plugin */\n\t editor.content.insertBlock(blockData);\n\t\n\t /** increase input index */\n\t currentInputIndex++;\n\t }\n\t\n\t /** Fire tool append callback */\n\t appendCallback = tool.appendCallback;\n\t\n\t if (appendCallback && typeof appendCallback == 'function') {\n\t\n\t appendCallback.call(event);\n\t }\n\t\n\t window.setTimeout(function () {\n\t\n\t /** Set caret to current block */\n\t editor.caret.setToBlock(currentInputIndex);\n\t }, 10);\n\t\n\t /**\r\n\t * Changing current Node\r\n\t */\n\t editor.content.workingNodeChanged();\n\t\n\t /**\r\n\t * Move toolbar when node is changed\r\n\t */\n\t editor.toolbar.move();\n\t };\n\t\n\t return toolbox;\n\t}({});\n\n/***/ },\n/* 12 */\n/***/ function(module, exports) {\n\n\t'use strict';\n\t\n\t/**\r\n\t * @module Codex Editor Callbacks module\r\n\t * @description Module works with editor added Elements\r\n\t *\r\n\t * @author Codex Team\r\n\t * @version 1.4.0\r\n\t */\n\t\n\tmodule.exports = function (callbacks) {\n\t\n\t var editor = codex.editor;\n\t\n\t /**\r\n\t * used by UI module\r\n\t * @description Routes all keydowns on document\r\n\t * @param {Object} event\r\n\t */\n\t callbacks.globalKeydown = function (event) {\n\t\n\t switch (event.keyCode) {\n\t case editor.core.keys.ENTER:\n\t enterKeyPressed_(event);break;\n\t }\n\t };\n\t\n\t /**\r\n\t * used by UI module\r\n\t * @description Routes all keydowns on redactors area\r\n\t * @param {Object} event\r\n\t */\n\t callbacks.redactorKeyDown = function (event) {\n\t\n\t switch (event.keyCode) {\n\t case editor.core.keys.TAB:\n\t tabKeyPressedOnRedactorsZone_(event);break;\n\t case editor.core.keys.ENTER:\n\t enterKeyPressedOnRedactorsZone_(event);break;\n\t case editor.core.keys.ESC:\n\t escapeKeyPressedOnRedactorsZone_(event);break;\n\t default:\n\t defaultKeyPressedOnRedactorsZone_(event);break;\n\t }\n\t };\n\t\n\t /**\r\n\t * used by UI module\r\n\t * @description Routes all keyup events\r\n\t * @param {Object} event\r\n\t */\n\t callbacks.globalKeyup = function (event) {\n\t\n\t switch (event.keyCode) {\n\t case editor.core.keys.UP:\n\t case editor.core.keys.LEFT:\n\t case editor.core.keys.RIGHT:\n\t case editor.core.keys.DOWN:\n\t arrowKeyPressed_(event);break;\n\t }\n\t };\n\t\n\t /**\r\n\t * @param {Object} event\r\n\t * @private\r\n\t *\r\n\t * Handles behaviour when tab pressed\r\n\t * @description if Content is empty show toolbox (if it is closed) or leaf tools\r\n\t * uses Toolbars toolbox module to handle the situation\r\n\t */\n\t var tabKeyPressedOnRedactorsZone_ = function tabKeyPressedOnRedactorsZone_(event) {\n\t\n\t /**\r\n\t * Wait for solution. Would like to know the behaviour\r\n\t * @todo Add spaces\r\n\t */\n\t event.preventDefault();\n\t\n\t var nativeInputs = editor.content.currentNode.querySelectorAll('textarea, input'),\n\t nativeInputsAreEmpty = true,\n\t textContentIsEmpty = !editor.content.currentNode.textContent.trim();\n\t\n\t Array.prototype.map.call(nativeInputs, function (input) {\n\t\n\t if (input.type == 'textarea' || input.type == 'text') {\n\t\n\t nativeInputsAreEmpty = nativeInputsAreEmpty && !input.value.trim();\n\t }\n\t });\n\t\n\t var blockIsEmpty = textContentIsEmpty && nativeInputsAreEmpty;\n\t\n\t if (!blockIsEmpty) {\n\t\n\t return;\n\t }\n\t\n\t if (!editor.toolbar.opened) {\n\t\n\t editor.toolbar.open();\n\t }\n\t\n\t if (editor.toolbar.opened && !editor.toolbar.toolbox.opened) {\n\t\n\t editor.toolbar.toolbox.open();\n\t } else {\n\t\n\t editor.toolbar.toolbox.leaf();\n\t }\n\t };\n\t\n\t /**\r\n\t * Handles global EnterKey Press\r\n\t * @see enterPressedOnBlock_\r\n\t * @param {Object} event\r\n\t */\n\t var enterKeyPressed_ = function enterKeyPressed_() {\n\t\n\t if (editor.content.editorAreaHightlighted) {\n\t\n\t /**\r\n\t * it means that we lose input index, saved index before is not correct\r\n\t * therefore we need to set caret when we insert new block\r\n\t */\n\t editor.caret.inputIndex = -1;\n\t\n\t enterPressedOnBlock_();\n\t }\n\t };\n\t\n\t /**\r\n\t * Callback for enter key pressing in first-level block area\r\n\t *\r\n\t * @param {Event} event\r\n\t * @private\r\n\t *\r\n\t * @description Inserts new block with initial type from settings\r\n\t */\n\t var enterPressedOnBlock_ = function enterPressedOnBlock_() {\n\t\n\t var NEW_BLOCK_TYPE = editor.settings.initialBlockPlugin;\n\t\n\t editor.content.insertBlock({\n\t type: NEW_BLOCK_TYPE,\n\t block: editor.tools[NEW_BLOCK_TYPE].render()\n\t }, true);\n\t\n\t editor.toolbar.move();\n\t editor.toolbar.open();\n\t };\n\t\n\t /**\r\n\t * ENTER key handler\r\n\t *\r\n\t * @param {Object} event\r\n\t * @private\r\n\t *\r\n\t * @description Makes new block with initial type from settings\r\n\t */\n\t var enterKeyPressedOnRedactorsZone_ = function enterKeyPressedOnRedactorsZone_(event) {\n\t\n\t if (event.target.contentEditable == 'true') {\n\t\n\t /** Update input index */\n\t editor.caret.saveCurrentInputIndex();\n\t }\n\t\n\t var currentInputIndex = editor.caret.getCurrentInputIndex() || 0,\n\t workingNode = editor.content.currentNode,\n\t tool = workingNode.dataset.tool,\n\t isEnterPressedOnToolbar = editor.toolbar.opened && editor.toolbar.current && event.target == editor.state.inputs[currentInputIndex];\n\t\n\t /** The list of tools which needs the default browser behaviour */\n\t var enableLineBreaks = editor.tools[tool].enableLineBreaks;\n\t\n\t /** This type of block creates when enter is pressed */\n\t var NEW_BLOCK_TYPE = editor.settings.initialBlockPlugin;\n\t\n\t /**\r\n\t * When toolbar is opened, select tool instead of making new paragraph\r\n\t */\n\t if (isEnterPressedOnToolbar) {\n\t\n\t event.preventDefault();\n\t\n\t editor.toolbar.toolbox.toolClicked(event);\n\t\n\t editor.toolbar.close();\n\t\n\t /**\r\n\t * Stop other listeners callback executions\r\n\t */\n\t event.stopPropagation();\n\t event.stopImmediatePropagation();\n\t\n\t return;\n\t }\n\t\n\t /**\r\n\t * Allow paragraph lineBreaks with shift enter\r\n\t * Or if shiftkey pressed and enter and enabledLineBreaks, the let new block creation\r\n\t */\n\t if (event.shiftKey || enableLineBreaks) {\n\t\n\t event.stopPropagation();\n\t event.stopImmediatePropagation();\n\t return;\n\t }\n\t\n\t var currentSelection = window.getSelection(),\n\t currentSelectedNode = currentSelection.anchorNode,\n\t caretAtTheEndOfText = editor.caret.position.atTheEnd(),\n\t isTextNodeHasParentBetweenContenteditable = false;\n\t\n\t /**\r\n\t * Allow making new

    in same block by SHIFT+ENTER and forbids to prevent default browser behaviour\r\n\t */\n\t if (event.shiftKey && !enableLineBreaks) {\n\t\n\t editor.callback.enterPressedOnBlock(editor.content.currentBlock, event);\n\t event.preventDefault();\n\t return;\n\t }\n\t\n\t /**\r\n\t * Workaround situation when caret at the Text node that has some wrapper Elements\r\n\t * Split block cant handle this.\r\n\t * We need to save default behavior\r\n\t */\n\t isTextNodeHasParentBetweenContenteditable = currentSelectedNode && currentSelectedNode.parentNode.contentEditable != 'true';\n\t\n\t /**\r\n\t * Split blocks when input has several nodes and caret placed in textNode\r\n\t */\n\t if (currentSelectedNode.nodeType == editor.core.nodeTypes.TEXT && !isTextNodeHasParentBetweenContenteditable && !caretAtTheEndOfText) {\n\t\n\t event.preventDefault();\n\t\n\t editor.core.log('Splitting Text node...');\n\t\n\t editor.content.splitBlock(currentInputIndex);\n\t\n\t /** Show plus button when next input after split is empty*/\n\t if (!editor.state.inputs[currentInputIndex + 1].textContent.trim()) {\n\t\n\t editor.toolbar.showPlusButton();\n\t }\n\t } else {\n\t\n\t var islastNode = editor.content.isLastNode(currentSelectedNode);\n\t\n\t if (islastNode && caretAtTheEndOfText) {\n\t\n\t event.preventDefault();\n\t event.stopPropagation();\n\t event.stopImmediatePropagation();\n\t\n\t editor.core.log('ENTER clicked in last textNode. Create new BLOCK');\n\t\n\t editor.content.insertBlock({\n\t type: NEW_BLOCK_TYPE,\n\t block: editor.tools[NEW_BLOCK_TYPE].render()\n\t }, true);\n\t\n\t editor.toolbar.move();\n\t editor.toolbar.open();\n\t\n\t /** Show plus button with empty block */\n\t editor.toolbar.showPlusButton();\n\t }\n\t }\n\t\n\t /** get all inputs after new appending block */\n\t editor.ui.saveInputs();\n\t };\n\t\n\t /**\r\n\t * Escape behaviour\r\n\t * @param event\r\n\t * @private\r\n\t *\r\n\t * @description Closes toolbox and toolbar. Prevents default behaviour\r\n\t */\n\t var escapeKeyPressedOnRedactorsZone_ = function escapeKeyPressedOnRedactorsZone_(event) {\n\t\n\t /** Close all toolbar */\n\t editor.toolbar.close();\n\t\n\t /** Close toolbox */\n\t editor.toolbar.toolbox.close();\n\t\n\t event.preventDefault();\n\t };\n\t\n\t /**\r\n\t * @param {Event} event\r\n\t * @private\r\n\t *\r\n\t * closes and moves toolbar\r\n\t */\n\t var arrowKeyPressed_ = function arrowKeyPressed_(event) {\n\t\n\t editor.content.workingNodeChanged();\n\t\n\t /* Closing toolbar */\n\t editor.toolbar.close();\n\t editor.toolbar.move();\n\t };\n\t\n\t /**\r\n\t * @private\r\n\t * @param {Event} event\r\n\t *\r\n\t * @description Closes all opened bars from toolbar.\r\n\t * If block is mark, clears highlightning\r\n\t */\n\t var defaultKeyPressedOnRedactorsZone_ = function defaultKeyPressedOnRedactorsZone_() {\n\t\n\t editor.toolbar.close();\n\t\n\t if (!editor.toolbar.inline.actionsOpened) {\n\t\n\t editor.toolbar.inline.close();\n\t editor.content.clearMark();\n\t }\n\t };\n\t\n\t /**\r\n\t * Handler when clicked on redactors area\r\n\t *\r\n\t * @protected\r\n\t * @param event\r\n\t *\r\n\t * @description Detects clicked area. If it is first-level block area, marks as detected and\r\n\t * on next enter press will be inserted new block\r\n\t * Otherwise, save carets position (input index) and put caret to the editable zone.\r\n\t *\r\n\t * @see detectWhenClickedOnFirstLevelBlockArea_\r\n\t *\r\n\t */\n\t callbacks.redactorClicked = function (event) {\n\t\n\t detectWhenClickedOnFirstLevelBlockArea_();\n\t\n\t editor.content.workingNodeChanged(event.target);\n\t editor.ui.saveInputs();\n\t\n\t var selectedText = editor.toolbar.inline.getSelectionText(),\n\t firstLevelBlock;\n\t\n\t /** If selection range took off, then we hide inline toolbar */\n\t if (selectedText.length === 0) {\n\t\n\t editor.toolbar.inline.close();\n\t }\n\t\n\t /** Update current input index in memory when caret focused into existed input */\n\t if (event.target.contentEditable == 'true') {\n\t\n\t editor.caret.saveCurrentInputIndex();\n\t }\n\t\n\t if (editor.content.currentNode === null) {\n\t\n\t /**\r\n\t * If inputs in redactor does not exits, then we put input index 0 not -1\r\n\t */\n\t var indexOfLastInput = editor.state.inputs.length > 0 ? editor.state.inputs.length - 1 : 0;\n\t\n\t /** If we have any inputs */\n\t if (editor.state.inputs.length) {\n\t\n\t /** getting firstlevel parent of input */\n\t firstLevelBlock = editor.content.getFirstLevelBlock(editor.state.inputs[indexOfLastInput]);\n\t }\n\t\n\t /** If input is empty, then we set caret to the last input */\n\t if (editor.state.inputs.length && editor.state.inputs[indexOfLastInput].textContent === '' && firstLevelBlock.dataset.tool == editor.settings.initialBlockPlugin) {\n\t\n\t editor.caret.setToBlock(indexOfLastInput);\n\t } else {\n\t\n\t /** Create new input when caret clicked in redactors area */\n\t var NEW_BLOCK_TYPE = editor.settings.initialBlockPlugin;\n\t\n\t editor.content.insertBlock({\n\t type: NEW_BLOCK_TYPE,\n\t block: editor.tools[NEW_BLOCK_TYPE].render()\n\t });\n\t\n\t /** If there is no inputs except inserted */\n\t if (editor.state.inputs.length === 1) {\n\t\n\t editor.caret.setToBlock(indexOfLastInput);\n\t } else {\n\t\n\t /** Set caret to this appended input */\n\t editor.caret.setToNextBlock(indexOfLastInput);\n\t }\n\t }\n\t } else {\n\t\n\t /** Close all panels */\n\t editor.toolbar.settings.close();\n\t editor.toolbar.toolbox.close();\n\t }\n\t\n\t /**\r\n\t * Move toolbar and open\r\n\t */\n\t editor.toolbar.move();\n\t editor.toolbar.open();\n\t\n\t var inputIsEmpty = !editor.content.currentNode.textContent.trim(),\n\t currentNodeType = editor.content.currentNode.dataset.tool,\n\t isInitialType = currentNodeType == editor.settings.initialBlockPlugin;\n\t\n\t /** Hide plus buttons */\n\t editor.toolbar.hidePlusButton();\n\t\n\t if (!inputIsEmpty) {\n\t\n\t /** Mark current block */\n\t editor.content.markBlock();\n\t }\n\t\n\t if (isInitialType && inputIsEmpty) {\n\t\n\t /** Show plus button */\n\t editor.toolbar.showPlusButton();\n\t }\n\t };\n\t\n\t /**\r\n\t * This method allows to define, is caret in contenteditable element or not.\r\n\t *\r\n\t * @private\r\n\t *\r\n\t * @description Otherwise, if we get TEXT node from range container, that will means we have input index.\r\n\t * In this case we use default browsers behaviour (if plugin allows that) or overwritten action.\r\n\t * Therefore, to be sure that we've clicked first-level block area, we should have currentNode, which always\r\n\t * specifies to the first-level block. Other cases we just ignore.\r\n\t */\n\t var detectWhenClickedOnFirstLevelBlockArea_ = function detectWhenClickedOnFirstLevelBlockArea_() {\n\t\n\t var selection = window.getSelection(),\n\t anchorNode = selection.anchorNode,\n\t flag = false;\n\t\n\t if (selection.rangeCount === 0) {\n\t\n\t editor.content.editorAreaHightlighted = true;\n\t } else {\n\t\n\t if (!editor.core.isDomNode(anchorNode)) {\n\t\n\t anchorNode = anchorNode.parentNode;\n\t }\n\t\n\t /** Already founded, without loop */\n\t if (anchorNode.contentEditable == 'true') {\n\t\n\t flag = true;\n\t }\n\t\n\t while (anchorNode.contentEditable != 'true') {\n\t\n\t anchorNode = anchorNode.parentNode;\n\t\n\t if (anchorNode.contentEditable == 'true') {\n\t\n\t flag = true;\n\t }\n\t\n\t if (anchorNode == document.body) {\n\t\n\t break;\n\t }\n\t }\n\t\n\t /** If editable element founded, flag is \"TRUE\", Therefore we return \"FALSE\" */\n\t editor.content.editorAreaHightlighted = !flag;\n\t }\n\t };\n\t\n\t /**\r\n\t * Toolbar button click handler\r\n\t *\r\n\t * @param {Object} event - cursor to the button\r\n\t * @protected\r\n\t *\r\n\t * @description gets current tool and calls render method\r\n\t */\n\t callbacks.toolbarButtonClicked = function (event) {\n\t\n\t var button = this;\n\t\n\t editor.toolbar.current = button.dataset.type;\n\t\n\t editor.toolbar.toolbox.toolClicked(event);\n\t editor.toolbar.close();\n\t };\n\t\n\t /**\r\n\t * Show or Hide toolbox when plus button is clicked\r\n\t */\n\t callbacks.plusButtonClicked = function () {\n\t\n\t if (!editor.nodes.toolbox.classList.contains('opened')) {\n\t\n\t editor.toolbar.toolbox.open();\n\t } else {\n\t\n\t editor.toolbar.toolbox.close();\n\t }\n\t };\n\t\n\t /**\r\n\t * Block handlers for KeyDown events\r\n\t *\r\n\t * @protected\r\n\t * @param {Object} event\r\n\t *\r\n\t * Handles keydowns on block\r\n\t * @see blockRightOrDownArrowPressed_\r\n\t * @see backspacePressed_\r\n\t * @see blockLeftOrUpArrowPressed_\r\n\t */\n\t callbacks.blockKeydown = function (event) {\n\t\n\t var block = event.target; // event.target is input\n\t\n\t switch (event.keyCode) {\n\t\n\t case editor.core.keys.DOWN:\n\t case editor.core.keys.RIGHT:\n\t blockRightOrDownArrowPressed_(event);\n\t break;\n\t\n\t case editor.core.keys.BACKSPACE:\n\t backspacePressed_(block, event);\n\t break;\n\t\n\t case editor.core.keys.UP:\n\t case editor.core.keys.LEFT:\n\t blockLeftOrUpArrowPressed_(event);\n\t break;\n\t\n\t }\n\t };\n\t\n\t /**\r\n\t * RIGHT or DOWN keydowns on block\r\n\t *\r\n\t * @param {Object} event\r\n\t * @private\r\n\t *\r\n\t * @description watches the selection and gets closest editable element.\r\n\t * Uses method getDeepestTextNodeFromPosition to get the last node of next block\r\n\t * Sets caret if it is contenteditable\r\n\t */\n\t var blockRightOrDownArrowPressed_ = function blockRightOrDownArrowPressed_(event) {\n\t\n\t var selection = window.getSelection(),\n\t inputs = editor.state.inputs,\n\t focusedNode = selection.anchorNode,\n\t focusedNodeHolder;\n\t\n\t /** Check for caret existance */\n\t if (!focusedNode) {\n\t\n\t return false;\n\t }\n\t\n\t /** Looking for closest (parent) contentEditable element of focused node */\n\t while (focusedNode.contentEditable != 'true') {\n\t\n\t focusedNodeHolder = focusedNode.parentNode;\n\t focusedNode = focusedNodeHolder;\n\t }\n\t\n\t /** Input index in DOM level */\n\t var editableElementIndex = 0;\n\t\n\t while (focusedNode != inputs[editableElementIndex]) {\n\t\n\t editableElementIndex++;\n\t }\n\t\n\t /**\r\n\t * Founded contentEditable element doesn't have childs\r\n\t * Or maybe New created block\r\n\t */\n\t if (!focusedNode.textContent) {\n\t\n\t editor.caret.setToNextBlock(editableElementIndex);\n\t return;\n\t }\n\t\n\t /**\r\n\t * Do nothing when caret doesn not reaches the end of last child\r\n\t */\n\t var caretInLastChild = false,\n\t caretAtTheEndOfText = false;\n\t\n\t var lastChild, deepestTextnode;\n\t\n\t lastChild = focusedNode.childNodes[focusedNode.childNodes.length - 1];\n\t\n\t if (editor.core.isDomNode(lastChild)) {\n\t\n\t deepestTextnode = editor.content.getDeepestTextNodeFromPosition(lastChild, lastChild.childNodes.length);\n\t } else {\n\t\n\t deepestTextnode = lastChild;\n\t }\n\t\n\t caretInLastChild = selection.anchorNode == deepestTextnode;\n\t caretAtTheEndOfText = deepestTextnode.length == selection.anchorOffset;\n\t\n\t if (!caretInLastChild || !caretAtTheEndOfText) {\n\t\n\t editor.core.log('arrow [down|right] : caret does not reached the end');\n\t return false;\n\t }\n\t\n\t editor.caret.setToNextBlock(editableElementIndex);\n\t };\n\t\n\t /**\r\n\t * LEFT or UP keydowns on block\r\n\t *\r\n\t * @param {Object} event\r\n\t * @private\r\n\t *\r\n\t * watches the selection and gets closest editable element.\r\n\t * Uses method getDeepestTextNodeFromPosition to get the last node of previous block\r\n\t * Sets caret if it is contenteditable\r\n\t *\r\n\t */\n\t var blockLeftOrUpArrowPressed_ = function blockLeftOrUpArrowPressed_(event) {\n\t\n\t var selection = window.getSelection(),\n\t inputs = editor.state.inputs,\n\t focusedNode = selection.anchorNode,\n\t focusedNodeHolder;\n\t\n\t /** Check for caret existance */\n\t if (!focusedNode) {\n\t\n\t return false;\n\t }\n\t\n\t /**\r\n\t * LEFT or UP not at the beginning\r\n\t */\n\t if (selection.anchorOffset !== 0) {\n\t\n\t return false;\n\t }\n\t\n\t /** Looking for parent contentEditable block */\n\t while (focusedNode.contentEditable != 'true') {\n\t\n\t focusedNodeHolder = focusedNode.parentNode;\n\t focusedNode = focusedNodeHolder;\n\t }\n\t\n\t /** Input index in DOM level */\n\t var editableElementIndex = 0;\n\t\n\t while (focusedNode != inputs[editableElementIndex]) {\n\t\n\t editableElementIndex++;\n\t }\n\t\n\t /**\r\n\t * Do nothing if caret is not at the beginning of first child\r\n\t */\n\t var caretInFirstChild = false,\n\t caretAtTheBeginning = false;\n\t\n\t var firstChild, deepestTextnode;\n\t\n\t /**\r\n\t * Founded contentEditable element doesn't have childs\r\n\t * Or maybe New created block\r\n\t */\n\t if (!focusedNode.textContent) {\n\t\n\t editor.caret.setToPreviousBlock(editableElementIndex);\n\t return;\n\t }\n\t\n\t firstChild = focusedNode.childNodes[0];\n\t\n\t if (editor.core.isDomNode(firstChild)) {\n\t\n\t deepestTextnode = editor.content.getDeepestTextNodeFromPosition(firstChild, 0);\n\t } else {\n\t\n\t deepestTextnode = firstChild;\n\t }\n\t\n\t caretInFirstChild = selection.anchorNode == deepestTextnode;\n\t caretAtTheBeginning = selection.anchorOffset === 0;\n\t\n\t if (caretInFirstChild && caretAtTheBeginning) {\n\t\n\t editor.caret.setToPreviousBlock(editableElementIndex);\n\t }\n\t };\n\t\n\t /**\r\n\t * Handles backspace keydown\r\n\t *\r\n\t * @param {Element} block\r\n\t * @param {Object} event\r\n\t * @private\r\n\t *\r\n\t * @description if block is empty, delete the block and set caret to the previous block\r\n\t * If block is not empty, try to merge two blocks - current and previous\r\n\t * But it we try'n to remove first block, then we should set caret to the next block, not previous.\r\n\t * If we removed the last block, create new one\r\n\t */\n\t var backspacePressed_ = function backspacePressed_(block, event) {\n\t\n\t var currentInputIndex = editor.caret.getCurrentInputIndex(),\n\t range,\n\t selectionLength,\n\t firstLevelBlocksCount;\n\t\n\t if (isNativeInput_(event.target)) {\n\t\n\t /** If input value is empty - remove block */\n\t if (event.target.value.trim() == '') {\n\t\n\t block.remove();\n\t } else {\n\t\n\t return;\n\t }\n\t }\n\t\n\t if (block.textContent.trim()) {\n\t\n\t range = editor.content.getRange();\n\t selectionLength = range.endOffset - range.startOffset;\n\t\n\t if (editor.caret.position.atStart() && !selectionLength && editor.state.inputs[currentInputIndex - 1]) {\n\t\n\t editor.content.mergeBlocks(currentInputIndex);\n\t } else {\n\t\n\t return;\n\t }\n\t }\n\t\n\t if (!selectionLength) {\n\t\n\t block.remove();\n\t }\n\t\n\t firstLevelBlocksCount = editor.nodes.redactor.childNodes.length;\n\t\n\t /**\r\n\t * If all blocks are removed\r\n\t */\n\t if (firstLevelBlocksCount === 0) {\n\t\n\t /** update currentNode variable */\n\t editor.content.currentNode = null;\n\t\n\t /** Inserting new empty initial block */\n\t editor.ui.addInitialBlock();\n\t\n\t /** Updating inputs state after deleting last block */\n\t editor.ui.saveInputs();\n\t\n\t /** Set to current appended block */\n\t window.setTimeout(function () {\n\t\n\t editor.caret.setToPreviousBlock(1);\n\t }, 10);\n\t } else {\n\t\n\t if (editor.caret.inputIndex !== 0) {\n\t\n\t /** Target block is not first */\n\t editor.caret.setToPreviousBlock(editor.caret.inputIndex);\n\t } else {\n\t\n\t /** If we try to delete first block */\n\t editor.caret.setToNextBlock(editor.caret.inputIndex);\n\t }\n\t }\n\t\n\t editor.toolbar.move();\n\t\n\t if (!editor.toolbar.opened) {\n\t\n\t editor.toolbar.open();\n\t }\n\t\n\t /** Updating inputs state */\n\t editor.ui.saveInputs();\n\t\n\t /** Prevent default browser behaviour */\n\t event.preventDefault();\n\t };\n\t\n\t /**\r\n\t * This method prevents default behaviour.\r\n\t *\r\n\t * @param {Object} event\r\n\t * @protected\r\n\t *\r\n\t * @description We get from clipboard pasted data, sanitize, make a fragment that contains of this sanitized nodes.\r\n\t * Firstly, we need to memorize the caret position. We can do that by getting the range of selection.\r\n\t * After all, we insert clear fragment into caret placed position. Then, we should move the caret to the last node\r\n\t */\n\t callbacks.blockPasteCallback = function (event) {\n\t\n\t /** If area is input or textarea then allow default behaviour */\n\t if (isNativeInput_(event.target)) {\n\t\n\t return;\n\t }\n\t\n\t /** Prevent default behaviour */\n\t event.preventDefault();\n\t\n\t var editableParent = editor.content.getEditableParent(event.target);\n\t\n\t /** Allow paste when event target placed in Editable element */\n\t if (!editableParent) {\n\t\n\t return;\n\t }\n\t\n\t /** get html pasted data - dirty data */\n\t var htmlData = event.clipboardData.getData('text/html'),\n\t plainData = event.clipboardData.getData('text/plain');\n\t\n\t /** Temporary DIV that is used to work with childs as arrays item */\n\t var div = editor.draw.node('DIV', '', {}),\n\t cleanData,\n\t wrappedData;\n\t\n\t /** Create fragment, that we paste to range after proccesing */\n\t cleanData = editor.sanitizer.clean(htmlData);\n\t\n\t /**\r\n\t * We wrap pasted text with

    tags to split it logically\r\n\t *\r\n\t * @type {string}\r\n\t */\n\t wrappedData = editor.content.wrapTextWithParagraphs(cleanData, plainData);\n\t div.innerHTML = wrappedData;\n\t\n\t var NEW_BLOCK_TYPE = editor.settings.initialBlockPlugin,\n\t currentBlockContent = editor.content.currentNode.firstChild.firstChild;\n\t\n\t /**\r\n\t * If there only one paragraph, just insert it\r\n\t */\n\t if (div.childNodes.length == 1) {\n\t\n\t editor.caret.insertNode(document.createTextNode(div.firstChild.innerHTML));\n\t return;\n\t }\n\t\n\t div.childNodes.forEach(function (paragraph, index) {\n\t\n\t /**\r\n\t * If there was no data in working node, replace it with first paragraph of pasted text\r\n\t */\n\t if (index == 0 && currentBlockContent.innerHTML.trim() === '') {\n\t\n\t editor.content.switchBlock(editor.content.currentNode, editor.tools[NEW_BLOCK_TYPE].render({\n\t text: paragraph.innerHTML\n\t }), NEW_BLOCK_TYPE);\n\t\n\t return;\n\t }\n\t\n\t editor.content.insertBlock({\n\t type: NEW_BLOCK_TYPE,\n\t block: editor.tools[NEW_BLOCK_TYPE].render({\n\t text: paragraph.innerHTML\n\t })\n\t });\n\t\n\t editor.caret.inputIndex++;\n\t });\n\t\n\t editor.caret.setToPreviousBlock(editor.caret.getCurrentInputIndex() + 1);\n\t };\n\t\n\t /**\r\n\t * used by UI module\r\n\t * Clicks on block settings button\r\n\t *\r\n\t * @param {Object} event\r\n\t * @protected\r\n\t * @description Opens toolbar settings\r\n\t */\n\t callbacks.showSettingsButtonClicked = function (event) {\n\t\n\t /**\r\n\t * Get type of current block\r\n\t * It uses to append settings from tool.settings property.\r\n\t * ...\r\n\t * Type is stored in data-type attribute on block\r\n\t */\n\t var currentToolType = editor.content.currentNode.dataset.tool;\n\t\n\t editor.toolbar.settings.toggle(currentToolType);\n\t\n\t /** Close toolbox when settings button is active */\n\t editor.toolbar.toolbox.close();\n\t editor.toolbar.settings.hideRemoveActions();\n\t };\n\t\n\t /**\r\n\t * Check block\r\n\t * @param target\r\n\t * @private\r\n\t *\r\n\t * @description Checks target is it native input\r\n\t */\n\t var isNativeInput_ = function isNativeInput_(target) {\n\t\n\t var nativeInputAreas = ['INPUT', 'TEXTAREA'];\n\t\n\t return nativeInputAreas.indexOf(target.tagName) != -1;\n\t };\n\t\n\t return callbacks;\n\t}({});\n\n/***/ },\n/* 13 */\n/***/ function(module, exports) {\n\n\t'use strict';\n\t\n\t/**\r\n\t * Codex Editor Draw module\r\n\t *\r\n\t * @author Codex Team\r\n\t * @version 1.0.\r\n\t */\n\t\n\tmodule.exports = function (draw) {\n\t\n\t /**\r\n\t * Base editor wrapper\r\n\t */\n\t draw.wrapper = function () {\n\t\n\t var wrapper = document.createElement('div');\n\t\n\t wrapper.className += 'codex-editor';\n\t\n\t return wrapper;\n\t };\n\t\n\t /**\r\n\t * Content-editable holder\r\n\t */\n\t draw.redactor = function () {\n\t\n\t var redactor = document.createElement('div');\n\t\n\t redactor.className += 'ce-redactor';\n\t\n\t return redactor;\n\t };\n\t\n\t draw.ceBlock = function () {\n\t\n\t var block = document.createElement('DIV');\n\t\n\t block.className += 'ce_block';\n\t\n\t return block;\n\t };\n\t\n\t /**\r\n\t * Empty toolbar with toggler\r\n\t */\n\t draw.toolbar = function () {\n\t\n\t var bar = document.createElement('div');\n\t\n\t bar.className += 'ce-toolbar';\n\t\n\t return bar;\n\t };\n\t\n\t draw.toolbarContent = function () {\n\t\n\t var wrapper = document.createElement('DIV');\n\t\n\t wrapper.classList.add('ce-toolbar__content');\n\t\n\t return wrapper;\n\t };\n\t\n\t /**\r\n\t * Inline toolbar\r\n\t */\n\t draw.inlineToolbar = function () {\n\t\n\t var bar = document.createElement('DIV');\n\t\n\t bar.className += 'ce-toolbar-inline';\n\t\n\t return bar;\n\t };\n\t\n\t /**\r\n\t * Wrapper for inline toobar buttons\r\n\t */\n\t draw.inlineToolbarButtons = function () {\n\t\n\t var wrapper = document.createElement('DIV');\n\t\n\t wrapper.className += 'ce-toolbar-inline__buttons';\n\t\n\t return wrapper;\n\t };\n\t\n\t /**\r\n\t * For some actions\r\n\t */\n\t draw.inlineToolbarActions = function () {\n\t\n\t var wrapper = document.createElement('DIV');\n\t\n\t wrapper.className += 'ce-toolbar-inline__actions';\n\t\n\t return wrapper;\n\t };\n\t\n\t draw.inputForLink = function () {\n\t\n\t var input = document.createElement('INPUT');\n\t\n\t input.type = 'input';\n\t input.className += 'inputForLink';\n\t input.placeholder = 'Вставьте ссылку ...';\n\t input.setAttribute('form', 'defaultForm');\n\t\n\t input.setAttribute('autofocus', 'autofocus');\n\t\n\t return input;\n\t };\n\t\n\t /**\r\n\t * @todo Desc\r\n\t */\n\t draw.blockButtons = function () {\n\t\n\t var block = document.createElement('div');\n\t\n\t block.className += 'ce-toolbar__actions';\n\t\n\t return block;\n\t };\n\t\n\t /**\r\n\t * Block settings panel\r\n\t */\n\t draw.blockSettings = function () {\n\t\n\t var settings = document.createElement('div');\n\t\n\t settings.className += 'ce-settings';\n\t\n\t return settings;\n\t };\n\t\n\t draw.defaultSettings = function () {\n\t\n\t var div = document.createElement('div');\n\t\n\t div.classList.add('ce-settings_default');\n\t\n\t return div;\n\t };\n\t\n\t draw.pluginsSettings = function () {\n\t\n\t var div = document.createElement('div');\n\t\n\t div.classList.add('ce-settings_plugin');\n\t\n\t return div;\n\t };\n\t\n\t draw.plusButton = function () {\n\t\n\t var button = document.createElement('span');\n\t\n\t button.className = 'ce-toolbar__plus';\n\t // button.innerHTML = '';\n\t\n\t return button;\n\t };\n\t\n\t /**\r\n\t * Settings button in toolbar\r\n\t */\n\t draw.settingsButton = function () {\n\t\n\t var toggler = document.createElement('span');\n\t\n\t toggler.className = 'ce-toolbar__settings-btn';\n\t\n\t /** Toggler button*/\n\t toggler.innerHTML = '';\n\t\n\t return toggler;\n\t };\n\t\n\t /**\r\n\t * Redactor tools wrapper\r\n\t */\n\t\n\t draw.toolbox = function () {\n\t\n\t var wrapper = document.createElement('div');\n\t\n\t wrapper.className = 'ce-toolbar__tools';\n\t\n\t return wrapper;\n\t };\n\t\n\t /**\r\n\t * @protected\r\n\t *\r\n\t * Draws tool buttons for toolbox\r\n\t *\r\n\t * @param {String} type\r\n\t * @param {String} classname\r\n\t * @returns {Element}\r\n\t */\n\t draw.toolbarButton = function (type, classname) {\n\t\n\t var button = document.createElement('li'),\n\t toolIcon = document.createElement('i'),\n\t toolTitle = document.createElement('span');\n\t\n\t button.dataset.type = type;\n\t button.setAttribute('title', type);\n\t\n\t toolIcon.classList.add(classname);\n\t toolTitle.classList.add('ce_toolbar_tools--title');\n\t\n\t button.appendChild(toolIcon);\n\t button.appendChild(toolTitle);\n\t\n\t return button;\n\t };\n\t\n\t /**\r\n\t * @protected\r\n\t *\r\n\t * Draws tools for inline toolbar\r\n\t *\r\n\t * @param {String} type\r\n\t * @param {String} classname\r\n\t */\n\t draw.toolbarButtonInline = function (type, classname) {\n\t\n\t var button = document.createElement('BUTTON'),\n\t toolIcon = document.createElement('I');\n\t\n\t button.type = 'button';\n\t button.dataset.type = type;\n\t toolIcon.classList.add(classname);\n\t\n\t button.appendChild(toolIcon);\n\t\n\t return button;\n\t };\n\t\n\t /**\r\n\t * Redactor block\r\n\t */\n\t draw.block = function (tagName, content) {\n\t\n\t var node = document.createElement(tagName);\n\t\n\t node.innerHTML = content || '';\n\t\n\t return node;\n\t };\n\t\n\t /**\r\n\t * Creates Node with passed tagName and className\r\n\t * @param {string} tagName\r\n\t * @param {string} className\r\n\t * @param {object} properties - allow to assign properties\r\n\t */\n\t draw.node = function (tagName, className, properties) {\n\t\n\t var el = document.createElement(tagName);\n\t\n\t if (className) el.className = className;\n\t\n\t if (properties) {\n\t\n\t for (var name in properties) {\n\t\n\t el[name] = properties[name];\n\t }\n\t }\n\t\n\t return el;\n\t };\n\t\n\t /**\r\n\t * Unavailable plugin block\r\n\t */\n\t draw.unavailableBlock = function () {\n\t\n\t var wrapper = document.createElement('DIV');\n\t\n\t wrapper.classList.add('cdx-unavailable-block');\n\t\n\t return wrapper;\n\t };\n\t\n\t return draw;\n\t}({});\n\n/***/ },\n/* 14 */\n/***/ function(module, exports) {\n\n\t'use strict';\n\t\n\t/**\r\n\t * Codex Editor Caret Module\r\n\t *\r\n\t * @author Codex Team\r\n\t * @version 1.0\r\n\t */\n\t\n\tmodule.exports = function (caret) {\n\t\n\t var editor = codex.editor;\n\t\n\t /**\r\n\t * @var {int} InputIndex - editable element in DOM\r\n\t */\n\t caret.inputIndex = null;\n\t\n\t /**\r\n\t * @var {int} offset - caret position in a text node.\r\n\t */\n\t caret.offset = null;\n\t\n\t /**\r\n\t * @var {int} focusedNodeIndex - we get index of child node from first-level block\r\n\t */\n\t caret.focusedNodeIndex = null;\n\t\n\t /**\r\n\t * Creates Document Range and sets caret to the element.\r\n\t * @protected\r\n\t * @uses caret.save — if you need to save caret position\r\n\t * @param {Element} el - Changed Node.\r\n\t */\n\t caret.set = function (el, index, offset) {\n\t\n\t offset = offset || caret.offset || 0;\n\t index = index || caret.focusedNodeIndex || 0;\n\t\n\t var childs = el.childNodes,\n\t nodeToSet;\n\t\n\t if (childs.length === 0) {\n\t\n\t nodeToSet = el;\n\t } else {\n\t\n\t nodeToSet = childs[index];\n\t }\n\t\n\t /** If Element is INPUT */\n\t if (el.contentEditable != 'true') {\n\t\n\t el.focus();\n\t return;\n\t }\n\t\n\t if (editor.core.isDomNode(nodeToSet)) {\n\t\n\t nodeToSet = editor.content.getDeepestTextNodeFromPosition(nodeToSet, nodeToSet.childNodes.length);\n\t }\n\t\n\t var range = document.createRange(),\n\t selection = window.getSelection();\n\t\n\t window.setTimeout(function () {\n\t\n\t range.setStart(nodeToSet, offset);\n\t range.setEnd(nodeToSet, offset);\n\t\n\t selection.removeAllRanges();\n\t selection.addRange(range);\n\t\n\t editor.caret.saveCurrentInputIndex();\n\t }, 20);\n\t };\n\t\n\t /**\r\n\t * @protected\r\n\t * Updates index of input and saves it in caret object\r\n\t */\n\t caret.saveCurrentInputIndex = function () {\n\t\n\t /** Index of Input that we paste sanitized content */\n\t var selection = window.getSelection(),\n\t inputs = editor.state.inputs,\n\t focusedNode = selection.anchorNode,\n\t focusedNodeHolder;\n\t\n\t if (!focusedNode) {\n\t\n\t return;\n\t }\n\t\n\t /** Looking for parent contentEditable block */\n\t while (focusedNode.contentEditable != 'true') {\n\t\n\t focusedNodeHolder = focusedNode.parentNode;\n\t focusedNode = focusedNodeHolder;\n\t }\n\t\n\t /** Input index in DOM level */\n\t var editableElementIndex = 0;\n\t\n\t while (focusedNode != inputs[editableElementIndex]) {\n\t\n\t editableElementIndex++;\n\t }\n\t\n\t caret.inputIndex = editableElementIndex;\n\t };\n\t\n\t /**\r\n\t * Returns current input index (caret object)\r\n\t */\n\t caret.getCurrentInputIndex = function () {\n\t\n\t return caret.inputIndex;\n\t };\n\t\n\t /**\r\n\t * @param {int} index - index of first-level block after that we set caret into next input\r\n\t */\n\t caret.setToNextBlock = function (index) {\n\t\n\t var inputs = editor.state.inputs,\n\t nextInput = inputs[index + 1];\n\t\n\t if (!nextInput) {\n\t\n\t editor.core.log('We are reached the end');\n\t return;\n\t }\n\t\n\t /**\r\n\t * When new Block created or deleted content of input\r\n\t * We should add some text node to set caret\r\n\t */\n\t if (!nextInput.childNodes.length) {\n\t\n\t var emptyTextElement = document.createTextNode('');\n\t\n\t nextInput.appendChild(emptyTextElement);\n\t }\n\t\n\t editor.caret.inputIndex = index + 1;\n\t editor.caret.set(nextInput, 0, 0);\n\t editor.content.workingNodeChanged(nextInput);\n\t };\n\t\n\t /**\r\n\t * @param {int} index - index of target input.\r\n\t * Sets caret to input with this index\r\n\t */\n\t caret.setToBlock = function (index) {\n\t\n\t var inputs = editor.state.inputs,\n\t targetInput = inputs[index];\n\t\n\t if (!targetInput) {\n\t\n\t return;\n\t }\n\t\n\t /**\r\n\t * When new Block created or deleted content of input\r\n\t * We should add some text node to set caret\r\n\t */\n\t if (!targetInput.childNodes.length) {\n\t\n\t var emptyTextElement = document.createTextNode('');\n\t\n\t targetInput.appendChild(emptyTextElement);\n\t }\n\t\n\t editor.caret.inputIndex = index;\n\t editor.caret.set(targetInput, 0, 0);\n\t editor.content.workingNodeChanged(targetInput);\n\t };\n\t\n\t /**\r\n\t * @param {int} index - index of input\r\n\t */\n\t caret.setToPreviousBlock = function (index) {\n\t\n\t index = index || 0;\n\t\n\t var inputs = editor.state.inputs,\n\t previousInput = inputs[index - 1],\n\t lastChildNode,\n\t lengthOfLastChildNode,\n\t emptyTextElement;\n\t\n\t if (!previousInput) {\n\t\n\t editor.core.log('We are reached first node');\n\t return;\n\t }\n\t\n\t lastChildNode = editor.content.getDeepestTextNodeFromPosition(previousInput, previousInput.childNodes.length);\n\t lengthOfLastChildNode = lastChildNode.length;\n\t\n\t /**\r\n\t * When new Block created or deleted content of input\r\n\t * We should add some text node to set caret\r\n\t */\n\t if (!previousInput.childNodes.length) {\n\t\n\t emptyTextElement = document.createTextNode('');\n\t previousInput.appendChild(emptyTextElement);\n\t }\n\t editor.caret.inputIndex = index - 1;\n\t editor.caret.set(previousInput, previousInput.childNodes.length - 1, lengthOfLastChildNode);\n\t editor.content.workingNodeChanged(inputs[index - 1]);\n\t };\n\t\n\t caret.position = {\n\t\n\t atStart: function atStart() {\n\t\n\t var selection = window.getSelection(),\n\t anchorOffset = selection.anchorOffset,\n\t anchorNode = selection.anchorNode,\n\t firstLevelBlock = editor.content.getFirstLevelBlock(anchorNode),\n\t pluginsRender = firstLevelBlock.childNodes[0];\n\t\n\t if (!editor.core.isDomNode(anchorNode)) {\n\t\n\t anchorNode = anchorNode.parentNode;\n\t }\n\t\n\t var isFirstNode = anchorNode === pluginsRender.childNodes[0],\n\t isOffsetZero = anchorOffset === 0;\n\t\n\t return isFirstNode && isOffsetZero;\n\t },\n\t\n\t atTheEnd: function atTheEnd() {\n\t\n\t var selection = window.getSelection(),\n\t anchorOffset = selection.anchorOffset,\n\t anchorNode = selection.anchorNode;\n\t\n\t /** Caret is at the end of input */\n\t return !anchorNode || !anchorNode.length || anchorOffset === anchorNode.length;\n\t }\n\t };\n\t\n\t /**\r\n\t * Inserts node at the caret location\r\n\t * @param node\r\n\t */\n\t caret.insertNode = function (node) {\n\t\n\t var selection, range;\n\t\n\t selection = window.getSelection();\n\t\n\t range = selection.getRangeAt(0);\n\t range.deleteContents();\n\t\n\t range.insertNode(node);\n\t\n\t range.setStartAfter(node);\n\t range.collapse(true);\n\t\n\t selection.removeAllRanges();\n\t selection.addRange(range);\n\t };\n\t\n\t return caret;\n\t}({});\n\n/***/ },\n/* 15 */\n/***/ function(module, exports) {\n\n\t'use strict';\n\t\n\t/**\r\n\t * Codex Editor Notification Module\r\n\t *\r\n\t * @author Codex Team\r\n\t * @version 1.0\r\n\t */\n\t\n\tmodule.exports = function (notifications) {\n\t\n\t var editor = codex.editor;\n\t\n\t var queue = [];\n\t\n\t var addToQueue = function addToQueue(settings) {\n\t\n\t queue.push(settings);\n\t\n\t var index = 0;\n\t\n\t while (index < queue.length && queue.length > 5) {\n\t\n\t if (queue[index].type == 'confirm' || queue[index].type == 'prompt') {\n\t\n\t index++;\n\t continue;\n\t }\n\t\n\t queue[index].close();\n\t queue.splice(index, 1);\n\t }\n\t };\n\t\n\t notifications.createHolder = function () {\n\t\n\t var holder = editor.draw.node('DIV', 'cdx-notifications-block');\n\t\n\t editor.nodes.notifications = document.body.appendChild(holder);\n\t\n\t return holder;\n\t };\n\t\n\t /**\r\n\t * Error notificator. Shows block with message\r\n\t * @protected\r\n\t */\n\t notifications.errorThrown = function (errorMsg, event) {\n\t\n\t editor.notifications.notification({ message: 'This action is not available currently', type: event.type });\n\t };\n\t\n\t /**\r\n\t *\r\n\t * Appends notification\r\n\t *\r\n\t * settings = {\r\n\t * type - notification type (reserved types: alert, confirm, prompt). Just add class 'cdx-notification-'+type\r\n\t * message - notification message\r\n\t * okMsg - confirm button text (default - 'Ok')\r\n\t * cancelBtn - cancel button text (default - 'Cancel'). Only for confirm and prompt types\r\n\t * confirm - function-handler for ok button click\r\n\t * cancel - function-handler for cancel button click. Only for confirm and prompt types\r\n\t * time - time (in seconds) after which notification will close (default - 10s)\r\n\t * }\r\n\t *\r\n\t * @param settings\r\n\t */\n\t notifications.notification = function (constructorSettings) {\n\t\n\t /** Private vars and methods */\n\t var notification = null,\n\t cancel = null,\n\t type = null,\n\t confirm = null,\n\t inputField = null;\n\t\n\t var confirmHandler = function confirmHandler() {\n\t\n\t close();\n\t\n\t if (typeof confirm !== 'function') {\n\t\n\t return;\n\t }\n\t\n\t if (type == 'prompt') {\n\t\n\t confirm(inputField.value);\n\t return;\n\t }\n\t\n\t confirm();\n\t };\n\t\n\t var cancelHandler = function cancelHandler() {\n\t\n\t close();\n\t\n\t if (typeof cancel !== 'function') {\n\t\n\t return;\n\t }\n\t\n\t cancel();\n\t };\n\t\n\t /** Public methods */\n\t function create(settings) {\n\t\n\t if (!(settings && settings.message)) {\n\t\n\t editor.core.log('Can\\'t create notification. Message is missed');\n\t return;\n\t }\n\t\n\t settings.type = settings.type || 'alert';\n\t settings.time = settings.time * 1000 || 10000;\n\t\n\t var wrapper = editor.draw.node('DIV', 'cdx-notification'),\n\t message = editor.draw.node('DIV', 'cdx-notification__message'),\n\t input = editor.draw.node('INPUT', 'cdx-notification__input'),\n\t okBtn = editor.draw.node('SPAN', 'cdx-notification__ok-btn'),\n\t cancelBtn = editor.draw.node('SPAN', 'cdx-notification__cancel-btn');\n\t\n\t message.textContent = settings.message;\n\t okBtn.textContent = settings.okMsg || 'ОК';\n\t cancelBtn.textContent = settings.cancelMsg || 'Отмена';\n\t\n\t editor.listeners.add(okBtn, 'click', confirmHandler);\n\t editor.listeners.add(cancelBtn, 'click', cancelHandler);\n\t\n\t wrapper.appendChild(message);\n\t\n\t if (settings.type == 'prompt') {\n\t\n\t wrapper.appendChild(input);\n\t }\n\t\n\t wrapper.appendChild(okBtn);\n\t\n\t if (settings.type == 'prompt' || settings.type == 'confirm') {\n\t\n\t wrapper.appendChild(cancelBtn);\n\t }\n\t\n\t wrapper.classList.add('cdx-notification-' + settings.type);\n\t wrapper.dataset.type = settings.type;\n\t\n\t notification = wrapper;\n\t type = settings.type;\n\t confirm = settings.confirm;\n\t cancel = settings.cancel;\n\t inputField = input;\n\t\n\t if (settings.type != 'prompt' && settings.type != 'confirm') {\n\t\n\t window.setTimeout(close, settings.time);\n\t }\n\t };\n\t\n\t function send() {\n\t\n\t editor.nodes.notifications.appendChild(notification);\n\t inputField.focus();\n\t\n\t editor.nodes.notifications.classList.add('cdx-notification__notification-appending');\n\t\n\t window.setTimeout(function () {\n\t\n\t editor.nodes.notifications.classList.remove('cdx-notification__notification-appending');\n\t }, 100);\n\t\n\t addToQueue({ type: type, close: close });\n\t };\n\t\n\t function close() {\n\t\n\t notification.remove();\n\t };\n\t\n\t if (constructorSettings) {\n\t\n\t create(constructorSettings);\n\t send();\n\t }\n\t\n\t return {\n\t create: create,\n\t send: send,\n\t close: close\n\t };\n\t };\n\t\n\t notifications.clear = function () {\n\t\n\t editor.nodes.notifications.innerHTML = '';\n\t queue = [];\n\t };\n\t\n\t return notifications;\n\t}({});\n\n/***/ },\n/* 16 */\n/***/ function(module, exports) {\n\n\t\"use strict\";\n\t\n\t/**\r\n\t * Codex Editor Parser Module\r\n\t *\r\n\t * @author Codex Team\r\n\t * @version 1.1\r\n\t */\n\t\n\tmodule.exports = function (parser) {\n\t\n\t var editor = codex.editor;\n\t\n\t /** inserting text */\n\t parser.insertPastedContent = function (blockType, tag) {\n\t\n\t editor.content.insertBlock({\n\t type: blockType.type,\n\t block: blockType.render({\n\t text: tag.innerHTML\n\t })\n\t });\n\t };\n\t\n\t /**\r\n\t * Check DOM node for display style: separated block or child-view\r\n\t */\n\t parser.isFirstLevelBlock = function (node) {\n\t\n\t return node.nodeType == editor.core.nodeTypes.TAG && node.classList.contains(editor.ui.className.BLOCK_CLASSNAME);\n\t };\n\t\n\t return parser;\n\t}({});\n\n/***/ },\n/* 17 */\n/***/ function(module, exports, __webpack_require__) {\n\n\t'use strict';\n\t\n\t/**\r\n\t * Codex Sanitizer\r\n\t */\n\t\n\tmodule.exports = function (sanitizer) {\n\t\n\t /** HTML Janitor library */\n\t var janitor = __webpack_require__(18);\n\t\n\t /** Codex Editor */\n\t var editor = codex.editor;\n\t\n\t sanitizer.prepare = function () {\n\t\n\t if (editor.settings.sanitizer && !editor.core.isEmpty(editor.settings.sanitizer)) {\n\t\n\t Config.CUSTOM = editor.settings.sanitizer;\n\t }\n\t };\n\t\n\t /**\r\n\t * Basic config\r\n\t */\n\t var Config = {\n\t\n\t /** User configuration */\n\t CUSTOM: null,\n\t\n\t BASIC: {\n\t\n\t tags: {\n\t p: {},\n\t a: {\n\t href: true,\n\t target: '_blank',\n\t rel: 'nofollow'\n\t },\n\t i: {},\n\t b: {},\n\t strong: {},\n\t em: {},\n\t span: {}\n\t }\n\t }\n\t };\n\t\n\t sanitizer.Config = Config;\n\t\n\t /**\r\n\t *\r\n\t * @param userCustomConfig\r\n\t * @returns {*}\r\n\t * @private\r\n\t *\r\n\t * @description If developer uses editor's API, then he can customize sane restrictions.\r\n\t * Or, sane config can be defined globally in editors initialization. That config will be used everywhere\r\n\t * At least, if there is no config overrides, that API uses BASIC Default configation\r\n\t */\n\t var init_ = function init_(userCustomConfig) {\n\t\n\t var configuration = userCustomConfig || Config.CUSTOM || Config.BASIC;\n\t\n\t return new janitor(configuration);\n\t };\n\t\n\t /**\r\n\t * Cleans string from unwanted tags\r\n\t * @protected\r\n\t * @param {String} dirtyString - taint string\r\n\t * @param {Object} customConfig - allowed tags\r\n\t */\n\t sanitizer.clean = function (dirtyString, customConfig) {\n\t\n\t var janitorInstance = init_(customConfig);\n\t\n\t return janitorInstance.clean(dirtyString);\n\t };\n\t\n\t return sanitizer;\n\t}({});\n\n/***/ },\n/* 18 */\n/***/ function(module, exports, __webpack_require__) {\n\n\tvar __WEBPACK_AMD_DEFINE_FACTORY__, __WEBPACK_AMD_DEFINE_RESULT__;(function (root, factory) {\n\t if (true) {\n\t !(__WEBPACK_AMD_DEFINE_FACTORY__ = (factory), __WEBPACK_AMD_DEFINE_RESULT__ = (typeof __WEBPACK_AMD_DEFINE_FACTORY__ === 'function' ? (__WEBPACK_AMD_DEFINE_FACTORY__.call(exports, __webpack_require__, exports, module)) : __WEBPACK_AMD_DEFINE_FACTORY__), __WEBPACK_AMD_DEFINE_RESULT__ !== undefined && (module.exports = __WEBPACK_AMD_DEFINE_RESULT__));\n\t } else if (typeof exports === 'object') {\n\t module.exports = factory();\n\t } else {\n\t root.HTMLJanitor = factory();\n\t }\n\t}(this, function () {\n\t\n\t /**\n\t * @param {Object} config.tags Dictionary of allowed tags.\n\t * @param {boolean} config.keepNestedBlockElements Default false.\n\t */\n\t function HTMLJanitor(config) {\n\t\n\t var tagDefinitions = config['tags'];\n\t var tags = Object.keys(tagDefinitions);\n\t\n\t var validConfigValues = tags\n\t .map(function(k) { return typeof tagDefinitions[k]; })\n\t .every(function(type) { return type === 'object' || type === 'boolean' || type === 'function'; });\n\t\n\t if(!validConfigValues) {\n\t throw new Error(\"The configuration was invalid\");\n\t }\n\t\n\t this.config = config;\n\t }\n\t\n\t // TODO: not exhaustive?\n\t var blockElementNames = ['P', 'LI', 'TD', 'TH', 'DIV', 'H1', 'H2', 'H3', 'H4', 'H5', 'H6', 'PRE'];\n\t function isBlockElement(node) {\n\t return blockElementNames.indexOf(node.nodeName) !== -1;\n\t }\n\t\n\t var inlineElementNames = ['A', 'B', 'STRONG', 'I', 'EM', 'SUB', 'SUP', 'U', 'STRIKE'];\n\t function isInlineElement(node) {\n\t return inlineElementNames.indexOf(node.nodeName) !== -1;\n\t }\n\t\n\t HTMLJanitor.prototype.clean = function (html) {\n\t var sandbox = document.createElement('div');\n\t sandbox.innerHTML = html;\n\t\n\t this._sanitize(sandbox);\n\t\n\t return sandbox.innerHTML;\n\t };\n\t\n\t HTMLJanitor.prototype._sanitize = function (parentNode) {\n\t var treeWalker = createTreeWalker(parentNode);\n\t var node = treeWalker.firstChild();\n\t if (!node) { return; }\n\t\n\t do {\n\t // Ignore nodes that have already been sanitized\n\t if (node._sanitized) {\n\t continue;\n\t }\n\t\n\t if (node.nodeType === Node.TEXT_NODE) {\n\t // If this text node is just whitespace and the previous or next element\n\t // sibling is a block element, remove it\n\t // N.B.: This heuristic could change. Very specific to a bug with\n\t // `contenteditable` in Firefox: http://jsbin.com/EyuKase/1/edit?js,output\n\t // FIXME: make this an option?\n\t if (node.data.trim() === ''\n\t && ((node.previousElementSibling && isBlockElement(node.previousElementSibling))\n\t || (node.nextElementSibling && isBlockElement(node.nextElementSibling)))) {\n\t parentNode.removeChild(node);\n\t this._sanitize(parentNode);\n\t break;\n\t } else {\n\t continue;\n\t }\n\t }\n\t\n\t // Remove all comments\n\t if (node.nodeType === Node.COMMENT_NODE) {\n\t parentNode.removeChild(node);\n\t this._sanitize(parentNode);\n\t break;\n\t }\n\t\n\t var isInline = isInlineElement(node);\n\t var containsBlockElement;\n\t if (isInline) {\n\t containsBlockElement = Array.prototype.some.call(node.childNodes, isBlockElement);\n\t }\n\t\n\t // Block elements should not be nested (e.g.

  • ...); if\n\t // they are, we want to unwrap the inner block element.\n\t var isNotTopContainer = !! parentNode.parentNode;\n\t var isNestedBlockElement =\n\t isBlockElement(parentNode) &&\n\t isBlockElement(node) &&\n\t isNotTopContainer;\n\t\n\t var nodeName = node.nodeName.toLowerCase();\n\t\n\t var allowedAttrs = getAllowedAttrs(this.config, nodeName, node);\n\t\n\t var isInvalid = isInline && containsBlockElement;\n\t\n\t // Drop tag entirely according to the whitelist *and* if the markup\n\t // is invalid.\n\t if (isInvalid || shouldRejectNode(node, allowedAttrs)\n\t || (!this.config.keepNestedBlockElements && isNestedBlockElement)) {\n\t // Do not keep the inner text of SCRIPT/STYLE elements.\n\t if (! (node.nodeName === 'SCRIPT' || node.nodeName === 'STYLE')) {\n\t while (node.childNodes.length > 0) {\n\t parentNode.insertBefore(node.childNodes[0], node);\n\t }\n\t }\n\t parentNode.removeChild(node);\n\t\n\t this._sanitize(parentNode);\n\t break;\n\t }\n\t\n\t // Sanitize attributes\n\t for (var a = 0; a < node.attributes.length; a += 1) {\n\t var attr = node.attributes[a];\n\t\n\t if (shouldRejectAttr(attr, allowedAttrs, node)) {\n\t node.removeAttribute(attr.name);\n\t // Shift the array to continue looping.\n\t a = a - 1;\n\t }\n\t }\n\t\n\t // Sanitize children\n\t this._sanitize(node);\n\t\n\t // Mark node as sanitized so it's ignored in future runs\n\t node._sanitized = true;\n\t } while ((node = treeWalker.nextSibling()));\n\t };\n\t\n\t function createTreeWalker(node) {\n\t return document.createTreeWalker(node,\n\t NodeFilter.SHOW_TEXT | NodeFilter.SHOW_ELEMENT | NodeFilter.SHOW_COMMENT,\n\t null, false);\n\t }\n\t\n\t function getAllowedAttrs(config, nodeName, node){\n\t if (typeof config.tags[nodeName] === 'function') {\n\t return config.tags[nodeName](node);\n\t } else {\n\t return config.tags[nodeName];\n\t }\n\t }\n\t\n\t function shouldRejectNode(node, allowedAttrs){\n\t if (typeof allowedAttrs === 'undefined') {\n\t return true;\n\t } else if (typeof allowedAttrs === 'boolean') {\n\t return !allowedAttrs;\n\t }\n\t\n\t return false;\n\t }\n\t\n\t function shouldRejectAttr(attr, allowedAttrs, node){\n\t var attrName = attr.name.toLowerCase();\n\t\n\t if (allowedAttrs === true){\n\t return false;\n\t } else if (typeof allowedAttrs[attrName] === 'function'){\n\t return !allowedAttrs[attrName](attr.value, node);\n\t } else if (typeof allowedAttrs[attrName] === 'undefined'){\n\t return true;\n\t } else if (allowedAttrs[attrName] === false) {\n\t return true;\n\t } else if (typeof allowedAttrs[attrName] === 'string') {\n\t return (allowedAttrs[attrName] !== attr.value);\n\t }\n\t\n\t return false;\n\t }\n\t\n\t return HTMLJanitor;\n\t\n\t}));\n\n\n/***/ },\n/* 19 */\n/***/ function(module, exports) {\n\n\t\"use strict\";\n\t\n\t/**\r\n\t * Codex Editor Listeners module\r\n\t *\r\n\t * @author Codex Team\r\n\t * @version 1.0\r\n\t */\n\t\n\t/**\r\n\t * Module-decorator for event listeners assignment\r\n\t */\n\tmodule.exports = function (listeners) {\n\t\n\t var allListeners = [];\n\t\n\t /**\r\n\t * Search methods\r\n\t *\r\n\t * byElement, byType and byHandler returns array of suitable listeners\r\n\t * one and all takes element, eventType, and handler and returns first (all) suitable listener\r\n\t *\r\n\t */\n\t listeners.search = function () {\n\t\n\t var byElement = function byElement(element, context) {\n\t\n\t var listenersOnElement = [];\n\t\n\t context = context || allListeners;\n\t\n\t for (var i = 0; i < context.length; i++) {\n\t\n\t var listener = context[i];\n\t\n\t if (listener.element === element) {\n\t\n\t listenersOnElement.push(listener);\n\t }\n\t }\n\t\n\t return listenersOnElement;\n\t };\n\t\n\t var byType = function byType(eventType, context) {\n\t\n\t var listenersWithType = [];\n\t\n\t context = context || allListeners;\n\t\n\t for (var i = 0; i < context.length; i++) {\n\t\n\t var listener = context[i];\n\t\n\t if (listener.type === eventType) {\n\t\n\t listenersWithType.push(listener);\n\t }\n\t }\n\t\n\t return listenersWithType;\n\t };\n\t\n\t var byHandler = function byHandler(handler, context) {\n\t\n\t var listenersWithHandler = [];\n\t\n\t context = context || allListeners;\n\t\n\t for (var i = 0; i < context.length; i++) {\n\t\n\t var listener = context[i];\n\t\n\t if (listener.handler === handler) {\n\t\n\t listenersWithHandler.push(listener);\n\t }\n\t }\n\t\n\t return listenersWithHandler;\n\t };\n\t\n\t var one = function one(element, eventType, handler) {\n\t\n\t var result = allListeners;\n\t\n\t if (element) result = byElement(element, result);\n\t\n\t if (eventType) result = byType(eventType, result);\n\t\n\t if (handler) result = byHandler(handler, result);\n\t\n\t return result[0];\n\t };\n\t\n\t var all = function all(element, eventType, handler) {\n\t\n\t var result = allListeners;\n\t\n\t if (element) result = byElement(element, result);\n\t\n\t if (eventType) result = byType(eventType, result);\n\t\n\t if (handler) result = byHandler(handler, result);\n\t\n\t return result;\n\t };\n\t\n\t return {\n\t byElement: byElement,\n\t byType: byType,\n\t byHandler: byHandler,\n\t one: one,\n\t all: all\n\t };\n\t }();\n\t\n\t listeners.add = function (element, eventType, handler, isCapture) {\n\t\n\t element.addEventListener(eventType, handler, isCapture);\n\t\n\t var data = {\n\t element: element,\n\t type: eventType,\n\t handler: handler\n\t };\n\t\n\t var alreadyAddedListener = listeners.search.one(element, eventType, handler);\n\t\n\t if (!alreadyAddedListener) {\n\t\n\t allListeners.push(data);\n\t }\n\t };\n\t\n\t listeners.remove = function (element, eventType, handler) {\n\t\n\t element.removeEventListener(eventType, handler);\n\t\n\t var existingListeners = listeners.search.all(element, eventType, handler);\n\t\n\t for (var i = 0; i < existingListeners.length; i++) {\n\t\n\t var index = allListeners.indexOf(existingListeners[i]);\n\t\n\t if (index > 0) {\n\t\n\t allListeners.splice(index, 1);\n\t }\n\t }\n\t };\n\t\n\t listeners.removeAll = function () {\n\t\n\t allListeners.map(function (current) {\n\t\n\t listeners.remove(current.element, current.type, current.handler);\n\t });\n\t };\n\t\n\t listeners.get = function (element, eventType, handler) {\n\t\n\t return listeners.search.all(element, eventType, handler);\n\t };\n\t\n\t return listeners;\n\t}({});\n\n/***/ },\n/* 20 */\n/***/ function(module, exports) {\n\n\t'use strict';\n\t\n\tvar _typeof = typeof Symbol === \"function\" && typeof Symbol.iterator === \"symbol\" ? function (obj) { return typeof obj; } : function (obj) { return obj && typeof Symbol === \"function\" && obj.constructor === Symbol && obj !== Symbol.prototype ? \"symbol\" : typeof obj; };\n\t\n\t/**\r\n\t * Codex Editor Destroyer module\r\n\t *\r\n\t * @auhor Codex Team\r\n\t * @version 1.0\r\n\t */\n\t\n\tmodule.exports = function (destroyer) {\n\t\n\t var editor = codex.editor;\n\t\n\t destroyer.removeNodes = function () {\n\t\n\t editor.nodes.wrapper.remove();\n\t editor.nodes.notifications.remove();\n\t };\n\t\n\t destroyer.destroyPlugins = function () {\n\t\n\t for (var tool in editor.tools) {\n\t\n\t if (typeof editor.tools[tool].destroy === 'function') {\n\t\n\t editor.tools[tool].destroy();\n\t }\n\t }\n\t };\n\t\n\t destroyer.destroyScripts = function () {\n\t\n\t var scripts = document.getElementsByTagName('SCRIPT');\n\t\n\t for (var i = 0; i < scripts.length; i++) {\n\t\n\t if (scripts[i].id.indexOf(editor.scriptPrefix) + 1) {\n\t\n\t scripts[i].remove();\n\t i--;\n\t }\n\t }\n\t };\n\t\n\t /**\r\n\t * Delete editor data from webpage.\r\n\t * You should send settings argument with boolean flags:\r\n\t * @param settings.ui- remove redactor event listeners and DOM nodes\r\n\t * @param settings.scripts - remove redactor scripts from DOM\r\n\t * @param settings.plugins - remove plugin's objects\r\n\t * @param settings.core - remove editor core. You can remove core only if UI and scripts flags is true\r\n\t * }\r\n\t *\r\n\t */\n\t destroyer.destroy = function (settings) {\n\t\n\t if (!settings || (typeof settings === 'undefined' ? 'undefined' : _typeof(settings)) !== 'object') {\n\t\n\t return;\n\t }\n\t\n\t if (settings.ui) {\n\t\n\t destroyer.removeNodes();\n\t editor.listeners.removeAll();\n\t }\n\t\n\t if (settings.scripts) {\n\t\n\t destroyer.destroyScripts();\n\t }\n\t\n\t if (settings.plugins) {\n\t\n\t destroyer.destroyPlugins();\n\t }\n\t\n\t if (settings.ui && settings.scripts && settings.core) {\n\t\n\t delete codex.editor;\n\t }\n\t };\n\t\n\t return destroyer;\n\t}({});\n\n/***/ },\n/* 21 */\n/***/ function(module, exports) {\n\n\t'use strict';\n\t\n\t/**\r\n\t * Codex Editor Paste module\r\n\t *\r\n\t * @author Codex Team\r\n\t * @version 1.1.1\r\n\t */\n\t\n\tmodule.exports = function (paste) {\n\t\n\t var editor = codex.editor;\n\t\n\t var patterns = [];\n\t\n\t paste.prepare = function () {\n\t\n\t var tools = editor.tools;\n\t\n\t for (var tool in tools) {\n\t\n\t if (!tools[tool].renderOnPastePatterns || !Array.isArray(tools[tool].renderOnPastePatterns)) {\n\t\n\t continue;\n\t }\n\t\n\t tools[tool].renderOnPastePatterns.map(function (pattern) {\n\t\n\t patterns.push(pattern);\n\t });\n\t }\n\t\n\t return Promise.resolve();\n\t };\n\t\n\t /**\r\n\t * Saves data\r\n\t * @param event\r\n\t */\n\t paste.pasted = function (event) {\n\t\n\t var clipBoardData = event.clipboardData || window.clipboardData,\n\t content = clipBoardData.getData('Text');\n\t\n\t var result = analize(content);\n\t\n\t if (result) {\n\t\n\t event.preventDefault();\n\t event.stopImmediatePropagation();\n\t }\n\t\n\t return result;\n\t };\n\t\n\t /**\r\n\t * Analizes pated string and calls necessary method\r\n\t */\n\t\n\t var analize = function analize(string) {\n\t\n\t var result = false,\n\t content = editor.content.currentNode,\n\t plugin = content.dataset.tool;\n\t\n\t patterns.map(function (pattern) {\n\t\n\t var execArray = pattern.regex.exec(string),\n\t match = execArray && execArray[0];\n\t\n\t if (match && match === string.trim()) {\n\t\n\t /** current block is not empty */\n\t if (content.textContent.trim() && plugin == editor.settings.initialBlockPlugin) {\n\t\n\t pasteToNewBlock_();\n\t }\n\t\n\t pattern.callback(string, pattern);\n\t result = true;\n\t }\n\t });\n\t\n\t return result;\n\t };\n\t\n\t var pasteToNewBlock_ = function pasteToNewBlock_() {\n\t\n\t /** Create new initial block */\n\t editor.content.insertBlock({\n\t\n\t type: editor.settings.initialBlockPlugin,\n\t block: editor.tools[editor.settings.initialBlockPlugin].render({\n\t text: ''\n\t })\n\t\n\t }, false);\n\t };\n\t\n\t return paste;\n\t}({});\n\n/***/ }\n/******/ ]);\n\n\n// WEBPACK FOOTER //\n// codex-editor.js"," \t// The module cache\n \tvar installedModules = {};\n\n \t// The require function\n \tfunction __webpack_require__(moduleId) {\n\n \t\t// Check if module is in cache\n \t\tif(installedModules[moduleId])\n \t\t\treturn installedModules[moduleId].exports;\n\n \t\t// Create a new module (and put it into the cache)\n \t\tvar module = installedModules[moduleId] = {\n \t\t\texports: {},\n \t\t\tid: moduleId,\n \t\t\tloaded: false\n \t\t};\n\n \t\t// Execute the module function\n \t\tmodules[moduleId].call(module.exports, module, module.exports, __webpack_require__);\n\n \t\t// Flag the module as loaded\n \t\tmodule.loaded = true;\n\n \t\t// Return the exports of the module\n \t\treturn module.exports;\n \t}\n\n\n \t// expose the modules object (__webpack_modules__)\n \t__webpack_require__.m = modules;\n\n \t// expose the module cache\n \t__webpack_require__.c = installedModules;\n\n \t// __webpack_public_path__\n \t__webpack_require__.p = \"\";\n\n \t// Load entry module and return exports\n \treturn __webpack_require__(0);\n\n\n\n// WEBPACK FOOTER //\n// webpack/bootstrap 4b9c333b21a67902b259","/**\r\n *\r\n * Codex Editor\r\n *\r\n * @author Codex Team\r\n */\r\n\r\nmodule.exports = (function (editor) {\r\n\r\n 'use strict';\r\n\r\n editor.version = VERSION;\r\n editor.scriptPrefix = 'cdx-script-';\r\n\r\n var init = function () {\r\n\r\n editor.core = require('./modules/core');\r\n editor.tools = require('./modules/tools');\r\n editor.ui = require('./modules/ui');\r\n editor.transport = require('./modules/transport');\r\n editor.renderer = require('./modules/renderer');\r\n editor.saver = require('./modules/saver');\r\n editor.content = require('./modules/content');\r\n editor.toolbar = require('./modules/toolbar/toolbar');\r\n editor.callback = require('./modules/callbacks');\r\n editor.draw = require('./modules/draw');\r\n editor.caret = require('./modules/caret');\r\n editor.notifications = require('./modules/notifications');\r\n editor.parser = require('./modules/parser');\r\n editor.sanitizer = require('./modules/sanitizer');\r\n editor.listeners = require('./modules/listeners');\r\n editor.destroyer = require('./modules/destroyer');\r\n editor.paste = require('./modules/paste');\r\n\r\n };\r\n\r\n /**\r\n * @public\r\n * holds initial settings\r\n */\r\n editor.settings = {\r\n tools : ['paragraph', 'header', 'picture', 'list', 'quote', 'code', 'twitter', 'instagram', 'smile'],\r\n holderId : 'codex-editor',\r\n\r\n // Type of block showing on empty editor\r\n initialBlockPlugin: 'paragraph'\r\n };\r\n\r\n /**\r\n * public\r\n *\r\n * Static nodes\r\n */\r\n editor.nodes = {\r\n holder : null,\r\n wrapper : null,\r\n toolbar : null,\r\n inlineToolbar : {\r\n wrapper : null,\r\n buttons : null,\r\n actions : null\r\n },\r\n toolbox : null,\r\n notifications : null,\r\n plusButton : null,\r\n showSettingsButton: null,\r\n showTrashButton : null,\r\n blockSettings : null,\r\n pluginSettings : null,\r\n defaultSettings : null,\r\n toolbarButtons : {}, // { type : DomEl, ... }\r\n redactor : null\r\n };\r\n\r\n /**\r\n * @public\r\n *\r\n * Output state\r\n */\r\n editor.state = {\r\n jsonOutput : [],\r\n blocks : [],\r\n inputs : []\r\n };\r\n\r\n /**\r\n * @public\r\n * Editor plugins\r\n */\r\n editor.tools = {};\r\n\r\n /**\r\n * Initialization\r\n * @uses Promise cEditor.core.prepare\r\n * @param {Object} userSettings\r\n * @param {Array} userSettings.tools list of plugins\r\n * @param {String} userSettings.holderId Element's id to append editor\r\n *\r\n * Load user defined tools\r\n * Tools must contain this important objects :\r\n * @param {String} type - this is a type of plugin. It can be used as plugin name\r\n * @param {String} iconClassname - this a icon in toolbar\r\n * @param {Object} make - what should plugin do, when it is clicked\r\n * @param {Object} appendCallback - callback after clicking\r\n * @param {Element} settings - what settings does it have\r\n * @param {Object} render - plugin get JSON, and should return HTML\r\n * @param {Object} save - plugin gets HTML content, returns JSON\r\n * @param {Boolean} displayInToolbox - will be displayed in toolbox. Default value is TRUE\r\n * @param {Boolean} enableLineBreaks - inserts new block or break lines. Default value is FALSE\r\n *\r\n * @example\r\n * - type : 'header',\r\n * - iconClassname : 'ce-icon-header',\r\n * - make : headerTool.make,\r\n * - appendCallback : headerTool.appendCallback,\r\n * - settings : headerTool.makeSettings(),\r\n * - render : headerTool.render,\r\n * - save : headerTool.save,\r\n * - displayInToolbox : true,\r\n * - enableLineBreaks : false\r\n */\r\n editor.start = function (userSettings) {\r\n\r\n init();\r\n\r\n editor.core.prepare(userSettings)\r\n\r\n // If all ok, make UI, bind events and parse initial-content\r\n .then(editor.ui.prepare)\r\n .then(editor.tools.prepare)\r\n .then(editor.sanitizer.prepare)\r\n .then(editor.paste.prepare)\r\n .then(editor.transport.prepare)\r\n .then(editor.renderer.makeBlocksFromData)\r\n .then(editor.ui.saveInputs)\r\n .catch(function (error) {\r\n\r\n editor.core.log('Initialization failed with error: %o', 'warn', error);\r\n\r\n });\r\n\r\n };\r\n\r\n return editor;\r\n\r\n})({});\r\n\n\n\n// WEBPACK FOOTER //\n// ./codex.js","/**\r\n * Codex Editor Core\r\n *\r\n * @author Codex Team\r\n * @version 1.1.3\r\n */\r\n\r\nmodule.exports = (function (core) {\r\n\r\n let editor = codex.editor;\r\n\r\n /**\r\n * @public\r\n *\r\n * Editor preparing method\r\n * @return Promise\r\n */\r\n core.prepare = function (userSettings) {\r\n\r\n return new Promise(function (resolve, reject) {\r\n\r\n if ( userSettings ) {\r\n\r\n editor.settings.tools = userSettings.tools || editor.settings.tools;\r\n\r\n }\r\n\r\n if (userSettings.data) {\r\n\r\n editor.state.blocks = userSettings.data;\r\n\r\n }\r\n\r\n if (userSettings.initialBlockPlugin) {\r\n\r\n editor.settings.initialBlockPlugin = userSettings.initialBlockPlugin;\r\n\r\n }\r\n\r\n if (userSettings.sanitizer) {\r\n\r\n editor.settings.sanitizer = userSettings.sanitizer;\r\n\r\n }\r\n\r\n editor.hideToolbar = userSettings.hideToolbar;\r\n\r\n editor.nodes.holder = document.getElementById(userSettings.holderId || editor.settings.holderId);\r\n\r\n if (typeof editor.nodes.holder === undefined || editor.nodes.holder === null) {\r\n\r\n reject(Error(\"Holder wasn't found by ID: #\" + userSettings.holderId));\r\n\r\n } else {\r\n\r\n resolve();\r\n\r\n }\r\n\r\n });\r\n\r\n };\r\n\r\n /**\r\n * Logging method\r\n * @param type = ['log', 'info', 'warn']\r\n */\r\n core.log = function (msg, type, arg) {\r\n\r\n type = type || 'log';\r\n\r\n if (!arg) {\r\n\r\n arg = msg || 'undefined';\r\n msg = '[codex-editor]: %o';\r\n\r\n } else {\r\n\r\n msg = '[codex-editor]: ' + msg;\r\n\r\n }\r\n\r\n try{\r\n\r\n if ( 'console' in window && window.console[ type ] ) {\r\n\r\n if ( arg ) window.console[ type ]( msg, arg );\r\n else window.console[ type ]( msg );\r\n\r\n }\r\n\r\n }catch(e) {}\r\n\r\n };\r\n\r\n /**\r\n * @protected\r\n *\r\n * Helper for insert one element after another\r\n */\r\n core.insertAfter = function (target, element) {\r\n\r\n target.parentNode.insertBefore(element, target.nextSibling);\r\n\r\n };\r\n\r\n /**\r\n * @const\r\n *\r\n * Readable DOM-node types map\r\n */\r\n core.nodeTypes = {\r\n TAG : 1,\r\n TEXT : 3,\r\n COMMENT : 8\r\n };\r\n\r\n /**\r\n * @const\r\n * Readable keys map\r\n */\r\n core.keys = { BACKSPACE: 8, TAB: 9, ENTER: 13, SHIFT: 16, CTRL: 17, ALT: 18, ESC: 27, SPACE: 32, LEFT: 37, UP: 38, DOWN: 40, RIGHT: 39, DELETE: 46, META: 91 };\r\n\r\n /**\r\n * @protected\r\n *\r\n * Check object for DOM node\r\n */\r\n core.isDomNode = function (el) {\r\n\r\n return el && typeof el === 'object' && el.nodeType && el.nodeType == this.nodeTypes.TAG;\r\n\r\n };\r\n\r\n /**\r\n * Checks passed object for emptiness\r\n * @require ES5 - Object.keys\r\n * @param {object}\r\n */\r\n core.isEmpty = function ( obj ) {\r\n\r\n return Object.keys(obj).length === 0;\r\n\r\n };\r\n\r\n /**\r\n * Native Ajax\r\n * @param {String} settings.url - request URL\r\n * @param {function} settings.beforeSend - returned value will be passed as context to the Success, Error and Progress callbacks\r\n * @param {function} settings.success\r\n * @param {function} settings.progress\r\n */\r\n core.ajax = function (settings) {\r\n\r\n if (!settings || !settings.url) {\r\n\r\n return;\r\n\r\n }\r\n\r\n var XMLHTTP = window.XMLHttpRequest ? new XMLHttpRequest() : new ActiveXObject('Microsoft.XMLHTTP'),\r\n encodedString,\r\n isFormData,\r\n prop;\r\n\r\n\r\n settings.async = true;\r\n settings.type = settings.type || 'GET';\r\n settings.data = settings.data || '';\r\n settings['content-type'] = settings['content-type'] || 'application/json; charset=utf-8';\r\n\r\n if (settings.type == 'GET' && settings.data) {\r\n\r\n settings.url = /\\?/.test(settings.url) ? settings.url + '&' + settings.data : settings.url + '?' + settings.data;\r\n\r\n } else {\r\n\r\n encodedString = '';\r\n for(prop in settings.data) {\r\n\r\n encodedString += (prop + '=' + encodeURIComponent(settings.data[prop]) + '&');\r\n\r\n }\r\n\r\n }\r\n\r\n if (settings.withCredentials) {\r\n\r\n XMLHTTP.withCredentials = true;\r\n\r\n }\r\n\r\n /**\r\n * Value returned in beforeSend funtion will be passed as context to the other response callbacks\r\n * If beforeSend returns false, AJAX will be blocked\r\n */\r\n let responseContext,\r\n beforeSendResult;\r\n\r\n if (typeof settings.beforeSend === 'function') {\r\n\r\n beforeSendResult = settings.beforeSend.call();\r\n\r\n if (beforeSendResult === false) {\r\n\r\n return;\r\n\r\n }\r\n\r\n }\r\n\r\n XMLHTTP.open( settings.type, settings.url, settings.async );\r\n\r\n /**\r\n * If we send FormData, we need no content-type header\r\n */\r\n isFormData = isFormData_(settings.data);\r\n\r\n if (!isFormData) {\r\n\r\n if (settings.type !== 'POST') {\r\n\r\n XMLHTTP.setRequestHeader('Content-type', settings['content-type']);\r\n\r\n } else {\r\n\r\n XMLHTTP.setRequestHeader('Content-type', 'application/x-www-form-urlencoded');\r\n\r\n }\r\n\r\n }\r\n\r\n XMLHTTP.setRequestHeader('X-Requested-With', 'XMLHttpRequest');\r\n\r\n responseContext = beforeSendResult || XMLHTTP;\r\n\r\n if (typeof settings.progress === 'function') {\r\n\r\n XMLHTTP.upload.onprogress = settings.progress.bind(responseContext);\r\n\r\n }\r\n\r\n XMLHTTP.onreadystatechange = function () {\r\n\r\n if (XMLHTTP.readyState === 4) {\r\n\r\n if (XMLHTTP.status === 200) {\r\n\r\n if (typeof settings.success === 'function') {\r\n\r\n settings.success.call(responseContext, XMLHTTP.responseText);\r\n\r\n }\r\n\r\n } else {\r\n\r\n if (typeof settings.error === 'function') {\r\n\r\n settings.error.call(responseContext, XMLHTTP.responseText, XMLHTTP.status);\r\n\r\n }\r\n\r\n }\r\n\r\n }\r\n\r\n };\r\n\r\n if (isFormData) {\r\n\r\n // Sending FormData\r\n XMLHTTP.send(settings.data);\r\n\r\n } else {\r\n\r\n // POST requests\r\n XMLHTTP.send(encodedString);\r\n\r\n }\r\n\r\n return XMLHTTP;\r\n\r\n };\r\n\r\n /**\r\n * Appends script to head of document\r\n * @return Promise\r\n */\r\n core.importScript = function (scriptPath, instanceName) {\r\n\r\n return new Promise(function (resolve, reject) {\r\n\r\n let script;\r\n\r\n /** Script is already loaded */\r\n if ( !instanceName ) {\r\n\r\n reject('Instance name is missed');\r\n\r\n } else if ( document.getElementById(editor.scriptPrefix + instanceName) ) {\r\n\r\n resolve(scriptPath);\r\n\r\n }\r\n\r\n script = document.createElement('SCRIPT');\r\n script.async = true;\r\n script.defer = true;\r\n script.id = editor.scriptPrefix + instanceName;\r\n\r\n script.onload = function () {\r\n\r\n resolve(scriptPath);\r\n\r\n };\r\n\r\n script.onerror = function () {\r\n\r\n reject(scriptPath);\r\n\r\n };\r\n\r\n script.src = scriptPath;\r\n document.head.appendChild(script);\r\n\r\n });\r\n\r\n };\r\n\r\n /**\r\n * Function for checking is it FormData object to send.\r\n * @param {Object} object to check\r\n * @return boolean\r\n */\r\n var isFormData_ = function (object) {\r\n\r\n return object instanceof FormData;\r\n\r\n };\r\n\r\n return core;\r\n\r\n})({});\r\n\n\n\n// WEBPACK FOOTER //\n// ./modules/core.js","/**\r\n* Module working with plugins\r\n*/\r\nmodule.exports = (function () {\r\n\r\n let editor = codex.editor;\r\n\r\n /**\r\n * Initialize plugins before using\r\n * Ex. Load scripts or call some internal methods\r\n * @return Promise\r\n */\r\n function prepare() {\r\n\r\n return new Promise(function (resolve_, reject_) {\r\n\r\n Promise.resolve()\r\n\r\n /**\r\n * Compose a sequence of plugins that requires preparation\r\n */\r\n .then(function () {\r\n\r\n let pluginsRequiresPreparation = [],\r\n allPlugins = editor.tools;\r\n\r\n for ( let pluginName in allPlugins ) {\r\n\r\n let plugin = allPlugins[pluginName];\r\n\r\n if (plugin.prepare && typeof plugin.prepare != 'function' || !plugin.prepare) {\r\n\r\n continue;\r\n\r\n }\r\n\r\n pluginsRequiresPreparation.push(plugin);\r\n\r\n }\r\n\r\n /**\r\n * If no one passed plugins requires preparation, finish prepare() and go ahead\r\n */\r\n if (!pluginsRequiresPreparation.length) {\r\n\r\n resolve_();\r\n\r\n }\r\n\r\n return pluginsRequiresPreparation;\r\n\r\n })\r\n\r\n /** Wait plugins while they prepares */\r\n .then(waitAllPluginsPreparation_)\r\n\r\n .then(function () {\r\n\r\n editor.core.log('Plugins loaded', 'info');\r\n resolve_();\r\n\r\n }).catch(function (error) {\r\n\r\n reject_(error);\r\n\r\n });\r\n\r\n });\r\n\r\n }\r\n\r\n /**\r\n * @param {array} plugins - list of tools that requires preparation\r\n * @return {Promise} resolved while all plugins will be ready or failed\r\n */\r\n function waitAllPluginsPreparation_(plugins) {\r\n\r\n /**\r\n * @calls allPluginsProcessed__ when all plugins prepared or failed\r\n */\r\n return new Promise (function (allPluginsProcessed__) {\r\n\r\n /**\r\n * pluck each element from queue\r\n * First, send resolved Promise as previous value\r\n * Each plugins \"prepare\" method returns a Promise, that's why\r\n * reduce current element will not be able to continue while can't get\r\n * a resolved Promise\r\n *\r\n * If last plugin is \"prepared\" then go to the next stage of initialization\r\n */\r\n plugins.reduce(function (previousValue, plugin, iteration) {\r\n\r\n return previousValue.then(function () {\r\n\r\n /**\r\n * Wait till plugins prepared\r\n * @calls pluginIsReady__ when plugin is ready or failed\r\n */\r\n return new Promise ( function (pluginIsReady__) {\r\n\r\n callPluginsPrepareMethod_( plugin )\r\n\r\n .then( pluginIsReady__ )\r\n .then( function () {\r\n\r\n plugin.available = true;\r\n\r\n })\r\n\r\n .catch(function (error) {\r\n\r\n editor.core.log(`Plugin «${plugin.type}» was not loaded. Preparation failed because %o`, 'warn', error);\r\n plugin.available = false;\r\n plugin.loadingMessage = error;\r\n\r\n /** Go ahead even some plugin has problems */\r\n pluginIsReady__();\r\n\r\n })\r\n\r\n .then(function () {\r\n\r\n /** If last plugin has problems then just ignore and continue */\r\n if (iteration == plugins.length - 1) {\r\n\r\n allPluginsProcessed__();\r\n\r\n }\r\n\r\n });\r\n\r\n });\r\n\r\n });\r\n\r\n }, Promise.resolve() );\r\n\r\n });\r\n\r\n }\r\n\r\n var callPluginsPrepareMethod_ = function (plugin) {\r\n\r\n return plugin.prepare( plugin.config || {} );\r\n\r\n };\r\n\r\n return {\r\n prepare: prepare\r\n };\r\n\r\n}());\n\n\n// WEBPACK FOOTER //\n// ./modules/tools.js","/**\r\n * Codex Editor UI module\r\n *\r\n * @author Codex Team\r\n * @version 1.2.0\r\n */\r\n\r\nmodule.exports = (function (ui) {\r\n\r\n let editor = codex.editor;\r\n\r\n /**\r\n * Basic editor classnames\r\n */\r\n ui.className = {\r\n\r\n /**\r\n * @const {string} BLOCK_CLASSNAME - redactor blocks name\r\n */\r\n BLOCK_CLASSNAME : 'ce-block',\r\n\r\n /**\r\n * @const {String} wrapper for plugins content\r\n */\r\n BLOCK_CONTENT : 'ce-block__content',\r\n\r\n /**\r\n * @const {String} BLOCK_STRETCHED - makes block stretched\r\n */\r\n BLOCK_STRETCHED : 'ce-block--stretched',\r\n\r\n /**\r\n * @const {String} BLOCK_HIGHLIGHTED - adds background\r\n */\r\n BLOCK_HIGHLIGHTED : 'ce-block--focused',\r\n\r\n /**\r\n * @const {String} - for all default settings\r\n */\r\n SETTINGS_ITEM : 'ce-settings__item'\r\n\r\n };\r\n\r\n /**\r\n * @protected\r\n *\r\n * Making main interface\r\n */\r\n ui.prepare = function () {\r\n\r\n return new Promise(function (resolve) {\r\n\r\n let wrapper = editor.draw.wrapper(),\r\n redactor = editor.draw.redactor(),\r\n toolbar = makeToolBar_();\r\n\r\n wrapper.appendChild(toolbar);\r\n wrapper.appendChild(redactor);\r\n\r\n /** Save created ui-elements to static nodes state */\r\n editor.nodes.wrapper = wrapper;\r\n editor.nodes.redactor = redactor;\r\n\r\n /** Append editor wrapper with redactor zone into holder */\r\n editor.nodes.holder.appendChild(wrapper);\r\n\r\n resolve();\r\n\r\n })\r\n\r\n /** Add toolbox tools */\r\n .then(addTools_)\r\n\r\n /** Make container for inline toolbar */\r\n .then(makeInlineToolbar_)\r\n\r\n /** Add inline toolbar tools */\r\n .then(addInlineToolbarTools_)\r\n\r\n /** Draw wrapper for notifications */\r\n .then(makeNotificationHolder_)\r\n\r\n /** Add eventlisteners to redactor elements */\r\n .then(bindEvents_)\r\n\r\n .catch( function () {\r\n\r\n editor.core.log(\"Can't draw editor interface\");\r\n\r\n });\r\n\r\n };\r\n\r\n /**\r\n * @private\r\n * Draws inline toolbar zone\r\n */\r\n var makeInlineToolbar_ = function () {\r\n\r\n var container = editor.draw.inlineToolbar();\r\n\r\n /** Append to redactor new inline block */\r\n editor.nodes.inlineToolbar.wrapper = container;\r\n\r\n /** Draw toolbar buttons */\r\n editor.nodes.inlineToolbar.buttons = editor.draw.inlineToolbarButtons();\r\n\r\n /** Buttons action or settings */\r\n editor.nodes.inlineToolbar.actions = editor.draw.inlineToolbarActions();\r\n\r\n /** Append to inline toolbar buttons as part of it */\r\n editor.nodes.inlineToolbar.wrapper.appendChild(editor.nodes.inlineToolbar.buttons);\r\n editor.nodes.inlineToolbar.wrapper.appendChild(editor.nodes.inlineToolbar.actions);\r\n\r\n editor.nodes.wrapper.appendChild(editor.nodes.inlineToolbar.wrapper);\r\n\r\n };\r\n\r\n var makeToolBar_ = function () {\r\n\r\n let toolbar = editor.draw.toolbar(),\r\n blockButtons = makeToolbarSettings_(),\r\n toolbarContent = makeToolbarContent_();\r\n\r\n /** Appending first-level block buttons */\r\n toolbar.appendChild(blockButtons);\r\n\r\n /** Append toolbarContent to toolbar */\r\n toolbar.appendChild(toolbarContent);\r\n\r\n /** Make toolbar global */\r\n editor.nodes.toolbar = toolbar;\r\n\r\n return toolbar;\r\n\r\n };\r\n\r\n var makeToolbarContent_ = function () {\r\n\r\n let toolbarContent = editor.draw.toolbarContent(),\r\n toolbox = editor.draw.toolbox(),\r\n plusButton = editor.draw.plusButton();\r\n\r\n /** Append plus button */\r\n toolbarContent.appendChild(plusButton);\r\n\r\n /** Appending toolbar tools */\r\n toolbarContent.appendChild(toolbox);\r\n\r\n /** Make Toolbox and plusButton global */\r\n editor.nodes.toolbox = toolbox;\r\n editor.nodes.plusButton = plusButton;\r\n\r\n return toolbarContent;\r\n\r\n };\r\n\r\n var makeToolbarSettings_ = function () {\r\n\r\n let blockSettings = editor.draw.blockSettings(),\r\n blockButtons = editor.draw.blockButtons(),\r\n defaultSettings = editor.draw.defaultSettings(),\r\n showSettingsButton = editor.draw.settingsButton(),\r\n showTrashButton = editor.toolbar.settings.makeRemoveBlockButton(),\r\n pluginSettings = editor.draw.pluginsSettings();\r\n\r\n /** Add default and plugins settings */\r\n blockSettings.appendChild(pluginSettings);\r\n blockSettings.appendChild(defaultSettings);\r\n\r\n /**\r\n * Make blocks buttons\r\n * This block contains settings button and remove block button\r\n */\r\n blockButtons.appendChild(showSettingsButton);\r\n blockButtons.appendChild(showTrashButton);\r\n blockButtons.appendChild(blockSettings);\r\n\r\n /** Make BlockSettings, PluginSettings, DefaultSettings global */\r\n editor.nodes.blockSettings = blockSettings;\r\n editor.nodes.pluginSettings = pluginSettings;\r\n editor.nodes.defaultSettings = defaultSettings;\r\n editor.nodes.showSettingsButton = showSettingsButton;\r\n editor.nodes.showTrashButton = showTrashButton;\r\n\r\n return blockButtons;\r\n\r\n };\r\n\r\n /** Draw notifications holder */\r\n var makeNotificationHolder_ = function () {\r\n\r\n /** Append block with notifications to the document */\r\n editor.nodes.notifications = editor.notifications.createHolder();\r\n\r\n };\r\n\r\n /**\r\n * @private\r\n * Append tools passed in editor.tools\r\n */\r\n var addTools_ = function () {\r\n\r\n var tool,\r\n toolName,\r\n toolButton;\r\n\r\n for ( toolName in editor.settings.tools ) {\r\n\r\n tool = editor.settings.tools[toolName];\r\n\r\n editor.tools[toolName] = tool;\r\n\r\n if (!tool.iconClassname && tool.displayInToolbox) {\r\n\r\n editor.core.log('Toolbar icon classname missed. Tool %o skipped', 'warn', toolName);\r\n continue;\r\n\r\n }\r\n\r\n if (typeof tool.render != 'function') {\r\n\r\n editor.core.log('render method missed. Tool %o skipped', 'warn', toolName);\r\n continue;\r\n\r\n }\r\n\r\n if (!tool.displayInToolbox) {\r\n\r\n continue;\r\n\r\n } else {\r\n\r\n /** if tools is for toolbox */\r\n toolButton = editor.draw.toolbarButton(toolName, tool.iconClassname);\r\n\r\n editor.nodes.toolbox.appendChild(toolButton);\r\n\r\n editor.nodes.toolbarButtons[toolName] = toolButton;\r\n\r\n }\r\n\r\n }\r\n\r\n };\r\n\r\n var addInlineToolbarTools_ = function () {\r\n\r\n var tools = {\r\n\r\n bold: {\r\n icon : 'ce-icon-bold',\r\n command : 'bold'\r\n },\r\n\r\n italic: {\r\n icon : 'ce-icon-italic',\r\n command : 'italic'\r\n },\r\n\r\n underline: {\r\n icon : 'ce-icon-underline',\r\n command : 'underline'\r\n },\r\n\r\n link: {\r\n icon : 'ce-icon-link',\r\n command : 'createLink'\r\n }\r\n };\r\n\r\n var toolButton,\r\n tool;\r\n\r\n for(var name in tools) {\r\n\r\n tool = tools[name];\r\n\r\n toolButton = editor.draw.toolbarButtonInline(name, tool.icon);\r\n\r\n editor.nodes.inlineToolbar.buttons.appendChild(toolButton);\r\n /**\r\n * Add callbacks to this buttons\r\n */\r\n editor.ui.setInlineToolbarButtonBehaviour(toolButton, tool.command);\r\n\r\n }\r\n\r\n };\r\n\r\n /**\r\n * @private\r\n * Bind editor UI events\r\n */\r\n var bindEvents_ = function () {\r\n\r\n editor.core.log('ui.bindEvents fired', 'info');\r\n\r\n // window.addEventListener('error', function (errorMsg, url, lineNumber) {\r\n // editor.notifications.errorThrown(errorMsg, event);\r\n // }, false );\r\n\r\n /** All keydowns on Document */\r\n editor.listeners.add(document, 'keydown', editor.callback.globalKeydown, false);\r\n\r\n /** All keydowns on Redactor zone */\r\n editor.listeners.add(editor.nodes.redactor, 'keydown', editor.callback.redactorKeyDown, false);\r\n\r\n /** All keydowns on Document */\r\n editor.listeners.add(document, 'keyup', editor.callback.globalKeyup, false );\r\n\r\n /**\r\n * Mouse click to radactor\r\n */\r\n editor.listeners.add(editor.nodes.redactor, 'click', editor.callback.redactorClicked, false );\r\n\r\n /**\r\n * Clicks to the Plus button\r\n */\r\n editor.listeners.add(editor.nodes.plusButton, 'click', editor.callback.plusButtonClicked, false);\r\n\r\n /**\r\n * Clicks to SETTINGS button in toolbar\r\n */\r\n editor.listeners.add(editor.nodes.showSettingsButton, 'click', editor.callback.showSettingsButtonClicked, false );\r\n\r\n /** Bind click listeners on toolbar buttons */\r\n for (var button in editor.nodes.toolbarButtons) {\r\n\r\n editor.listeners.add(editor.nodes.toolbarButtons[button], 'click', editor.callback.toolbarButtonClicked, false);\r\n\r\n }\r\n\r\n };\r\n\r\n ui.addBlockHandlers = function (block) {\r\n\r\n if (!block) return;\r\n\r\n /**\r\n * Block keydowns\r\n */\r\n editor.listeners.add(block, 'keydown', editor.callback.blockKeydown, false);\r\n\r\n /**\r\n * Pasting content from another source\r\n * We have two type of sanitization\r\n * First - uses deep-first search algorithm to get sub nodes,\r\n * sanitizes whole Block_content and replaces cleared nodes\r\n * This method is deprecated\r\n * Method is used in editor.callback.blockPaste(event)\r\n *\r\n * Secont - uses Mutation observer.\r\n * Observer \"observe\" DOM changes and send changings to callback.\r\n * Callback gets changed node, not whole Block_content.\r\n * Inserted or changed node, which we've gotten have been cleared and replaced with diry node\r\n *\r\n * Method is used in editor.callback.blockPasteViaSanitize(event)\r\n *\r\n * @uses html-janitor\r\n * @example editor.callback.blockPasteViaSanitize(event), the second method.\r\n *\r\n */\r\n editor.listeners.add(block, 'paste', editor.callback.blockPasteCallback, false);\r\n\r\n editor.listeners.add(block, 'mouseup', editor.toolbar.inline.show, false);\r\n\r\n };\r\n\r\n /** getting all contenteditable elements */\r\n ui.saveInputs = function () {\r\n\r\n var redactor = editor.nodes.redactor;\r\n\r\n editor.state.inputs = [];\r\n\r\n /** Save all inputs in global variable state */\r\n var inputs = redactor.querySelectorAll('[contenteditable], input, textarea');\r\n\r\n Array.prototype.map.call(inputs, function (current) {\r\n\r\n if (!current.type || current.type == 'text' || current.type == 'textarea') {\r\n\r\n editor.state.inputs.push(current);\r\n\r\n }\r\n\r\n });\r\n\r\n };\r\n\r\n /**\r\n * Adds first initial block on empty redactor\r\n */\r\n ui.addInitialBlock = function () {\r\n\r\n var initialBlockType = editor.settings.initialBlockPlugin,\r\n initialBlock;\r\n\r\n if ( !editor.tools[initialBlockType] ) {\r\n\r\n editor.core.log('Plugin %o was not implemented and can\\'t be used as initial block', 'warn', initialBlockType);\r\n return;\r\n\r\n }\r\n\r\n initialBlock = editor.tools[initialBlockType].render();\r\n\r\n initialBlock.setAttribute('data-placeholder', 'Расскажите свою историю...');\r\n\r\n editor.content.insertBlock({\r\n type : initialBlockType,\r\n block : initialBlock\r\n });\r\n\r\n editor.content.workingNodeChanged(initialBlock);\r\n\r\n };\r\n\r\n ui.setInlineToolbarButtonBehaviour = function (button, type) {\r\n\r\n editor.listeners.add(button, 'mousedown', function (event) {\r\n\r\n editor.toolbar.inline.toolClicked(event, type);\r\n\r\n }, false);\r\n\r\n };\r\n\r\n return ui;\r\n\r\n})({});\r\n\n\n\n// WEBPACK FOOTER //\n// ./modules/ui.js","/**\r\n *\r\n * Codex.Editor Transport Module\r\n *\r\n * @copyright 2017 Codex-Team\r\n * @version 1.2.0\r\n */\r\n\r\nmodule.exports = (function (transport) {\r\n\r\n let editor = codex.editor;\r\n\r\n\r\n /**\r\n * @private {Object} current XmlHttpRequest instance\r\n */\r\n var currentRequest = null;\r\n\r\n\r\n /**\r\n * @type {null} | {DOMElement} input - keeps input element in memory\r\n */\r\n transport.input = null;\r\n\r\n /**\r\n * @property {Object} arguments - keep plugin settings and defined callbacks\r\n */\r\n transport.arguments = null;\r\n\r\n /**\r\n * Prepares input element where will be files\r\n */\r\n transport.prepare = function () {\r\n\r\n let input = editor.draw.node( 'INPUT', '', { type : 'file' } );\r\n\r\n editor.listeners.add(input, 'change', editor.transport.fileSelected);\r\n editor.transport.input = input;\r\n\r\n };\r\n\r\n /** Clear input when files is uploaded */\r\n transport.clearInput = function () {\r\n\r\n /** Remove old input */\r\n transport.input = null;\r\n\r\n /** Prepare new one */\r\n transport.prepare();\r\n\r\n };\r\n\r\n /**\r\n * Callback for file selection\r\n * @param {Event} event\r\n */\r\n transport.fileSelected = function () {\r\n\r\n var input = this,\r\n i,\r\n files = input.files,\r\n formData = new FormData();\r\n\r\n if (editor.transport.arguments.multiple === true) {\r\n\r\n for ( i = 0; i < files.length; i++) {\r\n\r\n formData.append('files[]', files[i], files[i].name);\r\n\r\n }\r\n\r\n } else {\r\n\r\n formData.append('files', files[0], files[0].name);\r\n\r\n }\r\n\r\n currentRequest = editor.core.ajax({\r\n type : 'POST',\r\n data : formData,\r\n url : editor.transport.arguments.url,\r\n beforeSend : editor.transport.arguments.beforeSend,\r\n success : editor.transport.arguments.success,\r\n error : editor.transport.arguments.error,\r\n progress : editor.transport.arguments.progress\r\n });\r\n\r\n /** Clear input */\r\n transport.clearInput();\r\n\r\n };\r\n\r\n /**\r\n * Use plugin callbacks\r\n * @protected\r\n *\r\n * @param {Object} args - can have :\r\n * @param {String} args.url - fetch URL\r\n * @param {Function} args.beforeSend - function calls before sending ajax\r\n * @param {Function} args.success - success callback\r\n * @param {Function} args.error - on error handler\r\n * @param {Function} args.progress - xhr onprogress handler\r\n * @param {Boolean} args.multiple - allow select several files\r\n * @param {String} args.accept - adds accept attribute\r\n */\r\n transport.selectAndUpload = function (args) {\r\n\r\n transport.arguments = args;\r\n\r\n if ( args.multiple === true) {\r\n\r\n transport.input.setAttribute('multiple', 'multiple');\r\n\r\n }\r\n\r\n if ( args.accept ) {\r\n\r\n transport.input.setAttribute('accept', args.accept);\r\n\r\n }\r\n\r\n transport.input.click();\r\n\r\n };\r\n\r\n transport.abort = function () {\r\n\r\n currentRequest.abort();\r\n\r\n currentRequest = null;\r\n\r\n };\r\n\r\n return transport;\r\n\r\n})({});\n\n\n// WEBPACK FOOTER //\n// ./modules/transport.js","/**\r\n * Codex Editor Renderer Module\r\n *\r\n * @author Codex Team\r\n * @version 1.0\r\n */\r\n\r\nmodule.exports = (function (renderer) {\r\n\r\n let editor = codex.editor;\r\n\r\n /**\r\n * Asyncronously parses input JSON to redactor blocks\r\n */\r\n renderer.makeBlocksFromData = function () {\r\n\r\n /**\r\n * If redactor is empty, add first paragraph to start writing\r\n */\r\n if (editor.core.isEmpty(editor.state.blocks) || !editor.state.blocks.items.length) {\r\n\r\n editor.ui.addInitialBlock();\r\n return;\r\n\r\n }\r\n\r\n Promise.resolve()\r\n\r\n /** First, get JSON from state */\r\n .then(function () {\r\n\r\n return editor.state.blocks;\r\n\r\n })\r\n\r\n /** Then, start to iterate they */\r\n .then(editor.renderer.appendBlocks)\r\n\r\n /** Write log if something goes wrong */\r\n .catch(function (error) {\r\n\r\n editor.core.log('Error while parsing JSON: %o', 'error', error);\r\n\r\n });\r\n\r\n };\r\n\r\n /**\r\n * Parses JSON to blocks\r\n * @param {object} data\r\n * @return Primise -> nodeList\r\n */\r\n renderer.appendBlocks = function (data) {\r\n\r\n var blocks = data.items;\r\n\r\n /**\r\n * Sequence of one-by-one blocks appending\r\n * Uses to save blocks order after async-handler\r\n */\r\n var nodeSequence = Promise.resolve();\r\n\r\n for (var index = 0; index < blocks.length ; index++ ) {\r\n\r\n /** Add node to sequence at specified index */\r\n editor.renderer.appendNodeAtIndex(nodeSequence, blocks, index);\r\n\r\n }\r\n\r\n };\r\n\r\n /**\r\n * Append node at specified index\r\n */\r\n renderer.appendNodeAtIndex = function (nodeSequence, blocks, index) {\r\n\r\n /** We need to append node to sequence */\r\n nodeSequence\r\n\r\n /** first, get node async-aware */\r\n .then(function () {\r\n\r\n return editor.renderer.getNodeAsync(blocks, index);\r\n\r\n })\r\n\r\n /**\r\n * second, compose editor-block from JSON object\r\n */\r\n .then(editor.renderer.createBlockFromData)\r\n\r\n /**\r\n * now insert block to redactor\r\n */\r\n .then(function (blockData) {\r\n\r\n /**\r\n * blockData has 'block', 'type' and 'stretched' information\r\n */\r\n editor.content.insertBlock(blockData);\r\n\r\n /** Pass created block to next step */\r\n return blockData.block;\r\n\r\n })\r\n\r\n /** Log if something wrong with node */\r\n .catch(function (error) {\r\n\r\n editor.core.log('Node skipped while parsing because %o', 'error', error);\r\n\r\n });\r\n\r\n };\r\n\r\n /**\r\n * Asynchronously returns block data from blocksList by index\r\n * @return Promise to node\r\n */\r\n renderer.getNodeAsync = function (blocksList, index) {\r\n\r\n return Promise.resolve().then(function () {\r\n\r\n return {\r\n tool : blocksList[index],\r\n position : index\r\n };\r\n\r\n });\r\n\r\n };\r\n\r\n /**\r\n * Creates editor block by JSON-data\r\n *\r\n * @uses render method of each plugin\r\n *\r\n * @param {Object} toolData.tool\r\n * { header : {\r\n * text: '',\r\n * type: 'H3', ...\r\n * }\r\n * }\r\n * @param {Number} toolData.position - index in input-blocks array\r\n * @return {Object} with type and Element\r\n */\r\n renderer.createBlockFromData = function ( toolData ) {\r\n\r\n /** New parser */\r\n var block,\r\n tool = toolData.tool,\r\n pluginName = tool.type;\r\n\r\n /** Get first key of object that stores plugin name */\r\n // for (var pluginName in blockData) break;\r\n\r\n /** Check for plugin existance */\r\n if (!editor.tools[pluginName]) {\r\n\r\n throw Error(`Plugin «${pluginName}» not found`);\r\n\r\n }\r\n\r\n /** Check for plugin having render method */\r\n if (typeof editor.tools[pluginName].render != 'function') {\r\n\r\n throw Error(`Plugin «${pluginName}» must have «render» method`);\r\n\r\n }\r\n\r\n if ( editor.tools[pluginName].available === false ) {\r\n\r\n block = editor.draw.unavailableBlock();\r\n\r\n block.innerHTML = editor.tools[pluginName].loadingMessage;\r\n\r\n /**\r\n * Saver will extract data from initial block data by position in array\r\n */\r\n block.dataset.inputPosition = toolData.position;\r\n\r\n } else {\r\n\r\n /** New Parser */\r\n block = editor.tools[pluginName].render(tool.data);\r\n\r\n }\r\n\r\n /** is first-level block stretched */\r\n var stretched = editor.tools[pluginName].isStretched || false;\r\n\r\n /** Retrun type and block */\r\n return {\r\n type : pluginName,\r\n block : block,\r\n stretched : stretched\r\n };\r\n\r\n };\r\n\r\n return renderer;\r\n\r\n})({});\n\n\n// WEBPACK FOOTER //\n// ./modules/renderer.js","/**\r\n * Codex Editor Saver\r\n *\r\n * @author Codex Team\r\n * @version 1.0.2\r\n */\r\n\r\nmodule.exports = (function (saver) {\r\n\r\n let editor = codex.editor;\r\n\r\n /**\r\n * Saves blocks\r\n * @private\r\n */\r\n saver.saveBlocks = function () {\r\n\r\n /** Save html content of redactor to memory */\r\n editor.state.html = editor.nodes.redactor.innerHTML;\r\n\r\n /** Empty jsonOutput state */\r\n editor.state.jsonOutput = [];\r\n\r\n Promise.resolve()\r\n\r\n .then(function () {\r\n\r\n return editor.nodes.redactor.childNodes;\r\n\r\n })\r\n /** Making a sequence from separate blocks */\r\n .then(editor.saver.makeQueue)\r\n\r\n .then(function () {\r\n // editor.nodes.textarea.innerHTML = editor.state.html;\r\n })\r\n\r\n .catch( function (error) {\r\n\r\n editor.core.log(error);\r\n\r\n });\r\n\r\n };\r\n\r\n saver.makeQueue = function (blocks) {\r\n\r\n var queue = Promise.resolve();\r\n\r\n for(var index = 0; index < blocks.length; index++) {\r\n\r\n /** Add node to sequence at specified index */\r\n editor.saver.getBlockData(queue, blocks, index);\r\n\r\n }\r\n\r\n };\r\n\r\n /** Gets every block and makes From Data */\r\n saver.getBlockData = function (queue, blocks, index) {\r\n\r\n queue.then(function () {\r\n\r\n return editor.saver.getNodeAsync(blocks, index);\r\n\r\n })\r\n\r\n .then(editor.saver.makeFormDataFromBlocks);\r\n\r\n };\r\n\r\n\r\n /**\r\n * Asynchronously returns block data from blocksList by index\r\n * @return Promise to node\r\n */\r\n saver.getNodeAsync = function (blocksList, index) {\r\n\r\n return Promise.resolve().then(function () {\r\n\r\n return blocksList[index];\r\n\r\n });\r\n\r\n };\r\n\r\n saver.makeFormDataFromBlocks = function (block) {\r\n\r\n var pluginName = block.dataset.tool;\r\n\r\n /** Check for plugin existance */\r\n if (!editor.tools[pluginName]) {\r\n\r\n throw Error(`Plugin «${pluginName}» not found`);\r\n\r\n }\r\n\r\n /** Check for plugin having render method */\r\n if (typeof editor.tools[pluginName].save != 'function') {\r\n\r\n throw Error(`Plugin «${pluginName}» must have save method`);\r\n\r\n }\r\n\r\n /** Result saver */\r\n var blockContent = block.childNodes[0],\r\n pluginsContent = blockContent.childNodes[0],\r\n savedData,\r\n position,\r\n output;\r\n\r\n /** If plugin wasn't available then return data from cache */\r\n if ( editor.tools[pluginName].available === false ) {\r\n\r\n position = pluginsContent.dataset.inputPosition;\r\n savedData = codex.editor.state.blocks.items[position].data;\r\n\r\n } else {\r\n\r\n savedData = editor.tools[pluginName].save(pluginsContent);\r\n\r\n if (editor.tools[pluginName].validate) {\r\n\r\n var result = editor.tools[pluginName].validate(savedData);\r\n\r\n /**\r\n * Do not allow invalid data\r\n */\r\n if (!result)\r\n return;\r\n\r\n }\r\n\r\n }\r\n\r\n output = {\r\n type : pluginName,\r\n data : savedData\r\n };\r\n\r\n editor.state.jsonOutput.push(output);\r\n\r\n };\r\n\r\n return saver;\r\n\r\n})({});\n\n\n// WEBPACK FOOTER //\n// ./modules/saver.js","/**\r\n * Codex Editor Content Module\r\n * Works with DOM\r\n *\r\n * @module Codex Editor content module\r\n *\r\n * @author Codex Team\r\n * @version 1.3.13\r\n *\r\n * @description Module works with Elements that have been appended to the main DOM\r\n */\r\n\r\nmodule.exports = (function (content) {\r\n\r\n let editor = codex.editor;\r\n\r\n /**\r\n * Links to current active block\r\n * @type {null | Element}\r\n */\r\n content.currentNode = null;\r\n\r\n /**\r\n * clicked in redactor area\r\n * @type {null | Boolean}\r\n */\r\n content.editorAreaHightlighted = null;\r\n\r\n /**\r\n * @deprecated\r\n * Synchronizes redactor with original textarea\r\n */\r\n content.sync = function () {\r\n\r\n editor.core.log('syncing...');\r\n\r\n /**\r\n * Save redactor content to editor.state\r\n */\r\n editor.state.html = editor.nodes.redactor.innerHTML;\r\n\r\n };\r\n\r\n /**\r\n * Appends background to the block\r\n *\r\n * @description add CSS class to highlight visually first-level block area\r\n */\r\n content.markBlock = function () {\r\n\r\n editor.content.currentNode.classList.add(editor.ui.className.BLOCK_HIGHLIGHTED);\r\n\r\n };\r\n\r\n /**\r\n * Clear background\r\n *\r\n * @description clears styles that highlights block\r\n */\r\n content.clearMark = function () {\r\n\r\n if (editor.content.currentNode) {\r\n\r\n editor.content.currentNode.classList.remove(editor.ui.className.BLOCK_HIGHLIGHTED);\r\n\r\n }\r\n\r\n };\r\n\r\n /**\r\n * Finds first-level block\r\n *\r\n * @param {Element} node - selected or clicked in redactors area node\r\n * @protected\r\n *\r\n * @description looks for first-level block.\r\n * gets parent while node is not first-level\r\n */\r\n content.getFirstLevelBlock = function (node) {\r\n\r\n if (!editor.core.isDomNode(node)) {\r\n\r\n node = node.parentNode;\r\n\r\n }\r\n\r\n if (node === editor.nodes.redactor || node === document.body) {\r\n\r\n return null;\r\n\r\n } else {\r\n\r\n while(!node.classList.contains(editor.ui.className.BLOCK_CLASSNAME)) {\r\n\r\n node = node.parentNode;\r\n\r\n }\r\n\r\n return node;\r\n\r\n }\r\n\r\n };\r\n\r\n /**\r\n * Trigger this event when working node changed\r\n * @param {Element} targetNode - first-level of this node will be current\r\n * @protected\r\n *\r\n * @description If targetNode is first-level then we set it as current else we look for parents to find first-level\r\n */\r\n content.workingNodeChanged = function (targetNode) {\r\n\r\n /** Clear background from previous marked block before we change */\r\n editor.content.clearMark();\r\n\r\n if (!targetNode) {\r\n\r\n return;\r\n\r\n }\r\n\r\n content.currentNode = content.getFirstLevelBlock(targetNode);\r\n\r\n };\r\n\r\n /**\r\n * Replaces one redactor block with another\r\n * @protected\r\n * @param {Element} targetBlock - block to replace. Mostly currentNode.\r\n * @param {Element} newBlock\r\n * @param {string} newBlockType - type of new block; we need to store it to data-attribute\r\n *\r\n * [!] Function does not saves old block content.\r\n * You can get it manually and pass with newBlock.innerHTML\r\n */\r\n content.replaceBlock = function (targetBlock, newBlock) {\r\n\r\n if (!targetBlock || !newBlock) {\r\n\r\n editor.core.log('replaceBlock: missed params');\r\n return;\r\n\r\n }\r\n\r\n /** If target-block is not a frist-level block, then we iterate parents to find it */\r\n while(!targetBlock.classList.contains(editor.ui.className.BLOCK_CLASSNAME)) {\r\n\r\n targetBlock = targetBlock.parentNode;\r\n\r\n }\r\n\r\n /** Replacing */\r\n editor.nodes.redactor.replaceChild(newBlock, targetBlock);\r\n\r\n /**\r\n * Set new node as current\r\n */\r\n editor.content.workingNodeChanged(newBlock);\r\n\r\n /**\r\n * Add block handlers\r\n */\r\n editor.ui.addBlockHandlers(newBlock);\r\n\r\n /**\r\n * Save changes\r\n */\r\n editor.ui.saveInputs();\r\n\r\n };\r\n\r\n /**\r\n * @protected\r\n *\r\n * Inserts new block to redactor\r\n * Wrapps block into a DIV with BLOCK_CLASSNAME class\r\n *\r\n * @param blockData {object}\r\n * @param blockData.block {Element} element with block content\r\n * @param blockData.type {string} block plugin\r\n * @param needPlaceCaret {bool} pass true to set caret in new block\r\n *\r\n */\r\n content.insertBlock = function ( blockData, needPlaceCaret ) {\r\n\r\n var workingBlock = editor.content.currentNode,\r\n newBlockContent = blockData.block,\r\n blockType = blockData.type,\r\n isStretched = blockData.stretched;\r\n\r\n var newBlock = composeNewBlock_(newBlockContent, blockType, isStretched);\r\n\r\n if (workingBlock) {\r\n\r\n editor.core.insertAfter(workingBlock, newBlock);\r\n\r\n } else {\r\n\r\n /**\r\n * If redactor is empty, append as first child\r\n */\r\n editor.nodes.redactor.appendChild(newBlock);\r\n\r\n }\r\n\r\n /**\r\n * Block handler\r\n */\r\n editor.ui.addBlockHandlers(newBlock);\r\n\r\n /**\r\n * Set new node as current\r\n */\r\n editor.content.workingNodeChanged(newBlock);\r\n\r\n /**\r\n * Save changes\r\n */\r\n editor.ui.saveInputs();\r\n\r\n\r\n if ( needPlaceCaret ) {\r\n\r\n /**\r\n * If we don't know input index then we set default value -1\r\n */\r\n var currentInputIndex = editor.caret.getCurrentInputIndex() || -1;\r\n\r\n\r\n if (currentInputIndex == -1) {\r\n\r\n\r\n var editableElement = newBlock.querySelector('[contenteditable]'),\r\n emptyText = document.createTextNode('');\r\n\r\n editableElement.appendChild(emptyText);\r\n editor.caret.set(editableElement, 0, 0);\r\n\r\n editor.toolbar.move();\r\n editor.toolbar.showPlusButton();\r\n\r\n\r\n } else {\r\n\r\n if (currentInputIndex === editor.state.inputs.length - 1)\r\n return;\r\n\r\n /** Timeout for browsers execution */\r\n window.setTimeout(function () {\r\n\r\n /** Setting to the new input */\r\n editor.caret.setToNextBlock(currentInputIndex);\r\n editor.toolbar.move();\r\n editor.toolbar.open();\r\n\r\n }, 10);\r\n\r\n }\r\n\r\n }\r\n\r\n /**\r\n * Block is inserted, wait for new click that defined focusing on editors area\r\n * @type {boolean}\r\n */\r\n content.editorAreaHightlighted = false;\r\n\r\n };\r\n\r\n /**\r\n * Replaces blocks with saving content\r\n * @protected\r\n * @param {Element} noteToReplace\r\n * @param {Element} newNode\r\n * @param {Element} blockType\r\n */\r\n content.switchBlock = function (blockToReplace, newBlock, tool) {\r\n\r\n tool = tool || editor.content.currentNode.dataset.tool;\r\n var newBlockComposed = composeNewBlock_(newBlock, tool);\r\n\r\n /** Replacing */\r\n editor.content.replaceBlock(blockToReplace, newBlockComposed);\r\n\r\n /** Save new Inputs when block is changed */\r\n editor.ui.saveInputs();\r\n\r\n };\r\n\r\n /**\r\n * Iterates between child noted and looking for #text node on deepest level\r\n * @protected\r\n *\r\n * @param {Element} block - node where find\r\n * @param {int} postiton - starting postion\r\n * Example: childNodex.length to find from the end\r\n * or 0 to find from the start\r\n * @return {Text} block\r\n * @uses DFS\r\n */\r\n content.getDeepestTextNodeFromPosition = function (block, position) {\r\n\r\n /**\r\n * Clear Block from empty and useless spaces with trim.\r\n * Such nodes we should remove\r\n */\r\n var blockChilds = block.childNodes,\r\n index,\r\n node,\r\n text;\r\n\r\n for(index = 0; index < blockChilds.length; index++) {\r\n\r\n node = blockChilds[index];\r\n\r\n if (node.nodeType == editor.core.nodeTypes.TEXT) {\r\n\r\n text = node.textContent.trim();\r\n\r\n /** Text is empty. We should remove this child from node before we start DFS\r\n * decrease the quantity of childs.\r\n */\r\n if (text === '') {\r\n\r\n block.removeChild(node);\r\n position--;\r\n\r\n }\r\n\r\n }\r\n\r\n }\r\n\r\n if (block.childNodes.length === 0) {\r\n\r\n return document.createTextNode('');\r\n\r\n }\r\n\r\n /** Setting default position when we deleted all empty nodes */\r\n if ( position < 0 )\r\n position = 1;\r\n\r\n var lookingFromStart = false;\r\n\r\n /** For looking from START */\r\n if (position === 0) {\r\n\r\n lookingFromStart = true;\r\n position = 1;\r\n\r\n }\r\n\r\n while ( position ) {\r\n\r\n /** initial verticle of node. */\r\n if ( lookingFromStart ) {\r\n\r\n block = block.childNodes[0];\r\n\r\n } else {\r\n\r\n block = block.childNodes[position - 1];\r\n\r\n }\r\n\r\n if ( block.nodeType == editor.core.nodeTypes.TAG ) {\r\n\r\n position = block.childNodes.length;\r\n\r\n } else if (block.nodeType == editor.core.nodeTypes.TEXT ) {\r\n\r\n position = 0;\r\n\r\n }\r\n\r\n }\r\n\r\n return block;\r\n\r\n };\r\n\r\n /**\r\n * @private\r\n * @param {Element} block - current plugins render\r\n * @param {String} tool - plugins name\r\n * @param {Boolean} isStretched - make stretched block or not\r\n *\r\n * @description adds necessary information to wrap new created block by first-level holder\r\n */\r\n var composeNewBlock_ = function (block, tool, isStretched) {\r\n\r\n var newBlock = editor.draw.node('DIV', editor.ui.className.BLOCK_CLASSNAME, {}),\r\n blockContent = editor.draw.node('DIV', editor.ui.className.BLOCK_CONTENT, {});\r\n\r\n blockContent.appendChild(block);\r\n newBlock.appendChild(blockContent);\r\n\r\n if (isStretched) {\r\n\r\n blockContent.classList.add(editor.ui.className.BLOCK_STRETCHED);\r\n\r\n }\r\n\r\n newBlock.dataset.tool = tool;\r\n return newBlock;\r\n\r\n };\r\n\r\n /**\r\n * Returns Range object of current selection\r\n * @protected\r\n */\r\n content.getRange = function () {\r\n\r\n var selection = window.getSelection().getRangeAt(0);\r\n\r\n return selection;\r\n\r\n };\r\n\r\n /**\r\n * Divides block in two blocks (after and before caret)\r\n *\r\n * @protected\r\n * @param {int} inputIndex - target input index\r\n *\r\n * @description splits current input content to the separate blocks\r\n * When enter is pressed among the words, that text will be splited.\r\n */\r\n content.splitBlock = function (inputIndex) {\r\n\r\n var selection = window.getSelection(),\r\n anchorNode = selection.anchorNode,\r\n anchorNodeText = anchorNode.textContent,\r\n caretOffset = selection.anchorOffset,\r\n textBeforeCaret,\r\n textNodeBeforeCaret,\r\n textAfterCaret,\r\n textNodeAfterCaret;\r\n\r\n var currentBlock = editor.content.currentNode.querySelector('[contentEditable]');\r\n\r\n\r\n textBeforeCaret = anchorNodeText.substring(0, caretOffset);\r\n textAfterCaret = anchorNodeText.substring(caretOffset);\r\n\r\n textNodeBeforeCaret = document.createTextNode(textBeforeCaret);\r\n\r\n if (textAfterCaret) {\r\n\r\n textNodeAfterCaret = document.createTextNode(textAfterCaret);\r\n\r\n }\r\n\r\n var previousChilds = [],\r\n nextChilds = [],\r\n reachedCurrent = false;\r\n\r\n if (textNodeAfterCaret) {\r\n\r\n nextChilds.push(textNodeAfterCaret);\r\n\r\n }\r\n\r\n for ( var i = 0, child; !!(child = currentBlock.childNodes[i]); i++) {\r\n\r\n if ( child != anchorNode ) {\r\n\r\n if ( !reachedCurrent ) {\r\n\r\n previousChilds.push(child);\r\n\r\n } else {\r\n\r\n nextChilds.push(child);\r\n\r\n }\r\n\r\n } else {\r\n\r\n reachedCurrent = true;\r\n\r\n }\r\n\r\n }\r\n\r\n /** Clear current input */\r\n editor.state.inputs[inputIndex].innerHTML = '';\r\n\r\n /**\r\n * Append all childs founded before anchorNode\r\n */\r\n var previousChildsLength = previousChilds.length;\r\n\r\n for(i = 0; i < previousChildsLength; i++) {\r\n\r\n editor.state.inputs[inputIndex].appendChild(previousChilds[i]);\r\n\r\n }\r\n\r\n editor.state.inputs[inputIndex].appendChild(textNodeBeforeCaret);\r\n\r\n /**\r\n * Append text node which is after caret\r\n */\r\n var nextChildsLength = nextChilds.length,\r\n newNode = document.createElement('div');\r\n\r\n for(i = 0; i < nextChildsLength; i++) {\r\n\r\n newNode.appendChild(nextChilds[i]);\r\n\r\n }\r\n\r\n newNode = newNode.innerHTML;\r\n\r\n /** This type of block creates when enter is pressed */\r\n var NEW_BLOCK_TYPE = editor.settings.initialBlockPlugin;\r\n\r\n /**\r\n * Make new paragraph with text after caret\r\n */\r\n editor.content.insertBlock({\r\n type : NEW_BLOCK_TYPE,\r\n block : editor.tools[NEW_BLOCK_TYPE].render({\r\n text : newNode\r\n })\r\n }, true );\r\n\r\n };\r\n\r\n /**\r\n * Merges two blocks — current and target\r\n * If target index is not exist, then previous will be as target\r\n *\r\n * @protected\r\n * @param {int} currentInputIndex\r\n * @param {int} targetInputIndex\r\n *\r\n * @description gets two inputs indexes and merges into one\r\n */\r\n content.mergeBlocks = function (currentInputIndex, targetInputIndex) {\r\n\r\n /** If current input index is zero, then prevent method execution */\r\n if (currentInputIndex === 0) {\r\n\r\n return;\r\n\r\n }\r\n\r\n var targetInput,\r\n currentInputContent = editor.state.inputs[currentInputIndex].innerHTML;\r\n\r\n if (!targetInputIndex) {\r\n\r\n targetInput = editor.state.inputs[currentInputIndex - 1];\r\n\r\n } else {\r\n\r\n targetInput = editor.state.inputs[targetInputIndex];\r\n\r\n }\r\n\r\n targetInput.innerHTML += currentInputContent;\r\n\r\n };\r\n\r\n /**\r\n * Iterates all right siblings and parents, which has right siblings\r\n * while it does not reached the first-level block\r\n *\r\n * @param {Element} node\r\n * @return {boolean}\r\n */\r\n content.isLastNode = function (node) {\r\n\r\n // console.log('погнали перебор родителей');\r\n\r\n var allChecked = false;\r\n\r\n while ( !allChecked ) {\r\n\r\n // console.log('Смотрим на %o', node);\r\n // console.log('Проверим, пустые ли соседи справа');\r\n\r\n if ( !allSiblingsEmpty_(node) ) {\r\n\r\n // console.log('Есть непустые соседи. Узел не последний. Выходим.');\r\n return false;\r\n\r\n }\r\n\r\n node = node.parentNode;\r\n\r\n /**\r\n * Проверяем родителей до тех пор, пока не найдем блок первого уровня\r\n */\r\n if ( node.classList.contains(editor.ui.className.BLOCK_CONTENT) ) {\r\n\r\n allChecked = true;\r\n\r\n }\r\n\r\n }\r\n\r\n return true;\r\n\r\n };\r\n\r\n /**\r\n * Checks if all element right siblings is empty\r\n * @param node\r\n */\r\n var allSiblingsEmpty_ = function (node) {\r\n\r\n /**\r\n * Нужно убедиться, что после пустого соседа ничего нет\r\n */\r\n var sibling = node.nextSibling;\r\n\r\n while ( sibling ) {\r\n\r\n if (sibling.textContent.length) {\r\n\r\n return false;\r\n\r\n }\r\n\r\n sibling = sibling.nextSibling;\r\n\r\n }\r\n\r\n return true;\r\n\r\n };\r\n\r\n /**\r\n * @public\r\n *\r\n * @param {string} htmlData - html content as string\r\n * @param {string} plainData - plain text\r\n * @return {string} - html content as string\r\n */\r\n content.wrapTextWithParagraphs = function (htmlData, plainData) {\r\n\r\n if (!htmlData) {\r\n\r\n return '

    ' + plainData.split('\\n\\n').join('

    ') + '

    ';\r\n\r\n }\r\n\r\n var wrapper = document.createElement('DIV'),\r\n newWrapper = document.createElement('DIV'),\r\n i,\r\n paragraph,\r\n firstLevelBlocks = ['DIV', 'P'],\r\n blockTyped,\r\n node;\r\n\r\n /**\r\n * Make HTML Element to Wrap Text\r\n * It allows us to work with input data as HTML content\r\n */\r\n wrapper.innerHTML = htmlData;\r\n paragraph = document.createElement('P');\r\n\r\n for (i = 0; i < wrapper.childNodes.length; i++) {\r\n\r\n node = wrapper.childNodes[i];\r\n\r\n blockTyped = firstLevelBlocks.indexOf(node.tagName) != -1;\r\n\r\n /**\r\n * If node is first-levet\r\n * we add this node to our new wrapper\r\n */\r\n if ( blockTyped ) {\r\n\r\n /**\r\n * If we had splitted inline nodes to paragraph before\r\n */\r\n if ( paragraph.childNodes.length ) {\r\n\r\n newWrapper.appendChild(paragraph.cloneNode(true));\r\n\r\n /** empty paragraph */\r\n paragraph = null;\r\n paragraph = document.createElement('P');\r\n\r\n }\r\n\r\n newWrapper.appendChild(node.cloneNode(true));\r\n\r\n } else {\r\n\r\n /** Collect all inline nodes to one as paragraph */\r\n paragraph.appendChild(node.cloneNode(true));\r\n\r\n /** if node is last we should append this node to paragraph and paragraph to new wrapper */\r\n if ( i == wrapper.childNodes.length - 1 ) {\r\n\r\n newWrapper.appendChild(paragraph.cloneNode(true));\r\n\r\n }\r\n\r\n }\r\n\r\n }\r\n\r\n return newWrapper.innerHTML;\r\n\r\n };\r\n\r\n /**\r\n * Finds closest Contenteditable parent from Element\r\n * @param {Element} node element looking from\r\n * @return {Element} node contenteditable\r\n */\r\n content.getEditableParent = function (node) {\r\n\r\n while (node && node.contentEditable != 'true') {\r\n\r\n node = node.parentNode;\r\n\r\n }\r\n\r\n return node;\r\n\r\n };\r\n\r\n return content;\r\n\r\n})({});\n\n\n// WEBPACK FOOTER //\n// ./modules/content.js","/**\r\n * Codex Editor toolbar module\r\n *\r\n * Contains:\r\n * - Inline toolbox\r\n * - Toolbox within plus button\r\n * - Settings section\r\n *\r\n * @author Codex Team\r\n * @version 1.0\r\n */\r\n\r\nmodule.exports = (function (toolbar) {\r\n\r\n let editor = codex.editor;\r\n\r\n toolbar.settings = require('./settings');\r\n toolbar.inline = require('./inline');\r\n toolbar.toolbox = require('./toolbox');\r\n\r\n /**\r\n * Margin between focused node and toolbar\r\n */\r\n toolbar.defaultToolbarHeight = 49;\r\n\r\n toolbar.defaultOffset = 34;\r\n\r\n toolbar.opened = false;\r\n\r\n toolbar.current = null;\r\n\r\n /**\r\n * @protected\r\n */\r\n toolbar.open = function () {\r\n\r\n if (editor.hideToolbar) {\r\n\r\n return;\r\n\r\n }\r\n\r\n editor.nodes.toolbar.classList.add('opened');\r\n this.opened = true;\r\n\r\n };\r\n\r\n /**\r\n * @protected\r\n */\r\n toolbar.close = function () {\r\n\r\n editor.nodes.toolbar.classList.remove('opened');\r\n\r\n toolbar.opened = false;\r\n toolbar.current = null;\r\n\r\n for (var button in editor.nodes.toolbarButtons) {\r\n\r\n editor.nodes.toolbarButtons[button].classList.remove('selected');\r\n\r\n }\r\n\r\n /** Close toolbox when toolbar is not displayed */\r\n editor.toolbar.toolbox.close();\r\n editor.toolbar.settings.close();\r\n\r\n };\r\n\r\n toolbar.toggle = function () {\r\n\r\n if ( !this.opened ) {\r\n\r\n this.open();\r\n\r\n } else {\r\n\r\n this.close();\r\n\r\n }\r\n\r\n };\r\n\r\n toolbar.hidePlusButton = function () {\r\n\r\n editor.nodes.plusButton.classList.add('hide');\r\n\r\n };\r\n\r\n toolbar.showPlusButton = function () {\r\n\r\n editor.nodes.plusButton.classList.remove('hide');\r\n\r\n };\r\n\r\n /**\r\n * Moving toolbar to the specified node\r\n */\r\n toolbar.move = function () {\r\n\r\n /** Close Toolbox when we move toolbar */\r\n editor.toolbar.toolbox.close();\r\n\r\n if (!editor.content.currentNode) {\r\n\r\n return;\r\n\r\n }\r\n\r\n var newYCoordinate = editor.content.currentNode.offsetTop - (editor.toolbar.defaultToolbarHeight / 2) + editor.toolbar.defaultOffset;\r\n\r\n editor.nodes.toolbar.style.transform = `translate3D(0, ${Math.floor(newYCoordinate)}px, 0)`;\r\n\r\n /** Close trash actions */\r\n editor.toolbar.settings.hideRemoveActions();\r\n\r\n };\r\n\r\n return toolbar;\r\n\r\n})({});\n\n\n// WEBPACK FOOTER //\n// ./modules/toolbar/toolbar.js","/**\r\n * Toolbar settings\r\n *\r\n * @version 1.0.4\r\n */\r\n\r\nmodule.exports = (function (settings) {\r\n\r\n let editor = codex.editor;\r\n\r\n settings.opened = false;\r\n\r\n settings.setting = null;\r\n settings.actions = null;\r\n\r\n /**\r\n * Append and open settings\r\n */\r\n settings.open = function (toolType) {\r\n\r\n /**\r\n * Append settings content\r\n * It's stored in tool.settings\r\n */\r\n if (!editor.tools[toolType] || !editor.tools[toolType].makeSettings ) {\r\n\r\n editor.core.log(`Plugin «${toolType}» has no settings`, 'warn');\r\n editor.nodes.pluginSettings.innerHTML = `Плагин «${toolType}» не имеет настроек`;\r\n\r\n } else {\r\n\r\n /**\r\n * Draw settings block\r\n */\r\n var settingsBlock = editor.tools[toolType].makeSettings();\r\n\r\n editor.nodes.pluginSettings.appendChild(settingsBlock);\r\n\r\n }\r\n\r\n /** Open settings block */\r\n editor.nodes.blockSettings.classList.add('opened');\r\n this.opened = true;\r\n\r\n };\r\n\r\n /**\r\n * Close and clear settings\r\n */\r\n settings.close = function () {\r\n\r\n editor.nodes.blockSettings.classList.remove('opened');\r\n editor.nodes.pluginSettings.innerHTML = '';\r\n\r\n this.opened = false;\r\n\r\n };\r\n\r\n /**\r\n * @param {string} toolType - plugin type\r\n */\r\n settings.toggle = function ( toolType ) {\r\n\r\n if ( !this.opened ) {\r\n\r\n this.open(toolType);\r\n\r\n } else {\r\n\r\n this.close();\r\n\r\n }\r\n\r\n };\r\n\r\n /**\r\n * Here we will draw buttons and add listeners to components\r\n */\r\n settings.makeRemoveBlockButton = function () {\r\n\r\n var removeBlockWrapper = editor.draw.node('SPAN', 'ce-toolbar__remove-btn', {}),\r\n settingButton = editor.draw.node('SPAN', 'ce-toolbar__remove-setting', { innerHTML : '' }),\r\n actionWrapper = editor.draw.node('DIV', 'ce-toolbar__remove-confirmation', {}),\r\n confirmAction = editor.draw.node('DIV', 'ce-toolbar__remove-confirm', { textContent : 'Удалить блок' }),\r\n cancelAction = editor.draw.node('DIV', 'ce-toolbar__remove-cancel', { textContent : 'Отмена' });\r\n\r\n editor.listeners.add(settingButton, 'click', editor.toolbar.settings.removeButtonClicked, false);\r\n\r\n editor.listeners.add(confirmAction, 'click', editor.toolbar.settings.confirmRemovingRequest, false);\r\n\r\n editor.listeners.add(cancelAction, 'click', editor.toolbar.settings.cancelRemovingRequest, false);\r\n\r\n actionWrapper.appendChild(confirmAction);\r\n actionWrapper.appendChild(cancelAction);\r\n\r\n removeBlockWrapper.appendChild(settingButton);\r\n removeBlockWrapper.appendChild(actionWrapper);\r\n\r\n /** Save setting */\r\n editor.toolbar.settings.setting = settingButton;\r\n editor.toolbar.settings.actions = actionWrapper;\r\n\r\n return removeBlockWrapper;\r\n\r\n };\r\n\r\n settings.removeButtonClicked = function () {\r\n\r\n var action = editor.toolbar.settings.actions;\r\n\r\n if (action.classList.contains('opened')) {\r\n\r\n editor.toolbar.settings.hideRemoveActions();\r\n\r\n } else {\r\n\r\n editor.toolbar.settings.showRemoveActions();\r\n\r\n }\r\n\r\n editor.toolbar.toolbox.close();\r\n editor.toolbar.settings.close();\r\n\r\n };\r\n\r\n settings.cancelRemovingRequest = function () {\r\n\r\n editor.toolbar.settings.actions.classList.remove('opened');\r\n\r\n };\r\n\r\n settings.confirmRemovingRequest = function () {\r\n\r\n var currentBlock = editor.content.currentNode,\r\n firstLevelBlocksCount;\r\n\r\n currentBlock.remove();\r\n\r\n firstLevelBlocksCount = editor.nodes.redactor.childNodes.length;\r\n\r\n /**\r\n * If all blocks are removed\r\n */\r\n if (firstLevelBlocksCount === 0) {\r\n\r\n /** update currentNode variable */\r\n editor.content.currentNode = null;\r\n\r\n /** Inserting new empty initial block */\r\n editor.ui.addInitialBlock();\r\n\r\n }\r\n\r\n editor.ui.saveInputs();\r\n\r\n editor.toolbar.close();\r\n\r\n };\r\n\r\n settings.showRemoveActions = function () {\r\n\r\n editor.toolbar.settings.actions.classList.add('opened');\r\n\r\n };\r\n\r\n settings.hideRemoveActions = function () {\r\n\r\n editor.toolbar.settings.actions.classList.remove('opened');\r\n\r\n };\r\n\r\n return settings;\r\n\r\n})({});\n\n\n// WEBPACK FOOTER //\n// ./modules/toolbar/settings.js","/**\r\n * Inline toolbar\r\n *\r\n * Contains from tools:\r\n * Bold, Italic, Underline and Anchor\r\n *\r\n * @author Codex Team\r\n * @version 1.0\r\n */\r\n\r\nmodule.exports = (function (inline) {\r\n\r\n let editor = codex.editor;\r\n\r\n inline.buttonsOpened = null;\r\n inline.actionsOpened = null;\r\n inline.wrappersOffset = null;\r\n\r\n /**\r\n * saving selection that need for execCommand for styling\r\n *\r\n */\r\n inline.storedSelection = null;\r\n\r\n /**\r\n * @protected\r\n *\r\n * Open inline toobar\r\n */\r\n inline.show = function () {\r\n\r\n var currentNode = editor.content.currentNode,\r\n tool = currentNode.dataset.tool,\r\n plugin;\r\n\r\n /**\r\n * tool allowed to open inline toolbar\r\n */\r\n plugin = editor.tools[tool];\r\n\r\n if (!plugin.showInlineToolbar)\r\n return;\r\n\r\n var selectedText = inline.getSelectionText(),\r\n toolbar = editor.nodes.inlineToolbar.wrapper;\r\n\r\n if (selectedText.length > 0) {\r\n\r\n /** Move toolbar and open */\r\n editor.toolbar.inline.move();\r\n\r\n /** Open inline toolbar */\r\n toolbar.classList.add('opened');\r\n\r\n /** show buttons of inline toolbar */\r\n editor.toolbar.inline.showButtons();\r\n\r\n }\r\n\r\n };\r\n\r\n /**\r\n * @protected\r\n *\r\n * Closes inline toolbar\r\n */\r\n inline.close = function () {\r\n\r\n var toolbar = editor.nodes.inlineToolbar.wrapper;\r\n\r\n toolbar.classList.remove('opened');\r\n\r\n };\r\n\r\n /**\r\n * @private\r\n *\r\n * Moving toolbar\r\n */\r\n inline.move = function () {\r\n\r\n if (!this.wrappersOffset) {\r\n\r\n this.wrappersOffset = this.getWrappersOffset();\r\n\r\n }\r\n\r\n var coords = this.getSelectionCoords(),\r\n defaultOffset = 0,\r\n toolbar = editor.nodes.inlineToolbar.wrapper,\r\n newCoordinateX,\r\n newCoordinateY;\r\n\r\n if (toolbar.offsetHeight === 0) {\r\n\r\n defaultOffset = 40;\r\n\r\n }\r\n\r\n newCoordinateX = coords.x - this.wrappersOffset.left;\r\n newCoordinateY = coords.y + window.scrollY - this.wrappersOffset.top - defaultOffset - toolbar.offsetHeight;\r\n\r\n toolbar.style.transform = `translate3D(${Math.floor(newCoordinateX)}px, ${Math.floor(newCoordinateY)}px, 0)`;\r\n\r\n /** Close everything */\r\n editor.toolbar.inline.closeButtons();\r\n editor.toolbar.inline.closeAction();\r\n\r\n };\r\n\r\n /**\r\n * @private\r\n *\r\n * Tool Clicked\r\n */\r\n\r\n inline.toolClicked = function (event, type) {\r\n\r\n /**\r\n * For simple tools we use default browser function\r\n * For more complicated tools, we should write our own behavior\r\n */\r\n switch (type) {\r\n case 'createLink' : editor.toolbar.inline.createLinkAction(event, type); break;\r\n default : editor.toolbar.inline.defaultToolAction(type); break;\r\n }\r\n\r\n /**\r\n * highlight buttons\r\n * after making some action\r\n */\r\n editor.nodes.inlineToolbar.buttons.childNodes.forEach(editor.toolbar.inline.hightlight);\r\n\r\n };\r\n\r\n /**\r\n * @private\r\n *\r\n * Saving wrappers offset in DOM\r\n */\r\n inline.getWrappersOffset = function () {\r\n\r\n var wrapper = editor.nodes.wrapper,\r\n offset = this.getOffset(wrapper);\r\n\r\n this.wrappersOffset = offset;\r\n return offset;\r\n\r\n };\r\n\r\n /**\r\n * @private\r\n *\r\n * Calculates offset of DOM element\r\n *\r\n * @param el\r\n * @returns {{top: number, left: number}}\r\n */\r\n inline.getOffset = function ( el ) {\r\n\r\n var _x = 0;\r\n var _y = 0;\r\n\r\n while( el && !isNaN( el.offsetLeft ) && !isNaN( el.offsetTop ) ) {\r\n\r\n _x += (el.offsetLeft + el.clientLeft);\r\n _y += (el.offsetTop + el.clientTop);\r\n el = el.offsetParent;\r\n\r\n }\r\n return { top: _y, left: _x };\r\n\r\n };\r\n\r\n /**\r\n * @private\r\n *\r\n * Calculates position of selected text\r\n * @returns {{x: number, y: number}}\r\n */\r\n inline.getSelectionCoords = function () {\r\n\r\n var sel = document.selection, range;\r\n var x = 0, y = 0;\r\n\r\n if (sel) {\r\n\r\n if (sel.type != 'Control') {\r\n\r\n range = sel.createRange();\r\n range.collapse(true);\r\n x = range.boundingLeft;\r\n y = range.boundingTop;\r\n\r\n }\r\n\r\n } else if (window.getSelection) {\r\n\r\n sel = window.getSelection();\r\n\r\n if (sel.rangeCount) {\r\n\r\n range = sel.getRangeAt(0).cloneRange();\r\n if (range.getClientRects) {\r\n\r\n range.collapse(true);\r\n var rect = range.getClientRects()[0];\r\n\r\n if (!rect) {\r\n\r\n return;\r\n\r\n }\r\n\r\n x = rect.left;\r\n y = rect.top;\r\n\r\n }\r\n\r\n }\r\n\r\n }\r\n return { x: x, y: y };\r\n\r\n };\r\n\r\n /**\r\n * @private\r\n *\r\n * Returns selected text as String\r\n * @returns {string}\r\n */\r\n inline.getSelectionText = function () {\r\n\r\n var selectedText = '';\r\n\r\n // all modern browsers and IE9+\r\n if (window.getSelection) {\r\n\r\n selectedText = window.getSelection().toString();\r\n\r\n }\r\n\r\n return selectedText;\r\n\r\n };\r\n\r\n /** Opens buttons block */\r\n inline.showButtons = function () {\r\n\r\n var buttons = editor.nodes.inlineToolbar.buttons;\r\n\r\n buttons.classList.add('opened');\r\n\r\n editor.toolbar.inline.buttonsOpened = true;\r\n\r\n /** highlight buttons */\r\n editor.nodes.inlineToolbar.buttons.childNodes.forEach(editor.toolbar.inline.hightlight);\r\n\r\n };\r\n\r\n /** Makes buttons disappear */\r\n inline.closeButtons = function () {\r\n\r\n var buttons = editor.nodes.inlineToolbar.buttons;\r\n\r\n buttons.classList.remove('opened');\r\n\r\n editor.toolbar.inline.buttonsOpened = false;\r\n\r\n };\r\n\r\n /** Open buttons defined action if exist */\r\n inline.showActions = function () {\r\n\r\n var action = editor.nodes.inlineToolbar.actions;\r\n\r\n action.classList.add('opened');\r\n\r\n editor.toolbar.inline.actionsOpened = true;\r\n\r\n };\r\n\r\n /** Close actions block */\r\n inline.closeAction = function () {\r\n\r\n var action = editor.nodes.inlineToolbar.actions;\r\n\r\n action.innerHTML = '';\r\n action.classList.remove('opened');\r\n editor.toolbar.inline.actionsOpened = false;\r\n\r\n };\r\n\r\n\r\n /**\r\n * Callback for keydowns in inline toolbar \"Insert link...\" input\r\n */\r\n let inlineToolbarAnchorInputKeydown_ = function (event) {\r\n\r\n if (event.keyCode != editor.core.keys.ENTER) {\r\n\r\n return;\r\n\r\n }\r\n\r\n let editable = editor.content.currentNode,\r\n storedSelection = editor.toolbar.inline.storedSelection;\r\n\r\n editor.toolbar.inline.restoreSelection(editable, storedSelection);\r\n editor.toolbar.inline.setAnchor(this.value);\r\n\r\n /**\r\n * Preventing events that will be able to happen\r\n */\r\n event.preventDefault();\r\n event.stopImmediatePropagation();\r\n\r\n editor.toolbar.inline.clearRange();\r\n\r\n };\r\n\r\n /** Action for link creation or for setting anchor */\r\n inline.createLinkAction = function (event) {\r\n\r\n var isActive = this.isLinkActive();\r\n\r\n var editable = editor.content.currentNode,\r\n storedSelection = editor.toolbar.inline.saveSelection(editable);\r\n\r\n /** Save globally selection */\r\n editor.toolbar.inline.storedSelection = storedSelection;\r\n\r\n if (isActive) {\r\n\r\n\r\n /**\r\n * Changing stored selection. if we want to remove anchor from word\r\n * we should remove anchor from whole word, not only selected part.\r\n * The solution is than we get the length of current link\r\n * Change start position to - end of selection minus length of anchor\r\n */\r\n editor.toolbar.inline.restoreSelection(editable, storedSelection);\r\n\r\n editor.toolbar.inline.defaultToolAction('unlink');\r\n\r\n } else {\r\n\r\n /** Create input and close buttons */\r\n var action = editor.draw.inputForLink();\r\n\r\n editor.nodes.inlineToolbar.actions.appendChild(action);\r\n\r\n editor.toolbar.inline.closeButtons();\r\n editor.toolbar.inline.showActions();\r\n\r\n /**\r\n * focus to input\r\n * Solution: https://developer.mozilla.org/ru/docs/Web/API/HTMLElement/focus\r\n * Prevents event after showing input and when we need to focus an input which is in unexisted form\r\n */\r\n action.focus();\r\n event.preventDefault();\r\n\r\n /** Callback to link action */\r\n editor.listeners.add(action, 'keydown', inlineToolbarAnchorInputKeydown_, false);\r\n\r\n }\r\n\r\n };\r\n\r\n inline.isLinkActive = function () {\r\n\r\n var isActive = false;\r\n\r\n editor.nodes.inlineToolbar.buttons.childNodes.forEach(function (tool) {\r\n\r\n var dataType = tool.dataset.type;\r\n\r\n if (dataType == 'link' && tool.classList.contains('hightlighted')) {\r\n\r\n isActive = true;\r\n\r\n }\r\n\r\n });\r\n\r\n return isActive;\r\n\r\n };\r\n\r\n /** default action behavior of tool */\r\n inline.defaultToolAction = function (type) {\r\n\r\n document.execCommand(type, false, null);\r\n\r\n };\r\n\r\n /**\r\n * @private\r\n *\r\n * Sets URL\r\n *\r\n * @param {String} url - URL\r\n */\r\n inline.setAnchor = function (url) {\r\n\r\n document.execCommand('createLink', false, url);\r\n\r\n /** Close after URL inserting */\r\n editor.toolbar.inline.closeAction();\r\n\r\n };\r\n\r\n /**\r\n * @private\r\n *\r\n * Saves selection\r\n */\r\n inline.saveSelection = function (containerEl) {\r\n\r\n var range = window.getSelection().getRangeAt(0),\r\n preSelectionRange = range.cloneRange(),\r\n start;\r\n\r\n preSelectionRange.selectNodeContents(containerEl);\r\n preSelectionRange.setEnd(range.startContainer, range.startOffset);\r\n\r\n start = preSelectionRange.toString().length;\r\n\r\n return {\r\n start: start,\r\n end: start + range.toString().length\r\n };\r\n\r\n };\r\n\r\n /**\r\n * @private\r\n *\r\n * Sets to previous selection (Range)\r\n *\r\n * @param {Element} containerEl - editable element where we restore range\r\n * @param {Object} savedSel - range basic information to restore\r\n */\r\n inline.restoreSelection = function (containerEl, savedSel) {\r\n\r\n var range = document.createRange(),\r\n charIndex = 0;\r\n\r\n range.setStart(containerEl, 0);\r\n range.collapse(true);\r\n\r\n var nodeStack = [ containerEl ],\r\n node,\r\n foundStart = false,\r\n stop = false,\r\n nextCharIndex;\r\n\r\n while (!stop && (node = nodeStack.pop())) {\r\n\r\n if (node.nodeType == 3) {\r\n\r\n nextCharIndex = charIndex + node.length;\r\n\r\n if (!foundStart && savedSel.start >= charIndex && savedSel.start <= nextCharIndex) {\r\n\r\n range.setStart(node, savedSel.start - charIndex);\r\n foundStart = true;\r\n\r\n }\r\n if (foundStart && savedSel.end >= charIndex && savedSel.end <= nextCharIndex) {\r\n\r\n range.setEnd(node, savedSel.end - charIndex);\r\n stop = true;\r\n\r\n }\r\n charIndex = nextCharIndex;\r\n\r\n } else {\r\n\r\n var i = node.childNodes.length;\r\n\r\n while (i--) {\r\n\r\n nodeStack.push(node.childNodes[i]);\r\n\r\n }\r\n\r\n }\r\n\r\n }\r\n\r\n var sel = window.getSelection();\r\n\r\n sel.removeAllRanges();\r\n sel.addRange(range);\r\n\r\n };\r\n\r\n /**\r\n * @private\r\n *\r\n * Removes all ranges from window selection\r\n */\r\n inline.clearRange = function () {\r\n\r\n var selection = window.getSelection();\r\n\r\n selection.removeAllRanges();\r\n\r\n };\r\n\r\n /**\r\n * @private\r\n *\r\n * sets or removes hightlight\r\n */\r\n inline.hightlight = function (tool) {\r\n\r\n var dataType = tool.dataset.type;\r\n\r\n if (document.queryCommandState(dataType)) {\r\n\r\n editor.toolbar.inline.setButtonHighlighted(tool);\r\n\r\n } else {\r\n\r\n editor.toolbar.inline.removeButtonsHighLight(tool);\r\n\r\n }\r\n\r\n /**\r\n *\r\n * hightlight for anchors\r\n */\r\n var selection = window.getSelection(),\r\n tag = selection.anchorNode.parentNode;\r\n\r\n if (tag.tagName == 'A' && dataType == 'link') {\r\n\r\n editor.toolbar.inline.setButtonHighlighted(tool);\r\n\r\n }\r\n\r\n };\r\n\r\n /**\r\n * @private\r\n *\r\n * Mark button if text is already executed\r\n */\r\n inline.setButtonHighlighted = function (button) {\r\n\r\n button.classList.add('hightlighted');\r\n\r\n /** At link tool we also change icon */\r\n if (button.dataset.type == 'link') {\r\n\r\n var icon = button.childNodes[0];\r\n\r\n icon.classList.remove('ce-icon-link');\r\n icon.classList.add('ce-icon-unlink');\r\n\r\n }\r\n\r\n };\r\n\r\n /**\r\n * @private\r\n *\r\n * Removes hightlight\r\n */\r\n inline.removeButtonsHighLight = function (button) {\r\n\r\n button.classList.remove('hightlighted');\r\n\r\n /** At link tool we also change icon */\r\n if (button.dataset.type == 'link') {\r\n\r\n var icon = button.childNodes[0];\r\n\r\n icon.classList.remove('ce-icon-unlink');\r\n icon.classList.add('ce-icon-link');\r\n\r\n }\r\n\r\n };\r\n\r\n\r\n return inline;\r\n\r\n})({});\n\n\n// WEBPACK FOOTER //\n// ./modules/toolbar/inline.js","/**\r\n * Codex Editor toolbox\r\n *\r\n * All tools be able to appended here\r\n *\r\n * @author Codex Team\r\n * @version 1.0\r\n */\r\n\r\nmodule.exports = (function (toolbox) {\r\n\r\n let editor = codex.editor;\r\n\r\n toolbox.opened = false;\r\n\r\n /** Shows toolbox */\r\n toolbox.open = function () {\r\n\r\n /** Close setting if toolbox is opened */\r\n if (editor.toolbar.settings.opened) {\r\n\r\n editor.toolbar.settings.close();\r\n\r\n }\r\n\r\n /** display toolbox */\r\n editor.nodes.toolbox.classList.add('opened');\r\n\r\n /** Animate plus button */\r\n editor.nodes.plusButton.classList.add('clicked');\r\n\r\n /** toolbox state */\r\n editor.toolbar.toolbox.opened = true;\r\n\r\n };\r\n\r\n /** Closes toolbox */\r\n toolbox.close = function () {\r\n\r\n /** Makes toolbox disapear */\r\n editor.nodes.toolbox.classList.remove('opened');\r\n\r\n /** Rotate plus button */\r\n editor.nodes.plusButton.classList.remove('clicked');\r\n\r\n /** toolbox state */\r\n editor.toolbar.toolbox.opened = false;\r\n\r\n editor.toolbar.current = null;\r\n\r\n };\r\n\r\n toolbox.leaf = function () {\r\n\r\n let currentTool = editor.toolbar.current,\r\n tools = Object.keys(editor.tools),\r\n barButtons = editor.nodes.toolbarButtons,\r\n nextToolIndex = 0,\r\n toolToSelect,\r\n visibleTool,\r\n tool;\r\n\r\n if ( !currentTool ) {\r\n\r\n /** Get first tool from object*/\r\n for(tool in editor.tools) {\r\n\r\n if (editor.tools[tool].displayInToolbox) {\r\n\r\n break;\r\n\r\n }\r\n\r\n nextToolIndex ++;\r\n\r\n }\r\n\r\n } else {\r\n\r\n nextToolIndex = (tools.indexOf(currentTool) + 1) % tools.length;\r\n visibleTool = tools[nextToolIndex];\r\n\r\n while (!editor.tools[visibleTool].displayInToolbox) {\r\n\r\n nextToolIndex = (nextToolIndex + 1) % tools.length;\r\n visibleTool = tools[nextToolIndex];\r\n\r\n }\r\n\r\n }\r\n\r\n toolToSelect = tools[nextToolIndex];\r\n\r\n for ( var button in barButtons ) {\r\n\r\n barButtons[button].classList.remove('selected');\r\n\r\n }\r\n\r\n barButtons[toolToSelect].classList.add('selected');\r\n editor.toolbar.current = toolToSelect;\r\n\r\n };\r\n\r\n /**\r\n * Transforming selected node type into selected toolbar element type\r\n * @param {event} event\r\n */\r\n toolbox.toolClicked = function (event) {\r\n\r\n /**\r\n * UNREPLACEBLE_TOOLS this types of tools are forbidden to replace even they are empty\r\n */\r\n var UNREPLACEBLE_TOOLS = ['image', 'link', 'list', 'instagram', 'twitter', 'embed'],\r\n tool = editor.tools[editor.toolbar.current],\r\n workingNode = editor.content.currentNode,\r\n currentInputIndex = editor.caret.inputIndex,\r\n newBlockContent,\r\n appendCallback,\r\n blockData;\r\n\r\n /** Make block from plugin */\r\n newBlockContent = tool.render();\r\n\r\n /** information about block */\r\n blockData = {\r\n block : newBlockContent,\r\n type : tool.type,\r\n stretched : false\r\n };\r\n\r\n if (\r\n workingNode &&\r\n UNREPLACEBLE_TOOLS.indexOf(workingNode.dataset.tool) === -1 &&\r\n workingNode.textContent.trim() === ''\r\n ) {\r\n\r\n /** Replace current block */\r\n editor.content.switchBlock(workingNode, newBlockContent, tool.type);\r\n\r\n } else {\r\n\r\n /** Insert new Block from plugin */\r\n editor.content.insertBlock(blockData);\r\n\r\n /** increase input index */\r\n currentInputIndex++;\r\n\r\n }\r\n\r\n /** Fire tool append callback */\r\n appendCallback = tool.appendCallback;\r\n\r\n if (appendCallback && typeof appendCallback == 'function') {\r\n\r\n appendCallback.call(event);\r\n\r\n }\r\n\r\n window.setTimeout(function () {\r\n\r\n /** Set caret to current block */\r\n editor.caret.setToBlock(currentInputIndex);\r\n\r\n }, 10);\r\n\r\n\r\n /**\r\n * Changing current Node\r\n */\r\n editor.content.workingNodeChanged();\r\n\r\n /**\r\n * Move toolbar when node is changed\r\n */\r\n editor.toolbar.move();\r\n\r\n };\r\n\r\n return toolbox;\r\n\r\n})({});\n\n\n// WEBPACK FOOTER //\n// ./modules/toolbar/toolbox.js","/**\r\n * @module Codex Editor Callbacks module\r\n * @description Module works with editor added Elements\r\n *\r\n * @author Codex Team\r\n * @version 1.4.0\r\n */\r\n\r\nmodule.exports = (function (callbacks) {\r\n\r\n let editor = codex.editor;\r\n\r\n /**\r\n * used by UI module\r\n * @description Routes all keydowns on document\r\n * @param {Object} event\r\n */\r\n callbacks.globalKeydown = function (event) {\r\n\r\n switch (event.keyCode) {\r\n case editor.core.keys.ENTER : enterKeyPressed_(event); break;\r\n }\r\n\r\n };\r\n\r\n /**\r\n * used by UI module\r\n * @description Routes all keydowns on redactors area\r\n * @param {Object} event\r\n */\r\n callbacks.redactorKeyDown = function (event) {\r\n\r\n switch (event.keyCode) {\r\n case editor.core.keys.TAB : tabKeyPressedOnRedactorsZone_(event); break;\r\n case editor.core.keys.ENTER : enterKeyPressedOnRedactorsZone_(event); break;\r\n case editor.core.keys.ESC : escapeKeyPressedOnRedactorsZone_(event); break;\r\n default : defaultKeyPressedOnRedactorsZone_(event); break;\r\n }\r\n\r\n };\r\n\r\n /**\r\n * used by UI module\r\n * @description Routes all keyup events\r\n * @param {Object} event\r\n */\r\n callbacks.globalKeyup = function (event) {\r\n\r\n switch (event.keyCode) {\r\n case editor.core.keys.UP :\r\n case editor.core.keys.LEFT :\r\n case editor.core.keys.RIGHT :\r\n case editor.core.keys.DOWN : arrowKeyPressed_(event); break;\r\n }\r\n\r\n };\r\n\r\n /**\r\n * @param {Object} event\r\n * @private\r\n *\r\n * Handles behaviour when tab pressed\r\n * @description if Content is empty show toolbox (if it is closed) or leaf tools\r\n * uses Toolbars toolbox module to handle the situation\r\n */\r\n var tabKeyPressedOnRedactorsZone_ = function (event) {\r\n\r\n /**\r\n * Wait for solution. Would like to know the behaviour\r\n * @todo Add spaces\r\n */\r\n event.preventDefault();\r\n\r\n var nativeInputs = editor.content.currentNode.querySelectorAll('textarea, input'),\r\n nativeInputsAreEmpty = true,\r\n textContentIsEmpty = !editor.content.currentNode.textContent.trim();\r\n\r\n Array.prototype.map.call(nativeInputs, function (input) {\r\n\r\n if (input.type == 'textarea' || input.type == 'text') {\r\n\r\n nativeInputsAreEmpty = nativeInputsAreEmpty && !input.value.trim();\r\n\r\n }\r\n\r\n });\r\n\r\n var blockIsEmpty = textContentIsEmpty && nativeInputsAreEmpty;\r\n\r\n if (!blockIsEmpty) {\r\n\r\n return;\r\n\r\n }\r\n\r\n if ( !editor.toolbar.opened ) {\r\n\r\n editor.toolbar.open();\r\n\r\n }\r\n\r\n if (editor.toolbar.opened && !editor.toolbar.toolbox.opened) {\r\n\r\n editor.toolbar.toolbox.open();\r\n\r\n } else {\r\n\r\n editor.toolbar.toolbox.leaf();\r\n\r\n }\r\n\r\n };\r\n\r\n /**\r\n * Handles global EnterKey Press\r\n * @see enterPressedOnBlock_\r\n * @param {Object} event\r\n */\r\n var enterKeyPressed_ = function () {\r\n\r\n if (editor.content.editorAreaHightlighted) {\r\n\r\n /**\r\n * it means that we lose input index, saved index before is not correct\r\n * therefore we need to set caret when we insert new block\r\n */\r\n editor.caret.inputIndex = -1;\r\n\r\n enterPressedOnBlock_();\r\n\r\n }\r\n\r\n };\r\n\r\n /**\r\n * Callback for enter key pressing in first-level block area\r\n *\r\n * @param {Event} event\r\n * @private\r\n *\r\n * @description Inserts new block with initial type from settings\r\n */\r\n var enterPressedOnBlock_ = function () {\r\n\r\n var NEW_BLOCK_TYPE = editor.settings.initialBlockPlugin;\r\n\r\n editor.content.insertBlock({\r\n type : NEW_BLOCK_TYPE,\r\n block : editor.tools[NEW_BLOCK_TYPE].render()\r\n }, true );\r\n\r\n editor.toolbar.move();\r\n editor.toolbar.open();\r\n\r\n };\r\n\r\n\r\n /**\r\n * ENTER key handler\r\n *\r\n * @param {Object} event\r\n * @private\r\n *\r\n * @description Makes new block with initial type from settings\r\n */\r\n var enterKeyPressedOnRedactorsZone_ = function (event) {\r\n\r\n if (event.target.contentEditable == 'true') {\r\n\r\n /** Update input index */\r\n editor.caret.saveCurrentInputIndex();\r\n\r\n }\r\n\r\n var currentInputIndex = editor.caret.getCurrentInputIndex() || 0,\r\n workingNode = editor.content.currentNode,\r\n tool = workingNode.dataset.tool,\r\n isEnterPressedOnToolbar = editor.toolbar.opened &&\r\n editor.toolbar.current &&\r\n event.target == editor.state.inputs[currentInputIndex];\r\n\r\n /** The list of tools which needs the default browser behaviour */\r\n var enableLineBreaks = editor.tools[tool].enableLineBreaks;\r\n\r\n /** This type of block creates when enter is pressed */\r\n var NEW_BLOCK_TYPE = editor.settings.initialBlockPlugin;\r\n\r\n /**\r\n * When toolbar is opened, select tool instead of making new paragraph\r\n */\r\n if ( isEnterPressedOnToolbar ) {\r\n\r\n event.preventDefault();\r\n\r\n editor.toolbar.toolbox.toolClicked(event);\r\n\r\n editor.toolbar.close();\r\n\r\n /**\r\n * Stop other listeners callback executions\r\n */\r\n event.stopPropagation();\r\n event.stopImmediatePropagation();\r\n\r\n return;\r\n\r\n }\r\n\r\n /**\r\n * Allow paragraph lineBreaks with shift enter\r\n * Or if shiftkey pressed and enter and enabledLineBreaks, the let new block creation\r\n */\r\n if ( event.shiftKey || enableLineBreaks ) {\r\n\r\n event.stopPropagation();\r\n event.stopImmediatePropagation();\r\n return;\r\n\r\n }\r\n\r\n var currentSelection = window.getSelection(),\r\n currentSelectedNode = currentSelection.anchorNode,\r\n caretAtTheEndOfText = editor.caret.position.atTheEnd(),\r\n isTextNodeHasParentBetweenContenteditable = false;\r\n\r\n /**\r\n * Allow making new

    in same block by SHIFT+ENTER and forbids to prevent default browser behaviour\r\n */\r\n if ( event.shiftKey && !enableLineBreaks ) {\r\n\r\n editor.callback.enterPressedOnBlock(editor.content.currentBlock, event);\r\n event.preventDefault();\r\n return;\r\n\r\n }\r\n\r\n /**\r\n * Workaround situation when caret at the Text node that has some wrapper Elements\r\n * Split block cant handle this.\r\n * We need to save default behavior\r\n */\r\n isTextNodeHasParentBetweenContenteditable = currentSelectedNode && currentSelectedNode.parentNode.contentEditable != 'true';\r\n\r\n /**\r\n * Split blocks when input has several nodes and caret placed in textNode\r\n */\r\n if (\r\n currentSelectedNode.nodeType == editor.core.nodeTypes.TEXT &&\r\n !isTextNodeHasParentBetweenContenteditable &&\r\n !caretAtTheEndOfText\r\n ) {\r\n\r\n event.preventDefault();\r\n\r\n editor.core.log('Splitting Text node...');\r\n\r\n editor.content.splitBlock(currentInputIndex);\r\n\r\n /** Show plus button when next input after split is empty*/\r\n if (!editor.state.inputs[currentInputIndex + 1].textContent.trim()) {\r\n\r\n editor.toolbar.showPlusButton();\r\n\r\n }\r\n\r\n } else {\r\n\r\n var islastNode = editor.content.isLastNode(currentSelectedNode);\r\n\r\n if ( islastNode && caretAtTheEndOfText ) {\r\n\r\n event.preventDefault();\r\n event.stopPropagation();\r\n event.stopImmediatePropagation();\r\n\r\n editor.core.log('ENTER clicked in last textNode. Create new BLOCK');\r\n\r\n editor.content.insertBlock({\r\n type: NEW_BLOCK_TYPE,\r\n block: editor.tools[NEW_BLOCK_TYPE].render()\r\n }, true);\r\n\r\n editor.toolbar.move();\r\n editor.toolbar.open();\r\n\r\n /** Show plus button with empty block */\r\n editor.toolbar.showPlusButton();\r\n\r\n }\r\n\r\n }\r\n\r\n /** get all inputs after new appending block */\r\n editor.ui.saveInputs();\r\n\r\n };\r\n\r\n /**\r\n * Escape behaviour\r\n * @param event\r\n * @private\r\n *\r\n * @description Closes toolbox and toolbar. Prevents default behaviour\r\n */\r\n var escapeKeyPressedOnRedactorsZone_ = function (event) {\r\n\r\n /** Close all toolbar */\r\n editor.toolbar.close();\r\n\r\n /** Close toolbox */\r\n editor.toolbar.toolbox.close();\r\n\r\n event.preventDefault();\r\n\r\n };\r\n\r\n /**\r\n * @param {Event} event\r\n * @private\r\n *\r\n * closes and moves toolbar\r\n */\r\n var arrowKeyPressed_ = function (event) {\r\n\r\n editor.content.workingNodeChanged();\r\n\r\n /* Closing toolbar */\r\n editor.toolbar.close();\r\n editor.toolbar.move();\r\n\r\n };\r\n\r\n /**\r\n * @private\r\n * @param {Event} event\r\n *\r\n * @description Closes all opened bars from toolbar.\r\n * If block is mark, clears highlightning\r\n */\r\n var defaultKeyPressedOnRedactorsZone_ = function () {\r\n\r\n editor.toolbar.close();\r\n\r\n if (!editor.toolbar.inline.actionsOpened) {\r\n\r\n editor.toolbar.inline.close();\r\n editor.content.clearMark();\r\n\r\n }\r\n\r\n };\r\n\r\n /**\r\n * Handler when clicked on redactors area\r\n *\r\n * @protected\r\n * @param event\r\n *\r\n * @description Detects clicked area. If it is first-level block area, marks as detected and\r\n * on next enter press will be inserted new block\r\n * Otherwise, save carets position (input index) and put caret to the editable zone.\r\n *\r\n * @see detectWhenClickedOnFirstLevelBlockArea_\r\n *\r\n */\r\n callbacks.redactorClicked = function (event) {\r\n\r\n detectWhenClickedOnFirstLevelBlockArea_();\r\n\r\n editor.content.workingNodeChanged(event.target);\r\n editor.ui.saveInputs();\r\n\r\n var selectedText = editor.toolbar.inline.getSelectionText(),\r\n firstLevelBlock;\r\n\r\n /** If selection range took off, then we hide inline toolbar */\r\n if (selectedText.length === 0) {\r\n\r\n editor.toolbar.inline.close();\r\n\r\n }\r\n\r\n /** Update current input index in memory when caret focused into existed input */\r\n if (event.target.contentEditable == 'true') {\r\n\r\n editor.caret.saveCurrentInputIndex();\r\n\r\n }\r\n\r\n if (editor.content.currentNode === null) {\r\n\r\n /**\r\n * If inputs in redactor does not exits, then we put input index 0 not -1\r\n */\r\n var indexOfLastInput = editor.state.inputs.length > 0 ? editor.state.inputs.length - 1 : 0;\r\n\r\n /** If we have any inputs */\r\n if (editor.state.inputs.length) {\r\n\r\n /** getting firstlevel parent of input */\r\n firstLevelBlock = editor.content.getFirstLevelBlock(editor.state.inputs[indexOfLastInput]);\r\n\r\n }\r\n\r\n /** If input is empty, then we set caret to the last input */\r\n if (editor.state.inputs.length && editor.state.inputs[indexOfLastInput].textContent === '' && firstLevelBlock.dataset.tool == editor.settings.initialBlockPlugin) {\r\n\r\n editor.caret.setToBlock(indexOfLastInput);\r\n\r\n } else {\r\n\r\n /** Create new input when caret clicked in redactors area */\r\n var NEW_BLOCK_TYPE = editor.settings.initialBlockPlugin;\r\n\r\n editor.content.insertBlock({\r\n type : NEW_BLOCK_TYPE,\r\n block : editor.tools[NEW_BLOCK_TYPE].render()\r\n });\r\n\r\n /** If there is no inputs except inserted */\r\n if (editor.state.inputs.length === 1) {\r\n\r\n editor.caret.setToBlock(indexOfLastInput);\r\n\r\n } else {\r\n\r\n /** Set caret to this appended input */\r\n editor.caret.setToNextBlock(indexOfLastInput);\r\n\r\n }\r\n\r\n }\r\n\r\n } else {\r\n\r\n /** Close all panels */\r\n editor.toolbar.settings.close();\r\n editor.toolbar.toolbox.close();\r\n\r\n }\r\n\r\n /**\r\n * Move toolbar and open\r\n */\r\n editor.toolbar.move();\r\n editor.toolbar.open();\r\n\r\n var inputIsEmpty = !editor.content.currentNode.textContent.trim(),\r\n currentNodeType = editor.content.currentNode.dataset.tool,\r\n isInitialType = currentNodeType == editor.settings.initialBlockPlugin;\r\n\r\n\r\n /** Hide plus buttons */\r\n editor.toolbar.hidePlusButton();\r\n\r\n if (!inputIsEmpty) {\r\n\r\n /** Mark current block */\r\n editor.content.markBlock();\r\n\r\n }\r\n\r\n if ( isInitialType && inputIsEmpty ) {\r\n\r\n /** Show plus button */\r\n editor.toolbar.showPlusButton();\r\n\r\n }\r\n\r\n\r\n };\r\n\r\n /**\r\n * This method allows to define, is caret in contenteditable element or not.\r\n *\r\n * @private\r\n *\r\n * @description Otherwise, if we get TEXT node from range container, that will means we have input index.\r\n * In this case we use default browsers behaviour (if plugin allows that) or overwritten action.\r\n * Therefore, to be sure that we've clicked first-level block area, we should have currentNode, which always\r\n * specifies to the first-level block. Other cases we just ignore.\r\n */\r\n var detectWhenClickedOnFirstLevelBlockArea_ = function () {\r\n\r\n var selection = window.getSelection(),\r\n anchorNode = selection.anchorNode,\r\n flag = false;\r\n\r\n if (selection.rangeCount === 0) {\r\n\r\n editor.content.editorAreaHightlighted = true;\r\n\r\n } else {\r\n\r\n if (!editor.core.isDomNode(anchorNode)) {\r\n\r\n anchorNode = anchorNode.parentNode;\r\n\r\n }\r\n\r\n /** Already founded, without loop */\r\n if (anchorNode.contentEditable == 'true') {\r\n\r\n flag = true;\r\n\r\n }\r\n\r\n while (anchorNode.contentEditable != 'true') {\r\n\r\n anchorNode = anchorNode.parentNode;\r\n\r\n if (anchorNode.contentEditable == 'true') {\r\n\r\n flag = true;\r\n\r\n }\r\n\r\n if (anchorNode == document.body) {\r\n\r\n break;\r\n\r\n }\r\n\r\n }\r\n\r\n /** If editable element founded, flag is \"TRUE\", Therefore we return \"FALSE\" */\r\n editor.content.editorAreaHightlighted = !flag;\r\n\r\n }\r\n\r\n };\r\n\r\n /**\r\n * Toolbar button click handler\r\n *\r\n * @param {Object} event - cursor to the button\r\n * @protected\r\n *\r\n * @description gets current tool and calls render method\r\n */\r\n callbacks.toolbarButtonClicked = function (event) {\r\n\r\n var button = this;\r\n\r\n editor.toolbar.current = button.dataset.type;\r\n\r\n editor.toolbar.toolbox.toolClicked(event);\r\n editor.toolbar.close();\r\n\r\n };\r\n\r\n /**\r\n * Show or Hide toolbox when plus button is clicked\r\n */\r\n callbacks.plusButtonClicked = function () {\r\n\r\n if (!editor.nodes.toolbox.classList.contains('opened')) {\r\n\r\n editor.toolbar.toolbox.open();\r\n\r\n } else {\r\n\r\n editor.toolbar.toolbox.close();\r\n\r\n }\r\n\r\n };\r\n\r\n /**\r\n * Block handlers for KeyDown events\r\n *\r\n * @protected\r\n * @param {Object} event\r\n *\r\n * Handles keydowns on block\r\n * @see blockRightOrDownArrowPressed_\r\n * @see backspacePressed_\r\n * @see blockLeftOrUpArrowPressed_\r\n */\r\n callbacks.blockKeydown = function (event) {\r\n\r\n let block = event.target; // event.target is input\r\n\r\n switch (event.keyCode) {\r\n\r\n case editor.core.keys.DOWN:\r\n case editor.core.keys.RIGHT:\r\n blockRightOrDownArrowPressed_(event);\r\n break;\r\n\r\n case editor.core.keys.BACKSPACE:\r\n backspacePressed_(block, event);\r\n break;\r\n\r\n case editor.core.keys.UP:\r\n case editor.core.keys.LEFT:\r\n blockLeftOrUpArrowPressed_(event);\r\n break;\r\n\r\n }\r\n\r\n };\r\n\r\n /**\r\n * RIGHT or DOWN keydowns on block\r\n *\r\n * @param {Object} event\r\n * @private\r\n *\r\n * @description watches the selection and gets closest editable element.\r\n * Uses method getDeepestTextNodeFromPosition to get the last node of next block\r\n * Sets caret if it is contenteditable\r\n */\r\n var blockRightOrDownArrowPressed_ = function (event) {\r\n\r\n var selection = window.getSelection(),\r\n inputs = editor.state.inputs,\r\n focusedNode = selection.anchorNode,\r\n focusedNodeHolder;\r\n\r\n /** Check for caret existance */\r\n if (!focusedNode) {\r\n\r\n return false;\r\n\r\n }\r\n\r\n /** Looking for closest (parent) contentEditable element of focused node */\r\n while (focusedNode.contentEditable != 'true') {\r\n\r\n focusedNodeHolder = focusedNode.parentNode;\r\n focusedNode = focusedNodeHolder;\r\n\r\n }\r\n\r\n /** Input index in DOM level */\r\n var editableElementIndex = 0;\r\n\r\n while (focusedNode != inputs[editableElementIndex]) {\r\n\r\n editableElementIndex ++;\r\n\r\n }\r\n\r\n /**\r\n * Founded contentEditable element doesn't have childs\r\n * Or maybe New created block\r\n */\r\n if (!focusedNode.textContent) {\r\n\r\n editor.caret.setToNextBlock(editableElementIndex);\r\n return;\r\n\r\n }\r\n\r\n /**\r\n * Do nothing when caret doesn not reaches the end of last child\r\n */\r\n var caretInLastChild = false,\r\n caretAtTheEndOfText = false;\r\n\r\n var lastChild,\r\n deepestTextnode;\r\n\r\n lastChild = focusedNode.childNodes[focusedNode.childNodes.length - 1 ];\r\n\r\n if (editor.core.isDomNode(lastChild)) {\r\n\r\n deepestTextnode = editor.content.getDeepestTextNodeFromPosition(lastChild, lastChild.childNodes.length);\r\n\r\n } else {\r\n\r\n deepestTextnode = lastChild;\r\n\r\n }\r\n\r\n caretInLastChild = selection.anchorNode == deepestTextnode;\r\n caretAtTheEndOfText = deepestTextnode.length == selection.anchorOffset;\r\n\r\n if ( !caretInLastChild || !caretAtTheEndOfText ) {\r\n\r\n editor.core.log('arrow [down|right] : caret does not reached the end');\r\n return false;\r\n\r\n }\r\n\r\n editor.caret.setToNextBlock(editableElementIndex);\r\n\r\n };\r\n\r\n /**\r\n * LEFT or UP keydowns on block\r\n *\r\n * @param {Object} event\r\n * @private\r\n *\r\n * watches the selection and gets closest editable element.\r\n * Uses method getDeepestTextNodeFromPosition to get the last node of previous block\r\n * Sets caret if it is contenteditable\r\n *\r\n */\r\n var blockLeftOrUpArrowPressed_ = function (event) {\r\n\r\n var selection = window.getSelection(),\r\n inputs = editor.state.inputs,\r\n focusedNode = selection.anchorNode,\r\n focusedNodeHolder;\r\n\r\n /** Check for caret existance */\r\n if (!focusedNode) {\r\n\r\n return false;\r\n\r\n }\r\n\r\n /**\r\n * LEFT or UP not at the beginning\r\n */\r\n if ( selection.anchorOffset !== 0) {\r\n\r\n return false;\r\n\r\n }\r\n\r\n /** Looking for parent contentEditable block */\r\n while (focusedNode.contentEditable != 'true') {\r\n\r\n focusedNodeHolder = focusedNode.parentNode;\r\n focusedNode = focusedNodeHolder;\r\n\r\n }\r\n\r\n /** Input index in DOM level */\r\n var editableElementIndex = 0;\r\n\r\n while (focusedNode != inputs[editableElementIndex]) {\r\n\r\n editableElementIndex ++;\r\n\r\n }\r\n\r\n /**\r\n * Do nothing if caret is not at the beginning of first child\r\n */\r\n var caretInFirstChild = false,\r\n caretAtTheBeginning = false;\r\n\r\n var firstChild,\r\n deepestTextnode;\r\n\r\n /**\r\n * Founded contentEditable element doesn't have childs\r\n * Or maybe New created block\r\n */\r\n if (!focusedNode.textContent) {\r\n\r\n editor.caret.setToPreviousBlock(editableElementIndex);\r\n return;\r\n\r\n }\r\n\r\n firstChild = focusedNode.childNodes[0];\r\n\r\n if (editor.core.isDomNode(firstChild)) {\r\n\r\n deepestTextnode = editor.content.getDeepestTextNodeFromPosition(firstChild, 0);\r\n\r\n } else {\r\n\r\n deepestTextnode = firstChild;\r\n\r\n }\r\n\r\n caretInFirstChild = selection.anchorNode == deepestTextnode;\r\n caretAtTheBeginning = selection.anchorOffset === 0;\r\n\r\n if ( caretInFirstChild && caretAtTheBeginning ) {\r\n\r\n editor.caret.setToPreviousBlock(editableElementIndex);\r\n\r\n }\r\n\r\n };\r\n\r\n /**\r\n * Handles backspace keydown\r\n *\r\n * @param {Element} block\r\n * @param {Object} event\r\n * @private\r\n *\r\n * @description if block is empty, delete the block and set caret to the previous block\r\n * If block is not empty, try to merge two blocks - current and previous\r\n * But it we try'n to remove first block, then we should set caret to the next block, not previous.\r\n * If we removed the last block, create new one\r\n */\r\n var backspacePressed_ = function (block, event) {\r\n\r\n var currentInputIndex = editor.caret.getCurrentInputIndex(),\r\n range,\r\n selectionLength,\r\n firstLevelBlocksCount;\r\n\r\n if (isNativeInput_(event.target)) {\r\n\r\n /** If input value is empty - remove block */\r\n if (event.target.value.trim() == '') {\r\n\r\n block.remove();\r\n\r\n } else {\r\n\r\n return;\r\n\r\n }\r\n\r\n }\r\n\r\n if (block.textContent.trim()) {\r\n\r\n range = editor.content.getRange();\r\n selectionLength = range.endOffset - range.startOffset;\r\n\r\n if (editor.caret.position.atStart() && !selectionLength && editor.state.inputs[currentInputIndex - 1]) {\r\n\r\n editor.content.mergeBlocks(currentInputIndex);\r\n\r\n } else {\r\n\r\n return;\r\n\r\n }\r\n\r\n }\r\n\r\n if (!selectionLength) {\r\n\r\n block.remove();\r\n\r\n }\r\n\r\n\r\n firstLevelBlocksCount = editor.nodes.redactor.childNodes.length;\r\n\r\n /**\r\n * If all blocks are removed\r\n */\r\n if (firstLevelBlocksCount === 0) {\r\n\r\n /** update currentNode variable */\r\n editor.content.currentNode = null;\r\n\r\n /** Inserting new empty initial block */\r\n editor.ui.addInitialBlock();\r\n\r\n /** Updating inputs state after deleting last block */\r\n editor.ui.saveInputs();\r\n\r\n /** Set to current appended block */\r\n window.setTimeout(function () {\r\n\r\n editor.caret.setToPreviousBlock(1);\r\n\r\n }, 10);\r\n\r\n } else {\r\n\r\n if (editor.caret.inputIndex !== 0) {\r\n\r\n /** Target block is not first */\r\n editor.caret.setToPreviousBlock(editor.caret.inputIndex);\r\n\r\n } else {\r\n\r\n /** If we try to delete first block */\r\n editor.caret.setToNextBlock(editor.caret.inputIndex);\r\n\r\n }\r\n\r\n }\r\n\r\n editor.toolbar.move();\r\n\r\n if (!editor.toolbar.opened) {\r\n\r\n editor.toolbar.open();\r\n\r\n }\r\n\r\n /** Updating inputs state */\r\n editor.ui.saveInputs();\r\n\r\n /** Prevent default browser behaviour */\r\n event.preventDefault();\r\n\r\n };\r\n\r\n /**\r\n * This method prevents default behaviour.\r\n *\r\n * @param {Object} event\r\n * @protected\r\n *\r\n * @description We get from clipboard pasted data, sanitize, make a fragment that contains of this sanitized nodes.\r\n * Firstly, we need to memorize the caret position. We can do that by getting the range of selection.\r\n * After all, we insert clear fragment into caret placed position. Then, we should move the caret to the last node\r\n */\r\n callbacks.blockPasteCallback = function (event) {\r\n\r\n /** If area is input or textarea then allow default behaviour */\r\n if ( isNativeInput_(event.target) ) {\r\n\r\n return;\r\n\r\n }\r\n\r\n /** Prevent default behaviour */\r\n event.preventDefault();\r\n\r\n var editableParent = editor.content.getEditableParent(event.target);\r\n\r\n /** Allow paste when event target placed in Editable element */\r\n if (!editableParent) {\r\n\r\n return;\r\n\r\n }\r\n\r\n /** get html pasted data - dirty data */\r\n var htmlData = event.clipboardData.getData('text/html'),\r\n plainData = event.clipboardData.getData('text/plain');\r\n\r\n\r\n /** Temporary DIV that is used to work with childs as arrays item */\r\n var div = editor.draw.node('DIV', '', {}),\r\n cleanData,\r\n wrappedData;\r\n\r\n /** Create fragment, that we paste to range after proccesing */\r\n cleanData = editor.sanitizer.clean(htmlData);\r\n\r\n\r\n /**\r\n * We wrap pasted text with

    tags to split it logically\r\n *\r\n * @type {string}\r\n */\r\n wrappedData = editor.content.wrapTextWithParagraphs(cleanData, plainData);\r\n div.innerHTML = wrappedData;\r\n\r\n var NEW_BLOCK_TYPE = editor.settings.initialBlockPlugin,\r\n currentBlockContent = editor.content.currentNode.firstChild.firstChild;\r\n\r\n /**\r\n * If there only one paragraph, just insert it\r\n */\r\n if (div.childNodes.length == 1) {\r\n\r\n editor.caret.insertNode(document.createTextNode(div.firstChild.innerHTML));\r\n return;\r\n\r\n }\r\n\r\n div.childNodes.forEach(function (paragraph, index) {\r\n\r\n\r\n /**\r\n * If there was no data in working node, replace it with first paragraph of pasted text\r\n */\r\n if (index == 0 && currentBlockContent.innerHTML.trim() === '') {\r\n\r\n editor.content.switchBlock(editor.content.currentNode, editor.tools[NEW_BLOCK_TYPE].render({\r\n text : paragraph.innerHTML\r\n }), NEW_BLOCK_TYPE);\r\n\r\n return;\r\n\r\n }\r\n\r\n editor.content.insertBlock({\r\n type : NEW_BLOCK_TYPE,\r\n block : editor.tools[NEW_BLOCK_TYPE].render({\r\n text : paragraph.innerHTML\r\n })\r\n });\r\n\r\n editor.caret.inputIndex++;\r\n\r\n });\r\n\r\n editor.caret.setToPreviousBlock(editor.caret.getCurrentInputIndex() + 1);\r\n\r\n\r\n };\r\n\r\n /**\r\n * used by UI module\r\n * Clicks on block settings button\r\n *\r\n * @param {Object} event\r\n * @protected\r\n * @description Opens toolbar settings\r\n */\r\n callbacks.showSettingsButtonClicked = function (event) {\r\n\r\n /**\r\n * Get type of current block\r\n * It uses to append settings from tool.settings property.\r\n * ...\r\n * Type is stored in data-type attribute on block\r\n */\r\n var currentToolType = editor.content.currentNode.dataset.tool;\r\n\r\n editor.toolbar.settings.toggle(currentToolType);\r\n\r\n /** Close toolbox when settings button is active */\r\n editor.toolbar.toolbox.close();\r\n editor.toolbar.settings.hideRemoveActions();\r\n\r\n };\r\n\r\n /**\r\n * Check block\r\n * @param target\r\n * @private\r\n *\r\n * @description Checks target is it native input\r\n */\r\n var isNativeInput_ = function (target) {\r\n\r\n var nativeInputAreas = ['INPUT', 'TEXTAREA'];\r\n\r\n return (nativeInputAreas.indexOf(target.tagName) != -1);\r\n\r\n };\r\n\r\n return callbacks;\r\n\r\n})({});\n\n\n// WEBPACK FOOTER //\n// ./modules/callbacks.js","/**\r\n * Codex Editor Draw module\r\n *\r\n * @author Codex Team\r\n * @version 1.0.\r\n */\r\n\r\nmodule.exports = (function (draw) {\r\n\r\n /**\r\n * Base editor wrapper\r\n */\r\n draw.wrapper = function () {\r\n\r\n var wrapper = document.createElement('div');\r\n\r\n wrapper.className += 'codex-editor';\r\n\r\n return wrapper;\r\n\r\n };\r\n\r\n /**\r\n * Content-editable holder\r\n */\r\n draw.redactor = function () {\r\n\r\n var redactor = document.createElement('div');\r\n\r\n redactor.className += 'ce-redactor';\r\n\r\n return redactor;\r\n\r\n };\r\n\r\n draw.ceBlock = function () {\r\n\r\n var block = document.createElement('DIV');\r\n\r\n block.className += 'ce_block';\r\n\r\n return block;\r\n\r\n };\r\n\r\n /**\r\n * Empty toolbar with toggler\r\n */\r\n draw.toolbar = function () {\r\n\r\n var bar = document.createElement('div');\r\n\r\n bar.className += 'ce-toolbar';\r\n\r\n return bar;\r\n\r\n };\r\n\r\n draw.toolbarContent = function () {\r\n\r\n var wrapper = document.createElement('DIV');\r\n\r\n wrapper.classList.add('ce-toolbar__content');\r\n\r\n return wrapper;\r\n\r\n };\r\n\r\n /**\r\n * Inline toolbar\r\n */\r\n draw.inlineToolbar = function () {\r\n\r\n var bar = document.createElement('DIV');\r\n\r\n bar.className += 'ce-toolbar-inline';\r\n\r\n return bar;\r\n\r\n };\r\n\r\n /**\r\n * Wrapper for inline toobar buttons\r\n */\r\n draw.inlineToolbarButtons = function () {\r\n\r\n var wrapper = document.createElement('DIV');\r\n\r\n wrapper.className += 'ce-toolbar-inline__buttons';\r\n\r\n return wrapper;\r\n\r\n };\r\n\r\n /**\r\n * For some actions\r\n */\r\n draw.inlineToolbarActions = function () {\r\n\r\n var wrapper = document.createElement('DIV');\r\n\r\n wrapper.className += 'ce-toolbar-inline__actions';\r\n\r\n return wrapper;\r\n\r\n };\r\n\r\n draw.inputForLink = function () {\r\n\r\n var input = document.createElement('INPUT');\r\n\r\n input.type = 'input';\r\n input.className += 'inputForLink';\r\n input.placeholder = 'Вставьте ссылку ...';\r\n input.setAttribute('form', 'defaultForm');\r\n\r\n input.setAttribute('autofocus', 'autofocus');\r\n\r\n return input;\r\n\r\n };\r\n\r\n /**\r\n * @todo Desc\r\n */\r\n draw.blockButtons = function () {\r\n\r\n var block = document.createElement('div');\r\n\r\n block.className += 'ce-toolbar__actions';\r\n\r\n return block;\r\n\r\n };\r\n\r\n /**\r\n * Block settings panel\r\n */\r\n draw.blockSettings = function () {\r\n\r\n var settings = document.createElement('div');\r\n\r\n settings.className += 'ce-settings';\r\n\r\n return settings;\r\n\r\n };\r\n\r\n draw.defaultSettings = function () {\r\n\r\n var div = document.createElement('div');\r\n\r\n div.classList.add('ce-settings_default');\r\n\r\n return div;\r\n\r\n };\r\n\r\n draw.pluginsSettings = function () {\r\n\r\n var div = document.createElement('div');\r\n\r\n div.classList.add('ce-settings_plugin');\r\n\r\n return div;\r\n\r\n };\r\n\r\n draw.plusButton = function () {\r\n\r\n var button = document.createElement('span');\r\n\r\n button.className = 'ce-toolbar__plus';\r\n // button.innerHTML = '';\r\n\r\n return button;\r\n\r\n };\r\n\r\n /**\r\n * Settings button in toolbar\r\n */\r\n draw.settingsButton = function () {\r\n\r\n var toggler = document.createElement('span');\r\n\r\n toggler.className = 'ce-toolbar__settings-btn';\r\n\r\n /** Toggler button*/\r\n toggler.innerHTML = '';\r\n\r\n return toggler;\r\n\r\n };\r\n\r\n /**\r\n * Redactor tools wrapper\r\n */\r\n\r\n draw.toolbox = function () {\r\n\r\n var wrapper = document.createElement('div');\r\n\r\n wrapper.className = 'ce-toolbar__tools';\r\n\r\n return wrapper;\r\n\r\n };\r\n\r\n /**\r\n * @protected\r\n *\r\n * Draws tool buttons for toolbox\r\n *\r\n * @param {String} type\r\n * @param {String} classname\r\n * @returns {Element}\r\n */\r\n draw.toolbarButton = function (type, classname) {\r\n\r\n var button = document.createElement('li'),\r\n toolIcon = document.createElement('i'),\r\n toolTitle = document.createElement('span');\r\n\r\n button.dataset.type = type;\r\n button.setAttribute('title', type);\r\n\r\n toolIcon.classList.add(classname);\r\n toolTitle.classList.add('ce_toolbar_tools--title');\r\n\r\n\r\n button.appendChild(toolIcon);\r\n button.appendChild(toolTitle);\r\n\r\n return button;\r\n\r\n };\r\n\r\n /**\r\n * @protected\r\n *\r\n * Draws tools for inline toolbar\r\n *\r\n * @param {String} type\r\n * @param {String} classname\r\n */\r\n draw.toolbarButtonInline = function (type, classname) {\r\n\r\n var button = document.createElement('BUTTON'),\r\n toolIcon = document.createElement('I');\r\n\r\n button.type = 'button';\r\n button.dataset.type = type;\r\n toolIcon.classList.add(classname);\r\n\r\n button.appendChild(toolIcon);\r\n\r\n return button;\r\n\r\n };\r\n\r\n /**\r\n * Redactor block\r\n */\r\n draw.block = function (tagName, content) {\r\n\r\n var node = document.createElement(tagName);\r\n\r\n node.innerHTML = content || '';\r\n\r\n return node;\r\n\r\n };\r\n\r\n /**\r\n * Creates Node with passed tagName and className\r\n * @param {string} tagName\r\n * @param {string} className\r\n * @param {object} properties - allow to assign properties\r\n */\r\n draw.node = function ( tagName, className, properties ) {\r\n\r\n var el = document.createElement( tagName );\r\n\r\n if ( className ) el.className = className;\r\n\r\n if ( properties ) {\r\n\r\n for (var name in properties) {\r\n\r\n el[name] = properties[name];\r\n\r\n }\r\n\r\n }\r\n\r\n return el;\r\n\r\n };\r\n\r\n /**\r\n * Unavailable plugin block\r\n */\r\n draw.unavailableBlock = function () {\r\n\r\n var wrapper = document.createElement('DIV');\r\n\r\n wrapper.classList.add('cdx-unavailable-block');\r\n\r\n return wrapper;\r\n\r\n };\r\n\r\n return draw;\r\n\r\n})({});\n\n\n// WEBPACK FOOTER //\n// ./modules/draw.js","/**\r\n * Codex Editor Caret Module\r\n *\r\n * @author Codex Team\r\n * @version 1.0\r\n */\r\n\r\nmodule.exports = (function (caret) {\r\n\r\n let editor = codex.editor;\r\n\r\n /**\r\n * @var {int} InputIndex - editable element in DOM\r\n */\r\n caret.inputIndex = null;\r\n\r\n /**\r\n * @var {int} offset - caret position in a text node.\r\n */\r\n caret.offset = null;\r\n\r\n /**\r\n * @var {int} focusedNodeIndex - we get index of child node from first-level block\r\n */\r\n caret.focusedNodeIndex = null;\r\n\r\n /**\r\n * Creates Document Range and sets caret to the element.\r\n * @protected\r\n * @uses caret.save — if you need to save caret position\r\n * @param {Element} el - Changed Node.\r\n */\r\n caret.set = function ( el, index, offset) {\r\n\r\n offset = offset || caret.offset || 0;\r\n index = index || caret.focusedNodeIndex || 0;\r\n\r\n var childs = el.childNodes,\r\n nodeToSet;\r\n\r\n if ( childs.length === 0 ) {\r\n\r\n nodeToSet = el;\r\n\r\n } else {\r\n\r\n nodeToSet = childs[index];\r\n\r\n }\r\n\r\n /** If Element is INPUT */\r\n if (el.contentEditable != 'true') {\r\n\r\n el.focus();\r\n return;\r\n\r\n }\r\n\r\n if (editor.core.isDomNode(nodeToSet)) {\r\n\r\n nodeToSet = editor.content.getDeepestTextNodeFromPosition(nodeToSet, nodeToSet.childNodes.length);\r\n\r\n }\r\n\r\n var range = document.createRange(),\r\n selection = window.getSelection();\r\n\r\n window.setTimeout(function () {\r\n\r\n range.setStart(nodeToSet, offset);\r\n range.setEnd(nodeToSet, offset);\r\n\r\n selection.removeAllRanges();\r\n selection.addRange(range);\r\n\r\n editor.caret.saveCurrentInputIndex();\r\n\r\n }, 20);\r\n\r\n };\r\n\r\n /**\r\n * @protected\r\n * Updates index of input and saves it in caret object\r\n */\r\n caret.saveCurrentInputIndex = function () {\r\n\r\n /** Index of Input that we paste sanitized content */\r\n var selection = window.getSelection(),\r\n inputs = editor.state.inputs,\r\n focusedNode = selection.anchorNode,\r\n focusedNodeHolder;\r\n\r\n if (!focusedNode) {\r\n\r\n return;\r\n\r\n }\r\n\r\n /** Looking for parent contentEditable block */\r\n while (focusedNode.contentEditable != 'true') {\r\n\r\n focusedNodeHolder = focusedNode.parentNode;\r\n focusedNode = focusedNodeHolder;\r\n\r\n }\r\n\r\n /** Input index in DOM level */\r\n var editableElementIndex = 0;\r\n\r\n while (focusedNode != inputs[editableElementIndex]) {\r\n\r\n editableElementIndex ++;\r\n\r\n }\r\n\r\n caret.inputIndex = editableElementIndex;\r\n\r\n };\r\n\r\n /**\r\n * Returns current input index (caret object)\r\n */\r\n caret.getCurrentInputIndex = function () {\r\n\r\n return caret.inputIndex;\r\n\r\n };\r\n\r\n /**\r\n * @param {int} index - index of first-level block after that we set caret into next input\r\n */\r\n caret.setToNextBlock = function (index) {\r\n\r\n var inputs = editor.state.inputs,\r\n nextInput = inputs[index + 1];\r\n\r\n if (!nextInput) {\r\n\r\n editor.core.log('We are reached the end');\r\n return;\r\n\r\n }\r\n\r\n /**\r\n * When new Block created or deleted content of input\r\n * We should add some text node to set caret\r\n */\r\n if (!nextInput.childNodes.length) {\r\n\r\n var emptyTextElement = document.createTextNode('');\r\n\r\n nextInput.appendChild(emptyTextElement);\r\n\r\n }\r\n\r\n editor.caret.inputIndex = index + 1;\r\n editor.caret.set(nextInput, 0, 0);\r\n editor.content.workingNodeChanged(nextInput);\r\n\r\n };\r\n\r\n /**\r\n * @param {int} index - index of target input.\r\n * Sets caret to input with this index\r\n */\r\n caret.setToBlock = function (index) {\r\n\r\n var inputs = editor.state.inputs,\r\n targetInput = inputs[index];\r\n\r\n if ( !targetInput ) {\r\n\r\n return;\r\n\r\n }\r\n\r\n /**\r\n * When new Block created or deleted content of input\r\n * We should add some text node to set caret\r\n */\r\n if (!targetInput.childNodes.length) {\r\n\r\n var emptyTextElement = document.createTextNode('');\r\n\r\n targetInput.appendChild(emptyTextElement);\r\n\r\n }\r\n\r\n editor.caret.inputIndex = index;\r\n editor.caret.set(targetInput, 0, 0);\r\n editor.content.workingNodeChanged(targetInput);\r\n\r\n };\r\n\r\n /**\r\n * @param {int} index - index of input\r\n */\r\n caret.setToPreviousBlock = function (index) {\r\n\r\n index = index || 0;\r\n\r\n var inputs = editor.state.inputs,\r\n previousInput = inputs[index - 1],\r\n lastChildNode,\r\n lengthOfLastChildNode,\r\n emptyTextElement;\r\n\r\n\r\n if (!previousInput) {\r\n\r\n editor.core.log('We are reached first node');\r\n return;\r\n\r\n }\r\n\r\n lastChildNode = editor.content.getDeepestTextNodeFromPosition(previousInput, previousInput.childNodes.length);\r\n lengthOfLastChildNode = lastChildNode.length;\r\n\r\n /**\r\n * When new Block created or deleted content of input\r\n * We should add some text node to set caret\r\n */\r\n if (!previousInput.childNodes.length) {\r\n\r\n emptyTextElement = document.createTextNode('');\r\n previousInput.appendChild(emptyTextElement);\r\n\r\n }\r\n editor.caret.inputIndex = index - 1;\r\n editor.caret.set(previousInput, previousInput.childNodes.length - 1, lengthOfLastChildNode);\r\n editor.content.workingNodeChanged(inputs[index - 1]);\r\n\r\n };\r\n\r\n caret.position = {\r\n\r\n atStart : function () {\r\n\r\n var selection = window.getSelection(),\r\n anchorOffset = selection.anchorOffset,\r\n anchorNode = selection.anchorNode,\r\n firstLevelBlock = editor.content.getFirstLevelBlock(anchorNode),\r\n pluginsRender = firstLevelBlock.childNodes[0];\r\n\r\n if (!editor.core.isDomNode(anchorNode)) {\r\n\r\n anchorNode = anchorNode.parentNode;\r\n\r\n }\r\n\r\n var isFirstNode = anchorNode === pluginsRender.childNodes[0],\r\n isOffsetZero = anchorOffset === 0;\r\n\r\n return isFirstNode && isOffsetZero;\r\n\r\n },\r\n\r\n atTheEnd : function () {\r\n\r\n var selection = window.getSelection(),\r\n anchorOffset = selection.anchorOffset,\r\n anchorNode = selection.anchorNode;\r\n\r\n /** Caret is at the end of input */\r\n return !anchorNode || !anchorNode.length || anchorOffset === anchorNode.length;\r\n\r\n }\r\n };\r\n\r\n\r\n /**\r\n * Inserts node at the caret location\r\n * @param node\r\n */\r\n caret.insertNode = function (node) {\r\n\r\n var selection, range;\r\n\r\n selection = window.getSelection();\r\n\r\n range = selection.getRangeAt(0);\r\n range.deleteContents();\r\n\r\n range.insertNode(node);\r\n\r\n range.setStartAfter(node);\r\n range.collapse(true);\r\n\r\n selection.removeAllRanges();\r\n selection.addRange(range);\r\n\r\n\r\n };\r\n\r\n return caret;\r\n\r\n})({});\n\n\n// WEBPACK FOOTER //\n// ./modules/caret.js","/**\r\n * Codex Editor Notification Module\r\n *\r\n * @author Codex Team\r\n * @version 1.0\r\n */\r\n\r\nmodule.exports = (function (notifications) {\r\n\r\n let editor = codex.editor;\r\n\r\n var queue = [];\r\n\r\n var addToQueue = function (settings) {\r\n\r\n queue.push(settings);\r\n\r\n var index = 0;\r\n\r\n while ( index < queue.length && queue.length > 5) {\r\n\r\n if (queue[index].type == 'confirm' || queue[index].type == 'prompt') {\r\n\r\n index++;\r\n continue;\r\n\r\n }\r\n\r\n queue[index].close();\r\n queue.splice(index, 1);\r\n\r\n }\r\n\r\n };\r\n\r\n notifications.createHolder = function () {\r\n\r\n var holder = editor.draw.node('DIV', 'cdx-notifications-block');\r\n\r\n editor.nodes.notifications = document.body.appendChild(holder);\r\n\r\n return holder;\r\n\r\n };\r\n\r\n\r\n /**\r\n * Error notificator. Shows block with message\r\n * @protected\r\n */\r\n notifications.errorThrown = function (errorMsg, event) {\r\n\r\n editor.notifications.notification({message: 'This action is not available currently', type: event.type});\r\n\r\n };\r\n\r\n /**\r\n *\r\n * Appends notification\r\n *\r\n * settings = {\r\n * type - notification type (reserved types: alert, confirm, prompt). Just add class 'cdx-notification-'+type\r\n * message - notification message\r\n * okMsg - confirm button text (default - 'Ok')\r\n * cancelBtn - cancel button text (default - 'Cancel'). Only for confirm and prompt types\r\n * confirm - function-handler for ok button click\r\n * cancel - function-handler for cancel button click. Only for confirm and prompt types\r\n * time - time (in seconds) after which notification will close (default - 10s)\r\n * }\r\n *\r\n * @param settings\r\n */\r\n notifications.notification = function (constructorSettings) {\r\n\r\n /** Private vars and methods */\r\n var notification = null,\r\n cancel = null,\r\n type = null,\r\n confirm = null,\r\n inputField = null;\r\n\r\n var confirmHandler = function () {\r\n\r\n close();\r\n\r\n if (typeof confirm !== 'function' ) {\r\n\r\n return;\r\n\r\n }\r\n\r\n if (type == 'prompt') {\r\n\r\n confirm(inputField.value);\r\n return;\r\n\r\n }\r\n\r\n confirm();\r\n\r\n };\r\n\r\n var cancelHandler = function () {\r\n\r\n close();\r\n\r\n if (typeof cancel !== 'function' ) {\r\n\r\n return;\r\n\r\n }\r\n\r\n cancel();\r\n\r\n };\r\n\r\n\r\n /** Public methods */\r\n function create(settings) {\r\n\r\n if (!(settings && settings.message)) {\r\n\r\n editor.core.log('Can\\'t create notification. Message is missed');\r\n return;\r\n\r\n }\r\n\r\n settings.type = settings.type || 'alert';\r\n settings.time = settings.time*1000 || 10000;\r\n\r\n var wrapper = editor.draw.node('DIV', 'cdx-notification'),\r\n message = editor.draw.node('DIV', 'cdx-notification__message'),\r\n input = editor.draw.node('INPUT', 'cdx-notification__input'),\r\n okBtn = editor.draw.node('SPAN', 'cdx-notification__ok-btn'),\r\n cancelBtn = editor.draw.node('SPAN', 'cdx-notification__cancel-btn');\r\n\r\n message.textContent = settings.message;\r\n okBtn.textContent = settings.okMsg || 'ОК';\r\n cancelBtn.textContent = settings.cancelMsg || 'Отмена';\r\n\r\n editor.listeners.add(okBtn, 'click', confirmHandler);\r\n editor.listeners.add(cancelBtn, 'click', cancelHandler);\r\n\r\n wrapper.appendChild(message);\r\n\r\n if (settings.type == 'prompt') {\r\n\r\n wrapper.appendChild(input);\r\n\r\n }\r\n\r\n wrapper.appendChild(okBtn);\r\n\r\n if (settings.type == 'prompt' || settings.type == 'confirm') {\r\n\r\n wrapper.appendChild(cancelBtn);\r\n\r\n }\r\n\r\n wrapper.classList.add('cdx-notification-' + settings.type);\r\n wrapper.dataset.type = settings.type;\r\n\r\n notification = wrapper;\r\n type = settings.type;\r\n confirm = settings.confirm;\r\n cancel = settings.cancel;\r\n inputField = input;\r\n\r\n if (settings.type != 'prompt' && settings.type != 'confirm') {\r\n\r\n window.setTimeout(close, settings.time);\r\n\r\n }\r\n\r\n };\r\n\r\n function send() {\r\n\r\n editor.nodes.notifications.appendChild(notification);\r\n inputField.focus();\r\n\r\n editor.nodes.notifications.classList.add('cdx-notification__notification-appending');\r\n\r\n window.setTimeout(function () {\r\n\r\n editor.nodes.notifications.classList.remove('cdx-notification__notification-appending');\r\n\r\n }, 100);\r\n\r\n addToQueue({type: type, close: close});\r\n\r\n };\r\n\r\n function close() {\r\n\r\n notification.remove();\r\n\r\n };\r\n\r\n\r\n if (constructorSettings) {\r\n\r\n create(constructorSettings);\r\n send();\r\n\r\n }\r\n\r\n return {\r\n create: create,\r\n send: send,\r\n close: close\r\n };\r\n\r\n };\r\n\r\n notifications.clear = function () {\r\n\r\n editor.nodes.notifications.innerHTML = '';\r\n queue = [];\r\n\r\n };\r\n\r\n return notifications;\r\n\r\n})({});\n\n\n// WEBPACK FOOTER //\n// ./modules/notifications.js","/**\r\n * Codex Editor Parser Module\r\n *\r\n * @author Codex Team\r\n * @version 1.1\r\n */\r\n\r\nmodule.exports = (function (parser) {\r\n\r\n let editor = codex.editor;\r\n\r\n /** inserting text */\r\n parser.insertPastedContent = function (blockType, tag) {\r\n\r\n editor.content.insertBlock({\r\n type : blockType.type,\r\n block : blockType.render({\r\n text : tag.innerHTML\r\n })\r\n });\r\n\r\n };\r\n\r\n /**\r\n * Check DOM node for display style: separated block or child-view\r\n */\r\n parser.isFirstLevelBlock = function (node) {\r\n\r\n return node.nodeType == editor.core.nodeTypes.TAG &&\r\n node.classList.contains(editor.ui.className.BLOCK_CLASSNAME);\r\n\r\n };\r\n\r\n return parser;\r\n\r\n})({});\r\n\n\n\n// WEBPACK FOOTER //\n// ./modules/parser.js","/**\r\n * Codex Sanitizer\r\n */\r\n\r\nmodule.exports = (function (sanitizer) {\r\n\r\n /** HTML Janitor library */\r\n let janitor = require('html-janitor');\r\n\r\n /** Codex Editor */\r\n let editor = codex.editor;\r\n\r\n sanitizer.prepare = function () {\r\n\r\n if (editor.settings.sanitizer && !editor.core.isEmpty(editor.settings.sanitizer)) {\r\n\r\n Config.CUSTOM = editor.settings.sanitizer;\r\n\r\n }\r\n\r\n };\r\n\r\n /**\r\n * Basic config\r\n */\r\n var Config = {\r\n\r\n /** User configuration */\r\n CUSTOM : null,\r\n\r\n BASIC : {\r\n\r\n tags: {\r\n p: {},\r\n a: {\r\n href: true,\r\n target: '_blank',\r\n rel: 'nofollow'\r\n },\r\n i: {},\r\n b: {},\r\n strong: {},\r\n em: {},\r\n span: {}\r\n }\r\n }\r\n };\r\n\r\n sanitizer.Config = Config;\r\n\r\n /**\r\n *\r\n * @param userCustomConfig\r\n * @returns {*}\r\n * @private\r\n *\r\n * @description If developer uses editor's API, then he can customize sane restrictions.\r\n * Or, sane config can be defined globally in editors initialization. That config will be used everywhere\r\n * At least, if there is no config overrides, that API uses BASIC Default configation\r\n */\r\n let init_ = function (userCustomConfig) {\r\n\r\n let configuration = userCustomConfig || Config.CUSTOM || Config.BASIC;\r\n\r\n return new janitor(configuration);\r\n\r\n };\r\n\r\n /**\r\n * Cleans string from unwanted tags\r\n * @protected\r\n * @param {String} dirtyString - taint string\r\n * @param {Object} customConfig - allowed tags\r\n */\r\n sanitizer.clean = function (dirtyString, customConfig) {\r\n\r\n let janitorInstance = init_(customConfig);\r\n\r\n return janitorInstance.clean(dirtyString);\r\n\r\n };\r\n\r\n return sanitizer;\r\n\r\n})({});\n\n\n// WEBPACK FOOTER //\n// ./modules/sanitizer.js","(function (root, factory) {\n if (typeof define === 'function' && define.amd) {\n define('html-janitor', factory);\n } else if (typeof exports === 'object') {\n module.exports = factory();\n } else {\n root.HTMLJanitor = factory();\n }\n}(this, function () {\n\n /**\n * @param {Object} config.tags Dictionary of allowed tags.\n * @param {boolean} config.keepNestedBlockElements Default false.\n */\n function HTMLJanitor(config) {\n\n var tagDefinitions = config['tags'];\n var tags = Object.keys(tagDefinitions);\n\n var validConfigValues = tags\n .map(function(k) { return typeof tagDefinitions[k]; })\n .every(function(type) { return type === 'object' || type === 'boolean' || type === 'function'; });\n\n if(!validConfigValues) {\n throw new Error(\"The configuration was invalid\");\n }\n\n this.config = config;\n }\n\n // TODO: not exhaustive?\n var blockElementNames = ['P', 'LI', 'TD', 'TH', 'DIV', 'H1', 'H2', 'H3', 'H4', 'H5', 'H6', 'PRE'];\n function isBlockElement(node) {\n return blockElementNames.indexOf(node.nodeName) !== -1;\n }\n\n var inlineElementNames = ['A', 'B', 'STRONG', 'I', 'EM', 'SUB', 'SUP', 'U', 'STRIKE'];\n function isInlineElement(node) {\n return inlineElementNames.indexOf(node.nodeName) !== -1;\n }\n\n HTMLJanitor.prototype.clean = function (html) {\n var sandbox = document.createElement('div');\n sandbox.innerHTML = html;\n\n this._sanitize(sandbox);\n\n return sandbox.innerHTML;\n };\n\n HTMLJanitor.prototype._sanitize = function (parentNode) {\n var treeWalker = createTreeWalker(parentNode);\n var node = treeWalker.firstChild();\n if (!node) { return; }\n\n do {\n // Ignore nodes that have already been sanitized\n if (node._sanitized) {\n continue;\n }\n\n if (node.nodeType === Node.TEXT_NODE) {\n // If this text node is just whitespace and the previous or next element\n // sibling is a block element, remove it\n // N.B.: This heuristic could change. Very specific to a bug with\n // `contenteditable` in Firefox: http://jsbin.com/EyuKase/1/edit?js,output\n // FIXME: make this an option?\n if (node.data.trim() === ''\n && ((node.previousElementSibling && isBlockElement(node.previousElementSibling))\n || (node.nextElementSibling && isBlockElement(node.nextElementSibling)))) {\n parentNode.removeChild(node);\n this._sanitize(parentNode);\n break;\n } else {\n continue;\n }\n }\n\n // Remove all comments\n if (node.nodeType === Node.COMMENT_NODE) {\n parentNode.removeChild(node);\n this._sanitize(parentNode);\n break;\n }\n\n var isInline = isInlineElement(node);\n var containsBlockElement;\n if (isInline) {\n containsBlockElement = Array.prototype.some.call(node.childNodes, isBlockElement);\n }\n\n // Block elements should not be nested (e.g.

  • ...); if\n // they are, we want to unwrap the inner block element.\n var isNotTopContainer = !! parentNode.parentNode;\n var isNestedBlockElement =\n isBlockElement(parentNode) &&\n isBlockElement(node) &&\n isNotTopContainer;\n\n var nodeName = node.nodeName.toLowerCase();\n\n var allowedAttrs = getAllowedAttrs(this.config, nodeName, node);\n\n var isInvalid = isInline && containsBlockElement;\n\n // Drop tag entirely according to the whitelist *and* if the markup\n // is invalid.\n if (isInvalid || shouldRejectNode(node, allowedAttrs)\n || (!this.config.keepNestedBlockElements && isNestedBlockElement)) {\n // Do not keep the inner text of SCRIPT/STYLE elements.\n if (! (node.nodeName === 'SCRIPT' || node.nodeName === 'STYLE')) {\n while (node.childNodes.length > 0) {\n parentNode.insertBefore(node.childNodes[0], node);\n }\n }\n parentNode.removeChild(node);\n\n this._sanitize(parentNode);\n break;\n }\n\n // Sanitize attributes\n for (var a = 0; a < node.attributes.length; a += 1) {\n var attr = node.attributes[a];\n\n if (shouldRejectAttr(attr, allowedAttrs, node)) {\n node.removeAttribute(attr.name);\n // Shift the array to continue looping.\n a = a - 1;\n }\n }\n\n // Sanitize children\n this._sanitize(node);\n\n // Mark node as sanitized so it's ignored in future runs\n node._sanitized = true;\n } while ((node = treeWalker.nextSibling()));\n };\n\n function createTreeWalker(node) {\n return document.createTreeWalker(node,\n NodeFilter.SHOW_TEXT | NodeFilter.SHOW_ELEMENT | NodeFilter.SHOW_COMMENT,\n null, false);\n }\n\n function getAllowedAttrs(config, nodeName, node){\n if (typeof config.tags[nodeName] === 'function') {\n return config.tags[nodeName](node);\n } else {\n return config.tags[nodeName];\n }\n }\n\n function shouldRejectNode(node, allowedAttrs){\n if (typeof allowedAttrs === 'undefined') {\n return true;\n } else if (typeof allowedAttrs === 'boolean') {\n return !allowedAttrs;\n }\n\n return false;\n }\n\n function shouldRejectAttr(attr, allowedAttrs, node){\n var attrName = attr.name.toLowerCase();\n\n if (allowedAttrs === true){\n return false;\n } else if (typeof allowedAttrs[attrName] === 'function'){\n return !allowedAttrs[attrName](attr.value, node);\n } else if (typeof allowedAttrs[attrName] === 'undefined'){\n return true;\n } else if (allowedAttrs[attrName] === false) {\n return true;\n } else if (typeof allowedAttrs[attrName] === 'string') {\n return (allowedAttrs[attrName] !== attr.value);\n }\n\n return false;\n }\n\n return HTMLJanitor;\n\n}));\n\n\n\n//////////////////\n// WEBPACK FOOTER\n// ./~/html-janitor/src/html-janitor.js\n// module id = 18\n// module chunks = 0","/**\r\n * Codex Editor Listeners module\r\n *\r\n * @author Codex Team\r\n * @version 1.0\r\n */\r\n\r\n/**\r\n * Module-decorator for event listeners assignment\r\n */\r\nmodule.exports = function (listeners) {\r\n\r\n var allListeners = [];\r\n\r\n /**\r\n * Search methods\r\n *\r\n * byElement, byType and byHandler returns array of suitable listeners\r\n * one and all takes element, eventType, and handler and returns first (all) suitable listener\r\n *\r\n */\r\n listeners.search = function () {\r\n\r\n var byElement = function (element, context) {\r\n\r\n var listenersOnElement = [];\r\n\r\n context = context || allListeners;\r\n\r\n for (var i = 0; i < context.length; i++) {\r\n\r\n var listener = context[i];\r\n\r\n if (listener.element === element) {\r\n\r\n listenersOnElement.push(listener);\r\n\r\n }\r\n\r\n }\r\n\r\n return listenersOnElement;\r\n\r\n };\r\n\r\n var byType = function (eventType, context) {\r\n\r\n var listenersWithType = [];\r\n\r\n context = context || allListeners;\r\n\r\n for (var i = 0; i < context.length; i++) {\r\n\r\n var listener = context[i];\r\n\r\n if (listener.type === eventType) {\r\n\r\n listenersWithType.push(listener);\r\n\r\n }\r\n\r\n }\r\n\r\n return listenersWithType;\r\n\r\n };\r\n\r\n var byHandler = function (handler, context) {\r\n\r\n var listenersWithHandler = [];\r\n\r\n context = context || allListeners;\r\n\r\n for (var i = 0; i < context.length; i++) {\r\n\r\n var listener = context[i];\r\n\r\n if (listener.handler === handler) {\r\n\r\n listenersWithHandler.push(listener);\r\n\r\n }\r\n\r\n }\r\n\r\n return listenersWithHandler;\r\n\r\n };\r\n\r\n var one = function (element, eventType, handler) {\r\n\r\n var result = allListeners;\r\n\r\n if (element)\r\n result = byElement(element, result);\r\n\r\n if (eventType)\r\n result = byType(eventType, result);\r\n\r\n if (handler)\r\n result = byHandler(handler, result);\r\n\r\n return result[0];\r\n\r\n };\r\n\r\n var all = function (element, eventType, handler) {\r\n\r\n var result = allListeners;\r\n\r\n if (element)\r\n result = byElement(element, result);\r\n\r\n if (eventType)\r\n result = byType(eventType, result);\r\n\r\n if (handler)\r\n result = byHandler(handler, result);\r\n\r\n return result;\r\n\r\n };\r\n\r\n return {\r\n byElement : byElement,\r\n byType : byType,\r\n byHandler : byHandler,\r\n one : one,\r\n all : all\r\n };\r\n\r\n }();\r\n\r\n listeners.add = function (element, eventType, handler, isCapture) {\r\n\r\n element.addEventListener(eventType, handler, isCapture);\r\n\r\n var data = {\r\n element: element,\r\n type: eventType,\r\n handler: handler\r\n };\r\n\r\n var alreadyAddedListener = listeners.search.one(element, eventType, handler);\r\n\r\n if (!alreadyAddedListener) {\r\n\r\n allListeners.push(data);\r\n\r\n }\r\n\r\n };\r\n\r\n listeners.remove = function (element, eventType, handler) {\r\n\r\n element.removeEventListener(eventType, handler);\r\n\r\n var existingListeners = listeners.search.all(element, eventType, handler);\r\n\r\n for (var i = 0; i < existingListeners.length; i++) {\r\n\r\n var index = allListeners.indexOf(existingListeners[i]);\r\n\r\n if (index > 0) {\r\n\r\n allListeners.splice(index, 1);\r\n\r\n }\r\n\r\n }\r\n\r\n };\r\n\r\n listeners.removeAll = function () {\r\n\r\n allListeners.map(function (current) {\r\n\r\n listeners.remove(current.element, current.type, current.handler);\r\n\r\n });\r\n\r\n };\r\n\r\n listeners.get = function (element, eventType, handler) {\r\n\r\n return listeners.search.all(element, eventType, handler);\r\n\r\n };\r\n\r\n return listeners;\r\n\r\n}({});\n\n\n// WEBPACK FOOTER //\n// ./modules/listeners.js","/**\r\n * Codex Editor Destroyer module\r\n *\r\n * @auhor Codex Team\r\n * @version 1.0\r\n */\r\n\r\nmodule.exports = function (destroyer) {\r\n\r\n let editor = codex.editor;\r\n\r\n destroyer.removeNodes = function () {\r\n\r\n editor.nodes.wrapper.remove();\r\n editor.nodes.notifications.remove();\r\n\r\n };\r\n\r\n destroyer.destroyPlugins = function () {\r\n\r\n for (var tool in editor.tools) {\r\n\r\n if (typeof editor.tools[tool].destroy === 'function') {\r\n\r\n editor.tools[tool].destroy();\r\n\r\n }\r\n\r\n }\r\n\r\n };\r\n\r\n destroyer.destroyScripts = function () {\r\n\r\n var scripts = document.getElementsByTagName('SCRIPT');\r\n\r\n for (var i = 0; i < scripts.length; i++) {\r\n\r\n if (scripts[i].id.indexOf(editor.scriptPrefix) + 1) {\r\n\r\n scripts[i].remove();\r\n i--;\r\n\r\n }\r\n\r\n }\r\n\r\n };\r\n\r\n\r\n /**\r\n * Delete editor data from webpage.\r\n * You should send settings argument with boolean flags:\r\n * @param settings.ui- remove redactor event listeners and DOM nodes\r\n * @param settings.scripts - remove redactor scripts from DOM\r\n * @param settings.plugins - remove plugin's objects\r\n * @param settings.core - remove editor core. You can remove core only if UI and scripts flags is true\r\n * }\r\n *\r\n */\r\n destroyer.destroy = function (settings) {\r\n\r\n if (!settings || typeof settings !== 'object') {\r\n\r\n return;\r\n\r\n }\r\n\r\n if (settings.ui) {\r\n\r\n destroyer.removeNodes();\r\n editor.listeners.removeAll();\r\n\r\n }\r\n\r\n if (settings.scripts) {\r\n\r\n destroyer.destroyScripts();\r\n\r\n }\r\n\r\n if (settings.plugins) {\r\n\r\n destroyer.destroyPlugins();\r\n\r\n }\r\n\r\n if (settings.ui && settings.scripts && settings.core) {\r\n\r\n delete codex.editor;\r\n\r\n }\r\n\r\n };\r\n\r\n return destroyer;\r\n\r\n}({});\n\n\n// WEBPACK FOOTER //\n// ./modules/destroyer.js","/**\r\n * Codex Editor Paste module\r\n *\r\n * @author Codex Team\r\n * @version 1.1.1\r\n */\r\n\r\nmodule.exports = function (paste) {\r\n\r\n let editor = codex.editor;\r\n\r\n var patterns = [];\r\n\r\n paste.prepare = function () {\r\n\r\n var tools = editor.tools;\r\n\r\n for (var tool in tools) {\r\n\r\n if (!tools[tool].renderOnPastePatterns || !Array.isArray(tools[tool].renderOnPastePatterns)) {\r\n\r\n continue;\r\n\r\n }\r\n\r\n tools[tool].renderOnPastePatterns.map(function (pattern) {\r\n\r\n\r\n patterns.push(pattern);\r\n\r\n });\r\n\r\n }\r\n\r\n return Promise.resolve();\r\n\r\n };\r\n\r\n /**\r\n * Saves data\r\n * @param event\r\n */\r\n paste.pasted = function (event) {\r\n\r\n var clipBoardData = event.clipboardData || window.clipboardData,\r\n content = clipBoardData.getData('Text');\r\n\r\n var result = analize(content);\r\n\r\n if (result) {\r\n\r\n event.preventDefault();\r\n event.stopImmediatePropagation();\r\n\r\n }\r\n\r\n return result;\r\n\r\n };\r\n\r\n /**\r\n * Analizes pated string and calls necessary method\r\n */\r\n\r\n var analize = function (string) {\r\n\r\n var result = false,\r\n content = editor.content.currentNode,\r\n plugin = content.dataset.tool;\r\n\r\n patterns.map( function (pattern) {\r\n\r\n var execArray = pattern.regex.exec(string),\r\n match = execArray && execArray[0];\r\n\r\n if ( match && match === string.trim()) {\r\n\r\n /** current block is not empty */\r\n if ( content.textContent.trim() && plugin == editor.settings.initialBlockPlugin ) {\r\n\r\n pasteToNewBlock_();\r\n\r\n }\r\n\r\n pattern.callback(string, pattern);\r\n result = true;\r\n\r\n }\r\n\r\n });\r\n\r\n return result;\r\n\r\n };\r\n\r\n var pasteToNewBlock_ = function () {\r\n\r\n /** Create new initial block */\r\n editor.content.insertBlock({\r\n\r\n type : editor.settings.initialBlockPlugin,\r\n block : editor.tools[editor.settings.initialBlockPlugin].render({\r\n text : ''\r\n })\r\n\r\n }, false);\r\n\r\n };\r\n\r\n return paste;\r\n\r\n}({});\n\n\n// WEBPACK FOOTER //\n// ./modules/paste.js"],"sourceRoot":""} \ No newline at end of file diff --git a/package.json b/package.json index 872f57d6..fa3b007c 100644 --- a/package.json +++ b/package.json @@ -1,6 +1,6 @@ { "name": "codex.editor", - "version": "1.6.3", + "version": "1.6.4", "description": "Codex Editor. Native JS, based on API and Open Source", "main": "index.js", "scripts": { From 8fac7517c690d897977b168b59342665405823e7 Mon Sep 17 00:00:00 2001 From: George Berezhnoy Date: Mon, 24 Apr 2017 02:26:08 +0300 Subject: [PATCH 03/12] Move paste event handler to paste.js --- codex-editor.js | 4 +- codex-editor.js.map | 2 +- modules/callbacks.js | 115 +---------------------------------------- modules/core.js | 14 +++++ modules/paste.js | 119 +++++++++++++++++++++++++++++++++++++++++++ modules/ui.js | 2 +- 6 files changed, 138 insertions(+), 118 deletions(-) diff --git a/codex-editor.js b/codex-editor.js index 6cef92d1..8f816f10 100644 --- a/codex-editor.js +++ b/codex-editor.js @@ -1,3 +1,3 @@ -var codex=codex||{};codex.editor=function(e){function t(o){if(n[o])return n[o].exports;var r=n[o]={exports:{},id:o,loaded:!1};return e[o].call(r.exports,r,r.exports,t),r.loaded=!0,r.exports}var n={};return t.m=e,t.c=n,t.p="",t(0)}([function(e,t,n){"use strict";e.exports=function(e){e.version="1.6.4",e.scriptPrefix="cdx-script-";var t=function(){e.core=n(1),e.tools=n(2),e.ui=n(3),e.transport=n(4),e.renderer=n(5),e.saver=n(6),e.content=n(7),e.toolbar=n(8),e.callback=n(12),e.draw=n(13),e.caret=n(14),e.notifications=n(15),e.parser=n(16),e.sanitizer=n(17),e.listeners=n(19),e.destroyer=n(20),e.paste=n(21)};return e.settings={tools:["paragraph","header","picture","list","quote","code","twitter","instagram","smile"],holderId:"codex-editor",initialBlockPlugin:"paragraph"},e.nodes={holder:null,wrapper:null,toolbar:null,inlineToolbar:{wrapper:null,buttons:null,actions:null},toolbox:null,notifications:null,plusButton:null,showSettingsButton:null,showTrashButton:null,blockSettings:null,pluginSettings:null,defaultSettings:null,toolbarButtons:{},redactor:null},e.state={jsonOutput:[],blocks:[],inputs:[]},e.tools={},e.start=function(n){t(),e.core.prepare(n).then(e.ui.prepare).then(e.tools.prepare).then(e.sanitizer.prepare).then(e.paste.prepare).then(e.transport.prepare).then(e.renderer.makeBlocksFromData).then(e.ui.saveInputs).catch(function(t){e.core.log("Initialization failed with error: %o","warn",t)})},e}({})},function(e,t){"use strict";var n="function"==typeof Symbol&&"symbol"==typeof Symbol.iterator?function(e){return typeof e}:function(e){return e&&"function"==typeof Symbol&&e.constructor===Symbol&&e!==Symbol.prototype?"symbol":typeof e};e.exports=function(e){var t=codex.editor;e.prepare=function(e){return new Promise(function(o,r){e&&(t.settings.tools=e.tools||t.settings.tools),e.data&&(t.state.blocks=e.data),e.initialBlockPlugin&&(t.settings.initialBlockPlugin=e.initialBlockPlugin),e.sanitizer&&(t.settings.sanitizer=e.sanitizer),t.hideToolbar=e.hideToolbar,t.nodes.holder=document.getElementById(e.holderId||t.settings.holderId),void 0===n(t.nodes.holder)||null===t.nodes.holder?r(Error("Holder wasn't found by ID: #"+e.holderId)):o()})},e.log=function(e,t,n){t=t||"log",n?e="[codex-editor]: "+e:(n=e||"undefined",e="[codex-editor]: %o");try{"console"in window&&window.console[t]&&(n?window.console[t](e,n):window.console[t](e))}catch(e){}},e.insertAfter=function(e,t){e.parentNode.insertBefore(t,e.nextSibling)},e.nodeTypes={TAG:1,TEXT:3,COMMENT:8},e.keys={BACKSPACE:8,TAB:9,ENTER:13,SHIFT:16,CTRL:17,ALT:18,ESC:27,SPACE:32,LEFT:37,UP:38,DOWN:40,RIGHT:39,DELETE:46,META:91},e.isDomNode=function(e){return e&&"object"===("undefined"==typeof e?"undefined":n(e))&&e.nodeType&&e.nodeType==this.nodeTypes.TAG},e.isEmpty=function(e){return 0===Object.keys(e).length},e.ajax=function(e){if(e&&e.url){var t,n,r,i=window.XMLHttpRequest?new XMLHttpRequest:new ActiveXObject("Microsoft.XMLHTTP");if(e.async=!0,e.type=e.type||"GET",e.data=e.data||"",e["content-type"]=e["content-type"]||"application/json; charset=utf-8","GET"==e.type&&e.data)e.url=/\?/.test(e.url)?e.url+"&"+e.data:e.url+"?"+e.data;else{t="";for(r in e.data)t+=r+"="+encodeURIComponent(e.data[r])+"&"}e.withCredentials&&(i.withCredentials=!0);var a=void 0,s=void 0;if("function"!=typeof e.beforeSend||(s=e.beforeSend.call(),s!==!1))return i.open(e.type,e.url,e.async),n=o(e.data),n||("POST"!==e.type?i.setRequestHeader("Content-type",e["content-type"]):i.setRequestHeader("Content-type","application/x-www-form-urlencoded")),i.setRequestHeader("X-Requested-With","XMLHttpRequest"),a=s||i,"function"==typeof e.progress&&(i.upload.onprogress=e.progress.bind(a)),i.onreadystatechange=function(){4===i.readyState&&(200===i.status?"function"==typeof e.success&&e.success.call(a,i.responseText):"function"==typeof e.error&&e.error.call(a,i.responseText,i.status))},n?i.send(e.data):i.send(t),i}},e.importScript=function(e,n){return new Promise(function(o,r){var i=void 0;n?document.getElementById(t.scriptPrefix+n)&&o(e):r("Instance name is missed"),i=document.createElement("SCRIPT"),i.async=!0,i.defer=!0,i.id=t.scriptPrefix+n,i.onload=function(){o(e)},i.onerror=function(){r(e)},i.src=e,document.head.appendChild(i)})};var o=function(e){return e instanceof FormData};return e}({})},function(e,t){"use strict";e.exports=function(){function e(){return new Promise(function(e,o){Promise.resolve().then(function(){var t=[],o=n.tools;for(var r in o){var i=o[r];i.prepare&&"function"!=typeof i.prepare||!i.prepare||t.push(i)}return t.length||e(),t}).then(t).then(function(){n.core.log("Plugins loaded","info"),e()}).catch(function(e){o(e)})})}function t(e){return new Promise(function(t){e.reduce(function(r,i,a){return r.then(function(){return new Promise(function(r){o(i).then(r).then(function(){i.available=!0}).catch(function(e){n.core.log("Plugin «"+i.type+"» was not loaded. Preparation failed because %o","warn",e),i.available=!1,i.loadingMessage=e,r()}).then(function(){a==e.length-1&&t()})})})},Promise.resolve())})}var n=codex.editor,o=function(e){return e.prepare(e.config||{})};return{prepare:e}}()},function(e,t){"use strict";e.exports=function(e){var t=codex.editor;e.className={BLOCK_CLASSNAME:"ce-block",BLOCK_CONTENT:"ce-block__content",BLOCK_STRETCHED:"ce-block--stretched",BLOCK_HIGHLIGHTED:"ce-block--focused",SETTINGS_ITEM:"ce-settings__item"},e.prepare=function(){return new Promise(function(e){var n=t.draw.wrapper(),r=t.draw.redactor(),i=o();n.appendChild(i),n.appendChild(r),t.nodes.wrapper=n,t.nodes.redactor=r,t.nodes.holder.appendChild(n),e()}).then(s).then(n).then(c).then(a).then(l).catch(function(){t.core.log("Can't draw editor interface")})};var n=function(){var e=t.draw.inlineToolbar();t.nodes.inlineToolbar.wrapper=e,t.nodes.inlineToolbar.buttons=t.draw.inlineToolbarButtons(),t.nodes.inlineToolbar.actions=t.draw.inlineToolbarActions(),t.nodes.inlineToolbar.wrapper.appendChild(t.nodes.inlineToolbar.buttons),t.nodes.inlineToolbar.wrapper.appendChild(t.nodes.inlineToolbar.actions),t.nodes.wrapper.appendChild(t.nodes.inlineToolbar.wrapper)},o=function(){var e=t.draw.toolbar(),n=i(),o=r();return e.appendChild(n),e.appendChild(o),t.nodes.toolbar=e,e},r=function(){var e=t.draw.toolbarContent(),n=t.draw.toolbox(),o=t.draw.plusButton();return e.appendChild(o),e.appendChild(n),t.nodes.toolbox=n,t.nodes.plusButton=o,e},i=function(){var e=t.draw.blockSettings(),n=t.draw.blockButtons(),o=t.draw.defaultSettings(),r=t.draw.settingsButton(),i=t.toolbar.settings.makeRemoveBlockButton(),a=t.draw.pluginsSettings();return e.appendChild(a),e.appendChild(o),n.appendChild(r),n.appendChild(i),n.appendChild(e),t.nodes.blockSettings=e,t.nodes.pluginSettings=a,t.nodes.defaultSettings=o,t.nodes.showSettingsButton=r,t.nodes.showTrashButton=i,n},a=function(){t.nodes.notifications=t.notifications.createHolder()},s=function(){var e,n,o;for(n in t.settings.tools)e=t.settings.tools[n],t.tools[n]=e,e.iconClassname||!e.displayInToolbox?"function"==typeof e.render?e.displayInToolbox&&(o=t.draw.toolbarButton(n,e.iconClassname),t.nodes.toolbox.appendChild(o),t.nodes.toolbarButtons[n]=o):t.core.log("render method missed. Tool %o skipped","warn",n):t.core.log("Toolbar icon classname missed. Tool %o skipped","warn",n)},c=function(){var e,n,o={bold:{icon:"ce-icon-bold",command:"bold"},italic:{icon:"ce-icon-italic",command:"italic"},underline:{icon:"ce-icon-underline",command:"underline"},link:{icon:"ce-icon-link",command:"createLink"}};for(var r in o)n=o[r],e=t.draw.toolbarButtonInline(r,n.icon),t.nodes.inlineToolbar.buttons.appendChild(e),t.ui.setInlineToolbarButtonBehaviour(e,n.command)},l=function(){t.core.log("ui.bindEvents fired","info"),t.listeners.add(document,"keydown",t.callback.globalKeydown,!1),t.listeners.add(t.nodes.redactor,"keydown",t.callback.redactorKeyDown,!1),t.listeners.add(document,"keyup",t.callback.globalKeyup,!1),t.listeners.add(t.nodes.redactor,"click",t.callback.redactorClicked,!1),t.listeners.add(t.nodes.plusButton,"click",t.callback.plusButtonClicked,!1),t.listeners.add(t.nodes.showSettingsButton,"click",t.callback.showSettingsButtonClicked,!1);for(var e in t.nodes.toolbarButtons)t.listeners.add(t.nodes.toolbarButtons[e],"click",t.callback.toolbarButtonClicked,!1)};return e.addBlockHandlers=function(e){e&&(t.listeners.add(e,"keydown",t.callback.blockKeydown,!1),t.listeners.add(e,"paste",t.callback.blockPasteCallback,!1),t.listeners.add(e,"mouseup",t.toolbar.inline.show,!1))},e.saveInputs=function(){var e=t.nodes.redactor;t.state.inputs=[];var n=e.querySelectorAll("[contenteditable], input, textarea");Array.prototype.map.call(n,function(e){e.type&&"text"!=e.type&&"textarea"!=e.type||t.state.inputs.push(e)})},e.addInitialBlock=function(){var e,n=t.settings.initialBlockPlugin;return t.tools[n]?(e=t.tools[n].render(),e.setAttribute("data-placeholder","Расскажите свою историю..."),t.content.insertBlock({type:n,block:e}),void t.content.workingNodeChanged(e)):void t.core.log("Plugin %o was not implemented and can't be used as initial block","warn",n)},e.setInlineToolbarButtonBehaviour=function(e,n){t.listeners.add(e,"mousedown",function(e){t.toolbar.inline.toolClicked(e,n)},!1)},e}({})},function(e,t){"use strict";e.exports=function(e){var t=codex.editor,n=null;return e.input=null,e.arguments=null,e.prepare=function(){var e=t.draw.node("INPUT","",{type:"file"});t.listeners.add(e,"change",t.transport.fileSelected),t.transport.input=e},e.clearInput=function(){e.input=null,e.prepare()},e.fileSelected=function(){var o,r=this,i=r.files,a=new FormData;if(t.transport.arguments.multiple===!0)for(o=0;o"+t.split("\n\n").join("

    ")+"

    ";var n,o,r,i,a=document.createElement("DIV"),s=document.createElement("DIV"),c=["DIV","P"];for(a.innerHTML=e,o=document.createElement("P"),n=0;n'}),o=t.draw.node("DIV","ce-toolbar__remove-confirmation",{}),r=t.draw.node("DIV","ce-toolbar__remove-confirm",{textContent:"Удалить блок"}),i=t.draw.node("DIV","ce-toolbar__remove-cancel",{textContent:"Отмена"});return t.listeners.add(n,"click",t.toolbar.settings.removeButtonClicked,!1),t.listeners.add(r,"click",t.toolbar.settings.confirmRemovingRequest,!1),t.listeners.add(i,"click",t.toolbar.settings.cancelRemovingRequest,!1),o.appendChild(r),o.appendChild(i),e.appendChild(n),e.appendChild(o),t.toolbar.settings.setting=n,t.toolbar.settings.actions=o,e},e.removeButtonClicked=function(){var e=t.toolbar.settings.actions;e.classList.contains("opened")?t.toolbar.settings.hideRemoveActions():t.toolbar.settings.showRemoveActions(),t.toolbar.toolbox.close(),t.toolbar.settings.close()},e.cancelRemovingRequest=function(){t.toolbar.settings.actions.classList.remove("opened")},e.confirmRemovingRequest=function(){var e,n=t.content.currentNode;n.remove(),e=t.nodes.redactor.childNodes.length,0===e&&(t.content.currentNode=null,t.ui.addInitialBlock()),t.ui.saveInputs(),t.toolbar.close()},e.showRemoveActions=function(){t.toolbar.settings.actions.classList.add("opened")},e.hideRemoveActions=function(){t.toolbar.settings.actions.classList.remove("opened")},e}({})},function(e,t){"use strict";e.exports=function(e){var t=codex.editor;e.buttonsOpened=null,e.actionsOpened=null,e.wrappersOffset=null,e.storedSelection=null,e.show=function(){var n,o=t.content.currentNode,r=o.dataset.tool;if(n=t.tools[r],n.showInlineToolbar){var i=e.getSelectionText(),a=t.nodes.inlineToolbar.wrapper;i.length>0&&(t.toolbar.inline.move(),a.classList.add("opened"),t.toolbar.inline.showButtons())}},e.close=function(){var e=t.nodes.inlineToolbar.wrapper;e.classList.remove("opened")},e.move=function(){this.wrappersOffset||(this.wrappersOffset=this.getWrappersOffset());var e,n,o=this.getSelectionCoords(),r=0,i=t.nodes.inlineToolbar.wrapper;0===i.offsetHeight&&(r=40),e=o.x-this.wrappersOffset.left,n=o.y+window.scrollY-this.wrappersOffset.top-r-i.offsetHeight,i.style.transform="translate3D("+Math.floor(e)+"px, "+Math.floor(n)+"px, 0)",t.toolbar.inline.closeButtons(),t.toolbar.inline.closeAction()},e.toolClicked=function(e,n){switch(n){case"createLink":t.toolbar.inline.createLinkAction(e,n);break;default:t.toolbar.inline.defaultToolAction(n)}t.nodes.inlineToolbar.buttons.childNodes.forEach(t.toolbar.inline.hightlight)},e.getWrappersOffset=function(){var e=t.nodes.wrapper,n=this.getOffset(e);return this.wrappersOffset=n,n},e.getOffset=function(e){for(var t=0,n=0;e&&!isNaN(e.offsetLeft)&&!isNaN(e.offsetTop);)t+=e.offsetLeft+e.clientLeft,n+=e.offsetTop+e.clientTop,e=e.offsetParent;return{top:n,left:t}},e.getSelectionCoords=function(){var e,t=document.selection,n=0,o=0;if(t)"Control"!=t.type&&(e=t.createRange(),e.collapse(!0),n=e.boundingLeft,o=e.boundingTop);else if(window.getSelection&&(t=window.getSelection(),t.rangeCount&&(e=t.getRangeAt(0).cloneRange(),e.getClientRects))){e.collapse(!0);var r=e.getClientRects()[0];if(!r)return;n=r.left,o=r.top}return{x:n,y:o}},e.getSelectionText=function(){var e="";return window.getSelection&&(e=window.getSelection().toString()),e},e.showButtons=function(){var e=t.nodes.inlineToolbar.buttons;e.classList.add("opened"),t.toolbar.inline.buttonsOpened=!0,t.nodes.inlineToolbar.buttons.childNodes.forEach(t.toolbar.inline.hightlight)},e.closeButtons=function(){var e=t.nodes.inlineToolbar.buttons;e.classList.remove("opened"),t.toolbar.inline.buttonsOpened=!1},e.showActions=function(){var e=t.nodes.inlineToolbar.actions;e.classList.add("opened"),t.toolbar.inline.actionsOpened=!0},e.closeAction=function(){var e=t.nodes.inlineToolbar.actions;e.innerHTML="",e.classList.remove("opened"),t.toolbar.inline.actionsOpened=!1};var n=function(e){if(e.keyCode==t.core.keys.ENTER){var n=t.content.currentNode,o=t.toolbar.inline.storedSelection;t.toolbar.inline.restoreSelection(n,o),t.toolbar.inline.setAnchor(this.value),e.preventDefault(),e.stopImmediatePropagation(),t.toolbar.inline.clearRange()}};return e.createLinkAction=function(e){var o=this.isLinkActive(),r=t.content.currentNode,i=t.toolbar.inline.saveSelection(r);if(t.toolbar.inline.storedSelection=i,o)t.toolbar.inline.restoreSelection(r,i),t.toolbar.inline.defaultToolAction("unlink");else{var a=t.draw.inputForLink();t.nodes.inlineToolbar.actions.appendChild(a),t.toolbar.inline.closeButtons(),t.toolbar.inline.showActions(),a.focus(),e.preventDefault(),t.listeners.add(a,"keydown",n,!1)}},e.isLinkActive=function(){var e=!1;return t.nodes.inlineToolbar.buttons.childNodes.forEach(function(t){var n=t.dataset.type;"link"==n&&t.classList.contains("hightlighted")&&(e=!0)}),e},e.defaultToolAction=function(e){document.execCommand(e,!1,null)},e.setAnchor=function(e){document.execCommand("createLink",!1,e),t.toolbar.inline.closeAction()},e.saveSelection=function(e){var t,n=window.getSelection().getRangeAt(0),o=n.cloneRange();return o.selectNodeContents(e),o.setEnd(n.startContainer,n.startOffset),t=o.toString().length,{start:t,end:t+n.toString().length}},e.restoreSelection=function(e,t){var n=document.createRange(),o=0;n.setStart(e,0),n.collapse(!0);for(var r,i,a=[e],s=!1,c=!1;!c&&(r=a.pop());)if(3==r.nodeType)i=o+r.length,!s&&t.start>=o&&t.start<=i&&(n.setStart(r,t.start-o),s=!0),s&&t.end>=o&&t.end<=i&&(n.setEnd(r,t.end-o),c=!0),o=i;else for(var l=r.childNodes.length;l--;)a.push(r.childNodes[l]);var d=window.getSelection();d.removeAllRanges(),d.addRange(n)},e.clearRange=function(){var e=window.getSelection();e.removeAllRanges()},e.hightlight=function(e){var n=e.dataset.type;document.queryCommandState(n)?t.toolbar.inline.setButtonHighlighted(e):t.toolbar.inline.removeButtonsHighLight(e);var o=window.getSelection(),r=o.anchorNode.parentNode;"A"==r.tagName&&"link"==n&&t.toolbar.inline.setButtonHighlighted(e)},e.setButtonHighlighted=function(e){if(e.classList.add("hightlighted"),"link"==e.dataset.type){var t=e.childNodes[0];t.classList.remove("ce-icon-link"),t.classList.add("ce-icon-unlink")}},e.removeButtonsHighLight=function(e){if(e.classList.remove("hightlighted"),"link"==e.dataset.type){var t=e.childNodes[0];t.classList.remove("ce-icon-unlink"),t.classList.add("ce-icon-link")}},e}({})},function(e,t){"use strict";e.exports=function(e){var t=codex.editor;return e.opened=!1,e.open=function(){t.toolbar.settings.opened&&t.toolbar.settings.close(),t.nodes.toolbox.classList.add("opened"),t.nodes.plusButton.classList.add("clicked"),t.toolbar.toolbox.opened=!0},e.close=function(){t.nodes.toolbox.classList.remove("opened"),t.nodes.plusButton.classList.remove("clicked"),t.toolbar.toolbox.opened=!1,t.toolbar.current=null},e.leaf=function(){var e=t.toolbar.current,n=Object.keys(t.tools),o=t.nodes.toolbarButtons,r=0,i=void 0,a=void 0,s=void 0;if(e)for(r=(n.indexOf(e)+1)%n.length,a=n[r];!t.tools[a].displayInToolbox;)r=(r+1)%n.length,a=n[r];else for(s in t.tools){if(t.tools[s].displayInToolbox)break;r++}i=n[r];for(var c in o)o[c].classList.remove("selected");o[i].classList.add("selected"),t.toolbar.current=i},e.toolClicked=function(e){var n,o,r,i=["image","link","list","instagram","twitter","embed"],a=t.tools[t.toolbar.current],s=t.content.currentNode,c=t.caret.inputIndex;n=a.render(),r={block:n,type:a.type,stretched:!1},s&&i.indexOf(s.dataset.tool)===-1&&""===s.textContent.trim()?t.content.switchBlock(s,n,a.type):(t.content.insertBlock(r),c++),o=a.appendCallback,o&&"function"==typeof o&&o.call(e),window.setTimeout(function(){t.caret.setToBlock(c)},10),t.content.workingNodeChanged(),t.toolbar.move()},e}({})},function(e,t){"use strict";e.exports=function(e){var t=codex.editor;e.globalKeydown=function(e){switch(e.keyCode){case t.core.keys.ENTER:o(e)}},e.redactorKeyDown=function(e){switch(e.keyCode){case t.core.keys.TAB:n(e);break;case t.core.keys.ENTER:i(e);break;case t.core.keys.ESC:a(e);break;default:c(e)}},e.globalKeyup=function(e){switch(e.keyCode){case t.core.keys.UP:case t.core.keys.LEFT:case t.core.keys.RIGHT:case t.core.keys.DOWN:s(e)}};var n=function(e){e.preventDefault();var n=t.content.currentNode.querySelectorAll("textarea, input"),o=!0,r=!t.content.currentNode.textContent.trim();Array.prototype.map.call(n,function(e){"textarea"!=e.type&&"text"!=e.type||(o=o&&!e.value.trim())});var i=r&&o;i&&(t.toolbar.opened||t.toolbar.open(),t.toolbar.opened&&!t.toolbar.toolbox.opened?t.toolbar.toolbox.open():t.toolbar.toolbox.leaf())},o=function(){t.content.editorAreaHightlighted&&(t.caret.inputIndex=-1,r())},r=function(){var e=t.settings.initialBlockPlugin;t.content.insertBlock({type:e,block:t.tools[e].render()},!0),t.toolbar.move(),t.toolbar.open()},i=function(e){"true"==e.target.contentEditable&&t.caret.saveCurrentInputIndex();var n=t.caret.getCurrentInputIndex()||0,o=t.content.currentNode,r=o.dataset.tool,i=t.toolbar.opened&&t.toolbar.current&&e.target==t.state.inputs[n],a=t.tools[r].enableLineBreaks,s=t.settings.initialBlockPlugin;if(i)return e.preventDefault(),t.toolbar.toolbox.toolClicked(e),t.toolbar.close(),e.stopPropagation(),void e.stopImmediatePropagation();if(e.shiftKey||a)return e.stopPropagation(),void e.stopImmediatePropagation();var c=window.getSelection(),l=c.anchorNode,d=t.caret.position.atTheEnd(),u=!1;if(e.shiftKey&&!a)return t.callback.enterPressedOnBlock(t.content.currentBlock,e),void e.preventDefault();if(u=l&&"true"!=l.parentNode.contentEditable,l.nodeType!=t.core.nodeTypes.TEXT||u||d){var p=t.content.isLastNode(l);p&&d&&(e.preventDefault(),e.stopPropagation(),e.stopImmediatePropagation(),t.core.log("ENTER clicked in last textNode. Create new BLOCK"),t.content.insertBlock({type:s,block:t.tools[s].render()},!0),t.toolbar.move(),t.toolbar.open(),t.toolbar.showPlusButton())}else e.preventDefault(),t.core.log("Splitting Text node..."),t.content.splitBlock(n),t.state.inputs[n+1].textContent.trim()||t.toolbar.showPlusButton();t.ui.saveInputs()},a=function(e){t.toolbar.close(),t.toolbar.toolbox.close(),e.preventDefault()},s=function(e){t.content.workingNodeChanged(),t.toolbar.close(),t.toolbar.move()},c=function(){t.toolbar.close(),t.toolbar.inline.actionsOpened||(t.toolbar.inline.close(),t.content.clearMark())};e.redactorClicked=function(e){l(),t.content.workingNodeChanged(e.target),t.ui.saveInputs();var n,o=t.toolbar.inline.getSelectionText();if(0===o.length&&t.toolbar.inline.close(),"true"==e.target.contentEditable&&t.caret.saveCurrentInputIndex(),null===t.content.currentNode){var r=t.state.inputs.length>0?t.state.inputs.length-1:0;if(t.state.inputs.length&&(n=t.content.getFirstLevelBlock(t.state.inputs[r])),t.state.inputs.length&&""===t.state.inputs[r].textContent&&n.dataset.tool==t.settings.initialBlockPlugin)t.caret.setToBlock(r);else{var i=t.settings.initialBlockPlugin;t.content.insertBlock({type:i,block:t.tools[i].render()}),1===t.state.inputs.length?t.caret.setToBlock(r):t.caret.setToNextBlock(r)}}else t.toolbar.settings.close(),t.toolbar.toolbox.close();t.toolbar.move(),t.toolbar.open();var a=!t.content.currentNode.textContent.trim(),s=t.content.currentNode.dataset.tool,c=s==t.settings.initialBlockPlugin;t.toolbar.hidePlusButton(),a||t.content.markBlock(),c&&a&&t.toolbar.showPlusButton()};var l=function(){var e=window.getSelection(),n=e.anchorNode,o=!1;if(0===e.rangeCount)t.content.editorAreaHightlighted=!0;else{for(t.core.isDomNode(n)||(n=n.parentNode),"true"==n.contentEditable&&(o=!0);"true"!=n.contentEditable&&(n=n.parentNode,"true"==n.contentEditable&&(o=!0),n!=document.body););t.content.editorAreaHightlighted=!o}};e.toolbarButtonClicked=function(e){var n=this;t.toolbar.current=n.dataset.type,t.toolbar.toolbox.toolClicked(e),t.toolbar.close()},e.plusButtonClicked=function(){t.nodes.toolbox.classList.contains("opened")?t.toolbar.toolbox.close():t.toolbar.toolbox.open()},e.blockKeydown=function(e){var n=e.target;switch(e.keyCode){case t.core.keys.DOWN:case t.core.keys.RIGHT:d(e);break;case t.core.keys.BACKSPACE:p(n,e);break;case t.core.keys.UP:case t.core.keys.LEFT:u(e)}};var d=function(e){var n,o=window.getSelection(),r=t.state.inputs,i=o.anchorNode;if(!i)return!1;for(;"true"!=i.contentEditable;)n=i.parentNode,i=n;for(var a=0;i!=r[a];)a++;if(!i.textContent)return void t.caret.setToNextBlock(a);var s,c,l=!1,d=!1;return s=i.childNodes[i.childNodes.length-1],c=t.core.isDomNode(s)?t.content.getDeepestTextNodeFromPosition(s,s.childNodes.length):s,l=o.anchorNode==c,d=c.length==o.anchorOffset,l&&d?void t.caret.setToNextBlock(a):(t.core.log("arrow [down|right] : caret does not reached the end"),!1)},u=function(e){var n,o=window.getSelection(),r=t.state.inputs,i=o.anchorNode;if(!i)return!1;if(0!==o.anchorOffset)return!1;for(;"true"!=i.contentEditable;)n=i.parentNode,i=n;for(var a=0;i!=r[a];)a++;var s,c,l=!1,d=!1;return i.textContent?(s=i.childNodes[0],c=t.core.isDomNode(s)?t.content.getDeepestTextNodeFromPosition(s,0):s,l=o.anchorNode==c,d=0===o.anchorOffset,void(l&&d&&t.caret.setToPreviousBlock(a))):void t.caret.setToPreviousBlock(a)},p=function(e,n){var o,r,i,a=t.caret.getCurrentInputIndex();if(f(n.target)){if(""!=n.target.value.trim())return;e.remove()}if(e.textContent.trim()){if(o=t.content.getRange(),r=o.endOffset-o.startOffset,!t.caret.position.atStart()||r||!t.state.inputs[a-1])return;t.content.mergeBlocks(a)}r||e.remove(),i=t.nodes.redactor.childNodes.length,0===i?(t.content.currentNode=null,t.ui.addInitialBlock(),t.ui.saveInputs(),window.setTimeout(function(){t.caret.setToPreviousBlock(1)},10)):0!==t.caret.inputIndex?t.caret.setToPreviousBlock(t.caret.inputIndex):t.caret.setToNextBlock(t.caret.inputIndex), -t.toolbar.move(),t.toolbar.opened||t.toolbar.open(),t.ui.saveInputs(),n.preventDefault()};e.blockPasteCallback=function(e){if(!f(e.target)){e.preventDefault();var n=t.content.getEditableParent(e.target);if(n){var o,r,i=e.clipboardData.getData("text/html"),a=e.clipboardData.getData("text/plain"),s=t.draw.node("DIV","",{});o=t.sanitizer.clean(i),r=t.content.wrapTextWithParagraphs(o,a),s.innerHTML=r;var c=t.settings.initialBlockPlugin,l=t.content.currentNode.firstChild.firstChild;if(1==s.childNodes.length)return void t.caret.insertNode(document.createTextNode(s.firstChild.innerHTML));s.childNodes.forEach(function(e,n){return 0==n&&""===l.innerHTML.trim()?void t.content.switchBlock(t.content.currentNode,t.tools[c].render({text:e.innerHTML}),c):(t.content.insertBlock({type:c,block:t.tools[c].render({text:e.innerHTML})}),void t.caret.inputIndex++)}),t.caret.setToPreviousBlock(t.caret.getCurrentInputIndex()+1)}}},e.showSettingsButtonClicked=function(e){var n=t.content.currentNode.dataset.tool;t.toolbar.settings.toggle(n),t.toolbar.toolbox.close(),t.toolbar.settings.hideRemoveActions()};var f=function(e){var t=["INPUT","TEXTAREA"];return t.indexOf(e.tagName)!=-1};return e}({})},function(e,t){"use strict";e.exports=function(e){return e.wrapper=function(){var e=document.createElement("div");return e.className+="codex-editor",e},e.redactor=function(){var e=document.createElement("div");return e.className+="ce-redactor",e},e.ceBlock=function(){var e=document.createElement("DIV");return e.className+="ce_block",e},e.toolbar=function(){var e=document.createElement("div");return e.className+="ce-toolbar",e},e.toolbarContent=function(){var e=document.createElement("DIV");return e.classList.add("ce-toolbar__content"),e},e.inlineToolbar=function(){var e=document.createElement("DIV");return e.className+="ce-toolbar-inline",e},e.inlineToolbarButtons=function(){var e=document.createElement("DIV");return e.className+="ce-toolbar-inline__buttons",e},e.inlineToolbarActions=function(){var e=document.createElement("DIV");return e.className+="ce-toolbar-inline__actions",e},e.inputForLink=function(){var e=document.createElement("INPUT");return e.type="input",e.className+="inputForLink",e.placeholder="Вставьте ссылку ...",e.setAttribute("form","defaultForm"),e.setAttribute("autofocus","autofocus"),e},e.blockButtons=function(){var e=document.createElement("div");return e.className+="ce-toolbar__actions",e},e.blockSettings=function(){var e=document.createElement("div");return e.className+="ce-settings",e},e.defaultSettings=function(){var e=document.createElement("div");return e.classList.add("ce-settings_default"),e},e.pluginsSettings=function(){var e=document.createElement("div");return e.classList.add("ce-settings_plugin"),e},e.plusButton=function(){var e=document.createElement("span");return e.className="ce-toolbar__plus",e},e.settingsButton=function(){var e=document.createElement("span");return e.className="ce-toolbar__settings-btn",e.innerHTML='',e},e.toolbox=function(){var e=document.createElement("div");return e.className="ce-toolbar__tools",e},e.toolbarButton=function(e,t){var n=document.createElement("li"),o=document.createElement("i"),r=document.createElement("span");return n.dataset.type=e,n.setAttribute("title",e),o.classList.add(t),r.classList.add("ce_toolbar_tools--title"),n.appendChild(o),n.appendChild(r),n},e.toolbarButtonInline=function(e,t){var n=document.createElement("BUTTON"),o=document.createElement("I");return n.type="button",n.dataset.type=e,o.classList.add(t),n.appendChild(o),n},e.block=function(e,t){var n=document.createElement(e);return n.innerHTML=t||"",n},e.node=function(e,t,n){var o=document.createElement(e);if(t&&(o.className=t),n)for(var r in n)o[r]=n[r];return o},e.unavailableBlock=function(){var e=document.createElement("DIV");return e.classList.add("cdx-unavailable-block"),e},e}({})},function(e,t){"use strict";e.exports=function(e){var t=codex.editor;return e.inputIndex=null,e.offset=null,e.focusedNodeIndex=null,e.set=function(n,o,r){r=r||e.offset||0,o=o||e.focusedNodeIndex||0;var i,a=n.childNodes;if(i=0===a.length?n:a[o],"true"!=n.contentEditable)return void n.focus();t.core.isDomNode(i)&&(i=t.content.getDeepestTextNodeFromPosition(i,i.childNodes.length));var s=document.createRange(),c=window.getSelection();window.setTimeout(function(){s.setStart(i,r),s.setEnd(i,r),c.removeAllRanges(),c.addRange(s),t.caret.saveCurrentInputIndex()},20)},e.saveCurrentInputIndex=function(){var n,o=window.getSelection(),r=t.state.inputs,i=o.anchorNode;if(i){for(;"true"!=i.contentEditable;)n=i.parentNode,i=n;for(var a=0;i!=r[a];)a++;e.inputIndex=a}},e.getCurrentInputIndex=function(){return e.inputIndex},e.setToNextBlock=function(e){var n=t.state.inputs,o=n[e+1];if(!o)return void t.core.log("We are reached the end");if(!o.childNodes.length){var r=document.createTextNode("");o.appendChild(r)}t.caret.inputIndex=e+1,t.caret.set(o,0,0),t.content.workingNodeChanged(o)},e.setToBlock=function(e){var n=t.state.inputs,o=n[e];if(o){if(!o.childNodes.length){var r=document.createTextNode("");o.appendChild(r)}t.caret.inputIndex=e,t.caret.set(o,0,0),t.content.workingNodeChanged(o)}},e.setToPreviousBlock=function(e){e=e||0;var n,o,r,i=t.state.inputs,a=i[e-1];return a?(n=t.content.getDeepestTextNodeFromPosition(a,a.childNodes.length),o=n.length,a.childNodes.length||(r=document.createTextNode(""),a.appendChild(r)),t.caret.inputIndex=e-1,t.caret.set(a,a.childNodes.length-1,o),void t.content.workingNodeChanged(i[e-1])):void t.core.log("We are reached first node")},e.position={atStart:function(){var e=window.getSelection(),n=e.anchorOffset,o=e.anchorNode,r=t.content.getFirstLevelBlock(o),i=r.childNodes[0];t.core.isDomNode(o)||(o=o.parentNode);var a=o===i.childNodes[0],s=0===n;return a&&s},atTheEnd:function(){var e=window.getSelection(),t=e.anchorOffset,n=e.anchorNode;return!n||!n.length||t===n.length}},e.insertNode=function(e){var t,n;t=window.getSelection(),n=t.getRangeAt(0),n.deleteContents(),n.insertNode(e),n.setStartAfter(e),n.collapse(!0),t.removeAllRanges(),t.addRange(n)},e}({})},function(e,t){"use strict";e.exports=function(e){var t=codex.editor,n=[],o=function(e){n.push(e);for(var t=0;t5;)"confirm"!=n[t].type&&"prompt"!=n[t].type?(n[t].close(),n.splice(t,1)):t++};return e.createHolder=function(){var e=t.draw.node("DIV","cdx-notifications-block");return t.nodes.notifications=document.body.appendChild(e),e},e.errorThrown=function(e,n){t.notifications.notification({message:"This action is not available currently",type:n.type})},e.notification=function(e){function n(e){if(!e||!e.message)return void t.core.log("Can't create notification. Message is missed");e.type=e.type||"alert",e.time=1e3*e.time||1e4;var n=t.draw.node("DIV","cdx-notification"),o=t.draw.node("DIV","cdx-notification__message"),r=t.draw.node("INPUT","cdx-notification__input"),f=t.draw.node("SPAN","cdx-notification__ok-btn"),g=t.draw.node("SPAN","cdx-notification__cancel-btn");o.textContent=e.message,f.textContent=e.okMsg||"ОК",g.textContent=e.cancelMsg||"Отмена",t.listeners.add(f,"click",u),t.listeners.add(g,"click",p),n.appendChild(o),"prompt"==e.type&&n.appendChild(r),n.appendChild(f),"prompt"!=e.type&&"confirm"!=e.type||n.appendChild(g),n.classList.add("cdx-notification-"+e.type),n.dataset.type=e.type,a=n,c=e.type,l=e.confirm,s=e.cancel,d=r,"prompt"!=e.type&&"confirm"!=e.type&&window.setTimeout(i,e.time)}function r(){t.nodes.notifications.appendChild(a),d.focus(),t.nodes.notifications.classList.add("cdx-notification__notification-appending"),window.setTimeout(function(){t.nodes.notifications.classList.remove("cdx-notification__notification-appending")},100),o({type:c,close:i})}function i(){a.remove()}var a=null,s=null,c=null,l=null,d=null,u=function(){if(i(),"function"==typeof l)return"prompt"==c?void l(d.value):void l()},p=function(){i(),"function"==typeof s&&s()};return e&&(n(e),r()),{create:n,send:r,close:i}},e.clear=function(){t.nodes.notifications.innerHTML="",n=[]},e}({})},function(e,t){"use strict";e.exports=function(e){var t=codex.editor;return e.insertPastedContent=function(e,n){t.content.insertBlock({type:e.type,block:e.render({text:n.innerHTML})})},e.isFirstLevelBlock=function(e){return e.nodeType==t.core.nodeTypes.TAG&&e.classList.contains(t.ui.className.BLOCK_CLASSNAME)},e}({})},function(e,t,n){"use strict";e.exports=function(e){var t=n(18),o=codex.editor;e.prepare=function(){o.settings.sanitizer&&!o.core.isEmpty(o.settings.sanitizer)&&(r.CUSTOM=o.settings.sanitizer)};var r={CUSTOM:null,BASIC:{tags:{p:{},a:{href:!0,target:"_blank",rel:"nofollow"},i:{},b:{},strong:{},em:{},span:{}}}};e.Config=r;var i=function(e){var n=e||r.CUSTOM||r.BASIC;return new t(n)};return e.clean=function(e,t){var n=i(t);return n.clean(e)},e}({})},function(e,t,n){var o,r;!function(i,a){o=a,r="function"==typeof o?o.call(t,n,t,e):o,!(void 0!==r&&(e.exports=r))}(this,function(){function e(e){var t=e.tags,n=Object.keys(t),o=n.map(function(e){return typeof t[e]}).every(function(e){return"object"===e||"boolean"===e||"function"===e});if(!o)throw new Error("The configuration was invalid");this.config=e}function t(e){return s.indexOf(e.nodeName)!==-1}function n(e){return c.indexOf(e.nodeName)!==-1}function o(e){return document.createTreeWalker(e,NodeFilter.SHOW_TEXT|NodeFilter.SHOW_ELEMENT|NodeFilter.SHOW_COMMENT,null,!1)}function r(e,t,n){return"function"==typeof e.tags[t]?e.tags[t](n):e.tags[t]}function i(e,t){return"undefined"==typeof t||"boolean"==typeof t&&!t}function a(e,t,n){var o=e.name.toLowerCase();return t!==!0&&("function"==typeof t[o]?!t[o](e.value,n):"undefined"==typeof t[o]||(t[o]===!1||"string"==typeof t[o]&&t[o]!==e.value))}var s=["P","LI","TD","TH","DIV","H1","H2","H3","H4","H5","H6","PRE"],c=["A","B","STRONG","I","EM","SUB","SUP","U","STRIKE"];return e.prototype.clean=function(e){var t=document.createElement("div");return t.innerHTML=e,this._sanitize(t),t.innerHTML},e.prototype._sanitize=function(e){var s=o(e),c=s.firstChild();if(c)do if(!c._sanitized)if(c.nodeType!==Node.TEXT_NODE){if(c.nodeType===Node.COMMENT_NODE){e.removeChild(c),this._sanitize(e);break}var l,d=n(c);d&&(l=Array.prototype.some.call(c.childNodes,t));var u=!!e.parentNode,p=t(e)&&t(c)&&u,f=c.nodeName.toLowerCase(),g=r(this.config,f,c),h=d&&l;if(h||i(c,g)||!this.config.keepNestedBlockElements&&p){if("SCRIPT"!==c.nodeName&&"STYLE"!==c.nodeName)for(;c.childNodes.length>0;)e.insertBefore(c.childNodes[0],c);e.removeChild(c),this._sanitize(e);break}for(var v=0;v0&&t.splice(s,1)}},e.removeAll=function(){t.map(function(t){e.remove(t.element,t.type,t.handler)})},e.get=function(t,n,o){return e.search.all(t,n,o)},e}({})},function(e,t){"use strict";var n="function"==typeof Symbol&&"symbol"==typeof Symbol.iterator?function(e){return typeof e}:function(e){return e&&"function"==typeof Symbol&&e.constructor===Symbol&&e!==Symbol.prototype?"symbol":typeof e};e.exports=function(e){var t=codex.editor;return e.removeNodes=function(){t.nodes.wrapper.remove(),t.nodes.notifications.remove()},e.destroyPlugins=function(){for(var e in t.tools)"function"==typeof t.tools[e].destroy&&t.tools[e].destroy()},e.destroyScripts=function(){for(var e=document.getElementsByTagName("SCRIPT"),n=0;n"+t.split("\n\n").join("

    ")+"

    ";var n,o,r,i,a=document.createElement("DIV"),s=document.createElement("DIV"),c=["DIV","P"];for(a.innerHTML=e,o=document.createElement("P"),n=0;n'}),o=t.draw.node("DIV","ce-toolbar__remove-confirmation",{}),r=t.draw.node("DIV","ce-toolbar__remove-confirm",{textContent:"Удалить блок"}),i=t.draw.node("DIV","ce-toolbar__remove-cancel",{textContent:"Отмена"});return t.listeners.add(n,"click",t.toolbar.settings.removeButtonClicked,!1),t.listeners.add(r,"click",t.toolbar.settings.confirmRemovingRequest,!1),t.listeners.add(i,"click",t.toolbar.settings.cancelRemovingRequest,!1),o.appendChild(r),o.appendChild(i),e.appendChild(n),e.appendChild(o),t.toolbar.settings.setting=n,t.toolbar.settings.actions=o,e},e.removeButtonClicked=function(){var e=t.toolbar.settings.actions;e.classList.contains("opened")?t.toolbar.settings.hideRemoveActions():t.toolbar.settings.showRemoveActions(),t.toolbar.toolbox.close(),t.toolbar.settings.close()},e.cancelRemovingRequest=function(){t.toolbar.settings.actions.classList.remove("opened")},e.confirmRemovingRequest=function(){var e,n=t.content.currentNode;n.remove(),e=t.nodes.redactor.childNodes.length,0===e&&(t.content.currentNode=null,t.ui.addInitialBlock()),t.ui.saveInputs(),t.toolbar.close()},e.showRemoveActions=function(){t.toolbar.settings.actions.classList.add("opened")},e.hideRemoveActions=function(){t.toolbar.settings.actions.classList.remove("opened")},e}({})},function(e,t){"use strict";e.exports=function(e){var t=codex.editor;e.buttonsOpened=null,e.actionsOpened=null,e.wrappersOffset=null,e.storedSelection=null,e.show=function(){var n,o=t.content.currentNode,r=o.dataset.tool;if(n=t.tools[r],n.showInlineToolbar){var i=e.getSelectionText(),a=t.nodes.inlineToolbar.wrapper;i.length>0&&(t.toolbar.inline.move(),a.classList.add("opened"),t.toolbar.inline.showButtons())}},e.close=function(){var e=t.nodes.inlineToolbar.wrapper;e.classList.remove("opened")},e.move=function(){this.wrappersOffset||(this.wrappersOffset=this.getWrappersOffset());var e,n,o=this.getSelectionCoords(),r=0,i=t.nodes.inlineToolbar.wrapper;0===i.offsetHeight&&(r=40),e=o.x-this.wrappersOffset.left,n=o.y+window.scrollY-this.wrappersOffset.top-r-i.offsetHeight,i.style.transform="translate3D("+Math.floor(e)+"px, "+Math.floor(n)+"px, 0)",t.toolbar.inline.closeButtons(),t.toolbar.inline.closeAction()},e.toolClicked=function(e,n){switch(n){case"createLink":t.toolbar.inline.createLinkAction(e,n);break;default:t.toolbar.inline.defaultToolAction(n)}t.nodes.inlineToolbar.buttons.childNodes.forEach(t.toolbar.inline.hightlight)},e.getWrappersOffset=function(){var e=t.nodes.wrapper,n=this.getOffset(e);return this.wrappersOffset=n,n},e.getOffset=function(e){for(var t=0,n=0;e&&!isNaN(e.offsetLeft)&&!isNaN(e.offsetTop);)t+=e.offsetLeft+e.clientLeft,n+=e.offsetTop+e.clientTop,e=e.offsetParent;return{top:n,left:t}},e.getSelectionCoords=function(){var e,t=document.selection,n=0,o=0;if(t)"Control"!=t.type&&(e=t.createRange(),e.collapse(!0),n=e.boundingLeft,o=e.boundingTop);else if(window.getSelection&&(t=window.getSelection(),t.rangeCount&&(e=t.getRangeAt(0).cloneRange(),e.getClientRects))){e.collapse(!0);var r=e.getClientRects()[0];if(!r)return;n=r.left,o=r.top}return{x:n,y:o}},e.getSelectionText=function(){var e="";return window.getSelection&&(e=window.getSelection().toString()),e},e.showButtons=function(){var e=t.nodes.inlineToolbar.buttons;e.classList.add("opened"),t.toolbar.inline.buttonsOpened=!0,t.nodes.inlineToolbar.buttons.childNodes.forEach(t.toolbar.inline.hightlight)},e.closeButtons=function(){var e=t.nodes.inlineToolbar.buttons;e.classList.remove("opened"),t.toolbar.inline.buttonsOpened=!1},e.showActions=function(){var e=t.nodes.inlineToolbar.actions;e.classList.add("opened"),t.toolbar.inline.actionsOpened=!0},e.closeAction=function(){var e=t.nodes.inlineToolbar.actions;e.innerHTML="",e.classList.remove("opened"),t.toolbar.inline.actionsOpened=!1};var n=function(e){if(e.keyCode==t.core.keys.ENTER){var n=t.content.currentNode,o=t.toolbar.inline.storedSelection;t.toolbar.inline.restoreSelection(n,o),t.toolbar.inline.setAnchor(this.value),e.preventDefault(),e.stopImmediatePropagation(),t.toolbar.inline.clearRange()}};return e.createLinkAction=function(e){var o=this.isLinkActive(),r=t.content.currentNode,i=t.toolbar.inline.saveSelection(r);if(t.toolbar.inline.storedSelection=i,o)t.toolbar.inline.restoreSelection(r,i),t.toolbar.inline.defaultToolAction("unlink");else{var a=t.draw.inputForLink();t.nodes.inlineToolbar.actions.appendChild(a),t.toolbar.inline.closeButtons(),t.toolbar.inline.showActions(),a.focus(),e.preventDefault(),t.listeners.add(a,"keydown",n,!1)}},e.isLinkActive=function(){var e=!1;return t.nodes.inlineToolbar.buttons.childNodes.forEach(function(t){var n=t.dataset.type;"link"==n&&t.classList.contains("hightlighted")&&(e=!0)}),e},e.defaultToolAction=function(e){document.execCommand(e,!1,null)},e.setAnchor=function(e){document.execCommand("createLink",!1,e),t.toolbar.inline.closeAction()},e.saveSelection=function(e){var t,n=window.getSelection().getRangeAt(0),o=n.cloneRange();return o.selectNodeContents(e),o.setEnd(n.startContainer,n.startOffset),t=o.toString().length,{start:t,end:t+n.toString().length}},e.restoreSelection=function(e,t){var n=document.createRange(),o=0;n.setStart(e,0),n.collapse(!0);for(var r,i,a=[e],s=!1,c=!1;!c&&(r=a.pop());)if(3==r.nodeType)i=o+r.length,!s&&t.start>=o&&t.start<=i&&(n.setStart(r,t.start-o),s=!0),s&&t.end>=o&&t.end<=i&&(n.setEnd(r,t.end-o),c=!0),o=i;else for(var l=r.childNodes.length;l--;)a.push(r.childNodes[l]);var d=window.getSelection();d.removeAllRanges(),d.addRange(n)},e.clearRange=function(){var e=window.getSelection();e.removeAllRanges()},e.hightlight=function(e){var n=e.dataset.type;document.queryCommandState(n)?t.toolbar.inline.setButtonHighlighted(e):t.toolbar.inline.removeButtonsHighLight(e);var o=window.getSelection(),r=o.anchorNode.parentNode;"A"==r.tagName&&"link"==n&&t.toolbar.inline.setButtonHighlighted(e)},e.setButtonHighlighted=function(e){if(e.classList.add("hightlighted"),"link"==e.dataset.type){var t=e.childNodes[0];t.classList.remove("ce-icon-link"),t.classList.add("ce-icon-unlink")}},e.removeButtonsHighLight=function(e){if(e.classList.remove("hightlighted"),"link"==e.dataset.type){var t=e.childNodes[0];t.classList.remove("ce-icon-unlink"),t.classList.add("ce-icon-link")}},e}({})},function(e,t){"use strict";e.exports=function(e){var t=codex.editor;return e.opened=!1,e.open=function(){t.toolbar.settings.opened&&t.toolbar.settings.close(),t.nodes.toolbox.classList.add("opened"),t.nodes.plusButton.classList.add("clicked"),t.toolbar.toolbox.opened=!0},e.close=function(){t.nodes.toolbox.classList.remove("opened"),t.nodes.plusButton.classList.remove("clicked"),t.toolbar.toolbox.opened=!1,t.toolbar.current=null},e.leaf=function(){var e=t.toolbar.current,n=Object.keys(t.tools),o=t.nodes.toolbarButtons,r=0,i=void 0,a=void 0,s=void 0;if(e)for(r=(n.indexOf(e)+1)%n.length,a=n[r];!t.tools[a].displayInToolbox;)r=(r+1)%n.length,a=n[r];else for(s in t.tools){if(t.tools[s].displayInToolbox)break;r++}i=n[r];for(var c in o)o[c].classList.remove("selected");o[i].classList.add("selected"),t.toolbar.current=i},e.toolClicked=function(e){var n,o,r,i=["image","link","list","instagram","twitter","embed"],a=t.tools[t.toolbar.current],s=t.content.currentNode,c=t.caret.inputIndex;n=a.render(),r={block:n,type:a.type,stretched:!1},s&&i.indexOf(s.dataset.tool)===-1&&""===s.textContent.trim()?t.content.switchBlock(s,n,a.type):(t.content.insertBlock(r),c++),o=a.appendCallback,o&&"function"==typeof o&&o.call(e),window.setTimeout(function(){t.caret.setToBlock(c)},10),t.content.workingNodeChanged(),t.toolbar.move()},e}({})},function(e,t){"use strict";e.exports=function(e){var t=codex.editor;e.globalKeydown=function(e){switch(e.keyCode){case t.core.keys.ENTER:o(e)}},e.redactorKeyDown=function(e){switch(e.keyCode){case t.core.keys.TAB:n(e);break;case t.core.keys.ENTER:i(e);break;case t.core.keys.ESC:a(e);break;default:c(e)}},e.globalKeyup=function(e){switch(e.keyCode){case t.core.keys.UP:case t.core.keys.LEFT:case t.core.keys.RIGHT:case t.core.keys.DOWN:s(e)}};var n=function(e){e.preventDefault();var n=t.content.currentNode.querySelectorAll("textarea, input"),o=!0,r=!t.content.currentNode.textContent.trim();Array.prototype.map.call(n,function(e){"textarea"!=e.type&&"text"!=e.type||(o=o&&!e.value.trim())});var i=r&&o;i&&(t.toolbar.opened||t.toolbar.open(),t.toolbar.opened&&!t.toolbar.toolbox.opened?t.toolbar.toolbox.open():t.toolbar.toolbox.leaf())},o=function(){t.content.editorAreaHightlighted&&(t.caret.inputIndex=-1,r())},r=function(){var e=t.settings.initialBlockPlugin;t.content.insertBlock({type:e,block:t.tools[e].render()},!0),t.toolbar.move(),t.toolbar.open()},i=function(e){"true"==e.target.contentEditable&&t.caret.saveCurrentInputIndex();var n=t.caret.getCurrentInputIndex()||0,o=t.content.currentNode,r=o.dataset.tool,i=t.toolbar.opened&&t.toolbar.current&&e.target==t.state.inputs[n],a=t.tools[r].enableLineBreaks,s=t.settings.initialBlockPlugin;if(i)return e.preventDefault(),t.toolbar.toolbox.toolClicked(e),t.toolbar.close(),e.stopPropagation(),void e.stopImmediatePropagation();if(e.shiftKey||a)return e.stopPropagation(),void e.stopImmediatePropagation();var c=window.getSelection(),l=c.anchorNode,d=t.caret.position.atTheEnd(),u=!1;if(e.shiftKey&&!a)return t.callback.enterPressedOnBlock(t.content.currentBlock,e),void e.preventDefault();if(u=l&&"true"!=l.parentNode.contentEditable,l.nodeType!=t.core.nodeTypes.TEXT||u||d){var p=t.content.isLastNode(l);p&&d&&(e.preventDefault(),e.stopPropagation(),e.stopImmediatePropagation(),t.core.log("ENTER clicked in last textNode. Create new BLOCK"),t.content.insertBlock({type:s,block:t.tools[s].render()},!0),t.toolbar.move(),t.toolbar.open(),t.toolbar.showPlusButton())}else e.preventDefault(),t.core.log("Splitting Text node..."),t.content.splitBlock(n),t.state.inputs[n+1].textContent.trim()||t.toolbar.showPlusButton();t.ui.saveInputs()},a=function(e){t.toolbar.close(),t.toolbar.toolbox.close(),e.preventDefault()},s=function(e){t.content.workingNodeChanged(),t.toolbar.close(),t.toolbar.move()},c=function(){t.toolbar.close(),t.toolbar.inline.actionsOpened||(t.toolbar.inline.close(),t.content.clearMark())};e.redactorClicked=function(e){l(),t.content.workingNodeChanged(e.target),t.ui.saveInputs();var n,o=t.toolbar.inline.getSelectionText();if(0===o.length&&t.toolbar.inline.close(),"true"==e.target.contentEditable&&t.caret.saveCurrentInputIndex(),null===t.content.currentNode){var r=t.state.inputs.length>0?t.state.inputs.length-1:0;if(t.state.inputs.length&&(n=t.content.getFirstLevelBlock(t.state.inputs[r])),t.state.inputs.length&&""===t.state.inputs[r].textContent&&n.dataset.tool==t.settings.initialBlockPlugin)t.caret.setToBlock(r);else{var i=t.settings.initialBlockPlugin;t.content.insertBlock({type:i,block:t.tools[i].render()}),1===t.state.inputs.length?t.caret.setToBlock(r):t.caret.setToNextBlock(r)}}else t.toolbar.settings.close(),t.toolbar.toolbox.close();t.toolbar.move(),t.toolbar.open();var a=!t.content.currentNode.textContent.trim(),s=t.content.currentNode.dataset.tool,c=s==t.settings.initialBlockPlugin;t.toolbar.hidePlusButton(),a||t.content.markBlock(),c&&a&&t.toolbar.showPlusButton()};var l=function(){var e=window.getSelection(),n=e.anchorNode,o=!1;if(0===e.rangeCount)t.content.editorAreaHightlighted=!0;else{for(t.core.isDomNode(n)||(n=n.parentNode),"true"==n.contentEditable&&(o=!0);"true"!=n.contentEditable&&(n=n.parentNode,"true"==n.contentEditable&&(o=!0),n!=document.body););t.content.editorAreaHightlighted=!o}};e.toolbarButtonClicked=function(e){var n=this;t.toolbar.current=n.dataset.type,t.toolbar.toolbox.toolClicked(e),t.toolbar.close()},e.plusButtonClicked=function(){t.nodes.toolbox.classList.contains("opened")?t.toolbar.toolbox.close():t.toolbar.toolbox.open()},e.blockKeydown=function(e){var n=e.target;switch(e.keyCode){case t.core.keys.DOWN:case t.core.keys.RIGHT:d(e);break;case t.core.keys.BACKSPACE:p(n,e);break;case t.core.keys.UP:case t.core.keys.LEFT:u(e)}};var d=function(e){var n,o=window.getSelection(),r=t.state.inputs,i=o.anchorNode;if(!i)return!1;for(;"true"!=i.contentEditable;)n=i.parentNode,i=n;for(var a=0;i!=r[a];)a++;if(!i.textContent)return void t.caret.setToNextBlock(a);var s,c,l=!1,d=!1;return s=i.childNodes[i.childNodes.length-1],c=t.core.isDomNode(s)?t.content.getDeepestTextNodeFromPosition(s,s.childNodes.length):s,l=o.anchorNode==c,d=c.length==o.anchorOffset,l&&d?void t.caret.setToNextBlock(a):(t.core.log("arrow [down|right] : caret does not reached the end"),!1)},u=function(e){var n,o=window.getSelection(),r=t.state.inputs,i=o.anchorNode;if(!i)return!1;if(0!==o.anchorOffset)return!1;for(;"true"!=i.contentEditable;)n=i.parentNode,i=n;for(var a=0;i!=r[a];)a++;var s,c,l=!1,d=!1;return i.textContent?(s=i.childNodes[0],c=t.core.isDomNode(s)?t.content.getDeepestTextNodeFromPosition(s,0):s,l=o.anchorNode==c,d=0===o.anchorOffset,void(l&&d&&t.caret.setToPreviousBlock(a))):void t.caret.setToPreviousBlock(a)},p=function(e,n){var o,r,i,a=t.caret.getCurrentInputIndex();if(t.core.isNativeInput(n.target)){if(""!=n.target.value.trim())return;e.remove()}if(e.textContent.trim()){if(o=t.content.getRange(),r=o.endOffset-o.startOffset,!t.caret.position.atStart()||r||!t.state.inputs[a-1])return;t.content.mergeBlocks(a)}r||e.remove(),i=t.nodes.redactor.childNodes.length,0===i?(t.content.currentNode=null,t.ui.addInitialBlock(),t.ui.saveInputs(),window.setTimeout(function(){ +t.caret.setToPreviousBlock(1)},10)):0!==t.caret.inputIndex?t.caret.setToPreviousBlock(t.caret.inputIndex):t.caret.setToNextBlock(t.caret.inputIndex),t.toolbar.move(),t.toolbar.opened||t.toolbar.open(),t.ui.saveInputs(),n.preventDefault()};return e.showSettingsButtonClicked=function(e){var n=t.content.currentNode.dataset.tool;t.toolbar.settings.toggle(n),t.toolbar.toolbox.close(),t.toolbar.settings.hideRemoveActions()},e}({})},function(e,t){"use strict";e.exports=function(e){return e.wrapper=function(){var e=document.createElement("div");return e.className+="codex-editor",e},e.redactor=function(){var e=document.createElement("div");return e.className+="ce-redactor",e},e.ceBlock=function(){var e=document.createElement("DIV");return e.className+="ce_block",e},e.toolbar=function(){var e=document.createElement("div");return e.className+="ce-toolbar",e},e.toolbarContent=function(){var e=document.createElement("DIV");return e.classList.add("ce-toolbar__content"),e},e.inlineToolbar=function(){var e=document.createElement("DIV");return e.className+="ce-toolbar-inline",e},e.inlineToolbarButtons=function(){var e=document.createElement("DIV");return e.className+="ce-toolbar-inline__buttons",e},e.inlineToolbarActions=function(){var e=document.createElement("DIV");return e.className+="ce-toolbar-inline__actions",e},e.inputForLink=function(){var e=document.createElement("INPUT");return e.type="input",e.className+="inputForLink",e.placeholder="Вставьте ссылку ...",e.setAttribute("form","defaultForm"),e.setAttribute("autofocus","autofocus"),e},e.blockButtons=function(){var e=document.createElement("div");return e.className+="ce-toolbar__actions",e},e.blockSettings=function(){var e=document.createElement("div");return e.className+="ce-settings",e},e.defaultSettings=function(){var e=document.createElement("div");return e.classList.add("ce-settings_default"),e},e.pluginsSettings=function(){var e=document.createElement("div");return e.classList.add("ce-settings_plugin"),e},e.plusButton=function(){var e=document.createElement("span");return e.className="ce-toolbar__plus",e},e.settingsButton=function(){var e=document.createElement("span");return e.className="ce-toolbar__settings-btn",e.innerHTML='',e},e.toolbox=function(){var e=document.createElement("div");return e.className="ce-toolbar__tools",e},e.toolbarButton=function(e,t){var n=document.createElement("li"),o=document.createElement("i"),r=document.createElement("span");return n.dataset.type=e,n.setAttribute("title",e),o.classList.add(t),r.classList.add("ce_toolbar_tools--title"),n.appendChild(o),n.appendChild(r),n},e.toolbarButtonInline=function(e,t){var n=document.createElement("BUTTON"),o=document.createElement("I");return n.type="button",n.dataset.type=e,o.classList.add(t),n.appendChild(o),n},e.block=function(e,t){var n=document.createElement(e);return n.innerHTML=t||"",n},e.node=function(e,t,n){var o=document.createElement(e);if(t&&(o.className=t),n)for(var r in n)o[r]=n[r];return o},e.unavailableBlock=function(){var e=document.createElement("DIV");return e.classList.add("cdx-unavailable-block"),e},e}({})},function(e,t){"use strict";e.exports=function(e){var t=codex.editor;return e.inputIndex=null,e.offset=null,e.focusedNodeIndex=null,e.set=function(n,o,r){r=r||e.offset||0,o=o||e.focusedNodeIndex||0;var i,a=n.childNodes;if(i=0===a.length?n:a[o],"true"!=n.contentEditable)return void n.focus();t.core.isDomNode(i)&&(i=t.content.getDeepestTextNodeFromPosition(i,i.childNodes.length));var s=document.createRange(),c=window.getSelection();window.setTimeout(function(){s.setStart(i,r),s.setEnd(i,r),c.removeAllRanges(),c.addRange(s),t.caret.saveCurrentInputIndex()},20)},e.saveCurrentInputIndex=function(){var n,o=window.getSelection(),r=t.state.inputs,i=o.anchorNode;if(i){for(;"true"!=i.contentEditable;)n=i.parentNode,i=n;for(var a=0;i!=r[a];)a++;e.inputIndex=a}},e.getCurrentInputIndex=function(){return e.inputIndex},e.setToNextBlock=function(e){var n=t.state.inputs,o=n[e+1];if(!o)return void t.core.log("We are reached the end");if(!o.childNodes.length){var r=document.createTextNode("");o.appendChild(r)}t.caret.inputIndex=e+1,t.caret.set(o,0,0),t.content.workingNodeChanged(o)},e.setToBlock=function(e){var n=t.state.inputs,o=n[e];if(o){if(!o.childNodes.length){var r=document.createTextNode("");o.appendChild(r)}t.caret.inputIndex=e,t.caret.set(o,0,0),t.content.workingNodeChanged(o)}},e.setToPreviousBlock=function(e){e=e||0;var n,o,r,i=t.state.inputs,a=i[e-1];return a?(n=t.content.getDeepestTextNodeFromPosition(a,a.childNodes.length),o=n.length,a.childNodes.length||(r=document.createTextNode(""),a.appendChild(r)),t.caret.inputIndex=e-1,t.caret.set(a,a.childNodes.length-1,o),void t.content.workingNodeChanged(i[e-1])):void t.core.log("We are reached first node")},e.position={atStart:function(){var e=window.getSelection(),n=e.anchorOffset,o=e.anchorNode,r=t.content.getFirstLevelBlock(o),i=r.childNodes[0];t.core.isDomNode(o)||(o=o.parentNode);var a=o===i.childNodes[0],s=0===n;return a&&s},atTheEnd:function(){var e=window.getSelection(),t=e.anchorOffset,n=e.anchorNode;return!n||!n.length||t===n.length}},e.insertNode=function(e){var t,n;t=window.getSelection(),n=t.getRangeAt(0),n.deleteContents(),n.insertNode(e),n.setStartAfter(e),n.collapse(!0),t.removeAllRanges(),t.addRange(n)},e}({})},function(e,t){"use strict";e.exports=function(e){var t=codex.editor,n=[],o=function(e){n.push(e);for(var t=0;t5;)"confirm"!=n[t].type&&"prompt"!=n[t].type?(n[t].close(),n.splice(t,1)):t++};return e.createHolder=function(){var e=t.draw.node("DIV","cdx-notifications-block");return t.nodes.notifications=document.body.appendChild(e),e},e.errorThrown=function(e,n){t.notifications.notification({message:"This action is not available currently",type:n.type})},e.notification=function(e){function n(e){if(!e||!e.message)return void t.core.log("Can't create notification. Message is missed");e.type=e.type||"alert",e.time=1e3*e.time||1e4;var n=t.draw.node("DIV","cdx-notification"),o=t.draw.node("DIV","cdx-notification__message"),r=t.draw.node("INPUT","cdx-notification__input"),f=t.draw.node("SPAN","cdx-notification__ok-btn"),g=t.draw.node("SPAN","cdx-notification__cancel-btn");o.textContent=e.message,f.textContent=e.okMsg||"ОК",g.textContent=e.cancelMsg||"Отмена",t.listeners.add(f,"click",u),t.listeners.add(g,"click",p),n.appendChild(o),"prompt"==e.type&&n.appendChild(r),n.appendChild(f),"prompt"!=e.type&&"confirm"!=e.type||n.appendChild(g),n.classList.add("cdx-notification-"+e.type),n.dataset.type=e.type,a=n,c=e.type,l=e.confirm,s=e.cancel,d=r,"prompt"!=e.type&&"confirm"!=e.type&&window.setTimeout(i,e.time)}function r(){t.nodes.notifications.appendChild(a),d.focus(),t.nodes.notifications.classList.add("cdx-notification__notification-appending"),window.setTimeout(function(){t.nodes.notifications.classList.remove("cdx-notification__notification-appending")},100),o({type:c,close:i})}function i(){a.remove()}var a=null,s=null,c=null,l=null,d=null,u=function(){if(i(),"function"==typeof l)return"prompt"==c?void l(d.value):void l()},p=function(){i(),"function"==typeof s&&s()};return e&&(n(e),r()),{create:n,send:r,close:i}},e.clear=function(){t.nodes.notifications.innerHTML="",n=[]},e}({})},function(e,t){"use strict";e.exports=function(e){var t=codex.editor;return e.insertPastedContent=function(e,n){t.content.insertBlock({type:e.type,block:e.render({text:n.innerHTML})})},e.isFirstLevelBlock=function(e){return e.nodeType==t.core.nodeTypes.TAG&&e.classList.contains(t.ui.className.BLOCK_CLASSNAME)},e}({})},function(e,t,n){"use strict";e.exports=function(e){var t=n(18),o=codex.editor;e.prepare=function(){o.settings.sanitizer&&!o.core.isEmpty(o.settings.sanitizer)&&(r.CUSTOM=o.settings.sanitizer)};var r={CUSTOM:null,BASIC:{tags:{p:{},a:{href:!0,target:"_blank",rel:"nofollow"},i:{},b:{},strong:{},em:{},span:{}}}};e.Config=r;var i=function(e){var n=e||r.CUSTOM||r.BASIC;return new t(n)};return e.clean=function(e,t){var n=i(t);return n.clean(e)},e}({})},function(e,t,n){var o,r;!function(i,a){o=a,r="function"==typeof o?o.call(t,n,t,e):o,!(void 0!==r&&(e.exports=r))}(this,function(){function e(e){var t=e.tags,n=Object.keys(t),o=n.map(function(e){return typeof t[e]}).every(function(e){return"object"===e||"boolean"===e||"function"===e});if(!o)throw new Error("The configuration was invalid");this.config=e}function t(e){return s.indexOf(e.nodeName)!==-1}function n(e){return c.indexOf(e.nodeName)!==-1}function o(e){return document.createTreeWalker(e,NodeFilter.SHOW_TEXT|NodeFilter.SHOW_ELEMENT|NodeFilter.SHOW_COMMENT,null,!1)}function r(e,t,n){return"function"==typeof e.tags[t]?e.tags[t](n):e.tags[t]}function i(e,t){return"undefined"==typeof t||"boolean"==typeof t&&!t}function a(e,t,n){var o=e.name.toLowerCase();return t!==!0&&("function"==typeof t[o]?!t[o](e.value,n):"undefined"==typeof t[o]||(t[o]===!1||"string"==typeof t[o]&&t[o]!==e.value))}var s=["P","LI","TD","TH","DIV","H1","H2","H3","H4","H5","H6","PRE"],c=["A","B","STRONG","I","EM","SUB","SUP","U","STRIKE"];return e.prototype.clean=function(e){var t=document.createElement("div");return t.innerHTML=e,this._sanitize(t),t.innerHTML},e.prototype._sanitize=function(e){var s=o(e),c=s.firstChild();if(c)do if(!c._sanitized)if(c.nodeType!==Node.TEXT_NODE){if(c.nodeType===Node.COMMENT_NODE){e.removeChild(c),this._sanitize(e);break}var l,d=n(c);d&&(l=Array.prototype.some.call(c.childNodes,t));var u=!!e.parentNode,p=t(e)&&t(c)&&u,f=c.nodeName.toLowerCase(),g=r(this.config,f,c),h=d&&l;if(h||i(c,g)||!this.config.keepNestedBlockElements&&p){if("SCRIPT"!==c.nodeName&&"STYLE"!==c.nodeName)for(;c.childNodes.length>0;)e.insertBefore(c.childNodes[0],c);e.removeChild(c),this._sanitize(e);break}for(var v=0;v0&&t.splice(s,1)}},e.removeAll=function(){t.map(function(t){e.remove(t.element,t.type,t.handler)})},e.get=function(t,n,o){return e.search.all(t,n,o)},e}({})},function(e,t){"use strict";var n="function"==typeof Symbol&&"symbol"==typeof Symbol.iterator?function(e){return typeof e}:function(e){return e&&"function"==typeof Symbol&&e.constructor===Symbol&&e!==Symbol.prototype?"symbol":typeof e};e.exports=function(e){var t=codex.editor;return e.removeNodes=function(){t.nodes.wrapper.remove(),t.nodes.notifications.remove()},e.destroyPlugins=function(){for(var e in t.tools)"function"==typeof t.tools[e].destroy&&t.tools[e].destroy()},e.destroyScripts=function(){for(var e=document.getElementsByTagName("SCRIPT"),n=0;n nodeList\r\n\t */\n\t renderer.appendBlocks = function (data) {\n\t\n\t var blocks = data.items;\n\t\n\t /**\r\n\t * Sequence of one-by-one blocks appending\r\n\t * Uses to save blocks order after async-handler\r\n\t */\n\t var nodeSequence = Promise.resolve();\n\t\n\t for (var index = 0; index < blocks.length; index++) {\n\t\n\t /** Add node to sequence at specified index */\n\t editor.renderer.appendNodeAtIndex(nodeSequence, blocks, index);\n\t }\n\t };\n\t\n\t /**\r\n\t * Append node at specified index\r\n\t */\n\t renderer.appendNodeAtIndex = function (nodeSequence, blocks, index) {\n\t\n\t /** We need to append node to sequence */\n\t nodeSequence\n\t\n\t /** first, get node async-aware */\n\t .then(function () {\n\t\n\t return editor.renderer.getNodeAsync(blocks, index);\n\t })\n\t\n\t /**\r\n\t * second, compose editor-block from JSON object\r\n\t */\n\t .then(editor.renderer.createBlockFromData)\n\t\n\t /**\r\n\t * now insert block to redactor\r\n\t */\n\t .then(function (blockData) {\n\t\n\t /**\r\n\t * blockData has 'block', 'type' and 'stretched' information\r\n\t */\n\t editor.content.insertBlock(blockData);\n\t\n\t /** Pass created block to next step */\n\t return blockData.block;\n\t })\n\t\n\t /** Log if something wrong with node */\n\t .catch(function (error) {\n\t\n\t editor.core.log('Node skipped while parsing because %o', 'error', error);\n\t });\n\t };\n\t\n\t /**\r\n\t * Asynchronously returns block data from blocksList by index\r\n\t * @return Promise to node\r\n\t */\n\t renderer.getNodeAsync = function (blocksList, index) {\n\t\n\t return Promise.resolve().then(function () {\n\t\n\t return {\n\t tool: blocksList[index],\n\t position: index\n\t };\n\t });\n\t };\n\t\n\t /**\r\n\t * Creates editor block by JSON-data\r\n\t *\r\n\t * @uses render method of each plugin\r\n\t *\r\n\t * @param {Object} toolData.tool\r\n\t * { header : {\r\n\t * text: '',\r\n\t * type: 'H3', ...\r\n\t * }\r\n\t * }\r\n\t * @param {Number} toolData.position - index in input-blocks array\r\n\t * @return {Object} with type and Element\r\n\t */\n\t renderer.createBlockFromData = function (toolData) {\n\t\n\t /** New parser */\n\t var block,\n\t tool = toolData.tool,\n\t pluginName = tool.type;\n\t\n\t /** Get first key of object that stores plugin name */\n\t // for (var pluginName in blockData) break;\n\t\n\t /** Check for plugin existance */\n\t if (!editor.tools[pluginName]) {\n\t\n\t throw Error('Plugin \\xAB' + pluginName + '\\xBB not found');\n\t }\n\t\n\t /** Check for plugin having render method */\n\t if (typeof editor.tools[pluginName].render != 'function') {\n\t\n\t throw Error('Plugin \\xAB' + pluginName + '\\xBB must have \\xABrender\\xBB method');\n\t }\n\t\n\t if (editor.tools[pluginName].available === false) {\n\t\n\t block = editor.draw.unavailableBlock();\n\t\n\t block.innerHTML = editor.tools[pluginName].loadingMessage;\n\t\n\t /**\r\n\t * Saver will extract data from initial block data by position in array\r\n\t */\n\t block.dataset.inputPosition = toolData.position;\n\t } else {\n\t\n\t /** New Parser */\n\t block = editor.tools[pluginName].render(tool.data);\n\t }\n\t\n\t /** is first-level block stretched */\n\t var stretched = editor.tools[pluginName].isStretched || false;\n\t\n\t /** Retrun type and block */\n\t return {\n\t type: pluginName,\n\t block: block,\n\t stretched: stretched\n\t };\n\t };\n\t\n\t return renderer;\n\t}({});\n\n/***/ },\n/* 6 */\n/***/ function(module, exports) {\n\n\t'use strict';\n\t\n\t/**\r\n\t * Codex Editor Saver\r\n\t *\r\n\t * @author Codex Team\r\n\t * @version 1.0.2\r\n\t */\n\t\n\tmodule.exports = function (saver) {\n\t\n\t var editor = codex.editor;\n\t\n\t /**\r\n\t * Saves blocks\r\n\t * @private\r\n\t */\n\t saver.saveBlocks = function () {\n\t\n\t /** Save html content of redactor to memory */\n\t editor.state.html = editor.nodes.redactor.innerHTML;\n\t\n\t /** Empty jsonOutput state */\n\t editor.state.jsonOutput = [];\n\t\n\t Promise.resolve().then(function () {\n\t\n\t return editor.nodes.redactor.childNodes;\n\t })\n\t /** Making a sequence from separate blocks */\n\t .then(editor.saver.makeQueue).then(function () {\n\t // editor.nodes.textarea.innerHTML = editor.state.html;\n\t }).catch(function (error) {\n\t\n\t editor.core.log(error);\n\t });\n\t };\n\t\n\t saver.makeQueue = function (blocks) {\n\t\n\t var queue = Promise.resolve();\n\t\n\t for (var index = 0; index < blocks.length; index++) {\n\t\n\t /** Add node to sequence at specified index */\n\t editor.saver.getBlockData(queue, blocks, index);\n\t }\n\t };\n\t\n\t /** Gets every block and makes From Data */\n\t saver.getBlockData = function (queue, blocks, index) {\n\t\n\t queue.then(function () {\n\t\n\t return editor.saver.getNodeAsync(blocks, index);\n\t }).then(editor.saver.makeFormDataFromBlocks);\n\t };\n\t\n\t /**\r\n\t * Asynchronously returns block data from blocksList by index\r\n\t * @return Promise to node\r\n\t */\n\t saver.getNodeAsync = function (blocksList, index) {\n\t\n\t return Promise.resolve().then(function () {\n\t\n\t return blocksList[index];\n\t });\n\t };\n\t\n\t saver.makeFormDataFromBlocks = function (block) {\n\t\n\t var pluginName = block.dataset.tool;\n\t\n\t /** Check for plugin existance */\n\t if (!editor.tools[pluginName]) {\n\t\n\t throw Error('Plugin \\xAB' + pluginName + '\\xBB not found');\n\t }\n\t\n\t /** Check for plugin having render method */\n\t if (typeof editor.tools[pluginName].save != 'function') {\n\t\n\t throw Error('Plugin \\xAB' + pluginName + '\\xBB must have save method');\n\t }\n\t\n\t /** Result saver */\n\t var blockContent = block.childNodes[0],\n\t pluginsContent = blockContent.childNodes[0],\n\t savedData,\n\t position,\n\t output;\n\t\n\t /** If plugin wasn't available then return data from cache */\n\t if (editor.tools[pluginName].available === false) {\n\t\n\t position = pluginsContent.dataset.inputPosition;\n\t savedData = codex.editor.state.blocks.items[position].data;\n\t } else {\n\t\n\t savedData = editor.tools[pluginName].save(pluginsContent);\n\t\n\t if (editor.tools[pluginName].validate) {\n\t\n\t var result = editor.tools[pluginName].validate(savedData);\n\t\n\t /**\r\n\t * Do not allow invalid data\r\n\t */\n\t if (!result) return;\n\t }\n\t }\n\t\n\t output = {\n\t type: pluginName,\n\t data: savedData\n\t };\n\t\n\t editor.state.jsonOutput.push(output);\n\t };\n\t\n\t return saver;\n\t}({});\n\n/***/ },\n/* 7 */\n/***/ function(module, exports) {\n\n\t'use strict';\n\t\n\t/**\r\n\t * Codex Editor Content Module\r\n\t * Works with DOM\r\n\t *\r\n\t * @module Codex Editor content module\r\n\t *\r\n\t * @author Codex Team\r\n\t * @version 1.3.13\r\n\t *\r\n\t * @description Module works with Elements that have been appended to the main DOM\r\n\t */\n\t\n\tmodule.exports = function (content) {\n\t\n\t var editor = codex.editor;\n\t\n\t /**\r\n\t * Links to current active block\r\n\t * @type {null | Element}\r\n\t */\n\t content.currentNode = null;\n\t\n\t /**\r\n\t * clicked in redactor area\r\n\t * @type {null | Boolean}\r\n\t */\n\t content.editorAreaHightlighted = null;\n\t\n\t /**\r\n\t * @deprecated\r\n\t * Synchronizes redactor with original textarea\r\n\t */\n\t content.sync = function () {\n\t\n\t editor.core.log('syncing...');\n\t\n\t /**\r\n\t * Save redactor content to editor.state\r\n\t */\n\t editor.state.html = editor.nodes.redactor.innerHTML;\n\t };\n\t\n\t /**\r\n\t * Appends background to the block\r\n\t *\r\n\t * @description add CSS class to highlight visually first-level block area\r\n\t */\n\t content.markBlock = function () {\n\t\n\t editor.content.currentNode.classList.add(editor.ui.className.BLOCK_HIGHLIGHTED);\n\t };\n\t\n\t /**\r\n\t * Clear background\r\n\t *\r\n\t * @description clears styles that highlights block\r\n\t */\n\t content.clearMark = function () {\n\t\n\t if (editor.content.currentNode) {\n\t\n\t editor.content.currentNode.classList.remove(editor.ui.className.BLOCK_HIGHLIGHTED);\n\t }\n\t };\n\t\n\t /**\r\n\t * Finds first-level block\r\n\t *\r\n\t * @param {Element} node - selected or clicked in redactors area node\r\n\t * @protected\r\n\t *\r\n\t * @description looks for first-level block.\r\n\t * gets parent while node is not first-level\r\n\t */\n\t content.getFirstLevelBlock = function (node) {\n\t\n\t if (!editor.core.isDomNode(node)) {\n\t\n\t node = node.parentNode;\n\t }\n\t\n\t if (node === editor.nodes.redactor || node === document.body) {\n\t\n\t return null;\n\t } else {\n\t\n\t while (!node.classList.contains(editor.ui.className.BLOCK_CLASSNAME)) {\n\t\n\t node = node.parentNode;\n\t }\n\t\n\t return node;\n\t }\n\t };\n\t\n\t /**\r\n\t * Trigger this event when working node changed\r\n\t * @param {Element} targetNode - first-level of this node will be current\r\n\t * @protected\r\n\t *\r\n\t * @description If targetNode is first-level then we set it as current else we look for parents to find first-level\r\n\t */\n\t content.workingNodeChanged = function (targetNode) {\n\t\n\t /** Clear background from previous marked block before we change */\n\t editor.content.clearMark();\n\t\n\t if (!targetNode) {\n\t\n\t return;\n\t }\n\t\n\t content.currentNode = content.getFirstLevelBlock(targetNode);\n\t };\n\t\n\t /**\r\n\t * Replaces one redactor block with another\r\n\t * @protected\r\n\t * @param {Element} targetBlock - block to replace. Mostly currentNode.\r\n\t * @param {Element} newBlock\r\n\t * @param {string} newBlockType - type of new block; we need to store it to data-attribute\r\n\t *\r\n\t * [!] Function does not saves old block content.\r\n\t * You can get it manually and pass with newBlock.innerHTML\r\n\t */\n\t content.replaceBlock = function (targetBlock, newBlock) {\n\t\n\t if (!targetBlock || !newBlock) {\n\t\n\t editor.core.log('replaceBlock: missed params');\n\t return;\n\t }\n\t\n\t /** If target-block is not a frist-level block, then we iterate parents to find it */\n\t while (!targetBlock.classList.contains(editor.ui.className.BLOCK_CLASSNAME)) {\n\t\n\t targetBlock = targetBlock.parentNode;\n\t }\n\t\n\t /** Replacing */\n\t editor.nodes.redactor.replaceChild(newBlock, targetBlock);\n\t\n\t /**\r\n\t * Set new node as current\r\n\t */\n\t editor.content.workingNodeChanged(newBlock);\n\t\n\t /**\r\n\t * Add block handlers\r\n\t */\n\t editor.ui.addBlockHandlers(newBlock);\n\t\n\t /**\r\n\t * Save changes\r\n\t */\n\t editor.ui.saveInputs();\n\t };\n\t\n\t /**\r\n\t * @protected\r\n\t *\r\n\t * Inserts new block to redactor\r\n\t * Wrapps block into a DIV with BLOCK_CLASSNAME class\r\n\t *\r\n\t * @param blockData {object}\r\n\t * @param blockData.block {Element} element with block content\r\n\t * @param blockData.type {string} block plugin\r\n\t * @param needPlaceCaret {bool} pass true to set caret in new block\r\n\t *\r\n\t */\n\t content.insertBlock = function (blockData, needPlaceCaret) {\n\t\n\t var workingBlock = editor.content.currentNode,\n\t newBlockContent = blockData.block,\n\t blockType = blockData.type,\n\t isStretched = blockData.stretched;\n\t\n\t var newBlock = composeNewBlock_(newBlockContent, blockType, isStretched);\n\t\n\t if (workingBlock) {\n\t\n\t editor.core.insertAfter(workingBlock, newBlock);\n\t } else {\n\t\n\t /**\r\n\t * If redactor is empty, append as first child\r\n\t */\n\t editor.nodes.redactor.appendChild(newBlock);\n\t }\n\t\n\t /**\r\n\t * Block handler\r\n\t */\n\t editor.ui.addBlockHandlers(newBlock);\n\t\n\t /**\r\n\t * Set new node as current\r\n\t */\n\t editor.content.workingNodeChanged(newBlock);\n\t\n\t /**\r\n\t * Save changes\r\n\t */\n\t editor.ui.saveInputs();\n\t\n\t if (needPlaceCaret) {\n\t\n\t /**\r\n\t * If we don't know input index then we set default value -1\r\n\t */\n\t var currentInputIndex = editor.caret.getCurrentInputIndex() || -1;\n\t\n\t if (currentInputIndex == -1) {\n\t\n\t var editableElement = newBlock.querySelector('[contenteditable]'),\n\t emptyText = document.createTextNode('');\n\t\n\t editableElement.appendChild(emptyText);\n\t editor.caret.set(editableElement, 0, 0);\n\t\n\t editor.toolbar.move();\n\t editor.toolbar.showPlusButton();\n\t } else {\n\t\n\t if (currentInputIndex === editor.state.inputs.length - 1) return;\n\t\n\t /** Timeout for browsers execution */\n\t window.setTimeout(function () {\n\t\n\t /** Setting to the new input */\n\t editor.caret.setToNextBlock(currentInputIndex);\n\t editor.toolbar.move();\n\t editor.toolbar.open();\n\t }, 10);\n\t }\n\t }\n\t\n\t /**\r\n\t * Block is inserted, wait for new click that defined focusing on editors area\r\n\t * @type {boolean}\r\n\t */\n\t content.editorAreaHightlighted = false;\n\t };\n\t\n\t /**\r\n\t * Replaces blocks with saving content\r\n\t * @protected\r\n\t * @param {Element} noteToReplace\r\n\t * @param {Element} newNode\r\n\t * @param {Element} blockType\r\n\t */\n\t content.switchBlock = function (blockToReplace, newBlock, tool) {\n\t\n\t tool = tool || editor.content.currentNode.dataset.tool;\n\t var newBlockComposed = composeNewBlock_(newBlock, tool);\n\t\n\t /** Replacing */\n\t editor.content.replaceBlock(blockToReplace, newBlockComposed);\n\t\n\t /** Save new Inputs when block is changed */\n\t editor.ui.saveInputs();\n\t };\n\t\n\t /**\r\n\t * Iterates between child noted and looking for #text node on deepest level\r\n\t * @protected\r\n\t *\r\n\t * @param {Element} block - node where find\r\n\t * @param {int} postiton - starting postion\r\n\t * Example: childNodex.length to find from the end\r\n\t * or 0 to find from the start\r\n\t * @return {Text} block\r\n\t * @uses DFS\r\n\t */\n\t content.getDeepestTextNodeFromPosition = function (block, position) {\n\t\n\t /**\r\n\t * Clear Block from empty and useless spaces with trim.\r\n\t * Such nodes we should remove\r\n\t */\n\t var blockChilds = block.childNodes,\n\t index,\n\t node,\n\t text;\n\t\n\t for (index = 0; index < blockChilds.length; index++) {\n\t\n\t node = blockChilds[index];\n\t\n\t if (node.nodeType == editor.core.nodeTypes.TEXT) {\n\t\n\t text = node.textContent.trim();\n\t\n\t /** Text is empty. We should remove this child from node before we start DFS\r\n\t * decrease the quantity of childs.\r\n\t */\n\t if (text === '') {\n\t\n\t block.removeChild(node);\n\t position--;\n\t }\n\t }\n\t }\n\t\n\t if (block.childNodes.length === 0) {\n\t\n\t return document.createTextNode('');\n\t }\n\t\n\t /** Setting default position when we deleted all empty nodes */\n\t if (position < 0) position = 1;\n\t\n\t var lookingFromStart = false;\n\t\n\t /** For looking from START */\n\t if (position === 0) {\n\t\n\t lookingFromStart = true;\n\t position = 1;\n\t }\n\t\n\t while (position) {\n\t\n\t /** initial verticle of node. */\n\t if (lookingFromStart) {\n\t\n\t block = block.childNodes[0];\n\t } else {\n\t\n\t block = block.childNodes[position - 1];\n\t }\n\t\n\t if (block.nodeType == editor.core.nodeTypes.TAG) {\n\t\n\t position = block.childNodes.length;\n\t } else if (block.nodeType == editor.core.nodeTypes.TEXT) {\n\t\n\t position = 0;\n\t }\n\t }\n\t\n\t return block;\n\t };\n\t\n\t /**\r\n\t * @private\r\n\t * @param {Element} block - current plugins render\r\n\t * @param {String} tool - plugins name\r\n\t * @param {Boolean} isStretched - make stretched block or not\r\n\t *\r\n\t * @description adds necessary information to wrap new created block by first-level holder\r\n\t */\n\t var composeNewBlock_ = function composeNewBlock_(block, tool, isStretched) {\n\t\n\t var newBlock = editor.draw.node('DIV', editor.ui.className.BLOCK_CLASSNAME, {}),\n\t blockContent = editor.draw.node('DIV', editor.ui.className.BLOCK_CONTENT, {});\n\t\n\t blockContent.appendChild(block);\n\t newBlock.appendChild(blockContent);\n\t\n\t if (isStretched) {\n\t\n\t blockContent.classList.add(editor.ui.className.BLOCK_STRETCHED);\n\t }\n\t\n\t newBlock.dataset.tool = tool;\n\t return newBlock;\n\t };\n\t\n\t /**\r\n\t * Returns Range object of current selection\r\n\t * @protected\r\n\t */\n\t content.getRange = function () {\n\t\n\t var selection = window.getSelection().getRangeAt(0);\n\t\n\t return selection;\n\t };\n\t\n\t /**\r\n\t * Divides block in two blocks (after and before caret)\r\n\t *\r\n\t * @protected\r\n\t * @param {int} inputIndex - target input index\r\n\t *\r\n\t * @description splits current input content to the separate blocks\r\n\t * When enter is pressed among the words, that text will be splited.\r\n\t */\n\t content.splitBlock = function (inputIndex) {\n\t\n\t var selection = window.getSelection(),\n\t anchorNode = selection.anchorNode,\n\t anchorNodeText = anchorNode.textContent,\n\t caretOffset = selection.anchorOffset,\n\t textBeforeCaret,\n\t textNodeBeforeCaret,\n\t textAfterCaret,\n\t textNodeAfterCaret;\n\t\n\t var currentBlock = editor.content.currentNode.querySelector('[contentEditable]');\n\t\n\t textBeforeCaret = anchorNodeText.substring(0, caretOffset);\n\t textAfterCaret = anchorNodeText.substring(caretOffset);\n\t\n\t textNodeBeforeCaret = document.createTextNode(textBeforeCaret);\n\t\n\t if (textAfterCaret) {\n\t\n\t textNodeAfterCaret = document.createTextNode(textAfterCaret);\n\t }\n\t\n\t var previousChilds = [],\n\t nextChilds = [],\n\t reachedCurrent = false;\n\t\n\t if (textNodeAfterCaret) {\n\t\n\t nextChilds.push(textNodeAfterCaret);\n\t }\n\t\n\t for (var i = 0, child; !!(child = currentBlock.childNodes[i]); i++) {\n\t\n\t if (child != anchorNode) {\n\t\n\t if (!reachedCurrent) {\n\t\n\t previousChilds.push(child);\n\t } else {\n\t\n\t nextChilds.push(child);\n\t }\n\t } else {\n\t\n\t reachedCurrent = true;\n\t }\n\t }\n\t\n\t /** Clear current input */\n\t editor.state.inputs[inputIndex].innerHTML = '';\n\t\n\t /**\r\n\t * Append all childs founded before anchorNode\r\n\t */\n\t var previousChildsLength = previousChilds.length;\n\t\n\t for (i = 0; i < previousChildsLength; i++) {\n\t\n\t editor.state.inputs[inputIndex].appendChild(previousChilds[i]);\n\t }\n\t\n\t editor.state.inputs[inputIndex].appendChild(textNodeBeforeCaret);\n\t\n\t /**\r\n\t * Append text node which is after caret\r\n\t */\n\t var nextChildsLength = nextChilds.length,\n\t newNode = document.createElement('div');\n\t\n\t for (i = 0; i < nextChildsLength; i++) {\n\t\n\t newNode.appendChild(nextChilds[i]);\n\t }\n\t\n\t newNode = newNode.innerHTML;\n\t\n\t /** This type of block creates when enter is pressed */\n\t var NEW_BLOCK_TYPE = editor.settings.initialBlockPlugin;\n\t\n\t /**\r\n\t * Make new paragraph with text after caret\r\n\t */\n\t editor.content.insertBlock({\n\t type: NEW_BLOCK_TYPE,\n\t block: editor.tools[NEW_BLOCK_TYPE].render({\n\t text: newNode\n\t })\n\t }, true);\n\t };\n\t\n\t /**\r\n\t * Merges two blocks — current and target\r\n\t * If target index is not exist, then previous will be as target\r\n\t *\r\n\t * @protected\r\n\t * @param {int} currentInputIndex\r\n\t * @param {int} targetInputIndex\r\n\t *\r\n\t * @description gets two inputs indexes and merges into one\r\n\t */\n\t content.mergeBlocks = function (currentInputIndex, targetInputIndex) {\n\t\n\t /** If current input index is zero, then prevent method execution */\n\t if (currentInputIndex === 0) {\n\t\n\t return;\n\t }\n\t\n\t var targetInput,\n\t currentInputContent = editor.state.inputs[currentInputIndex].innerHTML;\n\t\n\t if (!targetInputIndex) {\n\t\n\t targetInput = editor.state.inputs[currentInputIndex - 1];\n\t } else {\n\t\n\t targetInput = editor.state.inputs[targetInputIndex];\n\t }\n\t\n\t targetInput.innerHTML += currentInputContent;\n\t };\n\t\n\t /**\r\n\t * Iterates all right siblings and parents, which has right siblings\r\n\t * while it does not reached the first-level block\r\n\t *\r\n\t * @param {Element} node\r\n\t * @return {boolean}\r\n\t */\n\t content.isLastNode = function (node) {\n\t\n\t // console.log('погнали перебор родителей');\n\t\n\t var allChecked = false;\n\t\n\t while (!allChecked) {\n\t\n\t // console.log('Смотрим на %o', node);\n\t // console.log('Проверим, пустые ли соседи справа');\n\t\n\t if (!allSiblingsEmpty_(node)) {\n\t\n\t // console.log('Есть непустые соседи. Узел не последний. Выходим.');\n\t return false;\n\t }\n\t\n\t node = node.parentNode;\n\t\n\t /**\r\n\t * Проверяем родителей до тех пор, пока не найдем блок первого уровня\r\n\t */\n\t if (node.classList.contains(editor.ui.className.BLOCK_CONTENT)) {\n\t\n\t allChecked = true;\n\t }\n\t }\n\t\n\t return true;\n\t };\n\t\n\t /**\r\n\t * Checks if all element right siblings is empty\r\n\t * @param node\r\n\t */\n\t var allSiblingsEmpty_ = function allSiblingsEmpty_(node) {\n\t\n\t /**\r\n\t * Нужно убедиться, что после пустого соседа ничего нет\r\n\t */\n\t var sibling = node.nextSibling;\n\t\n\t while (sibling) {\n\t\n\t if (sibling.textContent.length) {\n\t\n\t return false;\n\t }\n\t\n\t sibling = sibling.nextSibling;\n\t }\n\t\n\t return true;\n\t };\n\t\n\t /**\r\n\t * @public\r\n\t *\r\n\t * @param {string} htmlData - html content as string\r\n\t * @param {string} plainData - plain text\r\n\t * @return {string} - html content as string\r\n\t */\n\t content.wrapTextWithParagraphs = function (htmlData, plainData) {\n\t\n\t if (!htmlData) {\n\t\n\t return '

    ' + plainData.split('\\n\\n').join('

    ') + '

    ';\n\t }\n\t\n\t var wrapper = document.createElement('DIV'),\n\t newWrapper = document.createElement('DIV'),\n\t i,\n\t paragraph,\n\t firstLevelBlocks = ['DIV', 'P'],\n\t blockTyped,\n\t node;\n\t\n\t /**\r\n\t * Make HTML Element to Wrap Text\r\n\t * It allows us to work with input data as HTML content\r\n\t */\n\t wrapper.innerHTML = htmlData;\n\t paragraph = document.createElement('P');\n\t\n\t for (i = 0; i < wrapper.childNodes.length; i++) {\n\t\n\t node = wrapper.childNodes[i];\n\t\n\t blockTyped = firstLevelBlocks.indexOf(node.tagName) != -1;\n\t\n\t /**\r\n\t * If node is first-levet\r\n\t * we add this node to our new wrapper\r\n\t */\n\t if (blockTyped) {\n\t\n\t /**\r\n\t * If we had splitted inline nodes to paragraph before\r\n\t */\n\t if (paragraph.childNodes.length) {\n\t\n\t newWrapper.appendChild(paragraph.cloneNode(true));\n\t\n\t /** empty paragraph */\n\t paragraph = null;\n\t paragraph = document.createElement('P');\n\t }\n\t\n\t newWrapper.appendChild(node.cloneNode(true));\n\t } else {\n\t\n\t /** Collect all inline nodes to one as paragraph */\n\t paragraph.appendChild(node.cloneNode(true));\n\t\n\t /** if node is last we should append this node to paragraph and paragraph to new wrapper */\n\t if (i == wrapper.childNodes.length - 1) {\n\t\n\t newWrapper.appendChild(paragraph.cloneNode(true));\n\t }\n\t }\n\t }\n\t\n\t return newWrapper.innerHTML;\n\t };\n\t\n\t /**\r\n\t * Finds closest Contenteditable parent from Element\r\n\t * @param {Element} node element looking from\r\n\t * @return {Element} node contenteditable\r\n\t */\n\t content.getEditableParent = function (node) {\n\t\n\t while (node && node.contentEditable != 'true') {\n\t\n\t node = node.parentNode;\n\t }\n\t\n\t return node;\n\t };\n\t\n\t return content;\n\t}({});\n\n/***/ },\n/* 8 */\n/***/ function(module, exports, __webpack_require__) {\n\n\t'use strict';\n\t\n\t/**\r\n\t * Codex Editor toolbar module\r\n\t *\r\n\t * Contains:\r\n\t * - Inline toolbox\r\n\t * - Toolbox within plus button\r\n\t * - Settings section\r\n\t *\r\n\t * @author Codex Team\r\n\t * @version 1.0\r\n\t */\n\t\n\tmodule.exports = function (toolbar) {\n\t\n\t var editor = codex.editor;\n\t\n\t toolbar.settings = __webpack_require__(9);\n\t toolbar.inline = __webpack_require__(10);\n\t toolbar.toolbox = __webpack_require__(11);\n\t\n\t /**\r\n\t * Margin between focused node and toolbar\r\n\t */\n\t toolbar.defaultToolbarHeight = 49;\n\t\n\t toolbar.defaultOffset = 34;\n\t\n\t toolbar.opened = false;\n\t\n\t toolbar.current = null;\n\t\n\t /**\r\n\t * @protected\r\n\t */\n\t toolbar.open = function () {\n\t\n\t if (editor.hideToolbar) {\n\t\n\t return;\n\t }\n\t\n\t editor.nodes.toolbar.classList.add('opened');\n\t this.opened = true;\n\t };\n\t\n\t /**\r\n\t * @protected\r\n\t */\n\t toolbar.close = function () {\n\t\n\t editor.nodes.toolbar.classList.remove('opened');\n\t\n\t toolbar.opened = false;\n\t toolbar.current = null;\n\t\n\t for (var button in editor.nodes.toolbarButtons) {\n\t\n\t editor.nodes.toolbarButtons[button].classList.remove('selected');\n\t }\n\t\n\t /** Close toolbox when toolbar is not displayed */\n\t editor.toolbar.toolbox.close();\n\t editor.toolbar.settings.close();\n\t };\n\t\n\t toolbar.toggle = function () {\n\t\n\t if (!this.opened) {\n\t\n\t this.open();\n\t } else {\n\t\n\t this.close();\n\t }\n\t };\n\t\n\t toolbar.hidePlusButton = function () {\n\t\n\t editor.nodes.plusButton.classList.add('hide');\n\t };\n\t\n\t toolbar.showPlusButton = function () {\n\t\n\t editor.nodes.plusButton.classList.remove('hide');\n\t };\n\t\n\t /**\r\n\t * Moving toolbar to the specified node\r\n\t */\n\t toolbar.move = function () {\n\t\n\t /** Close Toolbox when we move toolbar */\n\t editor.toolbar.toolbox.close();\n\t\n\t if (!editor.content.currentNode) {\n\t\n\t return;\n\t }\n\t\n\t var newYCoordinate = editor.content.currentNode.offsetTop - editor.toolbar.defaultToolbarHeight / 2 + editor.toolbar.defaultOffset;\n\t\n\t editor.nodes.toolbar.style.transform = 'translate3D(0, ' + Math.floor(newYCoordinate) + 'px, 0)';\n\t\n\t /** Close trash actions */\n\t editor.toolbar.settings.hideRemoveActions();\n\t };\n\t\n\t return toolbar;\n\t}({});\n\n/***/ },\n/* 9 */\n/***/ function(module, exports) {\n\n\t'use strict';\n\t\n\t/**\r\n\t * Toolbar settings\r\n\t *\r\n\t * @version 1.0.4\r\n\t */\n\t\n\tmodule.exports = function (settings) {\n\t\n\t var editor = codex.editor;\n\t\n\t settings.opened = false;\n\t\n\t settings.setting = null;\n\t settings.actions = null;\n\t\n\t /**\r\n\t * Append and open settings\r\n\t */\n\t settings.open = function (toolType) {\n\t\n\t /**\r\n\t * Append settings content\r\n\t * It's stored in tool.settings\r\n\t */\n\t if (!editor.tools[toolType] || !editor.tools[toolType].makeSettings) {\n\t\n\t editor.core.log('Plugin \\xAB' + toolType + '\\xBB has no settings', 'warn');\n\t editor.nodes.pluginSettings.innerHTML = '\\u041F\\u043B\\u0430\\u0433\\u0438\\u043D \\xAB' + toolType + '\\xBB \\u043D\\u0435 \\u0438\\u043C\\u0435\\u0435\\u0442 \\u043D\\u0430\\u0441\\u0442\\u0440\\u043E\\u0435\\u043A';\n\t } else {\n\t\n\t /**\r\n\t * Draw settings block\r\n\t */\n\t var settingsBlock = editor.tools[toolType].makeSettings();\n\t\n\t editor.nodes.pluginSettings.appendChild(settingsBlock);\n\t }\n\t\n\t /** Open settings block */\n\t editor.nodes.blockSettings.classList.add('opened');\n\t this.opened = true;\n\t };\n\t\n\t /**\r\n\t * Close and clear settings\r\n\t */\n\t settings.close = function () {\n\t\n\t editor.nodes.blockSettings.classList.remove('opened');\n\t editor.nodes.pluginSettings.innerHTML = '';\n\t\n\t this.opened = false;\n\t };\n\t\n\t /**\r\n\t * @param {string} toolType - plugin type\r\n\t */\n\t settings.toggle = function (toolType) {\n\t\n\t if (!this.opened) {\n\t\n\t this.open(toolType);\n\t } else {\n\t\n\t this.close();\n\t }\n\t };\n\t\n\t /**\r\n\t * Here we will draw buttons and add listeners to components\r\n\t */\n\t settings.makeRemoveBlockButton = function () {\n\t\n\t var removeBlockWrapper = editor.draw.node('SPAN', 'ce-toolbar__remove-btn', {}),\n\t settingButton = editor.draw.node('SPAN', 'ce-toolbar__remove-setting', { innerHTML: '' }),\n\t actionWrapper = editor.draw.node('DIV', 'ce-toolbar__remove-confirmation', {}),\n\t confirmAction = editor.draw.node('DIV', 'ce-toolbar__remove-confirm', { textContent: 'Удалить блок' }),\n\t cancelAction = editor.draw.node('DIV', 'ce-toolbar__remove-cancel', { textContent: 'Отмена' });\n\t\n\t editor.listeners.add(settingButton, 'click', editor.toolbar.settings.removeButtonClicked, false);\n\t\n\t editor.listeners.add(confirmAction, 'click', editor.toolbar.settings.confirmRemovingRequest, false);\n\t\n\t editor.listeners.add(cancelAction, 'click', editor.toolbar.settings.cancelRemovingRequest, false);\n\t\n\t actionWrapper.appendChild(confirmAction);\n\t actionWrapper.appendChild(cancelAction);\n\t\n\t removeBlockWrapper.appendChild(settingButton);\n\t removeBlockWrapper.appendChild(actionWrapper);\n\t\n\t /** Save setting */\n\t editor.toolbar.settings.setting = settingButton;\n\t editor.toolbar.settings.actions = actionWrapper;\n\t\n\t return removeBlockWrapper;\n\t };\n\t\n\t settings.removeButtonClicked = function () {\n\t\n\t var action = editor.toolbar.settings.actions;\n\t\n\t if (action.classList.contains('opened')) {\n\t\n\t editor.toolbar.settings.hideRemoveActions();\n\t } else {\n\t\n\t editor.toolbar.settings.showRemoveActions();\n\t }\n\t\n\t editor.toolbar.toolbox.close();\n\t editor.toolbar.settings.close();\n\t };\n\t\n\t settings.cancelRemovingRequest = function () {\n\t\n\t editor.toolbar.settings.actions.classList.remove('opened');\n\t };\n\t\n\t settings.confirmRemovingRequest = function () {\n\t\n\t var currentBlock = editor.content.currentNode,\n\t firstLevelBlocksCount;\n\t\n\t currentBlock.remove();\n\t\n\t firstLevelBlocksCount = editor.nodes.redactor.childNodes.length;\n\t\n\t /**\r\n\t * If all blocks are removed\r\n\t */\n\t if (firstLevelBlocksCount === 0) {\n\t\n\t /** update currentNode variable */\n\t editor.content.currentNode = null;\n\t\n\t /** Inserting new empty initial block */\n\t editor.ui.addInitialBlock();\n\t }\n\t\n\t editor.ui.saveInputs();\n\t\n\t editor.toolbar.close();\n\t };\n\t\n\t settings.showRemoveActions = function () {\n\t\n\t editor.toolbar.settings.actions.classList.add('opened');\n\t };\n\t\n\t settings.hideRemoveActions = function () {\n\t\n\t editor.toolbar.settings.actions.classList.remove('opened');\n\t };\n\t\n\t return settings;\n\t}({});\n\n/***/ },\n/* 10 */\n/***/ function(module, exports) {\n\n\t'use strict';\n\t\n\t/**\r\n\t * Inline toolbar\r\n\t *\r\n\t * Contains from tools:\r\n\t * Bold, Italic, Underline and Anchor\r\n\t *\r\n\t * @author Codex Team\r\n\t * @version 1.0\r\n\t */\n\t\n\tmodule.exports = function (inline) {\n\t\n\t var editor = codex.editor;\n\t\n\t inline.buttonsOpened = null;\n\t inline.actionsOpened = null;\n\t inline.wrappersOffset = null;\n\t\n\t /**\r\n\t * saving selection that need for execCommand for styling\r\n\t *\r\n\t */\n\t inline.storedSelection = null;\n\t\n\t /**\r\n\t * @protected\r\n\t *\r\n\t * Open inline toobar\r\n\t */\n\t inline.show = function () {\n\t\n\t var currentNode = editor.content.currentNode,\n\t tool = currentNode.dataset.tool,\n\t plugin;\n\t\n\t /**\r\n\t * tool allowed to open inline toolbar\r\n\t */\n\t plugin = editor.tools[tool];\n\t\n\t if (!plugin.showInlineToolbar) return;\n\t\n\t var selectedText = inline.getSelectionText(),\n\t toolbar = editor.nodes.inlineToolbar.wrapper;\n\t\n\t if (selectedText.length > 0) {\n\t\n\t /** Move toolbar and open */\n\t editor.toolbar.inline.move();\n\t\n\t /** Open inline toolbar */\n\t toolbar.classList.add('opened');\n\t\n\t /** show buttons of inline toolbar */\n\t editor.toolbar.inline.showButtons();\n\t }\n\t };\n\t\n\t /**\r\n\t * @protected\r\n\t *\r\n\t * Closes inline toolbar\r\n\t */\n\t inline.close = function () {\n\t\n\t var toolbar = editor.nodes.inlineToolbar.wrapper;\n\t\n\t toolbar.classList.remove('opened');\n\t };\n\t\n\t /**\r\n\t * @private\r\n\t *\r\n\t * Moving toolbar\r\n\t */\n\t inline.move = function () {\n\t\n\t if (!this.wrappersOffset) {\n\t\n\t this.wrappersOffset = this.getWrappersOffset();\n\t }\n\t\n\t var coords = this.getSelectionCoords(),\n\t defaultOffset = 0,\n\t toolbar = editor.nodes.inlineToolbar.wrapper,\n\t newCoordinateX,\n\t newCoordinateY;\n\t\n\t if (toolbar.offsetHeight === 0) {\n\t\n\t defaultOffset = 40;\n\t }\n\t\n\t newCoordinateX = coords.x - this.wrappersOffset.left;\n\t newCoordinateY = coords.y + window.scrollY - this.wrappersOffset.top - defaultOffset - toolbar.offsetHeight;\n\t\n\t toolbar.style.transform = 'translate3D(' + Math.floor(newCoordinateX) + 'px, ' + Math.floor(newCoordinateY) + 'px, 0)';\n\t\n\t /** Close everything */\n\t editor.toolbar.inline.closeButtons();\n\t editor.toolbar.inline.closeAction();\n\t };\n\t\n\t /**\r\n\t * @private\r\n\t *\r\n\t * Tool Clicked\r\n\t */\n\t\n\t inline.toolClicked = function (event, type) {\n\t\n\t /**\r\n\t * For simple tools we use default browser function\r\n\t * For more complicated tools, we should write our own behavior\r\n\t */\n\t switch (type) {\n\t case 'createLink':\n\t editor.toolbar.inline.createLinkAction(event, type);break;\n\t default:\n\t editor.toolbar.inline.defaultToolAction(type);break;\n\t }\n\t\n\t /**\r\n\t * highlight buttons\r\n\t * after making some action\r\n\t */\n\t editor.nodes.inlineToolbar.buttons.childNodes.forEach(editor.toolbar.inline.hightlight);\n\t };\n\t\n\t /**\r\n\t * @private\r\n\t *\r\n\t * Saving wrappers offset in DOM\r\n\t */\n\t inline.getWrappersOffset = function () {\n\t\n\t var wrapper = editor.nodes.wrapper,\n\t offset = this.getOffset(wrapper);\n\t\n\t this.wrappersOffset = offset;\n\t return offset;\n\t };\n\t\n\t /**\r\n\t * @private\r\n\t *\r\n\t * Calculates offset of DOM element\r\n\t *\r\n\t * @param el\r\n\t * @returns {{top: number, left: number}}\r\n\t */\n\t inline.getOffset = function (el) {\n\t\n\t var _x = 0;\n\t var _y = 0;\n\t\n\t while (el && !isNaN(el.offsetLeft) && !isNaN(el.offsetTop)) {\n\t\n\t _x += el.offsetLeft + el.clientLeft;\n\t _y += el.offsetTop + el.clientTop;\n\t el = el.offsetParent;\n\t }\n\t return { top: _y, left: _x };\n\t };\n\t\n\t /**\r\n\t * @private\r\n\t *\r\n\t * Calculates position of selected text\r\n\t * @returns {{x: number, y: number}}\r\n\t */\n\t inline.getSelectionCoords = function () {\n\t\n\t var sel = document.selection,\n\t range;\n\t var x = 0,\n\t y = 0;\n\t\n\t if (sel) {\n\t\n\t if (sel.type != 'Control') {\n\t\n\t range = sel.createRange();\n\t range.collapse(true);\n\t x = range.boundingLeft;\n\t y = range.boundingTop;\n\t }\n\t } else if (window.getSelection) {\n\t\n\t sel = window.getSelection();\n\t\n\t if (sel.rangeCount) {\n\t\n\t range = sel.getRangeAt(0).cloneRange();\n\t if (range.getClientRects) {\n\t\n\t range.collapse(true);\n\t var rect = range.getClientRects()[0];\n\t\n\t if (!rect) {\n\t\n\t return;\n\t }\n\t\n\t x = rect.left;\n\t y = rect.top;\n\t }\n\t }\n\t }\n\t return { x: x, y: y };\n\t };\n\t\n\t /**\r\n\t * @private\r\n\t *\r\n\t * Returns selected text as String\r\n\t * @returns {string}\r\n\t */\n\t inline.getSelectionText = function () {\n\t\n\t var selectedText = '';\n\t\n\t // all modern browsers and IE9+\n\t if (window.getSelection) {\n\t\n\t selectedText = window.getSelection().toString();\n\t }\n\t\n\t return selectedText;\n\t };\n\t\n\t /** Opens buttons block */\n\t inline.showButtons = function () {\n\t\n\t var buttons = editor.nodes.inlineToolbar.buttons;\n\t\n\t buttons.classList.add('opened');\n\t\n\t editor.toolbar.inline.buttonsOpened = true;\n\t\n\t /** highlight buttons */\n\t editor.nodes.inlineToolbar.buttons.childNodes.forEach(editor.toolbar.inline.hightlight);\n\t };\n\t\n\t /** Makes buttons disappear */\n\t inline.closeButtons = function () {\n\t\n\t var buttons = editor.nodes.inlineToolbar.buttons;\n\t\n\t buttons.classList.remove('opened');\n\t\n\t editor.toolbar.inline.buttonsOpened = false;\n\t };\n\t\n\t /** Open buttons defined action if exist */\n\t inline.showActions = function () {\n\t\n\t var action = editor.nodes.inlineToolbar.actions;\n\t\n\t action.classList.add('opened');\n\t\n\t editor.toolbar.inline.actionsOpened = true;\n\t };\n\t\n\t /** Close actions block */\n\t inline.closeAction = function () {\n\t\n\t var action = editor.nodes.inlineToolbar.actions;\n\t\n\t action.innerHTML = '';\n\t action.classList.remove('opened');\n\t editor.toolbar.inline.actionsOpened = false;\n\t };\n\t\n\t /**\r\n\t * Callback for keydowns in inline toolbar \"Insert link...\" input\r\n\t */\n\t var inlineToolbarAnchorInputKeydown_ = function inlineToolbarAnchorInputKeydown_(event) {\n\t\n\t if (event.keyCode != editor.core.keys.ENTER) {\n\t\n\t return;\n\t }\n\t\n\t var editable = editor.content.currentNode,\n\t storedSelection = editor.toolbar.inline.storedSelection;\n\t\n\t editor.toolbar.inline.restoreSelection(editable, storedSelection);\n\t editor.toolbar.inline.setAnchor(this.value);\n\t\n\t /**\r\n\t * Preventing events that will be able to happen\r\n\t */\n\t event.preventDefault();\n\t event.stopImmediatePropagation();\n\t\n\t editor.toolbar.inline.clearRange();\n\t };\n\t\n\t /** Action for link creation or for setting anchor */\n\t inline.createLinkAction = function (event) {\n\t\n\t var isActive = this.isLinkActive();\n\t\n\t var editable = editor.content.currentNode,\n\t storedSelection = editor.toolbar.inline.saveSelection(editable);\n\t\n\t /** Save globally selection */\n\t editor.toolbar.inline.storedSelection = storedSelection;\n\t\n\t if (isActive) {\n\t\n\t /**\r\n\t * Changing stored selection. if we want to remove anchor from word\r\n\t * we should remove anchor from whole word, not only selected part.\r\n\t * The solution is than we get the length of current link\r\n\t * Change start position to - end of selection minus length of anchor\r\n\t */\n\t editor.toolbar.inline.restoreSelection(editable, storedSelection);\n\t\n\t editor.toolbar.inline.defaultToolAction('unlink');\n\t } else {\n\t\n\t /** Create input and close buttons */\n\t var action = editor.draw.inputForLink();\n\t\n\t editor.nodes.inlineToolbar.actions.appendChild(action);\n\t\n\t editor.toolbar.inline.closeButtons();\n\t editor.toolbar.inline.showActions();\n\t\n\t /**\r\n\t * focus to input\r\n\t * Solution: https://developer.mozilla.org/ru/docs/Web/API/HTMLElement/focus\r\n\t * Prevents event after showing input and when we need to focus an input which is in unexisted form\r\n\t */\n\t action.focus();\n\t event.preventDefault();\n\t\n\t /** Callback to link action */\n\t editor.listeners.add(action, 'keydown', inlineToolbarAnchorInputKeydown_, false);\n\t }\n\t };\n\t\n\t inline.isLinkActive = function () {\n\t\n\t var isActive = false;\n\t\n\t editor.nodes.inlineToolbar.buttons.childNodes.forEach(function (tool) {\n\t\n\t var dataType = tool.dataset.type;\n\t\n\t if (dataType == 'link' && tool.classList.contains('hightlighted')) {\n\t\n\t isActive = true;\n\t }\n\t });\n\t\n\t return isActive;\n\t };\n\t\n\t /** default action behavior of tool */\n\t inline.defaultToolAction = function (type) {\n\t\n\t document.execCommand(type, false, null);\n\t };\n\t\n\t /**\r\n\t * @private\r\n\t *\r\n\t * Sets URL\r\n\t *\r\n\t * @param {String} url - URL\r\n\t */\n\t inline.setAnchor = function (url) {\n\t\n\t document.execCommand('createLink', false, url);\n\t\n\t /** Close after URL inserting */\n\t editor.toolbar.inline.closeAction();\n\t };\n\t\n\t /**\r\n\t * @private\r\n\t *\r\n\t * Saves selection\r\n\t */\n\t inline.saveSelection = function (containerEl) {\n\t\n\t var range = window.getSelection().getRangeAt(0),\n\t preSelectionRange = range.cloneRange(),\n\t start;\n\t\n\t preSelectionRange.selectNodeContents(containerEl);\n\t preSelectionRange.setEnd(range.startContainer, range.startOffset);\n\t\n\t start = preSelectionRange.toString().length;\n\t\n\t return {\n\t start: start,\n\t end: start + range.toString().length\n\t };\n\t };\n\t\n\t /**\r\n\t * @private\r\n\t *\r\n\t * Sets to previous selection (Range)\r\n\t *\r\n\t * @param {Element} containerEl - editable element where we restore range\r\n\t * @param {Object} savedSel - range basic information to restore\r\n\t */\n\t inline.restoreSelection = function (containerEl, savedSel) {\n\t\n\t var range = document.createRange(),\n\t charIndex = 0;\n\t\n\t range.setStart(containerEl, 0);\n\t range.collapse(true);\n\t\n\t var nodeStack = [containerEl],\n\t node,\n\t foundStart = false,\n\t stop = false,\n\t nextCharIndex;\n\t\n\t while (!stop && (node = nodeStack.pop())) {\n\t\n\t if (node.nodeType == 3) {\n\t\n\t nextCharIndex = charIndex + node.length;\n\t\n\t if (!foundStart && savedSel.start >= charIndex && savedSel.start <= nextCharIndex) {\n\t\n\t range.setStart(node, savedSel.start - charIndex);\n\t foundStart = true;\n\t }\n\t if (foundStart && savedSel.end >= charIndex && savedSel.end <= nextCharIndex) {\n\t\n\t range.setEnd(node, savedSel.end - charIndex);\n\t stop = true;\n\t }\n\t charIndex = nextCharIndex;\n\t } else {\n\t\n\t var i = node.childNodes.length;\n\t\n\t while (i--) {\n\t\n\t nodeStack.push(node.childNodes[i]);\n\t }\n\t }\n\t }\n\t\n\t var sel = window.getSelection();\n\t\n\t sel.removeAllRanges();\n\t sel.addRange(range);\n\t };\n\t\n\t /**\r\n\t * @private\r\n\t *\r\n\t * Removes all ranges from window selection\r\n\t */\n\t inline.clearRange = function () {\n\t\n\t var selection = window.getSelection();\n\t\n\t selection.removeAllRanges();\n\t };\n\t\n\t /**\r\n\t * @private\r\n\t *\r\n\t * sets or removes hightlight\r\n\t */\n\t inline.hightlight = function (tool) {\n\t\n\t var dataType = tool.dataset.type;\n\t\n\t if (document.queryCommandState(dataType)) {\n\t\n\t editor.toolbar.inline.setButtonHighlighted(tool);\n\t } else {\n\t\n\t editor.toolbar.inline.removeButtonsHighLight(tool);\n\t }\n\t\n\t /**\r\n\t *\r\n\t * hightlight for anchors\r\n\t */\n\t var selection = window.getSelection(),\n\t tag = selection.anchorNode.parentNode;\n\t\n\t if (tag.tagName == 'A' && dataType == 'link') {\n\t\n\t editor.toolbar.inline.setButtonHighlighted(tool);\n\t }\n\t };\n\t\n\t /**\r\n\t * @private\r\n\t *\r\n\t * Mark button if text is already executed\r\n\t */\n\t inline.setButtonHighlighted = function (button) {\n\t\n\t button.classList.add('hightlighted');\n\t\n\t /** At link tool we also change icon */\n\t if (button.dataset.type == 'link') {\n\t\n\t var icon = button.childNodes[0];\n\t\n\t icon.classList.remove('ce-icon-link');\n\t icon.classList.add('ce-icon-unlink');\n\t }\n\t };\n\t\n\t /**\r\n\t * @private\r\n\t *\r\n\t * Removes hightlight\r\n\t */\n\t inline.removeButtonsHighLight = function (button) {\n\t\n\t button.classList.remove('hightlighted');\n\t\n\t /** At link tool we also change icon */\n\t if (button.dataset.type == 'link') {\n\t\n\t var icon = button.childNodes[0];\n\t\n\t icon.classList.remove('ce-icon-unlink');\n\t icon.classList.add('ce-icon-link');\n\t }\n\t };\n\t\n\t return inline;\n\t}({});\n\n/***/ },\n/* 11 */\n/***/ function(module, exports) {\n\n\t'use strict';\n\t\n\t/**\r\n\t * Codex Editor toolbox\r\n\t *\r\n\t * All tools be able to appended here\r\n\t *\r\n\t * @author Codex Team\r\n\t * @version 1.0\r\n\t */\n\t\n\tmodule.exports = function (toolbox) {\n\t\n\t var editor = codex.editor;\n\t\n\t toolbox.opened = false;\n\t\n\t /** Shows toolbox */\n\t toolbox.open = function () {\n\t\n\t /** Close setting if toolbox is opened */\n\t if (editor.toolbar.settings.opened) {\n\t\n\t editor.toolbar.settings.close();\n\t }\n\t\n\t /** display toolbox */\n\t editor.nodes.toolbox.classList.add('opened');\n\t\n\t /** Animate plus button */\n\t editor.nodes.plusButton.classList.add('clicked');\n\t\n\t /** toolbox state */\n\t editor.toolbar.toolbox.opened = true;\n\t };\n\t\n\t /** Closes toolbox */\n\t toolbox.close = function () {\n\t\n\t /** Makes toolbox disapear */\n\t editor.nodes.toolbox.classList.remove('opened');\n\t\n\t /** Rotate plus button */\n\t editor.nodes.plusButton.classList.remove('clicked');\n\t\n\t /** toolbox state */\n\t editor.toolbar.toolbox.opened = false;\n\t\n\t editor.toolbar.current = null;\n\t };\n\t\n\t toolbox.leaf = function () {\n\t\n\t var currentTool = editor.toolbar.current,\n\t tools = Object.keys(editor.tools),\n\t barButtons = editor.nodes.toolbarButtons,\n\t nextToolIndex = 0,\n\t toolToSelect = void 0,\n\t visibleTool = void 0,\n\t tool = void 0;\n\t\n\t if (!currentTool) {\n\t\n\t /** Get first tool from object*/\n\t for (tool in editor.tools) {\n\t\n\t if (editor.tools[tool].displayInToolbox) {\n\t\n\t break;\n\t }\n\t\n\t nextToolIndex++;\n\t }\n\t } else {\n\t\n\t nextToolIndex = (tools.indexOf(currentTool) + 1) % tools.length;\n\t visibleTool = tools[nextToolIndex];\n\t\n\t while (!editor.tools[visibleTool].displayInToolbox) {\n\t\n\t nextToolIndex = (nextToolIndex + 1) % tools.length;\n\t visibleTool = tools[nextToolIndex];\n\t }\n\t }\n\t\n\t toolToSelect = tools[nextToolIndex];\n\t\n\t for (var button in barButtons) {\n\t\n\t barButtons[button].classList.remove('selected');\n\t }\n\t\n\t barButtons[toolToSelect].classList.add('selected');\n\t editor.toolbar.current = toolToSelect;\n\t };\n\t\n\t /**\r\n\t * Transforming selected node type into selected toolbar element type\r\n\t * @param {event} event\r\n\t */\n\t toolbox.toolClicked = function (event) {\n\t\n\t /**\r\n\t * UNREPLACEBLE_TOOLS this types of tools are forbidden to replace even they are empty\r\n\t */\n\t var UNREPLACEBLE_TOOLS = ['image', 'link', 'list', 'instagram', 'twitter', 'embed'],\n\t tool = editor.tools[editor.toolbar.current],\n\t workingNode = editor.content.currentNode,\n\t currentInputIndex = editor.caret.inputIndex,\n\t newBlockContent,\n\t appendCallback,\n\t blockData;\n\t\n\t /** Make block from plugin */\n\t newBlockContent = tool.render();\n\t\n\t /** information about block */\n\t blockData = {\n\t block: newBlockContent,\n\t type: tool.type,\n\t stretched: false\n\t };\n\t\n\t if (workingNode && UNREPLACEBLE_TOOLS.indexOf(workingNode.dataset.tool) === -1 && workingNode.textContent.trim() === '') {\n\t\n\t /** Replace current block */\n\t editor.content.switchBlock(workingNode, newBlockContent, tool.type);\n\t } else {\n\t\n\t /** Insert new Block from plugin */\n\t editor.content.insertBlock(blockData);\n\t\n\t /** increase input index */\n\t currentInputIndex++;\n\t }\n\t\n\t /** Fire tool append callback */\n\t appendCallback = tool.appendCallback;\n\t\n\t if (appendCallback && typeof appendCallback == 'function') {\n\t\n\t appendCallback.call(event);\n\t }\n\t\n\t window.setTimeout(function () {\n\t\n\t /** Set caret to current block */\n\t editor.caret.setToBlock(currentInputIndex);\n\t }, 10);\n\t\n\t /**\r\n\t * Changing current Node\r\n\t */\n\t editor.content.workingNodeChanged();\n\t\n\t /**\r\n\t * Move toolbar when node is changed\r\n\t */\n\t editor.toolbar.move();\n\t };\n\t\n\t return toolbox;\n\t}({});\n\n/***/ },\n/* 12 */\n/***/ function(module, exports) {\n\n\t'use strict';\n\t\n\t/**\r\n\t * @module Codex Editor Callbacks module\r\n\t * @description Module works with editor added Elements\r\n\t *\r\n\t * @author Codex Team\r\n\t * @version 1.4.0\r\n\t */\n\t\n\tmodule.exports = function (callbacks) {\n\t\n\t var editor = codex.editor;\n\t\n\t /**\r\n\t * used by UI module\r\n\t * @description Routes all keydowns on document\r\n\t * @param {Object} event\r\n\t */\n\t callbacks.globalKeydown = function (event) {\n\t\n\t switch (event.keyCode) {\n\t case editor.core.keys.ENTER:\n\t enterKeyPressed_(event);break;\n\t }\n\t };\n\t\n\t /**\r\n\t * used by UI module\r\n\t * @description Routes all keydowns on redactors area\r\n\t * @param {Object} event\r\n\t */\n\t callbacks.redactorKeyDown = function (event) {\n\t\n\t switch (event.keyCode) {\n\t case editor.core.keys.TAB:\n\t tabKeyPressedOnRedactorsZone_(event);break;\n\t case editor.core.keys.ENTER:\n\t enterKeyPressedOnRedactorsZone_(event);break;\n\t case editor.core.keys.ESC:\n\t escapeKeyPressedOnRedactorsZone_(event);break;\n\t default:\n\t defaultKeyPressedOnRedactorsZone_(event);break;\n\t }\n\t };\n\t\n\t /**\r\n\t * used by UI module\r\n\t * @description Routes all keyup events\r\n\t * @param {Object} event\r\n\t */\n\t callbacks.globalKeyup = function (event) {\n\t\n\t switch (event.keyCode) {\n\t case editor.core.keys.UP:\n\t case editor.core.keys.LEFT:\n\t case editor.core.keys.RIGHT:\n\t case editor.core.keys.DOWN:\n\t arrowKeyPressed_(event);break;\n\t }\n\t };\n\t\n\t /**\r\n\t * @param {Object} event\r\n\t * @private\r\n\t *\r\n\t * Handles behaviour when tab pressed\r\n\t * @description if Content is empty show toolbox (if it is closed) or leaf tools\r\n\t * uses Toolbars toolbox module to handle the situation\r\n\t */\n\t var tabKeyPressedOnRedactorsZone_ = function tabKeyPressedOnRedactorsZone_(event) {\n\t\n\t /**\r\n\t * Wait for solution. Would like to know the behaviour\r\n\t * @todo Add spaces\r\n\t */\n\t event.preventDefault();\n\t\n\t var nativeInputs = editor.content.currentNode.querySelectorAll('textarea, input'),\n\t nativeInputsAreEmpty = true,\n\t textContentIsEmpty = !editor.content.currentNode.textContent.trim();\n\t\n\t Array.prototype.map.call(nativeInputs, function (input) {\n\t\n\t if (input.type == 'textarea' || input.type == 'text') {\n\t\n\t nativeInputsAreEmpty = nativeInputsAreEmpty && !input.value.trim();\n\t }\n\t });\n\t\n\t var blockIsEmpty = textContentIsEmpty && nativeInputsAreEmpty;\n\t\n\t if (!blockIsEmpty) {\n\t\n\t return;\n\t }\n\t\n\t if (!editor.toolbar.opened) {\n\t\n\t editor.toolbar.open();\n\t }\n\t\n\t if (editor.toolbar.opened && !editor.toolbar.toolbox.opened) {\n\t\n\t editor.toolbar.toolbox.open();\n\t } else {\n\t\n\t editor.toolbar.toolbox.leaf();\n\t }\n\t };\n\t\n\t /**\r\n\t * Handles global EnterKey Press\r\n\t * @see enterPressedOnBlock_\r\n\t * @param {Object} event\r\n\t */\n\t var enterKeyPressed_ = function enterKeyPressed_() {\n\t\n\t if (editor.content.editorAreaHightlighted) {\n\t\n\t /**\r\n\t * it means that we lose input index, saved index before is not correct\r\n\t * therefore we need to set caret when we insert new block\r\n\t */\n\t editor.caret.inputIndex = -1;\n\t\n\t enterPressedOnBlock_();\n\t }\n\t };\n\t\n\t /**\r\n\t * Callback for enter key pressing in first-level block area\r\n\t *\r\n\t * @param {Event} event\r\n\t * @private\r\n\t *\r\n\t * @description Inserts new block with initial type from settings\r\n\t */\n\t var enterPressedOnBlock_ = function enterPressedOnBlock_() {\n\t\n\t var NEW_BLOCK_TYPE = editor.settings.initialBlockPlugin;\n\t\n\t editor.content.insertBlock({\n\t type: NEW_BLOCK_TYPE,\n\t block: editor.tools[NEW_BLOCK_TYPE].render()\n\t }, true);\n\t\n\t editor.toolbar.move();\n\t editor.toolbar.open();\n\t };\n\t\n\t /**\r\n\t * ENTER key handler\r\n\t *\r\n\t * @param {Object} event\r\n\t * @private\r\n\t *\r\n\t * @description Makes new block with initial type from settings\r\n\t */\n\t var enterKeyPressedOnRedactorsZone_ = function enterKeyPressedOnRedactorsZone_(event) {\n\t\n\t if (event.target.contentEditable == 'true') {\n\t\n\t /** Update input index */\n\t editor.caret.saveCurrentInputIndex();\n\t }\n\t\n\t var currentInputIndex = editor.caret.getCurrentInputIndex() || 0,\n\t workingNode = editor.content.currentNode,\n\t tool = workingNode.dataset.tool,\n\t isEnterPressedOnToolbar = editor.toolbar.opened && editor.toolbar.current && event.target == editor.state.inputs[currentInputIndex];\n\t\n\t /** The list of tools which needs the default browser behaviour */\n\t var enableLineBreaks = editor.tools[tool].enableLineBreaks;\n\t\n\t /** This type of block creates when enter is pressed */\n\t var NEW_BLOCK_TYPE = editor.settings.initialBlockPlugin;\n\t\n\t /**\r\n\t * When toolbar is opened, select tool instead of making new paragraph\r\n\t */\n\t if (isEnterPressedOnToolbar) {\n\t\n\t event.preventDefault();\n\t\n\t editor.toolbar.toolbox.toolClicked(event);\n\t\n\t editor.toolbar.close();\n\t\n\t /**\r\n\t * Stop other listeners callback executions\r\n\t */\n\t event.stopPropagation();\n\t event.stopImmediatePropagation();\n\t\n\t return;\n\t }\n\t\n\t /**\r\n\t * Allow paragraph lineBreaks with shift enter\r\n\t * Or if shiftkey pressed and enter and enabledLineBreaks, the let new block creation\r\n\t */\n\t if (event.shiftKey || enableLineBreaks) {\n\t\n\t event.stopPropagation();\n\t event.stopImmediatePropagation();\n\t return;\n\t }\n\t\n\t var currentSelection = window.getSelection(),\n\t currentSelectedNode = currentSelection.anchorNode,\n\t caretAtTheEndOfText = editor.caret.position.atTheEnd(),\n\t isTextNodeHasParentBetweenContenteditable = false;\n\t\n\t /**\r\n\t * Allow making new

    in same block by SHIFT+ENTER and forbids to prevent default browser behaviour\r\n\t */\n\t if (event.shiftKey && !enableLineBreaks) {\n\t\n\t editor.callback.enterPressedOnBlock(editor.content.currentBlock, event);\n\t event.preventDefault();\n\t return;\n\t }\n\t\n\t /**\r\n\t * Workaround situation when caret at the Text node that has some wrapper Elements\r\n\t * Split block cant handle this.\r\n\t * We need to save default behavior\r\n\t */\n\t isTextNodeHasParentBetweenContenteditable = currentSelectedNode && currentSelectedNode.parentNode.contentEditable != 'true';\n\t\n\t /**\r\n\t * Split blocks when input has several nodes and caret placed in textNode\r\n\t */\n\t if (currentSelectedNode.nodeType == editor.core.nodeTypes.TEXT && !isTextNodeHasParentBetweenContenteditable && !caretAtTheEndOfText) {\n\t\n\t event.preventDefault();\n\t\n\t editor.core.log('Splitting Text node...');\n\t\n\t editor.content.splitBlock(currentInputIndex);\n\t\n\t /** Show plus button when next input after split is empty*/\n\t if (!editor.state.inputs[currentInputIndex + 1].textContent.trim()) {\n\t\n\t editor.toolbar.showPlusButton();\n\t }\n\t } else {\n\t\n\t var islastNode = editor.content.isLastNode(currentSelectedNode);\n\t\n\t if (islastNode && caretAtTheEndOfText) {\n\t\n\t event.preventDefault();\n\t event.stopPropagation();\n\t event.stopImmediatePropagation();\n\t\n\t editor.core.log('ENTER clicked in last textNode. Create new BLOCK');\n\t\n\t editor.content.insertBlock({\n\t type: NEW_BLOCK_TYPE,\n\t block: editor.tools[NEW_BLOCK_TYPE].render()\n\t }, true);\n\t\n\t editor.toolbar.move();\n\t editor.toolbar.open();\n\t\n\t /** Show plus button with empty block */\n\t editor.toolbar.showPlusButton();\n\t }\n\t }\n\t\n\t /** get all inputs after new appending block */\n\t editor.ui.saveInputs();\n\t };\n\t\n\t /**\r\n\t * Escape behaviour\r\n\t * @param event\r\n\t * @private\r\n\t *\r\n\t * @description Closes toolbox and toolbar. Prevents default behaviour\r\n\t */\n\t var escapeKeyPressedOnRedactorsZone_ = function escapeKeyPressedOnRedactorsZone_(event) {\n\t\n\t /** Close all toolbar */\n\t editor.toolbar.close();\n\t\n\t /** Close toolbox */\n\t editor.toolbar.toolbox.close();\n\t\n\t event.preventDefault();\n\t };\n\t\n\t /**\r\n\t * @param {Event} event\r\n\t * @private\r\n\t *\r\n\t * closes and moves toolbar\r\n\t */\n\t var arrowKeyPressed_ = function arrowKeyPressed_(event) {\n\t\n\t editor.content.workingNodeChanged();\n\t\n\t /* Closing toolbar */\n\t editor.toolbar.close();\n\t editor.toolbar.move();\n\t };\n\t\n\t /**\r\n\t * @private\r\n\t * @param {Event} event\r\n\t *\r\n\t * @description Closes all opened bars from toolbar.\r\n\t * If block is mark, clears highlightning\r\n\t */\n\t var defaultKeyPressedOnRedactorsZone_ = function defaultKeyPressedOnRedactorsZone_() {\n\t\n\t editor.toolbar.close();\n\t\n\t if (!editor.toolbar.inline.actionsOpened) {\n\t\n\t editor.toolbar.inline.close();\n\t editor.content.clearMark();\n\t }\n\t };\n\t\n\t /**\r\n\t * Handler when clicked on redactors area\r\n\t *\r\n\t * @protected\r\n\t * @param event\r\n\t *\r\n\t * @description Detects clicked area. If it is first-level block area, marks as detected and\r\n\t * on next enter press will be inserted new block\r\n\t * Otherwise, save carets position (input index) and put caret to the editable zone.\r\n\t *\r\n\t * @see detectWhenClickedOnFirstLevelBlockArea_\r\n\t *\r\n\t */\n\t callbacks.redactorClicked = function (event) {\n\t\n\t detectWhenClickedOnFirstLevelBlockArea_();\n\t\n\t editor.content.workingNodeChanged(event.target);\n\t editor.ui.saveInputs();\n\t\n\t var selectedText = editor.toolbar.inline.getSelectionText(),\n\t firstLevelBlock;\n\t\n\t /** If selection range took off, then we hide inline toolbar */\n\t if (selectedText.length === 0) {\n\t\n\t editor.toolbar.inline.close();\n\t }\n\t\n\t /** Update current input index in memory when caret focused into existed input */\n\t if (event.target.contentEditable == 'true') {\n\t\n\t editor.caret.saveCurrentInputIndex();\n\t }\n\t\n\t if (editor.content.currentNode === null) {\n\t\n\t /**\r\n\t * If inputs in redactor does not exits, then we put input index 0 not -1\r\n\t */\n\t var indexOfLastInput = editor.state.inputs.length > 0 ? editor.state.inputs.length - 1 : 0;\n\t\n\t /** If we have any inputs */\n\t if (editor.state.inputs.length) {\n\t\n\t /** getting firstlevel parent of input */\n\t firstLevelBlock = editor.content.getFirstLevelBlock(editor.state.inputs[indexOfLastInput]);\n\t }\n\t\n\t /** If input is empty, then we set caret to the last input */\n\t if (editor.state.inputs.length && editor.state.inputs[indexOfLastInput].textContent === '' && firstLevelBlock.dataset.tool == editor.settings.initialBlockPlugin) {\n\t\n\t editor.caret.setToBlock(indexOfLastInput);\n\t } else {\n\t\n\t /** Create new input when caret clicked in redactors area */\n\t var NEW_BLOCK_TYPE = editor.settings.initialBlockPlugin;\n\t\n\t editor.content.insertBlock({\n\t type: NEW_BLOCK_TYPE,\n\t block: editor.tools[NEW_BLOCK_TYPE].render()\n\t });\n\t\n\t /** If there is no inputs except inserted */\n\t if (editor.state.inputs.length === 1) {\n\t\n\t editor.caret.setToBlock(indexOfLastInput);\n\t } else {\n\t\n\t /** Set caret to this appended input */\n\t editor.caret.setToNextBlock(indexOfLastInput);\n\t }\n\t }\n\t } else {\n\t\n\t /** Close all panels */\n\t editor.toolbar.settings.close();\n\t editor.toolbar.toolbox.close();\n\t }\n\t\n\t /**\r\n\t * Move toolbar and open\r\n\t */\n\t editor.toolbar.move();\n\t editor.toolbar.open();\n\t\n\t var inputIsEmpty = !editor.content.currentNode.textContent.trim(),\n\t currentNodeType = editor.content.currentNode.dataset.tool,\n\t isInitialType = currentNodeType == editor.settings.initialBlockPlugin;\n\t\n\t /** Hide plus buttons */\n\t editor.toolbar.hidePlusButton();\n\t\n\t if (!inputIsEmpty) {\n\t\n\t /** Mark current block */\n\t editor.content.markBlock();\n\t }\n\t\n\t if (isInitialType && inputIsEmpty) {\n\t\n\t /** Show plus button */\n\t editor.toolbar.showPlusButton();\n\t }\n\t };\n\t\n\t /**\r\n\t * This method allows to define, is caret in contenteditable element or not.\r\n\t *\r\n\t * @private\r\n\t *\r\n\t * @description Otherwise, if we get TEXT node from range container, that will means we have input index.\r\n\t * In this case we use default browsers behaviour (if plugin allows that) or overwritten action.\r\n\t * Therefore, to be sure that we've clicked first-level block area, we should have currentNode, which always\r\n\t * specifies to the first-level block. Other cases we just ignore.\r\n\t */\n\t var detectWhenClickedOnFirstLevelBlockArea_ = function detectWhenClickedOnFirstLevelBlockArea_() {\n\t\n\t var selection = window.getSelection(),\n\t anchorNode = selection.anchorNode,\n\t flag = false;\n\t\n\t if (selection.rangeCount === 0) {\n\t\n\t editor.content.editorAreaHightlighted = true;\n\t } else {\n\t\n\t if (!editor.core.isDomNode(anchorNode)) {\n\t\n\t anchorNode = anchorNode.parentNode;\n\t }\n\t\n\t /** Already founded, without loop */\n\t if (anchorNode.contentEditable == 'true') {\n\t\n\t flag = true;\n\t }\n\t\n\t while (anchorNode.contentEditable != 'true') {\n\t\n\t anchorNode = anchorNode.parentNode;\n\t\n\t if (anchorNode.contentEditable == 'true') {\n\t\n\t flag = true;\n\t }\n\t\n\t if (anchorNode == document.body) {\n\t\n\t break;\n\t }\n\t }\n\t\n\t /** If editable element founded, flag is \"TRUE\", Therefore we return \"FALSE\" */\n\t editor.content.editorAreaHightlighted = !flag;\n\t }\n\t };\n\t\n\t /**\r\n\t * Toolbar button click handler\r\n\t *\r\n\t * @param {Object} event - cursor to the button\r\n\t * @protected\r\n\t *\r\n\t * @description gets current tool and calls render method\r\n\t */\n\t callbacks.toolbarButtonClicked = function (event) {\n\t\n\t var button = this;\n\t\n\t editor.toolbar.current = button.dataset.type;\n\t\n\t editor.toolbar.toolbox.toolClicked(event);\n\t editor.toolbar.close();\n\t };\n\t\n\t /**\r\n\t * Show or Hide toolbox when plus button is clicked\r\n\t */\n\t callbacks.plusButtonClicked = function () {\n\t\n\t if (!editor.nodes.toolbox.classList.contains('opened')) {\n\t\n\t editor.toolbar.toolbox.open();\n\t } else {\n\t\n\t editor.toolbar.toolbox.close();\n\t }\n\t };\n\t\n\t /**\r\n\t * Block handlers for KeyDown events\r\n\t *\r\n\t * @protected\r\n\t * @param {Object} event\r\n\t *\r\n\t * Handles keydowns on block\r\n\t * @see blockRightOrDownArrowPressed_\r\n\t * @see backspacePressed_\r\n\t * @see blockLeftOrUpArrowPressed_\r\n\t */\n\t callbacks.blockKeydown = function (event) {\n\t\n\t var block = event.target; // event.target is input\n\t\n\t switch (event.keyCode) {\n\t\n\t case editor.core.keys.DOWN:\n\t case editor.core.keys.RIGHT:\n\t blockRightOrDownArrowPressed_(event);\n\t break;\n\t\n\t case editor.core.keys.BACKSPACE:\n\t backspacePressed_(block, event);\n\t break;\n\t\n\t case editor.core.keys.UP:\n\t case editor.core.keys.LEFT:\n\t blockLeftOrUpArrowPressed_(event);\n\t break;\n\t\n\t }\n\t };\n\t\n\t /**\r\n\t * RIGHT or DOWN keydowns on block\r\n\t *\r\n\t * @param {Object} event\r\n\t * @private\r\n\t *\r\n\t * @description watches the selection and gets closest editable element.\r\n\t * Uses method getDeepestTextNodeFromPosition to get the last node of next block\r\n\t * Sets caret if it is contenteditable\r\n\t */\n\t var blockRightOrDownArrowPressed_ = function blockRightOrDownArrowPressed_(event) {\n\t\n\t var selection = window.getSelection(),\n\t inputs = editor.state.inputs,\n\t focusedNode = selection.anchorNode,\n\t focusedNodeHolder;\n\t\n\t /** Check for caret existance */\n\t if (!focusedNode) {\n\t\n\t return false;\n\t }\n\t\n\t /** Looking for closest (parent) contentEditable element of focused node */\n\t while (focusedNode.contentEditable != 'true') {\n\t\n\t focusedNodeHolder = focusedNode.parentNode;\n\t focusedNode = focusedNodeHolder;\n\t }\n\t\n\t /** Input index in DOM level */\n\t var editableElementIndex = 0;\n\t\n\t while (focusedNode != inputs[editableElementIndex]) {\n\t\n\t editableElementIndex++;\n\t }\n\t\n\t /**\r\n\t * Founded contentEditable element doesn't have childs\r\n\t * Or maybe New created block\r\n\t */\n\t if (!focusedNode.textContent) {\n\t\n\t editor.caret.setToNextBlock(editableElementIndex);\n\t return;\n\t }\n\t\n\t /**\r\n\t * Do nothing when caret doesn not reaches the end of last child\r\n\t */\n\t var caretInLastChild = false,\n\t caretAtTheEndOfText = false;\n\t\n\t var lastChild, deepestTextnode;\n\t\n\t lastChild = focusedNode.childNodes[focusedNode.childNodes.length - 1];\n\t\n\t if (editor.core.isDomNode(lastChild)) {\n\t\n\t deepestTextnode = editor.content.getDeepestTextNodeFromPosition(lastChild, lastChild.childNodes.length);\n\t } else {\n\t\n\t deepestTextnode = lastChild;\n\t }\n\t\n\t caretInLastChild = selection.anchorNode == deepestTextnode;\n\t caretAtTheEndOfText = deepestTextnode.length == selection.anchorOffset;\n\t\n\t if (!caretInLastChild || !caretAtTheEndOfText) {\n\t\n\t editor.core.log('arrow [down|right] : caret does not reached the end');\n\t return false;\n\t }\n\t\n\t editor.caret.setToNextBlock(editableElementIndex);\n\t };\n\t\n\t /**\r\n\t * LEFT or UP keydowns on block\r\n\t *\r\n\t * @param {Object} event\r\n\t * @private\r\n\t *\r\n\t * watches the selection and gets closest editable element.\r\n\t * Uses method getDeepestTextNodeFromPosition to get the last node of previous block\r\n\t * Sets caret if it is contenteditable\r\n\t *\r\n\t */\n\t var blockLeftOrUpArrowPressed_ = function blockLeftOrUpArrowPressed_(event) {\n\t\n\t var selection = window.getSelection(),\n\t inputs = editor.state.inputs,\n\t focusedNode = selection.anchorNode,\n\t focusedNodeHolder;\n\t\n\t /** Check for caret existance */\n\t if (!focusedNode) {\n\t\n\t return false;\n\t }\n\t\n\t /**\r\n\t * LEFT or UP not at the beginning\r\n\t */\n\t if (selection.anchorOffset !== 0) {\n\t\n\t return false;\n\t }\n\t\n\t /** Looking for parent contentEditable block */\n\t while (focusedNode.contentEditable != 'true') {\n\t\n\t focusedNodeHolder = focusedNode.parentNode;\n\t focusedNode = focusedNodeHolder;\n\t }\n\t\n\t /** Input index in DOM level */\n\t var editableElementIndex = 0;\n\t\n\t while (focusedNode != inputs[editableElementIndex]) {\n\t\n\t editableElementIndex++;\n\t }\n\t\n\t /**\r\n\t * Do nothing if caret is not at the beginning of first child\r\n\t */\n\t var caretInFirstChild = false,\n\t caretAtTheBeginning = false;\n\t\n\t var firstChild, deepestTextnode;\n\t\n\t /**\r\n\t * Founded contentEditable element doesn't have childs\r\n\t * Or maybe New created block\r\n\t */\n\t if (!focusedNode.textContent) {\n\t\n\t editor.caret.setToPreviousBlock(editableElementIndex);\n\t return;\n\t }\n\t\n\t firstChild = focusedNode.childNodes[0];\n\t\n\t if (editor.core.isDomNode(firstChild)) {\n\t\n\t deepestTextnode = editor.content.getDeepestTextNodeFromPosition(firstChild, 0);\n\t } else {\n\t\n\t deepestTextnode = firstChild;\n\t }\n\t\n\t caretInFirstChild = selection.anchorNode == deepestTextnode;\n\t caretAtTheBeginning = selection.anchorOffset === 0;\n\t\n\t if (caretInFirstChild && caretAtTheBeginning) {\n\t\n\t editor.caret.setToPreviousBlock(editableElementIndex);\n\t }\n\t };\n\t\n\t /**\r\n\t * Handles backspace keydown\r\n\t *\r\n\t * @param {Element} block\r\n\t * @param {Object} event\r\n\t * @private\r\n\t *\r\n\t * @description if block is empty, delete the block and set caret to the previous block\r\n\t * If block is not empty, try to merge two blocks - current and previous\r\n\t * But it we try'n to remove first block, then we should set caret to the next block, not previous.\r\n\t * If we removed the last block, create new one\r\n\t */\n\t var backspacePressed_ = function backspacePressed_(block, event) {\n\t\n\t var currentInputIndex = editor.caret.getCurrentInputIndex(),\n\t range,\n\t selectionLength,\n\t firstLevelBlocksCount;\n\t\n\t if (isNativeInput_(event.target)) {\n\t\n\t /** If input value is empty - remove block */\n\t if (event.target.value.trim() == '') {\n\t\n\t block.remove();\n\t } else {\n\t\n\t return;\n\t }\n\t }\n\t\n\t if (block.textContent.trim()) {\n\t\n\t range = editor.content.getRange();\n\t selectionLength = range.endOffset - range.startOffset;\n\t\n\t if (editor.caret.position.atStart() && !selectionLength && editor.state.inputs[currentInputIndex - 1]) {\n\t\n\t editor.content.mergeBlocks(currentInputIndex);\n\t } else {\n\t\n\t return;\n\t }\n\t }\n\t\n\t if (!selectionLength) {\n\t\n\t block.remove();\n\t }\n\t\n\t firstLevelBlocksCount = editor.nodes.redactor.childNodes.length;\n\t\n\t /**\r\n\t * If all blocks are removed\r\n\t */\n\t if (firstLevelBlocksCount === 0) {\n\t\n\t /** update currentNode variable */\n\t editor.content.currentNode = null;\n\t\n\t /** Inserting new empty initial block */\n\t editor.ui.addInitialBlock();\n\t\n\t /** Updating inputs state after deleting last block */\n\t editor.ui.saveInputs();\n\t\n\t /** Set to current appended block */\n\t window.setTimeout(function () {\n\t\n\t editor.caret.setToPreviousBlock(1);\n\t }, 10);\n\t } else {\n\t\n\t if (editor.caret.inputIndex !== 0) {\n\t\n\t /** Target block is not first */\n\t editor.caret.setToPreviousBlock(editor.caret.inputIndex);\n\t } else {\n\t\n\t /** If we try to delete first block */\n\t editor.caret.setToNextBlock(editor.caret.inputIndex);\n\t }\n\t }\n\t\n\t editor.toolbar.move();\n\t\n\t if (!editor.toolbar.opened) {\n\t\n\t editor.toolbar.open();\n\t }\n\t\n\t /** Updating inputs state */\n\t editor.ui.saveInputs();\n\t\n\t /** Prevent default browser behaviour */\n\t event.preventDefault();\n\t };\n\t\n\t /**\r\n\t * This method prevents default behaviour.\r\n\t *\r\n\t * @param {Object} event\r\n\t * @protected\r\n\t *\r\n\t * @description We get from clipboard pasted data, sanitize, make a fragment that contains of this sanitized nodes.\r\n\t * Firstly, we need to memorize the caret position. We can do that by getting the range of selection.\r\n\t * After all, we insert clear fragment into caret placed position. Then, we should move the caret to the last node\r\n\t */\n\t callbacks.blockPasteCallback = function (event) {\n\t\n\t /** If area is input or textarea then allow default behaviour */\n\t if (isNativeInput_(event.target)) {\n\t\n\t return;\n\t }\n\t\n\t /** Prevent default behaviour */\n\t event.preventDefault();\n\t\n\t var editableParent = editor.content.getEditableParent(event.target);\n\t\n\t /** Allow paste when event target placed in Editable element */\n\t if (!editableParent) {\n\t\n\t return;\n\t }\n\t\n\t /** get html pasted data - dirty data */\n\t var htmlData = event.clipboardData.getData('text/html'),\n\t plainData = event.clipboardData.getData('text/plain');\n\t\n\t /** Temporary DIV that is used to work with childs as arrays item */\n\t var div = editor.draw.node('DIV', '', {}),\n\t cleanData,\n\t wrappedData;\n\t\n\t /** Create fragment, that we paste to range after proccesing */\n\t cleanData = editor.sanitizer.clean(htmlData);\n\t\n\t /**\r\n\t * We wrap pasted text with

    tags to split it logically\r\n\t *\r\n\t * @type {string}\r\n\t */\n\t wrappedData = editor.content.wrapTextWithParagraphs(cleanData, plainData);\n\t div.innerHTML = wrappedData;\n\t\n\t var NEW_BLOCK_TYPE = editor.settings.initialBlockPlugin,\n\t currentBlockContent = editor.content.currentNode.firstChild.firstChild;\n\t\n\t /**\r\n\t * If there only one paragraph, just insert it\r\n\t */\n\t if (div.childNodes.length == 1) {\n\t\n\t editor.caret.insertNode(document.createTextNode(div.firstChild.innerHTML));\n\t return;\n\t }\n\t\n\t div.childNodes.forEach(function (paragraph, index) {\n\t\n\t /**\r\n\t * If there was no data in working node, replace it with first paragraph of pasted text\r\n\t */\n\t if (index == 0 && currentBlockContent.innerHTML.trim() === '') {\n\t\n\t editor.content.switchBlock(editor.content.currentNode, editor.tools[NEW_BLOCK_TYPE].render({\n\t text: paragraph.innerHTML\n\t }), NEW_BLOCK_TYPE);\n\t\n\t return;\n\t }\n\t\n\t editor.content.insertBlock({\n\t type: NEW_BLOCK_TYPE,\n\t block: editor.tools[NEW_BLOCK_TYPE].render({\n\t text: paragraph.innerHTML\n\t })\n\t });\n\t\n\t editor.caret.inputIndex++;\n\t });\n\t\n\t editor.caret.setToPreviousBlock(editor.caret.getCurrentInputIndex() + 1);\n\t };\n\t\n\t /**\r\n\t * used by UI module\r\n\t * Clicks on block settings button\r\n\t *\r\n\t * @param {Object} event\r\n\t * @protected\r\n\t * @description Opens toolbar settings\r\n\t */\n\t callbacks.showSettingsButtonClicked = function (event) {\n\t\n\t /**\r\n\t * Get type of current block\r\n\t * It uses to append settings from tool.settings property.\r\n\t * ...\r\n\t * Type is stored in data-type attribute on block\r\n\t */\n\t var currentToolType = editor.content.currentNode.dataset.tool;\n\t\n\t editor.toolbar.settings.toggle(currentToolType);\n\t\n\t /** Close toolbox when settings button is active */\n\t editor.toolbar.toolbox.close();\n\t editor.toolbar.settings.hideRemoveActions();\n\t };\n\t\n\t /**\r\n\t * Check block\r\n\t * @param target\r\n\t * @private\r\n\t *\r\n\t * @description Checks target is it native input\r\n\t */\n\t var isNativeInput_ = function isNativeInput_(target) {\n\t\n\t var nativeInputAreas = ['INPUT', 'TEXTAREA'];\n\t\n\t return nativeInputAreas.indexOf(target.tagName) != -1;\n\t };\n\t\n\t return callbacks;\n\t}({});\n\n/***/ },\n/* 13 */\n/***/ function(module, exports) {\n\n\t'use strict';\n\t\n\t/**\r\n\t * Codex Editor Draw module\r\n\t *\r\n\t * @author Codex Team\r\n\t * @version 1.0.\r\n\t */\n\t\n\tmodule.exports = function (draw) {\n\t\n\t /**\r\n\t * Base editor wrapper\r\n\t */\n\t draw.wrapper = function () {\n\t\n\t var wrapper = document.createElement('div');\n\t\n\t wrapper.className += 'codex-editor';\n\t\n\t return wrapper;\n\t };\n\t\n\t /**\r\n\t * Content-editable holder\r\n\t */\n\t draw.redactor = function () {\n\t\n\t var redactor = document.createElement('div');\n\t\n\t redactor.className += 'ce-redactor';\n\t\n\t return redactor;\n\t };\n\t\n\t draw.ceBlock = function () {\n\t\n\t var block = document.createElement('DIV');\n\t\n\t block.className += 'ce_block';\n\t\n\t return block;\n\t };\n\t\n\t /**\r\n\t * Empty toolbar with toggler\r\n\t */\n\t draw.toolbar = function () {\n\t\n\t var bar = document.createElement('div');\n\t\n\t bar.className += 'ce-toolbar';\n\t\n\t return bar;\n\t };\n\t\n\t draw.toolbarContent = function () {\n\t\n\t var wrapper = document.createElement('DIV');\n\t\n\t wrapper.classList.add('ce-toolbar__content');\n\t\n\t return wrapper;\n\t };\n\t\n\t /**\r\n\t * Inline toolbar\r\n\t */\n\t draw.inlineToolbar = function () {\n\t\n\t var bar = document.createElement('DIV');\n\t\n\t bar.className += 'ce-toolbar-inline';\n\t\n\t return bar;\n\t };\n\t\n\t /**\r\n\t * Wrapper for inline toobar buttons\r\n\t */\n\t draw.inlineToolbarButtons = function () {\n\t\n\t var wrapper = document.createElement('DIV');\n\t\n\t wrapper.className += 'ce-toolbar-inline__buttons';\n\t\n\t return wrapper;\n\t };\n\t\n\t /**\r\n\t * For some actions\r\n\t */\n\t draw.inlineToolbarActions = function () {\n\t\n\t var wrapper = document.createElement('DIV');\n\t\n\t wrapper.className += 'ce-toolbar-inline__actions';\n\t\n\t return wrapper;\n\t };\n\t\n\t draw.inputForLink = function () {\n\t\n\t var input = document.createElement('INPUT');\n\t\n\t input.type = 'input';\n\t input.className += 'inputForLink';\n\t input.placeholder = 'Вставьте ссылку ...';\n\t input.setAttribute('form', 'defaultForm');\n\t\n\t input.setAttribute('autofocus', 'autofocus');\n\t\n\t return input;\n\t };\n\t\n\t /**\r\n\t * @todo Desc\r\n\t */\n\t draw.blockButtons = function () {\n\t\n\t var block = document.createElement('div');\n\t\n\t block.className += 'ce-toolbar__actions';\n\t\n\t return block;\n\t };\n\t\n\t /**\r\n\t * Block settings panel\r\n\t */\n\t draw.blockSettings = function () {\n\t\n\t var settings = document.createElement('div');\n\t\n\t settings.className += 'ce-settings';\n\t\n\t return settings;\n\t };\n\t\n\t draw.defaultSettings = function () {\n\t\n\t var div = document.createElement('div');\n\t\n\t div.classList.add('ce-settings_default');\n\t\n\t return div;\n\t };\n\t\n\t draw.pluginsSettings = function () {\n\t\n\t var div = document.createElement('div');\n\t\n\t div.classList.add('ce-settings_plugin');\n\t\n\t return div;\n\t };\n\t\n\t draw.plusButton = function () {\n\t\n\t var button = document.createElement('span');\n\t\n\t button.className = 'ce-toolbar__plus';\n\t // button.innerHTML = '';\n\t\n\t return button;\n\t };\n\t\n\t /**\r\n\t * Settings button in toolbar\r\n\t */\n\t draw.settingsButton = function () {\n\t\n\t var toggler = document.createElement('span');\n\t\n\t toggler.className = 'ce-toolbar__settings-btn';\n\t\n\t /** Toggler button*/\n\t toggler.innerHTML = '';\n\t\n\t return toggler;\n\t };\n\t\n\t /**\r\n\t * Redactor tools wrapper\r\n\t */\n\t\n\t draw.toolbox = function () {\n\t\n\t var wrapper = document.createElement('div');\n\t\n\t wrapper.className = 'ce-toolbar__tools';\n\t\n\t return wrapper;\n\t };\n\t\n\t /**\r\n\t * @protected\r\n\t *\r\n\t * Draws tool buttons for toolbox\r\n\t *\r\n\t * @param {String} type\r\n\t * @param {String} classname\r\n\t * @returns {Element}\r\n\t */\n\t draw.toolbarButton = function (type, classname) {\n\t\n\t var button = document.createElement('li'),\n\t toolIcon = document.createElement('i'),\n\t toolTitle = document.createElement('span');\n\t\n\t button.dataset.type = type;\n\t button.setAttribute('title', type);\n\t\n\t toolIcon.classList.add(classname);\n\t toolTitle.classList.add('ce_toolbar_tools--title');\n\t\n\t button.appendChild(toolIcon);\n\t button.appendChild(toolTitle);\n\t\n\t return button;\n\t };\n\t\n\t /**\r\n\t * @protected\r\n\t *\r\n\t * Draws tools for inline toolbar\r\n\t *\r\n\t * @param {String} type\r\n\t * @param {String} classname\r\n\t */\n\t draw.toolbarButtonInline = function (type, classname) {\n\t\n\t var button = document.createElement('BUTTON'),\n\t toolIcon = document.createElement('I');\n\t\n\t button.type = 'button';\n\t button.dataset.type = type;\n\t toolIcon.classList.add(classname);\n\t\n\t button.appendChild(toolIcon);\n\t\n\t return button;\n\t };\n\t\n\t /**\r\n\t * Redactor block\r\n\t */\n\t draw.block = function (tagName, content) {\n\t\n\t var node = document.createElement(tagName);\n\t\n\t node.innerHTML = content || '';\n\t\n\t return node;\n\t };\n\t\n\t /**\r\n\t * Creates Node with passed tagName and className\r\n\t * @param {string} tagName\r\n\t * @param {string} className\r\n\t * @param {object} properties - allow to assign properties\r\n\t */\n\t draw.node = function (tagName, className, properties) {\n\t\n\t var el = document.createElement(tagName);\n\t\n\t if (className) el.className = className;\n\t\n\t if (properties) {\n\t\n\t for (var name in properties) {\n\t\n\t el[name] = properties[name];\n\t }\n\t }\n\t\n\t return el;\n\t };\n\t\n\t /**\r\n\t * Unavailable plugin block\r\n\t */\n\t draw.unavailableBlock = function () {\n\t\n\t var wrapper = document.createElement('DIV');\n\t\n\t wrapper.classList.add('cdx-unavailable-block');\n\t\n\t return wrapper;\n\t };\n\t\n\t return draw;\n\t}({});\n\n/***/ },\n/* 14 */\n/***/ function(module, exports) {\n\n\t'use strict';\n\t\n\t/**\r\n\t * Codex Editor Caret Module\r\n\t *\r\n\t * @author Codex Team\r\n\t * @version 1.0\r\n\t */\n\t\n\tmodule.exports = function (caret) {\n\t\n\t var editor = codex.editor;\n\t\n\t /**\r\n\t * @var {int} InputIndex - editable element in DOM\r\n\t */\n\t caret.inputIndex = null;\n\t\n\t /**\r\n\t * @var {int} offset - caret position in a text node.\r\n\t */\n\t caret.offset = null;\n\t\n\t /**\r\n\t * @var {int} focusedNodeIndex - we get index of child node from first-level block\r\n\t */\n\t caret.focusedNodeIndex = null;\n\t\n\t /**\r\n\t * Creates Document Range and sets caret to the element.\r\n\t * @protected\r\n\t * @uses caret.save — if you need to save caret position\r\n\t * @param {Element} el - Changed Node.\r\n\t */\n\t caret.set = function (el, index, offset) {\n\t\n\t offset = offset || caret.offset || 0;\n\t index = index || caret.focusedNodeIndex || 0;\n\t\n\t var childs = el.childNodes,\n\t nodeToSet;\n\t\n\t if (childs.length === 0) {\n\t\n\t nodeToSet = el;\n\t } else {\n\t\n\t nodeToSet = childs[index];\n\t }\n\t\n\t /** If Element is INPUT */\n\t if (el.contentEditable != 'true') {\n\t\n\t el.focus();\n\t return;\n\t }\n\t\n\t if (editor.core.isDomNode(nodeToSet)) {\n\t\n\t nodeToSet = editor.content.getDeepestTextNodeFromPosition(nodeToSet, nodeToSet.childNodes.length);\n\t }\n\t\n\t var range = document.createRange(),\n\t selection = window.getSelection();\n\t\n\t window.setTimeout(function () {\n\t\n\t range.setStart(nodeToSet, offset);\n\t range.setEnd(nodeToSet, offset);\n\t\n\t selection.removeAllRanges();\n\t selection.addRange(range);\n\t\n\t editor.caret.saveCurrentInputIndex();\n\t }, 20);\n\t };\n\t\n\t /**\r\n\t * @protected\r\n\t * Updates index of input and saves it in caret object\r\n\t */\n\t caret.saveCurrentInputIndex = function () {\n\t\n\t /** Index of Input that we paste sanitized content */\n\t var selection = window.getSelection(),\n\t inputs = editor.state.inputs,\n\t focusedNode = selection.anchorNode,\n\t focusedNodeHolder;\n\t\n\t if (!focusedNode) {\n\t\n\t return;\n\t }\n\t\n\t /** Looking for parent contentEditable block */\n\t while (focusedNode.contentEditable != 'true') {\n\t\n\t focusedNodeHolder = focusedNode.parentNode;\n\t focusedNode = focusedNodeHolder;\n\t }\n\t\n\t /** Input index in DOM level */\n\t var editableElementIndex = 0;\n\t\n\t while (focusedNode != inputs[editableElementIndex]) {\n\t\n\t editableElementIndex++;\n\t }\n\t\n\t caret.inputIndex = editableElementIndex;\n\t };\n\t\n\t /**\r\n\t * Returns current input index (caret object)\r\n\t */\n\t caret.getCurrentInputIndex = function () {\n\t\n\t return caret.inputIndex;\n\t };\n\t\n\t /**\r\n\t * @param {int} index - index of first-level block after that we set caret into next input\r\n\t */\n\t caret.setToNextBlock = function (index) {\n\t\n\t var inputs = editor.state.inputs,\n\t nextInput = inputs[index + 1];\n\t\n\t if (!nextInput) {\n\t\n\t editor.core.log('We are reached the end');\n\t return;\n\t }\n\t\n\t /**\r\n\t * When new Block created or deleted content of input\r\n\t * We should add some text node to set caret\r\n\t */\n\t if (!nextInput.childNodes.length) {\n\t\n\t var emptyTextElement = document.createTextNode('');\n\t\n\t nextInput.appendChild(emptyTextElement);\n\t }\n\t\n\t editor.caret.inputIndex = index + 1;\n\t editor.caret.set(nextInput, 0, 0);\n\t editor.content.workingNodeChanged(nextInput);\n\t };\n\t\n\t /**\r\n\t * @param {int} index - index of target input.\r\n\t * Sets caret to input with this index\r\n\t */\n\t caret.setToBlock = function (index) {\n\t\n\t var inputs = editor.state.inputs,\n\t targetInput = inputs[index];\n\t\n\t if (!targetInput) {\n\t\n\t return;\n\t }\n\t\n\t /**\r\n\t * When new Block created or deleted content of input\r\n\t * We should add some text node to set caret\r\n\t */\n\t if (!targetInput.childNodes.length) {\n\t\n\t var emptyTextElement = document.createTextNode('');\n\t\n\t targetInput.appendChild(emptyTextElement);\n\t }\n\t\n\t editor.caret.inputIndex = index;\n\t editor.caret.set(targetInput, 0, 0);\n\t editor.content.workingNodeChanged(targetInput);\n\t };\n\t\n\t /**\r\n\t * @param {int} index - index of input\r\n\t */\n\t caret.setToPreviousBlock = function (index) {\n\t\n\t index = index || 0;\n\t\n\t var inputs = editor.state.inputs,\n\t previousInput = inputs[index - 1],\n\t lastChildNode,\n\t lengthOfLastChildNode,\n\t emptyTextElement;\n\t\n\t if (!previousInput) {\n\t\n\t editor.core.log('We are reached first node');\n\t return;\n\t }\n\t\n\t lastChildNode = editor.content.getDeepestTextNodeFromPosition(previousInput, previousInput.childNodes.length);\n\t lengthOfLastChildNode = lastChildNode.length;\n\t\n\t /**\r\n\t * When new Block created or deleted content of input\r\n\t * We should add some text node to set caret\r\n\t */\n\t if (!previousInput.childNodes.length) {\n\t\n\t emptyTextElement = document.createTextNode('');\n\t previousInput.appendChild(emptyTextElement);\n\t }\n\t editor.caret.inputIndex = index - 1;\n\t editor.caret.set(previousInput, previousInput.childNodes.length - 1, lengthOfLastChildNode);\n\t editor.content.workingNodeChanged(inputs[index - 1]);\n\t };\n\t\n\t caret.position = {\n\t\n\t atStart: function atStart() {\n\t\n\t var selection = window.getSelection(),\n\t anchorOffset = selection.anchorOffset,\n\t anchorNode = selection.anchorNode,\n\t firstLevelBlock = editor.content.getFirstLevelBlock(anchorNode),\n\t pluginsRender = firstLevelBlock.childNodes[0];\n\t\n\t if (!editor.core.isDomNode(anchorNode)) {\n\t\n\t anchorNode = anchorNode.parentNode;\n\t }\n\t\n\t var isFirstNode = anchorNode === pluginsRender.childNodes[0],\n\t isOffsetZero = anchorOffset === 0;\n\t\n\t return isFirstNode && isOffsetZero;\n\t },\n\t\n\t atTheEnd: function atTheEnd() {\n\t\n\t var selection = window.getSelection(),\n\t anchorOffset = selection.anchorOffset,\n\t anchorNode = selection.anchorNode;\n\t\n\t /** Caret is at the end of input */\n\t return !anchorNode || !anchorNode.length || anchorOffset === anchorNode.length;\n\t }\n\t };\n\t\n\t /**\r\n\t * Inserts node at the caret location\r\n\t * @param node\r\n\t */\n\t caret.insertNode = function (node) {\n\t\n\t var selection, range;\n\t\n\t selection = window.getSelection();\n\t\n\t range = selection.getRangeAt(0);\n\t range.deleteContents();\n\t\n\t range.insertNode(node);\n\t\n\t range.setStartAfter(node);\n\t range.collapse(true);\n\t\n\t selection.removeAllRanges();\n\t selection.addRange(range);\n\t };\n\t\n\t return caret;\n\t}({});\n\n/***/ },\n/* 15 */\n/***/ function(module, exports) {\n\n\t'use strict';\n\t\n\t/**\r\n\t * Codex Editor Notification Module\r\n\t *\r\n\t * @author Codex Team\r\n\t * @version 1.0\r\n\t */\n\t\n\tmodule.exports = function (notifications) {\n\t\n\t var editor = codex.editor;\n\t\n\t var queue = [];\n\t\n\t var addToQueue = function addToQueue(settings) {\n\t\n\t queue.push(settings);\n\t\n\t var index = 0;\n\t\n\t while (index < queue.length && queue.length > 5) {\n\t\n\t if (queue[index].type == 'confirm' || queue[index].type == 'prompt') {\n\t\n\t index++;\n\t continue;\n\t }\n\t\n\t queue[index].close();\n\t queue.splice(index, 1);\n\t }\n\t };\n\t\n\t notifications.createHolder = function () {\n\t\n\t var holder = editor.draw.node('DIV', 'cdx-notifications-block');\n\t\n\t editor.nodes.notifications = document.body.appendChild(holder);\n\t\n\t return holder;\n\t };\n\t\n\t /**\r\n\t * Error notificator. Shows block with message\r\n\t * @protected\r\n\t */\n\t notifications.errorThrown = function (errorMsg, event) {\n\t\n\t editor.notifications.notification({ message: 'This action is not available currently', type: event.type });\n\t };\n\t\n\t /**\r\n\t *\r\n\t * Appends notification\r\n\t *\r\n\t * settings = {\r\n\t * type - notification type (reserved types: alert, confirm, prompt). Just add class 'cdx-notification-'+type\r\n\t * message - notification message\r\n\t * okMsg - confirm button text (default - 'Ok')\r\n\t * cancelBtn - cancel button text (default - 'Cancel'). Only for confirm and prompt types\r\n\t * confirm - function-handler for ok button click\r\n\t * cancel - function-handler for cancel button click. Only for confirm and prompt types\r\n\t * time - time (in seconds) after which notification will close (default - 10s)\r\n\t * }\r\n\t *\r\n\t * @param settings\r\n\t */\n\t notifications.notification = function (constructorSettings) {\n\t\n\t /** Private vars and methods */\n\t var notification = null,\n\t cancel = null,\n\t type = null,\n\t confirm = null,\n\t inputField = null;\n\t\n\t var confirmHandler = function confirmHandler() {\n\t\n\t close();\n\t\n\t if (typeof confirm !== 'function') {\n\t\n\t return;\n\t }\n\t\n\t if (type == 'prompt') {\n\t\n\t confirm(inputField.value);\n\t return;\n\t }\n\t\n\t confirm();\n\t };\n\t\n\t var cancelHandler = function cancelHandler() {\n\t\n\t close();\n\t\n\t if (typeof cancel !== 'function') {\n\t\n\t return;\n\t }\n\t\n\t cancel();\n\t };\n\t\n\t /** Public methods */\n\t function create(settings) {\n\t\n\t if (!(settings && settings.message)) {\n\t\n\t editor.core.log('Can\\'t create notification. Message is missed');\n\t return;\n\t }\n\t\n\t settings.type = settings.type || 'alert';\n\t settings.time = settings.time * 1000 || 10000;\n\t\n\t var wrapper = editor.draw.node('DIV', 'cdx-notification'),\n\t message = editor.draw.node('DIV', 'cdx-notification__message'),\n\t input = editor.draw.node('INPUT', 'cdx-notification__input'),\n\t okBtn = editor.draw.node('SPAN', 'cdx-notification__ok-btn'),\n\t cancelBtn = editor.draw.node('SPAN', 'cdx-notification__cancel-btn');\n\t\n\t message.textContent = settings.message;\n\t okBtn.textContent = settings.okMsg || 'ОК';\n\t cancelBtn.textContent = settings.cancelMsg || 'Отмена';\n\t\n\t editor.listeners.add(okBtn, 'click', confirmHandler);\n\t editor.listeners.add(cancelBtn, 'click', cancelHandler);\n\t\n\t wrapper.appendChild(message);\n\t\n\t if (settings.type == 'prompt') {\n\t\n\t wrapper.appendChild(input);\n\t }\n\t\n\t wrapper.appendChild(okBtn);\n\t\n\t if (settings.type == 'prompt' || settings.type == 'confirm') {\n\t\n\t wrapper.appendChild(cancelBtn);\n\t }\n\t\n\t wrapper.classList.add('cdx-notification-' + settings.type);\n\t wrapper.dataset.type = settings.type;\n\t\n\t notification = wrapper;\n\t type = settings.type;\n\t confirm = settings.confirm;\n\t cancel = settings.cancel;\n\t inputField = input;\n\t\n\t if (settings.type != 'prompt' && settings.type != 'confirm') {\n\t\n\t window.setTimeout(close, settings.time);\n\t }\n\t };\n\t\n\t function send() {\n\t\n\t editor.nodes.notifications.appendChild(notification);\n\t inputField.focus();\n\t\n\t editor.nodes.notifications.classList.add('cdx-notification__notification-appending');\n\t\n\t window.setTimeout(function () {\n\t\n\t editor.nodes.notifications.classList.remove('cdx-notification__notification-appending');\n\t }, 100);\n\t\n\t addToQueue({ type: type, close: close });\n\t };\n\t\n\t function close() {\n\t\n\t notification.remove();\n\t };\n\t\n\t if (constructorSettings) {\n\t\n\t create(constructorSettings);\n\t send();\n\t }\n\t\n\t return {\n\t create: create,\n\t send: send,\n\t close: close\n\t };\n\t };\n\t\n\t notifications.clear = function () {\n\t\n\t editor.nodes.notifications.innerHTML = '';\n\t queue = [];\n\t };\n\t\n\t return notifications;\n\t}({});\n\n/***/ },\n/* 16 */\n/***/ function(module, exports) {\n\n\t\"use strict\";\n\t\n\t/**\r\n\t * Codex Editor Parser Module\r\n\t *\r\n\t * @author Codex Team\r\n\t * @version 1.1\r\n\t */\n\t\n\tmodule.exports = function (parser) {\n\t\n\t var editor = codex.editor;\n\t\n\t /** inserting text */\n\t parser.insertPastedContent = function (blockType, tag) {\n\t\n\t editor.content.insertBlock({\n\t type: blockType.type,\n\t block: blockType.render({\n\t text: tag.innerHTML\n\t })\n\t });\n\t };\n\t\n\t /**\r\n\t * Check DOM node for display style: separated block or child-view\r\n\t */\n\t parser.isFirstLevelBlock = function (node) {\n\t\n\t return node.nodeType == editor.core.nodeTypes.TAG && node.classList.contains(editor.ui.className.BLOCK_CLASSNAME);\n\t };\n\t\n\t return parser;\n\t}({});\n\n/***/ },\n/* 17 */\n/***/ function(module, exports, __webpack_require__) {\n\n\t'use strict';\n\t\n\t/**\r\n\t * Codex Sanitizer\r\n\t */\n\t\n\tmodule.exports = function (sanitizer) {\n\t\n\t /** HTML Janitor library */\n\t var janitor = __webpack_require__(18);\n\t\n\t /** Codex Editor */\n\t var editor = codex.editor;\n\t\n\t sanitizer.prepare = function () {\n\t\n\t if (editor.settings.sanitizer && !editor.core.isEmpty(editor.settings.sanitizer)) {\n\t\n\t Config.CUSTOM = editor.settings.sanitizer;\n\t }\n\t };\n\t\n\t /**\r\n\t * Basic config\r\n\t */\n\t var Config = {\n\t\n\t /** User configuration */\n\t CUSTOM: null,\n\t\n\t BASIC: {\n\t\n\t tags: {\n\t p: {},\n\t a: {\n\t href: true,\n\t target: '_blank',\n\t rel: 'nofollow'\n\t },\n\t i: {},\n\t b: {},\n\t strong: {},\n\t em: {},\n\t span: {}\n\t }\n\t }\n\t };\n\t\n\t sanitizer.Config = Config;\n\t\n\t /**\r\n\t *\r\n\t * @param userCustomConfig\r\n\t * @returns {*}\r\n\t * @private\r\n\t *\r\n\t * @description If developer uses editor's API, then he can customize sane restrictions.\r\n\t * Or, sane config can be defined globally in editors initialization. That config will be used everywhere\r\n\t * At least, if there is no config overrides, that API uses BASIC Default configation\r\n\t */\n\t var init_ = function init_(userCustomConfig) {\n\t\n\t var configuration = userCustomConfig || Config.CUSTOM || Config.BASIC;\n\t\n\t return new janitor(configuration);\n\t };\n\t\n\t /**\r\n\t * Cleans string from unwanted tags\r\n\t * @protected\r\n\t * @param {String} dirtyString - taint string\r\n\t * @param {Object} customConfig - allowed tags\r\n\t */\n\t sanitizer.clean = function (dirtyString, customConfig) {\n\t\n\t var janitorInstance = init_(customConfig);\n\t\n\t return janitorInstance.clean(dirtyString);\n\t };\n\t\n\t return sanitizer;\n\t}({});\n\n/***/ },\n/* 18 */\n/***/ function(module, exports, __webpack_require__) {\n\n\tvar __WEBPACK_AMD_DEFINE_FACTORY__, __WEBPACK_AMD_DEFINE_RESULT__;(function (root, factory) {\n\t if (true) {\n\t !(__WEBPACK_AMD_DEFINE_FACTORY__ = (factory), __WEBPACK_AMD_DEFINE_RESULT__ = (typeof __WEBPACK_AMD_DEFINE_FACTORY__ === 'function' ? (__WEBPACK_AMD_DEFINE_FACTORY__.call(exports, __webpack_require__, exports, module)) : __WEBPACK_AMD_DEFINE_FACTORY__), __WEBPACK_AMD_DEFINE_RESULT__ !== undefined && (module.exports = __WEBPACK_AMD_DEFINE_RESULT__));\n\t } else if (typeof exports === 'object') {\n\t module.exports = factory();\n\t } else {\n\t root.HTMLJanitor = factory();\n\t }\n\t}(this, function () {\n\t\n\t /**\n\t * @param {Object} config.tags Dictionary of allowed tags.\n\t * @param {boolean} config.keepNestedBlockElements Default false.\n\t */\n\t function HTMLJanitor(config) {\n\t\n\t var tagDefinitions = config['tags'];\n\t var tags = Object.keys(tagDefinitions);\n\t\n\t var validConfigValues = tags\n\t .map(function(k) { return typeof tagDefinitions[k]; })\n\t .every(function(type) { return type === 'object' || type === 'boolean' || type === 'function'; });\n\t\n\t if(!validConfigValues) {\n\t throw new Error(\"The configuration was invalid\");\n\t }\n\t\n\t this.config = config;\n\t }\n\t\n\t // TODO: not exhaustive?\n\t var blockElementNames = ['P', 'LI', 'TD', 'TH', 'DIV', 'H1', 'H2', 'H3', 'H4', 'H5', 'H6', 'PRE'];\n\t function isBlockElement(node) {\n\t return blockElementNames.indexOf(node.nodeName) !== -1;\n\t }\n\t\n\t var inlineElementNames = ['A', 'B', 'STRONG', 'I', 'EM', 'SUB', 'SUP', 'U', 'STRIKE'];\n\t function isInlineElement(node) {\n\t return inlineElementNames.indexOf(node.nodeName) !== -1;\n\t }\n\t\n\t HTMLJanitor.prototype.clean = function (html) {\n\t var sandbox = document.createElement('div');\n\t sandbox.innerHTML = html;\n\t\n\t this._sanitize(sandbox);\n\t\n\t return sandbox.innerHTML;\n\t };\n\t\n\t HTMLJanitor.prototype._sanitize = function (parentNode) {\n\t var treeWalker = createTreeWalker(parentNode);\n\t var node = treeWalker.firstChild();\n\t if (!node) { return; }\n\t\n\t do {\n\t // Ignore nodes that have already been sanitized\n\t if (node._sanitized) {\n\t continue;\n\t }\n\t\n\t if (node.nodeType === Node.TEXT_NODE) {\n\t // If this text node is just whitespace and the previous or next element\n\t // sibling is a block element, remove it\n\t // N.B.: This heuristic could change. Very specific to a bug with\n\t // `contenteditable` in Firefox: http://jsbin.com/EyuKase/1/edit?js,output\n\t // FIXME: make this an option?\n\t if (node.data.trim() === ''\n\t && ((node.previousElementSibling && isBlockElement(node.previousElementSibling))\n\t || (node.nextElementSibling && isBlockElement(node.nextElementSibling)))) {\n\t parentNode.removeChild(node);\n\t this._sanitize(parentNode);\n\t break;\n\t } else {\n\t continue;\n\t }\n\t }\n\t\n\t // Remove all comments\n\t if (node.nodeType === Node.COMMENT_NODE) {\n\t parentNode.removeChild(node);\n\t this._sanitize(parentNode);\n\t break;\n\t }\n\t\n\t var isInline = isInlineElement(node);\n\t var containsBlockElement;\n\t if (isInline) {\n\t containsBlockElement = Array.prototype.some.call(node.childNodes, isBlockElement);\n\t }\n\t\n\t // Block elements should not be nested (e.g.

  • ...); if\n\t // they are, we want to unwrap the inner block element.\n\t var isNotTopContainer = !! parentNode.parentNode;\n\t var isNestedBlockElement =\n\t isBlockElement(parentNode) &&\n\t isBlockElement(node) &&\n\t isNotTopContainer;\n\t\n\t var nodeName = node.nodeName.toLowerCase();\n\t\n\t var allowedAttrs = getAllowedAttrs(this.config, nodeName, node);\n\t\n\t var isInvalid = isInline && containsBlockElement;\n\t\n\t // Drop tag entirely according to the whitelist *and* if the markup\n\t // is invalid.\n\t if (isInvalid || shouldRejectNode(node, allowedAttrs)\n\t || (!this.config.keepNestedBlockElements && isNestedBlockElement)) {\n\t // Do not keep the inner text of SCRIPT/STYLE elements.\n\t if (! (node.nodeName === 'SCRIPT' || node.nodeName === 'STYLE')) {\n\t while (node.childNodes.length > 0) {\n\t parentNode.insertBefore(node.childNodes[0], node);\n\t }\n\t }\n\t parentNode.removeChild(node);\n\t\n\t this._sanitize(parentNode);\n\t break;\n\t }\n\t\n\t // Sanitize attributes\n\t for (var a = 0; a < node.attributes.length; a += 1) {\n\t var attr = node.attributes[a];\n\t\n\t if (shouldRejectAttr(attr, allowedAttrs, node)) {\n\t node.removeAttribute(attr.name);\n\t // Shift the array to continue looping.\n\t a = a - 1;\n\t }\n\t }\n\t\n\t // Sanitize children\n\t this._sanitize(node);\n\t\n\t // Mark node as sanitized so it's ignored in future runs\n\t node._sanitized = true;\n\t } while ((node = treeWalker.nextSibling()));\n\t };\n\t\n\t function createTreeWalker(node) {\n\t return document.createTreeWalker(node,\n\t NodeFilter.SHOW_TEXT | NodeFilter.SHOW_ELEMENT | NodeFilter.SHOW_COMMENT,\n\t null, false);\n\t }\n\t\n\t function getAllowedAttrs(config, nodeName, node){\n\t if (typeof config.tags[nodeName] === 'function') {\n\t return config.tags[nodeName](node);\n\t } else {\n\t return config.tags[nodeName];\n\t }\n\t }\n\t\n\t function shouldRejectNode(node, allowedAttrs){\n\t if (typeof allowedAttrs === 'undefined') {\n\t return true;\n\t } else if (typeof allowedAttrs === 'boolean') {\n\t return !allowedAttrs;\n\t }\n\t\n\t return false;\n\t }\n\t\n\t function shouldRejectAttr(attr, allowedAttrs, node){\n\t var attrName = attr.name.toLowerCase();\n\t\n\t if (allowedAttrs === true){\n\t return false;\n\t } else if (typeof allowedAttrs[attrName] === 'function'){\n\t return !allowedAttrs[attrName](attr.value, node);\n\t } else if (typeof allowedAttrs[attrName] === 'undefined'){\n\t return true;\n\t } else if (allowedAttrs[attrName] === false) {\n\t return true;\n\t } else if (typeof allowedAttrs[attrName] === 'string') {\n\t return (allowedAttrs[attrName] !== attr.value);\n\t }\n\t\n\t return false;\n\t }\n\t\n\t return HTMLJanitor;\n\t\n\t}));\n\n\n/***/ },\n/* 19 */\n/***/ function(module, exports) {\n\n\t\"use strict\";\n\t\n\t/**\r\n\t * Codex Editor Listeners module\r\n\t *\r\n\t * @author Codex Team\r\n\t * @version 1.0\r\n\t */\n\t\n\t/**\r\n\t * Module-decorator for event listeners assignment\r\n\t */\n\tmodule.exports = function (listeners) {\n\t\n\t var allListeners = [];\n\t\n\t /**\r\n\t * Search methods\r\n\t *\r\n\t * byElement, byType and byHandler returns array of suitable listeners\r\n\t * one and all takes element, eventType, and handler and returns first (all) suitable listener\r\n\t *\r\n\t */\n\t listeners.search = function () {\n\t\n\t var byElement = function byElement(element, context) {\n\t\n\t var listenersOnElement = [];\n\t\n\t context = context || allListeners;\n\t\n\t for (var i = 0; i < context.length; i++) {\n\t\n\t var listener = context[i];\n\t\n\t if (listener.element === element) {\n\t\n\t listenersOnElement.push(listener);\n\t }\n\t }\n\t\n\t return listenersOnElement;\n\t };\n\t\n\t var byType = function byType(eventType, context) {\n\t\n\t var listenersWithType = [];\n\t\n\t context = context || allListeners;\n\t\n\t for (var i = 0; i < context.length; i++) {\n\t\n\t var listener = context[i];\n\t\n\t if (listener.type === eventType) {\n\t\n\t listenersWithType.push(listener);\n\t }\n\t }\n\t\n\t return listenersWithType;\n\t };\n\t\n\t var byHandler = function byHandler(handler, context) {\n\t\n\t var listenersWithHandler = [];\n\t\n\t context = context || allListeners;\n\t\n\t for (var i = 0; i < context.length; i++) {\n\t\n\t var listener = context[i];\n\t\n\t if (listener.handler === handler) {\n\t\n\t listenersWithHandler.push(listener);\n\t }\n\t }\n\t\n\t return listenersWithHandler;\n\t };\n\t\n\t var one = function one(element, eventType, handler) {\n\t\n\t var result = allListeners;\n\t\n\t if (element) result = byElement(element, result);\n\t\n\t if (eventType) result = byType(eventType, result);\n\t\n\t if (handler) result = byHandler(handler, result);\n\t\n\t return result[0];\n\t };\n\t\n\t var all = function all(element, eventType, handler) {\n\t\n\t var result = allListeners;\n\t\n\t if (element) result = byElement(element, result);\n\t\n\t if (eventType) result = byType(eventType, result);\n\t\n\t if (handler) result = byHandler(handler, result);\n\t\n\t return result;\n\t };\n\t\n\t return {\n\t byElement: byElement,\n\t byType: byType,\n\t byHandler: byHandler,\n\t one: one,\n\t all: all\n\t };\n\t }();\n\t\n\t listeners.add = function (element, eventType, handler, isCapture) {\n\t\n\t element.addEventListener(eventType, handler, isCapture);\n\t\n\t var data = {\n\t element: element,\n\t type: eventType,\n\t handler: handler\n\t };\n\t\n\t var alreadyAddedListener = listeners.search.one(element, eventType, handler);\n\t\n\t if (!alreadyAddedListener) {\n\t\n\t allListeners.push(data);\n\t }\n\t };\n\t\n\t listeners.remove = function (element, eventType, handler) {\n\t\n\t element.removeEventListener(eventType, handler);\n\t\n\t var existingListeners = listeners.search.all(element, eventType, handler);\n\t\n\t for (var i = 0; i < existingListeners.length; i++) {\n\t\n\t var index = allListeners.indexOf(existingListeners[i]);\n\t\n\t if (index > 0) {\n\t\n\t allListeners.splice(index, 1);\n\t }\n\t }\n\t };\n\t\n\t listeners.removeAll = function () {\n\t\n\t allListeners.map(function (current) {\n\t\n\t listeners.remove(current.element, current.type, current.handler);\n\t });\n\t };\n\t\n\t listeners.get = function (element, eventType, handler) {\n\t\n\t return listeners.search.all(element, eventType, handler);\n\t };\n\t\n\t return listeners;\n\t}({});\n\n/***/ },\n/* 20 */\n/***/ function(module, exports) {\n\n\t'use strict';\n\t\n\tvar _typeof = typeof Symbol === \"function\" && typeof Symbol.iterator === \"symbol\" ? function (obj) { return typeof obj; } : function (obj) { return obj && typeof Symbol === \"function\" && obj.constructor === Symbol && obj !== Symbol.prototype ? \"symbol\" : typeof obj; };\n\t\n\t/**\r\n\t * Codex Editor Destroyer module\r\n\t *\r\n\t * @auhor Codex Team\r\n\t * @version 1.0\r\n\t */\n\t\n\tmodule.exports = function (destroyer) {\n\t\n\t var editor = codex.editor;\n\t\n\t destroyer.removeNodes = function () {\n\t\n\t editor.nodes.wrapper.remove();\n\t editor.nodes.notifications.remove();\n\t };\n\t\n\t destroyer.destroyPlugins = function () {\n\t\n\t for (var tool in editor.tools) {\n\t\n\t if (typeof editor.tools[tool].destroy === 'function') {\n\t\n\t editor.tools[tool].destroy();\n\t }\n\t }\n\t };\n\t\n\t destroyer.destroyScripts = function () {\n\t\n\t var scripts = document.getElementsByTagName('SCRIPT');\n\t\n\t for (var i = 0; i < scripts.length; i++) {\n\t\n\t if (scripts[i].id.indexOf(editor.scriptPrefix) + 1) {\n\t\n\t scripts[i].remove();\n\t i--;\n\t }\n\t }\n\t };\n\t\n\t /**\r\n\t * Delete editor data from webpage.\r\n\t * You should send settings argument with boolean flags:\r\n\t * @param settings.ui- remove redactor event listeners and DOM nodes\r\n\t * @param settings.scripts - remove redactor scripts from DOM\r\n\t * @param settings.plugins - remove plugin's objects\r\n\t * @param settings.core - remove editor core. You can remove core only if UI and scripts flags is true\r\n\t * }\r\n\t *\r\n\t */\n\t destroyer.destroy = function (settings) {\n\t\n\t if (!settings || (typeof settings === 'undefined' ? 'undefined' : _typeof(settings)) !== 'object') {\n\t\n\t return;\n\t }\n\t\n\t if (settings.ui) {\n\t\n\t destroyer.removeNodes();\n\t editor.listeners.removeAll();\n\t }\n\t\n\t if (settings.scripts) {\n\t\n\t destroyer.destroyScripts();\n\t }\n\t\n\t if (settings.plugins) {\n\t\n\t destroyer.destroyPlugins();\n\t }\n\t\n\t if (settings.ui && settings.scripts && settings.core) {\n\t\n\t delete codex.editor;\n\t }\n\t };\n\t\n\t return destroyer;\n\t}({});\n\n/***/ },\n/* 21 */\n/***/ function(module, exports) {\n\n\t'use strict';\n\t\n\t/**\r\n\t * Codex Editor Paste module\r\n\t *\r\n\t * @author Codex Team\r\n\t * @version 1.1.1\r\n\t */\n\t\n\tmodule.exports = function (paste) {\n\t\n\t var editor = codex.editor;\n\t\n\t var patterns = [];\n\t\n\t paste.prepare = function () {\n\t\n\t var tools = editor.tools;\n\t\n\t for (var tool in tools) {\n\t\n\t if (!tools[tool].renderOnPastePatterns || !Array.isArray(tools[tool].renderOnPastePatterns)) {\n\t\n\t continue;\n\t }\n\t\n\t tools[tool].renderOnPastePatterns.map(function (pattern) {\n\t\n\t patterns.push(pattern);\n\t });\n\t }\n\t\n\t return Promise.resolve();\n\t };\n\t\n\t /**\r\n\t * Saves data\r\n\t * @param event\r\n\t */\n\t paste.pasted = function (event) {\n\t\n\t var clipBoardData = event.clipboardData || window.clipboardData,\n\t content = clipBoardData.getData('Text');\n\t\n\t var result = analize(content);\n\t\n\t if (result) {\n\t\n\t event.preventDefault();\n\t event.stopImmediatePropagation();\n\t }\n\t\n\t return result;\n\t };\n\t\n\t /**\r\n\t * Analizes pated string and calls necessary method\r\n\t */\n\t\n\t var analize = function analize(string) {\n\t\n\t var result = false,\n\t content = editor.content.currentNode,\n\t plugin = content.dataset.tool;\n\t\n\t patterns.map(function (pattern) {\n\t\n\t var execArray = pattern.regex.exec(string),\n\t match = execArray && execArray[0];\n\t\n\t if (match && match === string.trim()) {\n\t\n\t /** current block is not empty */\n\t if (content.textContent.trim() && plugin == editor.settings.initialBlockPlugin) {\n\t\n\t pasteToNewBlock_();\n\t }\n\t\n\t pattern.callback(string, pattern);\n\t result = true;\n\t }\n\t });\n\t\n\t return result;\n\t };\n\t\n\t var pasteToNewBlock_ = function pasteToNewBlock_() {\n\t\n\t /** Create new initial block */\n\t editor.content.insertBlock({\n\t\n\t type: editor.settings.initialBlockPlugin,\n\t block: editor.tools[editor.settings.initialBlockPlugin].render({\n\t text: ''\n\t })\n\t\n\t }, false);\n\t };\n\t\n\t return paste;\n\t}({});\n\n/***/ }\n/******/ ]);\n\n\n// WEBPACK FOOTER //\n// codex-editor.js"," \t// The module cache\n \tvar installedModules = {};\n\n \t// The require function\n \tfunction __webpack_require__(moduleId) {\n\n \t\t// Check if module is in cache\n \t\tif(installedModules[moduleId])\n \t\t\treturn installedModules[moduleId].exports;\n\n \t\t// Create a new module (and put it into the cache)\n \t\tvar module = installedModules[moduleId] = {\n \t\t\texports: {},\n \t\t\tid: moduleId,\n \t\t\tloaded: false\n \t\t};\n\n \t\t// Execute the module function\n \t\tmodules[moduleId].call(module.exports, module, module.exports, __webpack_require__);\n\n \t\t// Flag the module as loaded\n \t\tmodule.loaded = true;\n\n \t\t// Return the exports of the module\n \t\treturn module.exports;\n \t}\n\n\n \t// expose the modules object (__webpack_modules__)\n \t__webpack_require__.m = modules;\n\n \t// expose the module cache\n \t__webpack_require__.c = installedModules;\n\n \t// __webpack_public_path__\n \t__webpack_require__.p = \"\";\n\n \t// Load entry module and return exports\n \treturn __webpack_require__(0);\n\n\n\n// WEBPACK FOOTER //\n// webpack/bootstrap 4b9c333b21a67902b259","/**\r\n *\r\n * Codex Editor\r\n *\r\n * @author Codex Team\r\n */\r\n\r\nmodule.exports = (function (editor) {\r\n\r\n 'use strict';\r\n\r\n editor.version = VERSION;\r\n editor.scriptPrefix = 'cdx-script-';\r\n\r\n var init = function () {\r\n\r\n editor.core = require('./modules/core');\r\n editor.tools = require('./modules/tools');\r\n editor.ui = require('./modules/ui');\r\n editor.transport = require('./modules/transport');\r\n editor.renderer = require('./modules/renderer');\r\n editor.saver = require('./modules/saver');\r\n editor.content = require('./modules/content');\r\n editor.toolbar = require('./modules/toolbar/toolbar');\r\n editor.callback = require('./modules/callbacks');\r\n editor.draw = require('./modules/draw');\r\n editor.caret = require('./modules/caret');\r\n editor.notifications = require('./modules/notifications');\r\n editor.parser = require('./modules/parser');\r\n editor.sanitizer = require('./modules/sanitizer');\r\n editor.listeners = require('./modules/listeners');\r\n editor.destroyer = require('./modules/destroyer');\r\n editor.paste = require('./modules/paste');\r\n\r\n };\r\n\r\n /**\r\n * @public\r\n * holds initial settings\r\n */\r\n editor.settings = {\r\n tools : ['paragraph', 'header', 'picture', 'list', 'quote', 'code', 'twitter', 'instagram', 'smile'],\r\n holderId : 'codex-editor',\r\n\r\n // Type of block showing on empty editor\r\n initialBlockPlugin: 'paragraph'\r\n };\r\n\r\n /**\r\n * public\r\n *\r\n * Static nodes\r\n */\r\n editor.nodes = {\r\n holder : null,\r\n wrapper : null,\r\n toolbar : null,\r\n inlineToolbar : {\r\n wrapper : null,\r\n buttons : null,\r\n actions : null\r\n },\r\n toolbox : null,\r\n notifications : null,\r\n plusButton : null,\r\n showSettingsButton: null,\r\n showTrashButton : null,\r\n blockSettings : null,\r\n pluginSettings : null,\r\n defaultSettings : null,\r\n toolbarButtons : {}, // { type : DomEl, ... }\r\n redactor : null\r\n };\r\n\r\n /**\r\n * @public\r\n *\r\n * Output state\r\n */\r\n editor.state = {\r\n jsonOutput : [],\r\n blocks : [],\r\n inputs : []\r\n };\r\n\r\n /**\r\n * @public\r\n * Editor plugins\r\n */\r\n editor.tools = {};\r\n\r\n /**\r\n * Initialization\r\n * @uses Promise cEditor.core.prepare\r\n * @param {Object} userSettings\r\n * @param {Array} userSettings.tools list of plugins\r\n * @param {String} userSettings.holderId Element's id to append editor\r\n *\r\n * Load user defined tools\r\n * Tools must contain this important objects :\r\n * @param {String} type - this is a type of plugin. It can be used as plugin name\r\n * @param {String} iconClassname - this a icon in toolbar\r\n * @param {Object} make - what should plugin do, when it is clicked\r\n * @param {Object} appendCallback - callback after clicking\r\n * @param {Element} settings - what settings does it have\r\n * @param {Object} render - plugin get JSON, and should return HTML\r\n * @param {Object} save - plugin gets HTML content, returns JSON\r\n * @param {Boolean} displayInToolbox - will be displayed in toolbox. Default value is TRUE\r\n * @param {Boolean} enableLineBreaks - inserts new block or break lines. Default value is FALSE\r\n *\r\n * @example\r\n * - type : 'header',\r\n * - iconClassname : 'ce-icon-header',\r\n * - make : headerTool.make,\r\n * - appendCallback : headerTool.appendCallback,\r\n * - settings : headerTool.makeSettings(),\r\n * - render : headerTool.render,\r\n * - save : headerTool.save,\r\n * - displayInToolbox : true,\r\n * - enableLineBreaks : false\r\n */\r\n editor.start = function (userSettings) {\r\n\r\n init();\r\n\r\n editor.core.prepare(userSettings)\r\n\r\n // If all ok, make UI, bind events and parse initial-content\r\n .then(editor.ui.prepare)\r\n .then(editor.tools.prepare)\r\n .then(editor.sanitizer.prepare)\r\n .then(editor.paste.prepare)\r\n .then(editor.transport.prepare)\r\n .then(editor.renderer.makeBlocksFromData)\r\n .then(editor.ui.saveInputs)\r\n .catch(function (error) {\r\n\r\n editor.core.log('Initialization failed with error: %o', 'warn', error);\r\n\r\n });\r\n\r\n };\r\n\r\n return editor;\r\n\r\n})({});\r\n\n\n\n// WEBPACK FOOTER //\n// ./codex.js","/**\r\n * Codex Editor Core\r\n *\r\n * @author Codex Team\r\n * @version 1.1.3\r\n */\r\n\r\nmodule.exports = (function (core) {\r\n\r\n let editor = codex.editor;\r\n\r\n /**\r\n * @public\r\n *\r\n * Editor preparing method\r\n * @return Promise\r\n */\r\n core.prepare = function (userSettings) {\r\n\r\n return new Promise(function (resolve, reject) {\r\n\r\n if ( userSettings ) {\r\n\r\n editor.settings.tools = userSettings.tools || editor.settings.tools;\r\n\r\n }\r\n\r\n if (userSettings.data) {\r\n\r\n editor.state.blocks = userSettings.data;\r\n\r\n }\r\n\r\n if (userSettings.initialBlockPlugin) {\r\n\r\n editor.settings.initialBlockPlugin = userSettings.initialBlockPlugin;\r\n\r\n }\r\n\r\n if (userSettings.sanitizer) {\r\n\r\n editor.settings.sanitizer = userSettings.sanitizer;\r\n\r\n }\r\n\r\n editor.hideToolbar = userSettings.hideToolbar;\r\n\r\n editor.nodes.holder = document.getElementById(userSettings.holderId || editor.settings.holderId);\r\n\r\n if (typeof editor.nodes.holder === undefined || editor.nodes.holder === null) {\r\n\r\n reject(Error(\"Holder wasn't found by ID: #\" + userSettings.holderId));\r\n\r\n } else {\r\n\r\n resolve();\r\n\r\n }\r\n\r\n });\r\n\r\n };\r\n\r\n /**\r\n * Logging method\r\n * @param type = ['log', 'info', 'warn']\r\n */\r\n core.log = function (msg, type, arg) {\r\n\r\n type = type || 'log';\r\n\r\n if (!arg) {\r\n\r\n arg = msg || 'undefined';\r\n msg = '[codex-editor]: %o';\r\n\r\n } else {\r\n\r\n msg = '[codex-editor]: ' + msg;\r\n\r\n }\r\n\r\n try{\r\n\r\n if ( 'console' in window && window.console[ type ] ) {\r\n\r\n if ( arg ) window.console[ type ]( msg, arg );\r\n else window.console[ type ]( msg );\r\n\r\n }\r\n\r\n }catch(e) {}\r\n\r\n };\r\n\r\n /**\r\n * @protected\r\n *\r\n * Helper for insert one element after another\r\n */\r\n core.insertAfter = function (target, element) {\r\n\r\n target.parentNode.insertBefore(element, target.nextSibling);\r\n\r\n };\r\n\r\n /**\r\n * @const\r\n *\r\n * Readable DOM-node types map\r\n */\r\n core.nodeTypes = {\r\n TAG : 1,\r\n TEXT : 3,\r\n COMMENT : 8\r\n };\r\n\r\n /**\r\n * @const\r\n * Readable keys map\r\n */\r\n core.keys = { BACKSPACE: 8, TAB: 9, ENTER: 13, SHIFT: 16, CTRL: 17, ALT: 18, ESC: 27, SPACE: 32, LEFT: 37, UP: 38, DOWN: 40, RIGHT: 39, DELETE: 46, META: 91 };\r\n\r\n /**\r\n * @protected\r\n *\r\n * Check object for DOM node\r\n */\r\n core.isDomNode = function (el) {\r\n\r\n return el && typeof el === 'object' && el.nodeType && el.nodeType == this.nodeTypes.TAG;\r\n\r\n };\r\n\r\n /**\r\n * Checks passed object for emptiness\r\n * @require ES5 - Object.keys\r\n * @param {object}\r\n */\r\n core.isEmpty = function ( obj ) {\r\n\r\n return Object.keys(obj).length === 0;\r\n\r\n };\r\n\r\n /**\r\n * Native Ajax\r\n * @param {String} settings.url - request URL\r\n * @param {function} settings.beforeSend - returned value will be passed as context to the Success, Error and Progress callbacks\r\n * @param {function} settings.success\r\n * @param {function} settings.progress\r\n */\r\n core.ajax = function (settings) {\r\n\r\n if (!settings || !settings.url) {\r\n\r\n return;\r\n\r\n }\r\n\r\n var XMLHTTP = window.XMLHttpRequest ? new XMLHttpRequest() : new ActiveXObject('Microsoft.XMLHTTP'),\r\n encodedString,\r\n isFormData,\r\n prop;\r\n\r\n\r\n settings.async = true;\r\n settings.type = settings.type || 'GET';\r\n settings.data = settings.data || '';\r\n settings['content-type'] = settings['content-type'] || 'application/json; charset=utf-8';\r\n\r\n if (settings.type == 'GET' && settings.data) {\r\n\r\n settings.url = /\\?/.test(settings.url) ? settings.url + '&' + settings.data : settings.url + '?' + settings.data;\r\n\r\n } else {\r\n\r\n encodedString = '';\r\n for(prop in settings.data) {\r\n\r\n encodedString += (prop + '=' + encodeURIComponent(settings.data[prop]) + '&');\r\n\r\n }\r\n\r\n }\r\n\r\n if (settings.withCredentials) {\r\n\r\n XMLHTTP.withCredentials = true;\r\n\r\n }\r\n\r\n /**\r\n * Value returned in beforeSend funtion will be passed as context to the other response callbacks\r\n * If beforeSend returns false, AJAX will be blocked\r\n */\r\n let responseContext,\r\n beforeSendResult;\r\n\r\n if (typeof settings.beforeSend === 'function') {\r\n\r\n beforeSendResult = settings.beforeSend.call();\r\n\r\n if (beforeSendResult === false) {\r\n\r\n return;\r\n\r\n }\r\n\r\n }\r\n\r\n XMLHTTP.open( settings.type, settings.url, settings.async );\r\n\r\n /**\r\n * If we send FormData, we need no content-type header\r\n */\r\n isFormData = isFormData_(settings.data);\r\n\r\n if (!isFormData) {\r\n\r\n if (settings.type !== 'POST') {\r\n\r\n XMLHTTP.setRequestHeader('Content-type', settings['content-type']);\r\n\r\n } else {\r\n\r\n XMLHTTP.setRequestHeader('Content-type', 'application/x-www-form-urlencoded');\r\n\r\n }\r\n\r\n }\r\n\r\n XMLHTTP.setRequestHeader('X-Requested-With', 'XMLHttpRequest');\r\n\r\n responseContext = beforeSendResult || XMLHTTP;\r\n\r\n if (typeof settings.progress === 'function') {\r\n\r\n XMLHTTP.upload.onprogress = settings.progress.bind(responseContext);\r\n\r\n }\r\n\r\n XMLHTTP.onreadystatechange = function () {\r\n\r\n if (XMLHTTP.readyState === 4) {\r\n\r\n if (XMLHTTP.status === 200) {\r\n\r\n if (typeof settings.success === 'function') {\r\n\r\n settings.success.call(responseContext, XMLHTTP.responseText);\r\n\r\n }\r\n\r\n } else {\r\n\r\n if (typeof settings.error === 'function') {\r\n\r\n settings.error.call(responseContext, XMLHTTP.responseText, XMLHTTP.status);\r\n\r\n }\r\n\r\n }\r\n\r\n }\r\n\r\n };\r\n\r\n if (isFormData) {\r\n\r\n // Sending FormData\r\n XMLHTTP.send(settings.data);\r\n\r\n } else {\r\n\r\n // POST requests\r\n XMLHTTP.send(encodedString);\r\n\r\n }\r\n\r\n return XMLHTTP;\r\n\r\n };\r\n\r\n /**\r\n * Appends script to head of document\r\n * @return Promise\r\n */\r\n core.importScript = function (scriptPath, instanceName) {\r\n\r\n return new Promise(function (resolve, reject) {\r\n\r\n let script;\r\n\r\n /** Script is already loaded */\r\n if ( !instanceName ) {\r\n\r\n reject('Instance name is missed');\r\n\r\n } else if ( document.getElementById(editor.scriptPrefix + instanceName) ) {\r\n\r\n resolve(scriptPath);\r\n\r\n }\r\n\r\n script = document.createElement('SCRIPT');\r\n script.async = true;\r\n script.defer = true;\r\n script.id = editor.scriptPrefix + instanceName;\r\n\r\n script.onload = function () {\r\n\r\n resolve(scriptPath);\r\n\r\n };\r\n\r\n script.onerror = function () {\r\n\r\n reject(scriptPath);\r\n\r\n };\r\n\r\n script.src = scriptPath;\r\n document.head.appendChild(script);\r\n\r\n });\r\n\r\n };\r\n\r\n /**\r\n * Function for checking is it FormData object to send.\r\n * @param {Object} object to check\r\n * @return boolean\r\n */\r\n var isFormData_ = function (object) {\r\n\r\n return object instanceof FormData;\r\n\r\n };\r\n\r\n return core;\r\n\r\n})({});\r\n\n\n\n// WEBPACK FOOTER //\n// ./modules/core.js","/**\r\n* Module working with plugins\r\n*/\r\nmodule.exports = (function () {\r\n\r\n let editor = codex.editor;\r\n\r\n /**\r\n * Initialize plugins before using\r\n * Ex. Load scripts or call some internal methods\r\n * @return Promise\r\n */\r\n function prepare() {\r\n\r\n return new Promise(function (resolve_, reject_) {\r\n\r\n Promise.resolve()\r\n\r\n /**\r\n * Compose a sequence of plugins that requires preparation\r\n */\r\n .then(function () {\r\n\r\n let pluginsRequiresPreparation = [],\r\n allPlugins = editor.tools;\r\n\r\n for ( let pluginName in allPlugins ) {\r\n\r\n let plugin = allPlugins[pluginName];\r\n\r\n if (plugin.prepare && typeof plugin.prepare != 'function' || !plugin.prepare) {\r\n\r\n continue;\r\n\r\n }\r\n\r\n pluginsRequiresPreparation.push(plugin);\r\n\r\n }\r\n\r\n /**\r\n * If no one passed plugins requires preparation, finish prepare() and go ahead\r\n */\r\n if (!pluginsRequiresPreparation.length) {\r\n\r\n resolve_();\r\n\r\n }\r\n\r\n return pluginsRequiresPreparation;\r\n\r\n })\r\n\r\n /** Wait plugins while they prepares */\r\n .then(waitAllPluginsPreparation_)\r\n\r\n .then(function () {\r\n\r\n editor.core.log('Plugins loaded', 'info');\r\n resolve_();\r\n\r\n }).catch(function (error) {\r\n\r\n reject_(error);\r\n\r\n });\r\n\r\n });\r\n\r\n }\r\n\r\n /**\r\n * @param {array} plugins - list of tools that requires preparation\r\n * @return {Promise} resolved while all plugins will be ready or failed\r\n */\r\n function waitAllPluginsPreparation_(plugins) {\r\n\r\n /**\r\n * @calls allPluginsProcessed__ when all plugins prepared or failed\r\n */\r\n return new Promise (function (allPluginsProcessed__) {\r\n\r\n /**\r\n * pluck each element from queue\r\n * First, send resolved Promise as previous value\r\n * Each plugins \"prepare\" method returns a Promise, that's why\r\n * reduce current element will not be able to continue while can't get\r\n * a resolved Promise\r\n *\r\n * If last plugin is \"prepared\" then go to the next stage of initialization\r\n */\r\n plugins.reduce(function (previousValue, plugin, iteration) {\r\n\r\n return previousValue.then(function () {\r\n\r\n /**\r\n * Wait till plugins prepared\r\n * @calls pluginIsReady__ when plugin is ready or failed\r\n */\r\n return new Promise ( function (pluginIsReady__) {\r\n\r\n callPluginsPrepareMethod_( plugin )\r\n\r\n .then( pluginIsReady__ )\r\n .then( function () {\r\n\r\n plugin.available = true;\r\n\r\n })\r\n\r\n .catch(function (error) {\r\n\r\n editor.core.log(`Plugin «${plugin.type}» was not loaded. Preparation failed because %o`, 'warn', error);\r\n plugin.available = false;\r\n plugin.loadingMessage = error;\r\n\r\n /** Go ahead even some plugin has problems */\r\n pluginIsReady__();\r\n\r\n })\r\n\r\n .then(function () {\r\n\r\n /** If last plugin has problems then just ignore and continue */\r\n if (iteration == plugins.length - 1) {\r\n\r\n allPluginsProcessed__();\r\n\r\n }\r\n\r\n });\r\n\r\n });\r\n\r\n });\r\n\r\n }, Promise.resolve() );\r\n\r\n });\r\n\r\n }\r\n\r\n var callPluginsPrepareMethod_ = function (plugin) {\r\n\r\n return plugin.prepare( plugin.config || {} );\r\n\r\n };\r\n\r\n return {\r\n prepare: prepare\r\n };\r\n\r\n}());\n\n\n// WEBPACK FOOTER //\n// ./modules/tools.js","/**\r\n * Codex Editor UI module\r\n *\r\n * @author Codex Team\r\n * @version 1.2.0\r\n */\r\n\r\nmodule.exports = (function (ui) {\r\n\r\n let editor = codex.editor;\r\n\r\n /**\r\n * Basic editor classnames\r\n */\r\n ui.className = {\r\n\r\n /**\r\n * @const {string} BLOCK_CLASSNAME - redactor blocks name\r\n */\r\n BLOCK_CLASSNAME : 'ce-block',\r\n\r\n /**\r\n * @const {String} wrapper for plugins content\r\n */\r\n BLOCK_CONTENT : 'ce-block__content',\r\n\r\n /**\r\n * @const {String} BLOCK_STRETCHED - makes block stretched\r\n */\r\n BLOCK_STRETCHED : 'ce-block--stretched',\r\n\r\n /**\r\n * @const {String} BLOCK_HIGHLIGHTED - adds background\r\n */\r\n BLOCK_HIGHLIGHTED : 'ce-block--focused',\r\n\r\n /**\r\n * @const {String} - for all default settings\r\n */\r\n SETTINGS_ITEM : 'ce-settings__item'\r\n\r\n };\r\n\r\n /**\r\n * @protected\r\n *\r\n * Making main interface\r\n */\r\n ui.prepare = function () {\r\n\r\n return new Promise(function (resolve) {\r\n\r\n let wrapper = editor.draw.wrapper(),\r\n redactor = editor.draw.redactor(),\r\n toolbar = makeToolBar_();\r\n\r\n wrapper.appendChild(toolbar);\r\n wrapper.appendChild(redactor);\r\n\r\n /** Save created ui-elements to static nodes state */\r\n editor.nodes.wrapper = wrapper;\r\n editor.nodes.redactor = redactor;\r\n\r\n /** Append editor wrapper with redactor zone into holder */\r\n editor.nodes.holder.appendChild(wrapper);\r\n\r\n resolve();\r\n\r\n })\r\n\r\n /** Add toolbox tools */\r\n .then(addTools_)\r\n\r\n /** Make container for inline toolbar */\r\n .then(makeInlineToolbar_)\r\n\r\n /** Add inline toolbar tools */\r\n .then(addInlineToolbarTools_)\r\n\r\n /** Draw wrapper for notifications */\r\n .then(makeNotificationHolder_)\r\n\r\n /** Add eventlisteners to redactor elements */\r\n .then(bindEvents_)\r\n\r\n .catch( function () {\r\n\r\n editor.core.log(\"Can't draw editor interface\");\r\n\r\n });\r\n\r\n };\r\n\r\n /**\r\n * @private\r\n * Draws inline toolbar zone\r\n */\r\n var makeInlineToolbar_ = function () {\r\n\r\n var container = editor.draw.inlineToolbar();\r\n\r\n /** Append to redactor new inline block */\r\n editor.nodes.inlineToolbar.wrapper = container;\r\n\r\n /** Draw toolbar buttons */\r\n editor.nodes.inlineToolbar.buttons = editor.draw.inlineToolbarButtons();\r\n\r\n /** Buttons action or settings */\r\n editor.nodes.inlineToolbar.actions = editor.draw.inlineToolbarActions();\r\n\r\n /** Append to inline toolbar buttons as part of it */\r\n editor.nodes.inlineToolbar.wrapper.appendChild(editor.nodes.inlineToolbar.buttons);\r\n editor.nodes.inlineToolbar.wrapper.appendChild(editor.nodes.inlineToolbar.actions);\r\n\r\n editor.nodes.wrapper.appendChild(editor.nodes.inlineToolbar.wrapper);\r\n\r\n };\r\n\r\n var makeToolBar_ = function () {\r\n\r\n let toolbar = editor.draw.toolbar(),\r\n blockButtons = makeToolbarSettings_(),\r\n toolbarContent = makeToolbarContent_();\r\n\r\n /** Appending first-level block buttons */\r\n toolbar.appendChild(blockButtons);\r\n\r\n /** Append toolbarContent to toolbar */\r\n toolbar.appendChild(toolbarContent);\r\n\r\n /** Make toolbar global */\r\n editor.nodes.toolbar = toolbar;\r\n\r\n return toolbar;\r\n\r\n };\r\n\r\n var makeToolbarContent_ = function () {\r\n\r\n let toolbarContent = editor.draw.toolbarContent(),\r\n toolbox = editor.draw.toolbox(),\r\n plusButton = editor.draw.plusButton();\r\n\r\n /** Append plus button */\r\n toolbarContent.appendChild(plusButton);\r\n\r\n /** Appending toolbar tools */\r\n toolbarContent.appendChild(toolbox);\r\n\r\n /** Make Toolbox and plusButton global */\r\n editor.nodes.toolbox = toolbox;\r\n editor.nodes.plusButton = plusButton;\r\n\r\n return toolbarContent;\r\n\r\n };\r\n\r\n var makeToolbarSettings_ = function () {\r\n\r\n let blockSettings = editor.draw.blockSettings(),\r\n blockButtons = editor.draw.blockButtons(),\r\n defaultSettings = editor.draw.defaultSettings(),\r\n showSettingsButton = editor.draw.settingsButton(),\r\n showTrashButton = editor.toolbar.settings.makeRemoveBlockButton(),\r\n pluginSettings = editor.draw.pluginsSettings();\r\n\r\n /** Add default and plugins settings */\r\n blockSettings.appendChild(pluginSettings);\r\n blockSettings.appendChild(defaultSettings);\r\n\r\n /**\r\n * Make blocks buttons\r\n * This block contains settings button and remove block button\r\n */\r\n blockButtons.appendChild(showSettingsButton);\r\n blockButtons.appendChild(showTrashButton);\r\n blockButtons.appendChild(blockSettings);\r\n\r\n /** Make BlockSettings, PluginSettings, DefaultSettings global */\r\n editor.nodes.blockSettings = blockSettings;\r\n editor.nodes.pluginSettings = pluginSettings;\r\n editor.nodes.defaultSettings = defaultSettings;\r\n editor.nodes.showSettingsButton = showSettingsButton;\r\n editor.nodes.showTrashButton = showTrashButton;\r\n\r\n return blockButtons;\r\n\r\n };\r\n\r\n /** Draw notifications holder */\r\n var makeNotificationHolder_ = function () {\r\n\r\n /** Append block with notifications to the document */\r\n editor.nodes.notifications = editor.notifications.createHolder();\r\n\r\n };\r\n\r\n /**\r\n * @private\r\n * Append tools passed in editor.tools\r\n */\r\n var addTools_ = function () {\r\n\r\n var tool,\r\n toolName,\r\n toolButton;\r\n\r\n for ( toolName in editor.settings.tools ) {\r\n\r\n tool = editor.settings.tools[toolName];\r\n\r\n editor.tools[toolName] = tool;\r\n\r\n if (!tool.iconClassname && tool.displayInToolbox) {\r\n\r\n editor.core.log('Toolbar icon classname missed. Tool %o skipped', 'warn', toolName);\r\n continue;\r\n\r\n }\r\n\r\n if (typeof tool.render != 'function') {\r\n\r\n editor.core.log('render method missed. Tool %o skipped', 'warn', toolName);\r\n continue;\r\n\r\n }\r\n\r\n if (!tool.displayInToolbox) {\r\n\r\n continue;\r\n\r\n } else {\r\n\r\n /** if tools is for toolbox */\r\n toolButton = editor.draw.toolbarButton(toolName, tool.iconClassname);\r\n\r\n editor.nodes.toolbox.appendChild(toolButton);\r\n\r\n editor.nodes.toolbarButtons[toolName] = toolButton;\r\n\r\n }\r\n\r\n }\r\n\r\n };\r\n\r\n var addInlineToolbarTools_ = function () {\r\n\r\n var tools = {\r\n\r\n bold: {\r\n icon : 'ce-icon-bold',\r\n command : 'bold'\r\n },\r\n\r\n italic: {\r\n icon : 'ce-icon-italic',\r\n command : 'italic'\r\n },\r\n\r\n underline: {\r\n icon : 'ce-icon-underline',\r\n command : 'underline'\r\n },\r\n\r\n link: {\r\n icon : 'ce-icon-link',\r\n command : 'createLink'\r\n }\r\n };\r\n\r\n var toolButton,\r\n tool;\r\n\r\n for(var name in tools) {\r\n\r\n tool = tools[name];\r\n\r\n toolButton = editor.draw.toolbarButtonInline(name, tool.icon);\r\n\r\n editor.nodes.inlineToolbar.buttons.appendChild(toolButton);\r\n /**\r\n * Add callbacks to this buttons\r\n */\r\n editor.ui.setInlineToolbarButtonBehaviour(toolButton, tool.command);\r\n\r\n }\r\n\r\n };\r\n\r\n /**\r\n * @private\r\n * Bind editor UI events\r\n */\r\n var bindEvents_ = function () {\r\n\r\n editor.core.log('ui.bindEvents fired', 'info');\r\n\r\n // window.addEventListener('error', function (errorMsg, url, lineNumber) {\r\n // editor.notifications.errorThrown(errorMsg, event);\r\n // }, false );\r\n\r\n /** All keydowns on Document */\r\n editor.listeners.add(document, 'keydown', editor.callback.globalKeydown, false);\r\n\r\n /** All keydowns on Redactor zone */\r\n editor.listeners.add(editor.nodes.redactor, 'keydown', editor.callback.redactorKeyDown, false);\r\n\r\n /** All keydowns on Document */\r\n editor.listeners.add(document, 'keyup', editor.callback.globalKeyup, false );\r\n\r\n /**\r\n * Mouse click to radactor\r\n */\r\n editor.listeners.add(editor.nodes.redactor, 'click', editor.callback.redactorClicked, false );\r\n\r\n /**\r\n * Clicks to the Plus button\r\n */\r\n editor.listeners.add(editor.nodes.plusButton, 'click', editor.callback.plusButtonClicked, false);\r\n\r\n /**\r\n * Clicks to SETTINGS button in toolbar\r\n */\r\n editor.listeners.add(editor.nodes.showSettingsButton, 'click', editor.callback.showSettingsButtonClicked, false );\r\n\r\n /** Bind click listeners on toolbar buttons */\r\n for (var button in editor.nodes.toolbarButtons) {\r\n\r\n editor.listeners.add(editor.nodes.toolbarButtons[button], 'click', editor.callback.toolbarButtonClicked, false);\r\n\r\n }\r\n\r\n };\r\n\r\n ui.addBlockHandlers = function (block) {\r\n\r\n if (!block) return;\r\n\r\n /**\r\n * Block keydowns\r\n */\r\n editor.listeners.add(block, 'keydown', editor.callback.blockKeydown, false);\r\n\r\n /**\r\n * Pasting content from another source\r\n * We have two type of sanitization\r\n * First - uses deep-first search algorithm to get sub nodes,\r\n * sanitizes whole Block_content and replaces cleared nodes\r\n * This method is deprecated\r\n * Method is used in editor.callback.blockPaste(event)\r\n *\r\n * Secont - uses Mutation observer.\r\n * Observer \"observe\" DOM changes and send changings to callback.\r\n * Callback gets changed node, not whole Block_content.\r\n * Inserted or changed node, which we've gotten have been cleared and replaced with diry node\r\n *\r\n * Method is used in editor.callback.blockPasteViaSanitize(event)\r\n *\r\n * @uses html-janitor\r\n * @example editor.callback.blockPasteViaSanitize(event), the second method.\r\n *\r\n */\r\n editor.listeners.add(block, 'paste', editor.callback.blockPasteCallback, false);\r\n\r\n editor.listeners.add(block, 'mouseup', editor.toolbar.inline.show, false);\r\n\r\n };\r\n\r\n /** getting all contenteditable elements */\r\n ui.saveInputs = function () {\r\n\r\n var redactor = editor.nodes.redactor;\r\n\r\n editor.state.inputs = [];\r\n\r\n /** Save all inputs in global variable state */\r\n var inputs = redactor.querySelectorAll('[contenteditable], input, textarea');\r\n\r\n Array.prototype.map.call(inputs, function (current) {\r\n\r\n if (!current.type || current.type == 'text' || current.type == 'textarea') {\r\n\r\n editor.state.inputs.push(current);\r\n\r\n }\r\n\r\n });\r\n\r\n };\r\n\r\n /**\r\n * Adds first initial block on empty redactor\r\n */\r\n ui.addInitialBlock = function () {\r\n\r\n var initialBlockType = editor.settings.initialBlockPlugin,\r\n initialBlock;\r\n\r\n if ( !editor.tools[initialBlockType] ) {\r\n\r\n editor.core.log('Plugin %o was not implemented and can\\'t be used as initial block', 'warn', initialBlockType);\r\n return;\r\n\r\n }\r\n\r\n initialBlock = editor.tools[initialBlockType].render();\r\n\r\n initialBlock.setAttribute('data-placeholder', 'Расскажите свою историю...');\r\n\r\n editor.content.insertBlock({\r\n type : initialBlockType,\r\n block : initialBlock\r\n });\r\n\r\n editor.content.workingNodeChanged(initialBlock);\r\n\r\n };\r\n\r\n ui.setInlineToolbarButtonBehaviour = function (button, type) {\r\n\r\n editor.listeners.add(button, 'mousedown', function (event) {\r\n\r\n editor.toolbar.inline.toolClicked(event, type);\r\n\r\n }, false);\r\n\r\n };\r\n\r\n return ui;\r\n\r\n})({});\r\n\n\n\n// WEBPACK FOOTER //\n// ./modules/ui.js","/**\r\n *\r\n * Codex.Editor Transport Module\r\n *\r\n * @copyright 2017 Codex-Team\r\n * @version 1.2.0\r\n */\r\n\r\nmodule.exports = (function (transport) {\r\n\r\n let editor = codex.editor;\r\n\r\n\r\n /**\r\n * @private {Object} current XmlHttpRequest instance\r\n */\r\n var currentRequest = null;\r\n\r\n\r\n /**\r\n * @type {null} | {DOMElement} input - keeps input element in memory\r\n */\r\n transport.input = null;\r\n\r\n /**\r\n * @property {Object} arguments - keep plugin settings and defined callbacks\r\n */\r\n transport.arguments = null;\r\n\r\n /**\r\n * Prepares input element where will be files\r\n */\r\n transport.prepare = function () {\r\n\r\n let input = editor.draw.node( 'INPUT', '', { type : 'file' } );\r\n\r\n editor.listeners.add(input, 'change', editor.transport.fileSelected);\r\n editor.transport.input = input;\r\n\r\n };\r\n\r\n /** Clear input when files is uploaded */\r\n transport.clearInput = function () {\r\n\r\n /** Remove old input */\r\n transport.input = null;\r\n\r\n /** Prepare new one */\r\n transport.prepare();\r\n\r\n };\r\n\r\n /**\r\n * Callback for file selection\r\n * @param {Event} event\r\n */\r\n transport.fileSelected = function () {\r\n\r\n var input = this,\r\n i,\r\n files = input.files,\r\n formData = new FormData();\r\n\r\n if (editor.transport.arguments.multiple === true) {\r\n\r\n for ( i = 0; i < files.length; i++) {\r\n\r\n formData.append('files[]', files[i], files[i].name);\r\n\r\n }\r\n\r\n } else {\r\n\r\n formData.append('files', files[0], files[0].name);\r\n\r\n }\r\n\r\n currentRequest = editor.core.ajax({\r\n type : 'POST',\r\n data : formData,\r\n url : editor.transport.arguments.url,\r\n beforeSend : editor.transport.arguments.beforeSend,\r\n success : editor.transport.arguments.success,\r\n error : editor.transport.arguments.error,\r\n progress : editor.transport.arguments.progress\r\n });\r\n\r\n /** Clear input */\r\n transport.clearInput();\r\n\r\n };\r\n\r\n /**\r\n * Use plugin callbacks\r\n * @protected\r\n *\r\n * @param {Object} args - can have :\r\n * @param {String} args.url - fetch URL\r\n * @param {Function} args.beforeSend - function calls before sending ajax\r\n * @param {Function} args.success - success callback\r\n * @param {Function} args.error - on error handler\r\n * @param {Function} args.progress - xhr onprogress handler\r\n * @param {Boolean} args.multiple - allow select several files\r\n * @param {String} args.accept - adds accept attribute\r\n */\r\n transport.selectAndUpload = function (args) {\r\n\r\n transport.arguments = args;\r\n\r\n if ( args.multiple === true) {\r\n\r\n transport.input.setAttribute('multiple', 'multiple');\r\n\r\n }\r\n\r\n if ( args.accept ) {\r\n\r\n transport.input.setAttribute('accept', args.accept);\r\n\r\n }\r\n\r\n transport.input.click();\r\n\r\n };\r\n\r\n transport.abort = function () {\r\n\r\n currentRequest.abort();\r\n\r\n currentRequest = null;\r\n\r\n };\r\n\r\n return transport;\r\n\r\n})({});\n\n\n// WEBPACK FOOTER //\n// ./modules/transport.js","/**\r\n * Codex Editor Renderer Module\r\n *\r\n * @author Codex Team\r\n * @version 1.0\r\n */\r\n\r\nmodule.exports = (function (renderer) {\r\n\r\n let editor = codex.editor;\r\n\r\n /**\r\n * Asyncronously parses input JSON to redactor blocks\r\n */\r\n renderer.makeBlocksFromData = function () {\r\n\r\n /**\r\n * If redactor is empty, add first paragraph to start writing\r\n */\r\n if (editor.core.isEmpty(editor.state.blocks) || !editor.state.blocks.items.length) {\r\n\r\n editor.ui.addInitialBlock();\r\n return;\r\n\r\n }\r\n\r\n Promise.resolve()\r\n\r\n /** First, get JSON from state */\r\n .then(function () {\r\n\r\n return editor.state.blocks;\r\n\r\n })\r\n\r\n /** Then, start to iterate they */\r\n .then(editor.renderer.appendBlocks)\r\n\r\n /** Write log if something goes wrong */\r\n .catch(function (error) {\r\n\r\n editor.core.log('Error while parsing JSON: %o', 'error', error);\r\n\r\n });\r\n\r\n };\r\n\r\n /**\r\n * Parses JSON to blocks\r\n * @param {object} data\r\n * @return Primise -> nodeList\r\n */\r\n renderer.appendBlocks = function (data) {\r\n\r\n var blocks = data.items;\r\n\r\n /**\r\n * Sequence of one-by-one blocks appending\r\n * Uses to save blocks order after async-handler\r\n */\r\n var nodeSequence = Promise.resolve();\r\n\r\n for (var index = 0; index < blocks.length ; index++ ) {\r\n\r\n /** Add node to sequence at specified index */\r\n editor.renderer.appendNodeAtIndex(nodeSequence, blocks, index);\r\n\r\n }\r\n\r\n };\r\n\r\n /**\r\n * Append node at specified index\r\n */\r\n renderer.appendNodeAtIndex = function (nodeSequence, blocks, index) {\r\n\r\n /** We need to append node to sequence */\r\n nodeSequence\r\n\r\n /** first, get node async-aware */\r\n .then(function () {\r\n\r\n return editor.renderer.getNodeAsync(blocks, index);\r\n\r\n })\r\n\r\n /**\r\n * second, compose editor-block from JSON object\r\n */\r\n .then(editor.renderer.createBlockFromData)\r\n\r\n /**\r\n * now insert block to redactor\r\n */\r\n .then(function (blockData) {\r\n\r\n /**\r\n * blockData has 'block', 'type' and 'stretched' information\r\n */\r\n editor.content.insertBlock(blockData);\r\n\r\n /** Pass created block to next step */\r\n return blockData.block;\r\n\r\n })\r\n\r\n /** Log if something wrong with node */\r\n .catch(function (error) {\r\n\r\n editor.core.log('Node skipped while parsing because %o', 'error', error);\r\n\r\n });\r\n\r\n };\r\n\r\n /**\r\n * Asynchronously returns block data from blocksList by index\r\n * @return Promise to node\r\n */\r\n renderer.getNodeAsync = function (blocksList, index) {\r\n\r\n return Promise.resolve().then(function () {\r\n\r\n return {\r\n tool : blocksList[index],\r\n position : index\r\n };\r\n\r\n });\r\n\r\n };\r\n\r\n /**\r\n * Creates editor block by JSON-data\r\n *\r\n * @uses render method of each plugin\r\n *\r\n * @param {Object} toolData.tool\r\n * { header : {\r\n * text: '',\r\n * type: 'H3', ...\r\n * }\r\n * }\r\n * @param {Number} toolData.position - index in input-blocks array\r\n * @return {Object} with type and Element\r\n */\r\n renderer.createBlockFromData = function ( toolData ) {\r\n\r\n /** New parser */\r\n var block,\r\n tool = toolData.tool,\r\n pluginName = tool.type;\r\n\r\n /** Get first key of object that stores plugin name */\r\n // for (var pluginName in blockData) break;\r\n\r\n /** Check for plugin existance */\r\n if (!editor.tools[pluginName]) {\r\n\r\n throw Error(`Plugin «${pluginName}» not found`);\r\n\r\n }\r\n\r\n /** Check for plugin having render method */\r\n if (typeof editor.tools[pluginName].render != 'function') {\r\n\r\n throw Error(`Plugin «${pluginName}» must have «render» method`);\r\n\r\n }\r\n\r\n if ( editor.tools[pluginName].available === false ) {\r\n\r\n block = editor.draw.unavailableBlock();\r\n\r\n block.innerHTML = editor.tools[pluginName].loadingMessage;\r\n\r\n /**\r\n * Saver will extract data from initial block data by position in array\r\n */\r\n block.dataset.inputPosition = toolData.position;\r\n\r\n } else {\r\n\r\n /** New Parser */\r\n block = editor.tools[pluginName].render(tool.data);\r\n\r\n }\r\n\r\n /** is first-level block stretched */\r\n var stretched = editor.tools[pluginName].isStretched || false;\r\n\r\n /** Retrun type and block */\r\n return {\r\n type : pluginName,\r\n block : block,\r\n stretched : stretched\r\n };\r\n\r\n };\r\n\r\n return renderer;\r\n\r\n})({});\n\n\n// WEBPACK FOOTER //\n// ./modules/renderer.js","/**\r\n * Codex Editor Saver\r\n *\r\n * @author Codex Team\r\n * @version 1.0.2\r\n */\r\n\r\nmodule.exports = (function (saver) {\r\n\r\n let editor = codex.editor;\r\n\r\n /**\r\n * Saves blocks\r\n * @private\r\n */\r\n saver.saveBlocks = function () {\r\n\r\n /** Save html content of redactor to memory */\r\n editor.state.html = editor.nodes.redactor.innerHTML;\r\n\r\n /** Empty jsonOutput state */\r\n editor.state.jsonOutput = [];\r\n\r\n Promise.resolve()\r\n\r\n .then(function () {\r\n\r\n return editor.nodes.redactor.childNodes;\r\n\r\n })\r\n /** Making a sequence from separate blocks */\r\n .then(editor.saver.makeQueue)\r\n\r\n .then(function () {\r\n // editor.nodes.textarea.innerHTML = editor.state.html;\r\n })\r\n\r\n .catch( function (error) {\r\n\r\n editor.core.log(error);\r\n\r\n });\r\n\r\n };\r\n\r\n saver.makeQueue = function (blocks) {\r\n\r\n var queue = Promise.resolve();\r\n\r\n for(var index = 0; index < blocks.length; index++) {\r\n\r\n /** Add node to sequence at specified index */\r\n editor.saver.getBlockData(queue, blocks, index);\r\n\r\n }\r\n\r\n };\r\n\r\n /** Gets every block and makes From Data */\r\n saver.getBlockData = function (queue, blocks, index) {\r\n\r\n queue.then(function () {\r\n\r\n return editor.saver.getNodeAsync(blocks, index);\r\n\r\n })\r\n\r\n .then(editor.saver.makeFormDataFromBlocks);\r\n\r\n };\r\n\r\n\r\n /**\r\n * Asynchronously returns block data from blocksList by index\r\n * @return Promise to node\r\n */\r\n saver.getNodeAsync = function (blocksList, index) {\r\n\r\n return Promise.resolve().then(function () {\r\n\r\n return blocksList[index];\r\n\r\n });\r\n\r\n };\r\n\r\n saver.makeFormDataFromBlocks = function (block) {\r\n\r\n var pluginName = block.dataset.tool;\r\n\r\n /** Check for plugin existance */\r\n if (!editor.tools[pluginName]) {\r\n\r\n throw Error(`Plugin «${pluginName}» not found`);\r\n\r\n }\r\n\r\n /** Check for plugin having render method */\r\n if (typeof editor.tools[pluginName].save != 'function') {\r\n\r\n throw Error(`Plugin «${pluginName}» must have save method`);\r\n\r\n }\r\n\r\n /** Result saver */\r\n var blockContent = block.childNodes[0],\r\n pluginsContent = blockContent.childNodes[0],\r\n savedData,\r\n position,\r\n output;\r\n\r\n /** If plugin wasn't available then return data from cache */\r\n if ( editor.tools[pluginName].available === false ) {\r\n\r\n position = pluginsContent.dataset.inputPosition;\r\n savedData = codex.editor.state.blocks.items[position].data;\r\n\r\n } else {\r\n\r\n savedData = editor.tools[pluginName].save(pluginsContent);\r\n\r\n if (editor.tools[pluginName].validate) {\r\n\r\n var result = editor.tools[pluginName].validate(savedData);\r\n\r\n /**\r\n * Do not allow invalid data\r\n */\r\n if (!result)\r\n return;\r\n\r\n }\r\n\r\n }\r\n\r\n output = {\r\n type : pluginName,\r\n data : savedData\r\n };\r\n\r\n editor.state.jsonOutput.push(output);\r\n\r\n };\r\n\r\n return saver;\r\n\r\n})({});\n\n\n// WEBPACK FOOTER //\n// ./modules/saver.js","/**\r\n * Codex Editor Content Module\r\n * Works with DOM\r\n *\r\n * @module Codex Editor content module\r\n *\r\n * @author Codex Team\r\n * @version 1.3.13\r\n *\r\n * @description Module works with Elements that have been appended to the main DOM\r\n */\r\n\r\nmodule.exports = (function (content) {\r\n\r\n let editor = codex.editor;\r\n\r\n /**\r\n * Links to current active block\r\n * @type {null | Element}\r\n */\r\n content.currentNode = null;\r\n\r\n /**\r\n * clicked in redactor area\r\n * @type {null | Boolean}\r\n */\r\n content.editorAreaHightlighted = null;\r\n\r\n /**\r\n * @deprecated\r\n * Synchronizes redactor with original textarea\r\n */\r\n content.sync = function () {\r\n\r\n editor.core.log('syncing...');\r\n\r\n /**\r\n * Save redactor content to editor.state\r\n */\r\n editor.state.html = editor.nodes.redactor.innerHTML;\r\n\r\n };\r\n\r\n /**\r\n * Appends background to the block\r\n *\r\n * @description add CSS class to highlight visually first-level block area\r\n */\r\n content.markBlock = function () {\r\n\r\n editor.content.currentNode.classList.add(editor.ui.className.BLOCK_HIGHLIGHTED);\r\n\r\n };\r\n\r\n /**\r\n * Clear background\r\n *\r\n * @description clears styles that highlights block\r\n */\r\n content.clearMark = function () {\r\n\r\n if (editor.content.currentNode) {\r\n\r\n editor.content.currentNode.classList.remove(editor.ui.className.BLOCK_HIGHLIGHTED);\r\n\r\n }\r\n\r\n };\r\n\r\n /**\r\n * Finds first-level block\r\n *\r\n * @param {Element} node - selected or clicked in redactors area node\r\n * @protected\r\n *\r\n * @description looks for first-level block.\r\n * gets parent while node is not first-level\r\n */\r\n content.getFirstLevelBlock = function (node) {\r\n\r\n if (!editor.core.isDomNode(node)) {\r\n\r\n node = node.parentNode;\r\n\r\n }\r\n\r\n if (node === editor.nodes.redactor || node === document.body) {\r\n\r\n return null;\r\n\r\n } else {\r\n\r\n while(!node.classList.contains(editor.ui.className.BLOCK_CLASSNAME)) {\r\n\r\n node = node.parentNode;\r\n\r\n }\r\n\r\n return node;\r\n\r\n }\r\n\r\n };\r\n\r\n /**\r\n * Trigger this event when working node changed\r\n * @param {Element} targetNode - first-level of this node will be current\r\n * @protected\r\n *\r\n * @description If targetNode is first-level then we set it as current else we look for parents to find first-level\r\n */\r\n content.workingNodeChanged = function (targetNode) {\r\n\r\n /** Clear background from previous marked block before we change */\r\n editor.content.clearMark();\r\n\r\n if (!targetNode) {\r\n\r\n return;\r\n\r\n }\r\n\r\n content.currentNode = content.getFirstLevelBlock(targetNode);\r\n\r\n };\r\n\r\n /**\r\n * Replaces one redactor block with another\r\n * @protected\r\n * @param {Element} targetBlock - block to replace. Mostly currentNode.\r\n * @param {Element} newBlock\r\n * @param {string} newBlockType - type of new block; we need to store it to data-attribute\r\n *\r\n * [!] Function does not saves old block content.\r\n * You can get it manually and pass with newBlock.innerHTML\r\n */\r\n content.replaceBlock = function (targetBlock, newBlock) {\r\n\r\n if (!targetBlock || !newBlock) {\r\n\r\n editor.core.log('replaceBlock: missed params');\r\n return;\r\n\r\n }\r\n\r\n /** If target-block is not a frist-level block, then we iterate parents to find it */\r\n while(!targetBlock.classList.contains(editor.ui.className.BLOCK_CLASSNAME)) {\r\n\r\n targetBlock = targetBlock.parentNode;\r\n\r\n }\r\n\r\n /** Replacing */\r\n editor.nodes.redactor.replaceChild(newBlock, targetBlock);\r\n\r\n /**\r\n * Set new node as current\r\n */\r\n editor.content.workingNodeChanged(newBlock);\r\n\r\n /**\r\n * Add block handlers\r\n */\r\n editor.ui.addBlockHandlers(newBlock);\r\n\r\n /**\r\n * Save changes\r\n */\r\n editor.ui.saveInputs();\r\n\r\n };\r\n\r\n /**\r\n * @protected\r\n *\r\n * Inserts new block to redactor\r\n * Wrapps block into a DIV with BLOCK_CLASSNAME class\r\n *\r\n * @param blockData {object}\r\n * @param blockData.block {Element} element with block content\r\n * @param blockData.type {string} block plugin\r\n * @param needPlaceCaret {bool} pass true to set caret in new block\r\n *\r\n */\r\n content.insertBlock = function ( blockData, needPlaceCaret ) {\r\n\r\n var workingBlock = editor.content.currentNode,\r\n newBlockContent = blockData.block,\r\n blockType = blockData.type,\r\n isStretched = blockData.stretched;\r\n\r\n var newBlock = composeNewBlock_(newBlockContent, blockType, isStretched);\r\n\r\n if (workingBlock) {\r\n\r\n editor.core.insertAfter(workingBlock, newBlock);\r\n\r\n } else {\r\n\r\n /**\r\n * If redactor is empty, append as first child\r\n */\r\n editor.nodes.redactor.appendChild(newBlock);\r\n\r\n }\r\n\r\n /**\r\n * Block handler\r\n */\r\n editor.ui.addBlockHandlers(newBlock);\r\n\r\n /**\r\n * Set new node as current\r\n */\r\n editor.content.workingNodeChanged(newBlock);\r\n\r\n /**\r\n * Save changes\r\n */\r\n editor.ui.saveInputs();\r\n\r\n\r\n if ( needPlaceCaret ) {\r\n\r\n /**\r\n * If we don't know input index then we set default value -1\r\n */\r\n var currentInputIndex = editor.caret.getCurrentInputIndex() || -1;\r\n\r\n\r\n if (currentInputIndex == -1) {\r\n\r\n\r\n var editableElement = newBlock.querySelector('[contenteditable]'),\r\n emptyText = document.createTextNode('');\r\n\r\n editableElement.appendChild(emptyText);\r\n editor.caret.set(editableElement, 0, 0);\r\n\r\n editor.toolbar.move();\r\n editor.toolbar.showPlusButton();\r\n\r\n\r\n } else {\r\n\r\n if (currentInputIndex === editor.state.inputs.length - 1)\r\n return;\r\n\r\n /** Timeout for browsers execution */\r\n window.setTimeout(function () {\r\n\r\n /** Setting to the new input */\r\n editor.caret.setToNextBlock(currentInputIndex);\r\n editor.toolbar.move();\r\n editor.toolbar.open();\r\n\r\n }, 10);\r\n\r\n }\r\n\r\n }\r\n\r\n /**\r\n * Block is inserted, wait for new click that defined focusing on editors area\r\n * @type {boolean}\r\n */\r\n content.editorAreaHightlighted = false;\r\n\r\n };\r\n\r\n /**\r\n * Replaces blocks with saving content\r\n * @protected\r\n * @param {Element} noteToReplace\r\n * @param {Element} newNode\r\n * @param {Element} blockType\r\n */\r\n content.switchBlock = function (blockToReplace, newBlock, tool) {\r\n\r\n tool = tool || editor.content.currentNode.dataset.tool;\r\n var newBlockComposed = composeNewBlock_(newBlock, tool);\r\n\r\n /** Replacing */\r\n editor.content.replaceBlock(blockToReplace, newBlockComposed);\r\n\r\n /** Save new Inputs when block is changed */\r\n editor.ui.saveInputs();\r\n\r\n };\r\n\r\n /**\r\n * Iterates between child noted and looking for #text node on deepest level\r\n * @protected\r\n *\r\n * @param {Element} block - node where find\r\n * @param {int} postiton - starting postion\r\n * Example: childNodex.length to find from the end\r\n * or 0 to find from the start\r\n * @return {Text} block\r\n * @uses DFS\r\n */\r\n content.getDeepestTextNodeFromPosition = function (block, position) {\r\n\r\n /**\r\n * Clear Block from empty and useless spaces with trim.\r\n * Such nodes we should remove\r\n */\r\n var blockChilds = block.childNodes,\r\n index,\r\n node,\r\n text;\r\n\r\n for(index = 0; index < blockChilds.length; index++) {\r\n\r\n node = blockChilds[index];\r\n\r\n if (node.nodeType == editor.core.nodeTypes.TEXT) {\r\n\r\n text = node.textContent.trim();\r\n\r\n /** Text is empty. We should remove this child from node before we start DFS\r\n * decrease the quantity of childs.\r\n */\r\n if (text === '') {\r\n\r\n block.removeChild(node);\r\n position--;\r\n\r\n }\r\n\r\n }\r\n\r\n }\r\n\r\n if (block.childNodes.length === 0) {\r\n\r\n return document.createTextNode('');\r\n\r\n }\r\n\r\n /** Setting default position when we deleted all empty nodes */\r\n if ( position < 0 )\r\n position = 1;\r\n\r\n var lookingFromStart = false;\r\n\r\n /** For looking from START */\r\n if (position === 0) {\r\n\r\n lookingFromStart = true;\r\n position = 1;\r\n\r\n }\r\n\r\n while ( position ) {\r\n\r\n /** initial verticle of node. */\r\n if ( lookingFromStart ) {\r\n\r\n block = block.childNodes[0];\r\n\r\n } else {\r\n\r\n block = block.childNodes[position - 1];\r\n\r\n }\r\n\r\n if ( block.nodeType == editor.core.nodeTypes.TAG ) {\r\n\r\n position = block.childNodes.length;\r\n\r\n } else if (block.nodeType == editor.core.nodeTypes.TEXT ) {\r\n\r\n position = 0;\r\n\r\n }\r\n\r\n }\r\n\r\n return block;\r\n\r\n };\r\n\r\n /**\r\n * @private\r\n * @param {Element} block - current plugins render\r\n * @param {String} tool - plugins name\r\n * @param {Boolean} isStretched - make stretched block or not\r\n *\r\n * @description adds necessary information to wrap new created block by first-level holder\r\n */\r\n var composeNewBlock_ = function (block, tool, isStretched) {\r\n\r\n var newBlock = editor.draw.node('DIV', editor.ui.className.BLOCK_CLASSNAME, {}),\r\n blockContent = editor.draw.node('DIV', editor.ui.className.BLOCK_CONTENT, {});\r\n\r\n blockContent.appendChild(block);\r\n newBlock.appendChild(blockContent);\r\n\r\n if (isStretched) {\r\n\r\n blockContent.classList.add(editor.ui.className.BLOCK_STRETCHED);\r\n\r\n }\r\n\r\n newBlock.dataset.tool = tool;\r\n return newBlock;\r\n\r\n };\r\n\r\n /**\r\n * Returns Range object of current selection\r\n * @protected\r\n */\r\n content.getRange = function () {\r\n\r\n var selection = window.getSelection().getRangeAt(0);\r\n\r\n return selection;\r\n\r\n };\r\n\r\n /**\r\n * Divides block in two blocks (after and before caret)\r\n *\r\n * @protected\r\n * @param {int} inputIndex - target input index\r\n *\r\n * @description splits current input content to the separate blocks\r\n * When enter is pressed among the words, that text will be splited.\r\n */\r\n content.splitBlock = function (inputIndex) {\r\n\r\n var selection = window.getSelection(),\r\n anchorNode = selection.anchorNode,\r\n anchorNodeText = anchorNode.textContent,\r\n caretOffset = selection.anchorOffset,\r\n textBeforeCaret,\r\n textNodeBeforeCaret,\r\n textAfterCaret,\r\n textNodeAfterCaret;\r\n\r\n var currentBlock = editor.content.currentNode.querySelector('[contentEditable]');\r\n\r\n\r\n textBeforeCaret = anchorNodeText.substring(0, caretOffset);\r\n textAfterCaret = anchorNodeText.substring(caretOffset);\r\n\r\n textNodeBeforeCaret = document.createTextNode(textBeforeCaret);\r\n\r\n if (textAfterCaret) {\r\n\r\n textNodeAfterCaret = document.createTextNode(textAfterCaret);\r\n\r\n }\r\n\r\n var previousChilds = [],\r\n nextChilds = [],\r\n reachedCurrent = false;\r\n\r\n if (textNodeAfterCaret) {\r\n\r\n nextChilds.push(textNodeAfterCaret);\r\n\r\n }\r\n\r\n for ( var i = 0, child; !!(child = currentBlock.childNodes[i]); i++) {\r\n\r\n if ( child != anchorNode ) {\r\n\r\n if ( !reachedCurrent ) {\r\n\r\n previousChilds.push(child);\r\n\r\n } else {\r\n\r\n nextChilds.push(child);\r\n\r\n }\r\n\r\n } else {\r\n\r\n reachedCurrent = true;\r\n\r\n }\r\n\r\n }\r\n\r\n /** Clear current input */\r\n editor.state.inputs[inputIndex].innerHTML = '';\r\n\r\n /**\r\n * Append all childs founded before anchorNode\r\n */\r\n var previousChildsLength = previousChilds.length;\r\n\r\n for(i = 0; i < previousChildsLength; i++) {\r\n\r\n editor.state.inputs[inputIndex].appendChild(previousChilds[i]);\r\n\r\n }\r\n\r\n editor.state.inputs[inputIndex].appendChild(textNodeBeforeCaret);\r\n\r\n /**\r\n * Append text node which is after caret\r\n */\r\n var nextChildsLength = nextChilds.length,\r\n newNode = document.createElement('div');\r\n\r\n for(i = 0; i < nextChildsLength; i++) {\r\n\r\n newNode.appendChild(nextChilds[i]);\r\n\r\n }\r\n\r\n newNode = newNode.innerHTML;\r\n\r\n /** This type of block creates when enter is pressed */\r\n var NEW_BLOCK_TYPE = editor.settings.initialBlockPlugin;\r\n\r\n /**\r\n * Make new paragraph with text after caret\r\n */\r\n editor.content.insertBlock({\r\n type : NEW_BLOCK_TYPE,\r\n block : editor.tools[NEW_BLOCK_TYPE].render({\r\n text : newNode\r\n })\r\n }, true );\r\n\r\n };\r\n\r\n /**\r\n * Merges two blocks — current and target\r\n * If target index is not exist, then previous will be as target\r\n *\r\n * @protected\r\n * @param {int} currentInputIndex\r\n * @param {int} targetInputIndex\r\n *\r\n * @description gets two inputs indexes and merges into one\r\n */\r\n content.mergeBlocks = function (currentInputIndex, targetInputIndex) {\r\n\r\n /** If current input index is zero, then prevent method execution */\r\n if (currentInputIndex === 0) {\r\n\r\n return;\r\n\r\n }\r\n\r\n var targetInput,\r\n currentInputContent = editor.state.inputs[currentInputIndex].innerHTML;\r\n\r\n if (!targetInputIndex) {\r\n\r\n targetInput = editor.state.inputs[currentInputIndex - 1];\r\n\r\n } else {\r\n\r\n targetInput = editor.state.inputs[targetInputIndex];\r\n\r\n }\r\n\r\n targetInput.innerHTML += currentInputContent;\r\n\r\n };\r\n\r\n /**\r\n * Iterates all right siblings and parents, which has right siblings\r\n * while it does not reached the first-level block\r\n *\r\n * @param {Element} node\r\n * @return {boolean}\r\n */\r\n content.isLastNode = function (node) {\r\n\r\n // console.log('погнали перебор родителей');\r\n\r\n var allChecked = false;\r\n\r\n while ( !allChecked ) {\r\n\r\n // console.log('Смотрим на %o', node);\r\n // console.log('Проверим, пустые ли соседи справа');\r\n\r\n if ( !allSiblingsEmpty_(node) ) {\r\n\r\n // console.log('Есть непустые соседи. Узел не последний. Выходим.');\r\n return false;\r\n\r\n }\r\n\r\n node = node.parentNode;\r\n\r\n /**\r\n * Проверяем родителей до тех пор, пока не найдем блок первого уровня\r\n */\r\n if ( node.classList.contains(editor.ui.className.BLOCK_CONTENT) ) {\r\n\r\n allChecked = true;\r\n\r\n }\r\n\r\n }\r\n\r\n return true;\r\n\r\n };\r\n\r\n /**\r\n * Checks if all element right siblings is empty\r\n * @param node\r\n */\r\n var allSiblingsEmpty_ = function (node) {\r\n\r\n /**\r\n * Нужно убедиться, что после пустого соседа ничего нет\r\n */\r\n var sibling = node.nextSibling;\r\n\r\n while ( sibling ) {\r\n\r\n if (sibling.textContent.length) {\r\n\r\n return false;\r\n\r\n }\r\n\r\n sibling = sibling.nextSibling;\r\n\r\n }\r\n\r\n return true;\r\n\r\n };\r\n\r\n /**\r\n * @public\r\n *\r\n * @param {string} htmlData - html content as string\r\n * @param {string} plainData - plain text\r\n * @return {string} - html content as string\r\n */\r\n content.wrapTextWithParagraphs = function (htmlData, plainData) {\r\n\r\n if (!htmlData) {\r\n\r\n return '

    ' + plainData.split('\\n\\n').join('

    ') + '

    ';\r\n\r\n }\r\n\r\n var wrapper = document.createElement('DIV'),\r\n newWrapper = document.createElement('DIV'),\r\n i,\r\n paragraph,\r\n firstLevelBlocks = ['DIV', 'P'],\r\n blockTyped,\r\n node;\r\n\r\n /**\r\n * Make HTML Element to Wrap Text\r\n * It allows us to work with input data as HTML content\r\n */\r\n wrapper.innerHTML = htmlData;\r\n paragraph = document.createElement('P');\r\n\r\n for (i = 0; i < wrapper.childNodes.length; i++) {\r\n\r\n node = wrapper.childNodes[i];\r\n\r\n blockTyped = firstLevelBlocks.indexOf(node.tagName) != -1;\r\n\r\n /**\r\n * If node is first-levet\r\n * we add this node to our new wrapper\r\n */\r\n if ( blockTyped ) {\r\n\r\n /**\r\n * If we had splitted inline nodes to paragraph before\r\n */\r\n if ( paragraph.childNodes.length ) {\r\n\r\n newWrapper.appendChild(paragraph.cloneNode(true));\r\n\r\n /** empty paragraph */\r\n paragraph = null;\r\n paragraph = document.createElement('P');\r\n\r\n }\r\n\r\n newWrapper.appendChild(node.cloneNode(true));\r\n\r\n } else {\r\n\r\n /** Collect all inline nodes to one as paragraph */\r\n paragraph.appendChild(node.cloneNode(true));\r\n\r\n /** if node is last we should append this node to paragraph and paragraph to new wrapper */\r\n if ( i == wrapper.childNodes.length - 1 ) {\r\n\r\n newWrapper.appendChild(paragraph.cloneNode(true));\r\n\r\n }\r\n\r\n }\r\n\r\n }\r\n\r\n return newWrapper.innerHTML;\r\n\r\n };\r\n\r\n /**\r\n * Finds closest Contenteditable parent from Element\r\n * @param {Element} node element looking from\r\n * @return {Element} node contenteditable\r\n */\r\n content.getEditableParent = function (node) {\r\n\r\n while (node && node.contentEditable != 'true') {\r\n\r\n node = node.parentNode;\r\n\r\n }\r\n\r\n return node;\r\n\r\n };\r\n\r\n return content;\r\n\r\n})({});\n\n\n// WEBPACK FOOTER //\n// ./modules/content.js","/**\r\n * Codex Editor toolbar module\r\n *\r\n * Contains:\r\n * - Inline toolbox\r\n * - Toolbox within plus button\r\n * - Settings section\r\n *\r\n * @author Codex Team\r\n * @version 1.0\r\n */\r\n\r\nmodule.exports = (function (toolbar) {\r\n\r\n let editor = codex.editor;\r\n\r\n toolbar.settings = require('./settings');\r\n toolbar.inline = require('./inline');\r\n toolbar.toolbox = require('./toolbox');\r\n\r\n /**\r\n * Margin between focused node and toolbar\r\n */\r\n toolbar.defaultToolbarHeight = 49;\r\n\r\n toolbar.defaultOffset = 34;\r\n\r\n toolbar.opened = false;\r\n\r\n toolbar.current = null;\r\n\r\n /**\r\n * @protected\r\n */\r\n toolbar.open = function () {\r\n\r\n if (editor.hideToolbar) {\r\n\r\n return;\r\n\r\n }\r\n\r\n editor.nodes.toolbar.classList.add('opened');\r\n this.opened = true;\r\n\r\n };\r\n\r\n /**\r\n * @protected\r\n */\r\n toolbar.close = function () {\r\n\r\n editor.nodes.toolbar.classList.remove('opened');\r\n\r\n toolbar.opened = false;\r\n toolbar.current = null;\r\n\r\n for (var button in editor.nodes.toolbarButtons) {\r\n\r\n editor.nodes.toolbarButtons[button].classList.remove('selected');\r\n\r\n }\r\n\r\n /** Close toolbox when toolbar is not displayed */\r\n editor.toolbar.toolbox.close();\r\n editor.toolbar.settings.close();\r\n\r\n };\r\n\r\n toolbar.toggle = function () {\r\n\r\n if ( !this.opened ) {\r\n\r\n this.open();\r\n\r\n } else {\r\n\r\n this.close();\r\n\r\n }\r\n\r\n };\r\n\r\n toolbar.hidePlusButton = function () {\r\n\r\n editor.nodes.plusButton.classList.add('hide');\r\n\r\n };\r\n\r\n toolbar.showPlusButton = function () {\r\n\r\n editor.nodes.plusButton.classList.remove('hide');\r\n\r\n };\r\n\r\n /**\r\n * Moving toolbar to the specified node\r\n */\r\n toolbar.move = function () {\r\n\r\n /** Close Toolbox when we move toolbar */\r\n editor.toolbar.toolbox.close();\r\n\r\n if (!editor.content.currentNode) {\r\n\r\n return;\r\n\r\n }\r\n\r\n var newYCoordinate = editor.content.currentNode.offsetTop - (editor.toolbar.defaultToolbarHeight / 2) + editor.toolbar.defaultOffset;\r\n\r\n editor.nodes.toolbar.style.transform = `translate3D(0, ${Math.floor(newYCoordinate)}px, 0)`;\r\n\r\n /** Close trash actions */\r\n editor.toolbar.settings.hideRemoveActions();\r\n\r\n };\r\n\r\n return toolbar;\r\n\r\n})({});\n\n\n// WEBPACK FOOTER //\n// ./modules/toolbar/toolbar.js","/**\r\n * Toolbar settings\r\n *\r\n * @version 1.0.4\r\n */\r\n\r\nmodule.exports = (function (settings) {\r\n\r\n let editor = codex.editor;\r\n\r\n settings.opened = false;\r\n\r\n settings.setting = null;\r\n settings.actions = null;\r\n\r\n /**\r\n * Append and open settings\r\n */\r\n settings.open = function (toolType) {\r\n\r\n /**\r\n * Append settings content\r\n * It's stored in tool.settings\r\n */\r\n if (!editor.tools[toolType] || !editor.tools[toolType].makeSettings ) {\r\n\r\n editor.core.log(`Plugin «${toolType}» has no settings`, 'warn');\r\n editor.nodes.pluginSettings.innerHTML = `Плагин «${toolType}» не имеет настроек`;\r\n\r\n } else {\r\n\r\n /**\r\n * Draw settings block\r\n */\r\n var settingsBlock = editor.tools[toolType].makeSettings();\r\n\r\n editor.nodes.pluginSettings.appendChild(settingsBlock);\r\n\r\n }\r\n\r\n /** Open settings block */\r\n editor.nodes.blockSettings.classList.add('opened');\r\n this.opened = true;\r\n\r\n };\r\n\r\n /**\r\n * Close and clear settings\r\n */\r\n settings.close = function () {\r\n\r\n editor.nodes.blockSettings.classList.remove('opened');\r\n editor.nodes.pluginSettings.innerHTML = '';\r\n\r\n this.opened = false;\r\n\r\n };\r\n\r\n /**\r\n * @param {string} toolType - plugin type\r\n */\r\n settings.toggle = function ( toolType ) {\r\n\r\n if ( !this.opened ) {\r\n\r\n this.open(toolType);\r\n\r\n } else {\r\n\r\n this.close();\r\n\r\n }\r\n\r\n };\r\n\r\n /**\r\n * Here we will draw buttons and add listeners to components\r\n */\r\n settings.makeRemoveBlockButton = function () {\r\n\r\n var removeBlockWrapper = editor.draw.node('SPAN', 'ce-toolbar__remove-btn', {}),\r\n settingButton = editor.draw.node('SPAN', 'ce-toolbar__remove-setting', { innerHTML : '' }),\r\n actionWrapper = editor.draw.node('DIV', 'ce-toolbar__remove-confirmation', {}),\r\n confirmAction = editor.draw.node('DIV', 'ce-toolbar__remove-confirm', { textContent : 'Удалить блок' }),\r\n cancelAction = editor.draw.node('DIV', 'ce-toolbar__remove-cancel', { textContent : 'Отмена' });\r\n\r\n editor.listeners.add(settingButton, 'click', editor.toolbar.settings.removeButtonClicked, false);\r\n\r\n editor.listeners.add(confirmAction, 'click', editor.toolbar.settings.confirmRemovingRequest, false);\r\n\r\n editor.listeners.add(cancelAction, 'click', editor.toolbar.settings.cancelRemovingRequest, false);\r\n\r\n actionWrapper.appendChild(confirmAction);\r\n actionWrapper.appendChild(cancelAction);\r\n\r\n removeBlockWrapper.appendChild(settingButton);\r\n removeBlockWrapper.appendChild(actionWrapper);\r\n\r\n /** Save setting */\r\n editor.toolbar.settings.setting = settingButton;\r\n editor.toolbar.settings.actions = actionWrapper;\r\n\r\n return removeBlockWrapper;\r\n\r\n };\r\n\r\n settings.removeButtonClicked = function () {\r\n\r\n var action = editor.toolbar.settings.actions;\r\n\r\n if (action.classList.contains('opened')) {\r\n\r\n editor.toolbar.settings.hideRemoveActions();\r\n\r\n } else {\r\n\r\n editor.toolbar.settings.showRemoveActions();\r\n\r\n }\r\n\r\n editor.toolbar.toolbox.close();\r\n editor.toolbar.settings.close();\r\n\r\n };\r\n\r\n settings.cancelRemovingRequest = function () {\r\n\r\n editor.toolbar.settings.actions.classList.remove('opened');\r\n\r\n };\r\n\r\n settings.confirmRemovingRequest = function () {\r\n\r\n var currentBlock = editor.content.currentNode,\r\n firstLevelBlocksCount;\r\n\r\n currentBlock.remove();\r\n\r\n firstLevelBlocksCount = editor.nodes.redactor.childNodes.length;\r\n\r\n /**\r\n * If all blocks are removed\r\n */\r\n if (firstLevelBlocksCount === 0) {\r\n\r\n /** update currentNode variable */\r\n editor.content.currentNode = null;\r\n\r\n /** Inserting new empty initial block */\r\n editor.ui.addInitialBlock();\r\n\r\n }\r\n\r\n editor.ui.saveInputs();\r\n\r\n editor.toolbar.close();\r\n\r\n };\r\n\r\n settings.showRemoveActions = function () {\r\n\r\n editor.toolbar.settings.actions.classList.add('opened');\r\n\r\n };\r\n\r\n settings.hideRemoveActions = function () {\r\n\r\n editor.toolbar.settings.actions.classList.remove('opened');\r\n\r\n };\r\n\r\n return settings;\r\n\r\n})({});\n\n\n// WEBPACK FOOTER //\n// ./modules/toolbar/settings.js","/**\r\n * Inline toolbar\r\n *\r\n * Contains from tools:\r\n * Bold, Italic, Underline and Anchor\r\n *\r\n * @author Codex Team\r\n * @version 1.0\r\n */\r\n\r\nmodule.exports = (function (inline) {\r\n\r\n let editor = codex.editor;\r\n\r\n inline.buttonsOpened = null;\r\n inline.actionsOpened = null;\r\n inline.wrappersOffset = null;\r\n\r\n /**\r\n * saving selection that need for execCommand for styling\r\n *\r\n */\r\n inline.storedSelection = null;\r\n\r\n /**\r\n * @protected\r\n *\r\n * Open inline toobar\r\n */\r\n inline.show = function () {\r\n\r\n var currentNode = editor.content.currentNode,\r\n tool = currentNode.dataset.tool,\r\n plugin;\r\n\r\n /**\r\n * tool allowed to open inline toolbar\r\n */\r\n plugin = editor.tools[tool];\r\n\r\n if (!plugin.showInlineToolbar)\r\n return;\r\n\r\n var selectedText = inline.getSelectionText(),\r\n toolbar = editor.nodes.inlineToolbar.wrapper;\r\n\r\n if (selectedText.length > 0) {\r\n\r\n /** Move toolbar and open */\r\n editor.toolbar.inline.move();\r\n\r\n /** Open inline toolbar */\r\n toolbar.classList.add('opened');\r\n\r\n /** show buttons of inline toolbar */\r\n editor.toolbar.inline.showButtons();\r\n\r\n }\r\n\r\n };\r\n\r\n /**\r\n * @protected\r\n *\r\n * Closes inline toolbar\r\n */\r\n inline.close = function () {\r\n\r\n var toolbar = editor.nodes.inlineToolbar.wrapper;\r\n\r\n toolbar.classList.remove('opened');\r\n\r\n };\r\n\r\n /**\r\n * @private\r\n *\r\n * Moving toolbar\r\n */\r\n inline.move = function () {\r\n\r\n if (!this.wrappersOffset) {\r\n\r\n this.wrappersOffset = this.getWrappersOffset();\r\n\r\n }\r\n\r\n var coords = this.getSelectionCoords(),\r\n defaultOffset = 0,\r\n toolbar = editor.nodes.inlineToolbar.wrapper,\r\n newCoordinateX,\r\n newCoordinateY;\r\n\r\n if (toolbar.offsetHeight === 0) {\r\n\r\n defaultOffset = 40;\r\n\r\n }\r\n\r\n newCoordinateX = coords.x - this.wrappersOffset.left;\r\n newCoordinateY = coords.y + window.scrollY - this.wrappersOffset.top - defaultOffset - toolbar.offsetHeight;\r\n\r\n toolbar.style.transform = `translate3D(${Math.floor(newCoordinateX)}px, ${Math.floor(newCoordinateY)}px, 0)`;\r\n\r\n /** Close everything */\r\n editor.toolbar.inline.closeButtons();\r\n editor.toolbar.inline.closeAction();\r\n\r\n };\r\n\r\n /**\r\n * @private\r\n *\r\n * Tool Clicked\r\n */\r\n\r\n inline.toolClicked = function (event, type) {\r\n\r\n /**\r\n * For simple tools we use default browser function\r\n * For more complicated tools, we should write our own behavior\r\n */\r\n switch (type) {\r\n case 'createLink' : editor.toolbar.inline.createLinkAction(event, type); break;\r\n default : editor.toolbar.inline.defaultToolAction(type); break;\r\n }\r\n\r\n /**\r\n * highlight buttons\r\n * after making some action\r\n */\r\n editor.nodes.inlineToolbar.buttons.childNodes.forEach(editor.toolbar.inline.hightlight);\r\n\r\n };\r\n\r\n /**\r\n * @private\r\n *\r\n * Saving wrappers offset in DOM\r\n */\r\n inline.getWrappersOffset = function () {\r\n\r\n var wrapper = editor.nodes.wrapper,\r\n offset = this.getOffset(wrapper);\r\n\r\n this.wrappersOffset = offset;\r\n return offset;\r\n\r\n };\r\n\r\n /**\r\n * @private\r\n *\r\n * Calculates offset of DOM element\r\n *\r\n * @param el\r\n * @returns {{top: number, left: number}}\r\n */\r\n inline.getOffset = function ( el ) {\r\n\r\n var _x = 0;\r\n var _y = 0;\r\n\r\n while( el && !isNaN( el.offsetLeft ) && !isNaN( el.offsetTop ) ) {\r\n\r\n _x += (el.offsetLeft + el.clientLeft);\r\n _y += (el.offsetTop + el.clientTop);\r\n el = el.offsetParent;\r\n\r\n }\r\n return { top: _y, left: _x };\r\n\r\n };\r\n\r\n /**\r\n * @private\r\n *\r\n * Calculates position of selected text\r\n * @returns {{x: number, y: number}}\r\n */\r\n inline.getSelectionCoords = function () {\r\n\r\n var sel = document.selection, range;\r\n var x = 0, y = 0;\r\n\r\n if (sel) {\r\n\r\n if (sel.type != 'Control') {\r\n\r\n range = sel.createRange();\r\n range.collapse(true);\r\n x = range.boundingLeft;\r\n y = range.boundingTop;\r\n\r\n }\r\n\r\n } else if (window.getSelection) {\r\n\r\n sel = window.getSelection();\r\n\r\n if (sel.rangeCount) {\r\n\r\n range = sel.getRangeAt(0).cloneRange();\r\n if (range.getClientRects) {\r\n\r\n range.collapse(true);\r\n var rect = range.getClientRects()[0];\r\n\r\n if (!rect) {\r\n\r\n return;\r\n\r\n }\r\n\r\n x = rect.left;\r\n y = rect.top;\r\n\r\n }\r\n\r\n }\r\n\r\n }\r\n return { x: x, y: y };\r\n\r\n };\r\n\r\n /**\r\n * @private\r\n *\r\n * Returns selected text as String\r\n * @returns {string}\r\n */\r\n inline.getSelectionText = function () {\r\n\r\n var selectedText = '';\r\n\r\n // all modern browsers and IE9+\r\n if (window.getSelection) {\r\n\r\n selectedText = window.getSelection().toString();\r\n\r\n }\r\n\r\n return selectedText;\r\n\r\n };\r\n\r\n /** Opens buttons block */\r\n inline.showButtons = function () {\r\n\r\n var buttons = editor.nodes.inlineToolbar.buttons;\r\n\r\n buttons.classList.add('opened');\r\n\r\n editor.toolbar.inline.buttonsOpened = true;\r\n\r\n /** highlight buttons */\r\n editor.nodes.inlineToolbar.buttons.childNodes.forEach(editor.toolbar.inline.hightlight);\r\n\r\n };\r\n\r\n /** Makes buttons disappear */\r\n inline.closeButtons = function () {\r\n\r\n var buttons = editor.nodes.inlineToolbar.buttons;\r\n\r\n buttons.classList.remove('opened');\r\n\r\n editor.toolbar.inline.buttonsOpened = false;\r\n\r\n };\r\n\r\n /** Open buttons defined action if exist */\r\n inline.showActions = function () {\r\n\r\n var action = editor.nodes.inlineToolbar.actions;\r\n\r\n action.classList.add('opened');\r\n\r\n editor.toolbar.inline.actionsOpened = true;\r\n\r\n };\r\n\r\n /** Close actions block */\r\n inline.closeAction = function () {\r\n\r\n var action = editor.nodes.inlineToolbar.actions;\r\n\r\n action.innerHTML = '';\r\n action.classList.remove('opened');\r\n editor.toolbar.inline.actionsOpened = false;\r\n\r\n };\r\n\r\n\r\n /**\r\n * Callback for keydowns in inline toolbar \"Insert link...\" input\r\n */\r\n let inlineToolbarAnchorInputKeydown_ = function (event) {\r\n\r\n if (event.keyCode != editor.core.keys.ENTER) {\r\n\r\n return;\r\n\r\n }\r\n\r\n let editable = editor.content.currentNode,\r\n storedSelection = editor.toolbar.inline.storedSelection;\r\n\r\n editor.toolbar.inline.restoreSelection(editable, storedSelection);\r\n editor.toolbar.inline.setAnchor(this.value);\r\n\r\n /**\r\n * Preventing events that will be able to happen\r\n */\r\n event.preventDefault();\r\n event.stopImmediatePropagation();\r\n\r\n editor.toolbar.inline.clearRange();\r\n\r\n };\r\n\r\n /** Action for link creation or for setting anchor */\r\n inline.createLinkAction = function (event) {\r\n\r\n var isActive = this.isLinkActive();\r\n\r\n var editable = editor.content.currentNode,\r\n storedSelection = editor.toolbar.inline.saveSelection(editable);\r\n\r\n /** Save globally selection */\r\n editor.toolbar.inline.storedSelection = storedSelection;\r\n\r\n if (isActive) {\r\n\r\n\r\n /**\r\n * Changing stored selection. if we want to remove anchor from word\r\n * we should remove anchor from whole word, not only selected part.\r\n * The solution is than we get the length of current link\r\n * Change start position to - end of selection minus length of anchor\r\n */\r\n editor.toolbar.inline.restoreSelection(editable, storedSelection);\r\n\r\n editor.toolbar.inline.defaultToolAction('unlink');\r\n\r\n } else {\r\n\r\n /** Create input and close buttons */\r\n var action = editor.draw.inputForLink();\r\n\r\n editor.nodes.inlineToolbar.actions.appendChild(action);\r\n\r\n editor.toolbar.inline.closeButtons();\r\n editor.toolbar.inline.showActions();\r\n\r\n /**\r\n * focus to input\r\n * Solution: https://developer.mozilla.org/ru/docs/Web/API/HTMLElement/focus\r\n * Prevents event after showing input and when we need to focus an input which is in unexisted form\r\n */\r\n action.focus();\r\n event.preventDefault();\r\n\r\n /** Callback to link action */\r\n editor.listeners.add(action, 'keydown', inlineToolbarAnchorInputKeydown_, false);\r\n\r\n }\r\n\r\n };\r\n\r\n inline.isLinkActive = function () {\r\n\r\n var isActive = false;\r\n\r\n editor.nodes.inlineToolbar.buttons.childNodes.forEach(function (tool) {\r\n\r\n var dataType = tool.dataset.type;\r\n\r\n if (dataType == 'link' && tool.classList.contains('hightlighted')) {\r\n\r\n isActive = true;\r\n\r\n }\r\n\r\n });\r\n\r\n return isActive;\r\n\r\n };\r\n\r\n /** default action behavior of tool */\r\n inline.defaultToolAction = function (type) {\r\n\r\n document.execCommand(type, false, null);\r\n\r\n };\r\n\r\n /**\r\n * @private\r\n *\r\n * Sets URL\r\n *\r\n * @param {String} url - URL\r\n */\r\n inline.setAnchor = function (url) {\r\n\r\n document.execCommand('createLink', false, url);\r\n\r\n /** Close after URL inserting */\r\n editor.toolbar.inline.closeAction();\r\n\r\n };\r\n\r\n /**\r\n * @private\r\n *\r\n * Saves selection\r\n */\r\n inline.saveSelection = function (containerEl) {\r\n\r\n var range = window.getSelection().getRangeAt(0),\r\n preSelectionRange = range.cloneRange(),\r\n start;\r\n\r\n preSelectionRange.selectNodeContents(containerEl);\r\n preSelectionRange.setEnd(range.startContainer, range.startOffset);\r\n\r\n start = preSelectionRange.toString().length;\r\n\r\n return {\r\n start: start,\r\n end: start + range.toString().length\r\n };\r\n\r\n };\r\n\r\n /**\r\n * @private\r\n *\r\n * Sets to previous selection (Range)\r\n *\r\n * @param {Element} containerEl - editable element where we restore range\r\n * @param {Object} savedSel - range basic information to restore\r\n */\r\n inline.restoreSelection = function (containerEl, savedSel) {\r\n\r\n var range = document.createRange(),\r\n charIndex = 0;\r\n\r\n range.setStart(containerEl, 0);\r\n range.collapse(true);\r\n\r\n var nodeStack = [ containerEl ],\r\n node,\r\n foundStart = false,\r\n stop = false,\r\n nextCharIndex;\r\n\r\n while (!stop && (node = nodeStack.pop())) {\r\n\r\n if (node.nodeType == 3) {\r\n\r\n nextCharIndex = charIndex + node.length;\r\n\r\n if (!foundStart && savedSel.start >= charIndex && savedSel.start <= nextCharIndex) {\r\n\r\n range.setStart(node, savedSel.start - charIndex);\r\n foundStart = true;\r\n\r\n }\r\n if (foundStart && savedSel.end >= charIndex && savedSel.end <= nextCharIndex) {\r\n\r\n range.setEnd(node, savedSel.end - charIndex);\r\n stop = true;\r\n\r\n }\r\n charIndex = nextCharIndex;\r\n\r\n } else {\r\n\r\n var i = node.childNodes.length;\r\n\r\n while (i--) {\r\n\r\n nodeStack.push(node.childNodes[i]);\r\n\r\n }\r\n\r\n }\r\n\r\n }\r\n\r\n var sel = window.getSelection();\r\n\r\n sel.removeAllRanges();\r\n sel.addRange(range);\r\n\r\n };\r\n\r\n /**\r\n * @private\r\n *\r\n * Removes all ranges from window selection\r\n */\r\n inline.clearRange = function () {\r\n\r\n var selection = window.getSelection();\r\n\r\n selection.removeAllRanges();\r\n\r\n };\r\n\r\n /**\r\n * @private\r\n *\r\n * sets or removes hightlight\r\n */\r\n inline.hightlight = function (tool) {\r\n\r\n var dataType = tool.dataset.type;\r\n\r\n if (document.queryCommandState(dataType)) {\r\n\r\n editor.toolbar.inline.setButtonHighlighted(tool);\r\n\r\n } else {\r\n\r\n editor.toolbar.inline.removeButtonsHighLight(tool);\r\n\r\n }\r\n\r\n /**\r\n *\r\n * hightlight for anchors\r\n */\r\n var selection = window.getSelection(),\r\n tag = selection.anchorNode.parentNode;\r\n\r\n if (tag.tagName == 'A' && dataType == 'link') {\r\n\r\n editor.toolbar.inline.setButtonHighlighted(tool);\r\n\r\n }\r\n\r\n };\r\n\r\n /**\r\n * @private\r\n *\r\n * Mark button if text is already executed\r\n */\r\n inline.setButtonHighlighted = function (button) {\r\n\r\n button.classList.add('hightlighted');\r\n\r\n /** At link tool we also change icon */\r\n if (button.dataset.type == 'link') {\r\n\r\n var icon = button.childNodes[0];\r\n\r\n icon.classList.remove('ce-icon-link');\r\n icon.classList.add('ce-icon-unlink');\r\n\r\n }\r\n\r\n };\r\n\r\n /**\r\n * @private\r\n *\r\n * Removes hightlight\r\n */\r\n inline.removeButtonsHighLight = function (button) {\r\n\r\n button.classList.remove('hightlighted');\r\n\r\n /** At link tool we also change icon */\r\n if (button.dataset.type == 'link') {\r\n\r\n var icon = button.childNodes[0];\r\n\r\n icon.classList.remove('ce-icon-unlink');\r\n icon.classList.add('ce-icon-link');\r\n\r\n }\r\n\r\n };\r\n\r\n\r\n return inline;\r\n\r\n})({});\n\n\n// WEBPACK FOOTER //\n// ./modules/toolbar/inline.js","/**\r\n * Codex Editor toolbox\r\n *\r\n * All tools be able to appended here\r\n *\r\n * @author Codex Team\r\n * @version 1.0\r\n */\r\n\r\nmodule.exports = (function (toolbox) {\r\n\r\n let editor = codex.editor;\r\n\r\n toolbox.opened = false;\r\n\r\n /** Shows toolbox */\r\n toolbox.open = function () {\r\n\r\n /** Close setting if toolbox is opened */\r\n if (editor.toolbar.settings.opened) {\r\n\r\n editor.toolbar.settings.close();\r\n\r\n }\r\n\r\n /** display toolbox */\r\n editor.nodes.toolbox.classList.add('opened');\r\n\r\n /** Animate plus button */\r\n editor.nodes.plusButton.classList.add('clicked');\r\n\r\n /** toolbox state */\r\n editor.toolbar.toolbox.opened = true;\r\n\r\n };\r\n\r\n /** Closes toolbox */\r\n toolbox.close = function () {\r\n\r\n /** Makes toolbox disapear */\r\n editor.nodes.toolbox.classList.remove('opened');\r\n\r\n /** Rotate plus button */\r\n editor.nodes.plusButton.classList.remove('clicked');\r\n\r\n /** toolbox state */\r\n editor.toolbar.toolbox.opened = false;\r\n\r\n editor.toolbar.current = null;\r\n\r\n };\r\n\r\n toolbox.leaf = function () {\r\n\r\n let currentTool = editor.toolbar.current,\r\n tools = Object.keys(editor.tools),\r\n barButtons = editor.nodes.toolbarButtons,\r\n nextToolIndex = 0,\r\n toolToSelect,\r\n visibleTool,\r\n tool;\r\n\r\n if ( !currentTool ) {\r\n\r\n /** Get first tool from object*/\r\n for(tool in editor.tools) {\r\n\r\n if (editor.tools[tool].displayInToolbox) {\r\n\r\n break;\r\n\r\n }\r\n\r\n nextToolIndex ++;\r\n\r\n }\r\n\r\n } else {\r\n\r\n nextToolIndex = (tools.indexOf(currentTool) + 1) % tools.length;\r\n visibleTool = tools[nextToolIndex];\r\n\r\n while (!editor.tools[visibleTool].displayInToolbox) {\r\n\r\n nextToolIndex = (nextToolIndex + 1) % tools.length;\r\n visibleTool = tools[nextToolIndex];\r\n\r\n }\r\n\r\n }\r\n\r\n toolToSelect = tools[nextToolIndex];\r\n\r\n for ( var button in barButtons ) {\r\n\r\n barButtons[button].classList.remove('selected');\r\n\r\n }\r\n\r\n barButtons[toolToSelect].classList.add('selected');\r\n editor.toolbar.current = toolToSelect;\r\n\r\n };\r\n\r\n /**\r\n * Transforming selected node type into selected toolbar element type\r\n * @param {event} event\r\n */\r\n toolbox.toolClicked = function (event) {\r\n\r\n /**\r\n * UNREPLACEBLE_TOOLS this types of tools are forbidden to replace even they are empty\r\n */\r\n var UNREPLACEBLE_TOOLS = ['image', 'link', 'list', 'instagram', 'twitter', 'embed'],\r\n tool = editor.tools[editor.toolbar.current],\r\n workingNode = editor.content.currentNode,\r\n currentInputIndex = editor.caret.inputIndex,\r\n newBlockContent,\r\n appendCallback,\r\n blockData;\r\n\r\n /** Make block from plugin */\r\n newBlockContent = tool.render();\r\n\r\n /** information about block */\r\n blockData = {\r\n block : newBlockContent,\r\n type : tool.type,\r\n stretched : false\r\n };\r\n\r\n if (\r\n workingNode &&\r\n UNREPLACEBLE_TOOLS.indexOf(workingNode.dataset.tool) === -1 &&\r\n workingNode.textContent.trim() === ''\r\n ) {\r\n\r\n /** Replace current block */\r\n editor.content.switchBlock(workingNode, newBlockContent, tool.type);\r\n\r\n } else {\r\n\r\n /** Insert new Block from plugin */\r\n editor.content.insertBlock(blockData);\r\n\r\n /** increase input index */\r\n currentInputIndex++;\r\n\r\n }\r\n\r\n /** Fire tool append callback */\r\n appendCallback = tool.appendCallback;\r\n\r\n if (appendCallback && typeof appendCallback == 'function') {\r\n\r\n appendCallback.call(event);\r\n\r\n }\r\n\r\n window.setTimeout(function () {\r\n\r\n /** Set caret to current block */\r\n editor.caret.setToBlock(currentInputIndex);\r\n\r\n }, 10);\r\n\r\n\r\n /**\r\n * Changing current Node\r\n */\r\n editor.content.workingNodeChanged();\r\n\r\n /**\r\n * Move toolbar when node is changed\r\n */\r\n editor.toolbar.move();\r\n\r\n };\r\n\r\n return toolbox;\r\n\r\n})({});\n\n\n// WEBPACK FOOTER //\n// ./modules/toolbar/toolbox.js","/**\r\n * @module Codex Editor Callbacks module\r\n * @description Module works with editor added Elements\r\n *\r\n * @author Codex Team\r\n * @version 1.4.0\r\n */\r\n\r\nmodule.exports = (function (callbacks) {\r\n\r\n let editor = codex.editor;\r\n\r\n /**\r\n * used by UI module\r\n * @description Routes all keydowns on document\r\n * @param {Object} event\r\n */\r\n callbacks.globalKeydown = function (event) {\r\n\r\n switch (event.keyCode) {\r\n case editor.core.keys.ENTER : enterKeyPressed_(event); break;\r\n }\r\n\r\n };\r\n\r\n /**\r\n * used by UI module\r\n * @description Routes all keydowns on redactors area\r\n * @param {Object} event\r\n */\r\n callbacks.redactorKeyDown = function (event) {\r\n\r\n switch (event.keyCode) {\r\n case editor.core.keys.TAB : tabKeyPressedOnRedactorsZone_(event); break;\r\n case editor.core.keys.ENTER : enterKeyPressedOnRedactorsZone_(event); break;\r\n case editor.core.keys.ESC : escapeKeyPressedOnRedactorsZone_(event); break;\r\n default : defaultKeyPressedOnRedactorsZone_(event); break;\r\n }\r\n\r\n };\r\n\r\n /**\r\n * used by UI module\r\n * @description Routes all keyup events\r\n * @param {Object} event\r\n */\r\n callbacks.globalKeyup = function (event) {\r\n\r\n switch (event.keyCode) {\r\n case editor.core.keys.UP :\r\n case editor.core.keys.LEFT :\r\n case editor.core.keys.RIGHT :\r\n case editor.core.keys.DOWN : arrowKeyPressed_(event); break;\r\n }\r\n\r\n };\r\n\r\n /**\r\n * @param {Object} event\r\n * @private\r\n *\r\n * Handles behaviour when tab pressed\r\n * @description if Content is empty show toolbox (if it is closed) or leaf tools\r\n * uses Toolbars toolbox module to handle the situation\r\n */\r\n var tabKeyPressedOnRedactorsZone_ = function (event) {\r\n\r\n /**\r\n * Wait for solution. Would like to know the behaviour\r\n * @todo Add spaces\r\n */\r\n event.preventDefault();\r\n\r\n var nativeInputs = editor.content.currentNode.querySelectorAll('textarea, input'),\r\n nativeInputsAreEmpty = true,\r\n textContentIsEmpty = !editor.content.currentNode.textContent.trim();\r\n\r\n Array.prototype.map.call(nativeInputs, function (input) {\r\n\r\n if (input.type == 'textarea' || input.type == 'text') {\r\n\r\n nativeInputsAreEmpty = nativeInputsAreEmpty && !input.value.trim();\r\n\r\n }\r\n\r\n });\r\n\r\n var blockIsEmpty = textContentIsEmpty && nativeInputsAreEmpty;\r\n\r\n if (!blockIsEmpty) {\r\n\r\n return;\r\n\r\n }\r\n\r\n if ( !editor.toolbar.opened ) {\r\n\r\n editor.toolbar.open();\r\n\r\n }\r\n\r\n if (editor.toolbar.opened && !editor.toolbar.toolbox.opened) {\r\n\r\n editor.toolbar.toolbox.open();\r\n\r\n } else {\r\n\r\n editor.toolbar.toolbox.leaf();\r\n\r\n }\r\n\r\n };\r\n\r\n /**\r\n * Handles global EnterKey Press\r\n * @see enterPressedOnBlock_\r\n * @param {Object} event\r\n */\r\n var enterKeyPressed_ = function () {\r\n\r\n if (editor.content.editorAreaHightlighted) {\r\n\r\n /**\r\n * it means that we lose input index, saved index before is not correct\r\n * therefore we need to set caret when we insert new block\r\n */\r\n editor.caret.inputIndex = -1;\r\n\r\n enterPressedOnBlock_();\r\n\r\n }\r\n\r\n };\r\n\r\n /**\r\n * Callback for enter key pressing in first-level block area\r\n *\r\n * @param {Event} event\r\n * @private\r\n *\r\n * @description Inserts new block with initial type from settings\r\n */\r\n var enterPressedOnBlock_ = function () {\r\n\r\n var NEW_BLOCK_TYPE = editor.settings.initialBlockPlugin;\r\n\r\n editor.content.insertBlock({\r\n type : NEW_BLOCK_TYPE,\r\n block : editor.tools[NEW_BLOCK_TYPE].render()\r\n }, true );\r\n\r\n editor.toolbar.move();\r\n editor.toolbar.open();\r\n\r\n };\r\n\r\n\r\n /**\r\n * ENTER key handler\r\n *\r\n * @param {Object} event\r\n * @private\r\n *\r\n * @description Makes new block with initial type from settings\r\n */\r\n var enterKeyPressedOnRedactorsZone_ = function (event) {\r\n\r\n if (event.target.contentEditable == 'true') {\r\n\r\n /** Update input index */\r\n editor.caret.saveCurrentInputIndex();\r\n\r\n }\r\n\r\n var currentInputIndex = editor.caret.getCurrentInputIndex() || 0,\r\n workingNode = editor.content.currentNode,\r\n tool = workingNode.dataset.tool,\r\n isEnterPressedOnToolbar = editor.toolbar.opened &&\r\n editor.toolbar.current &&\r\n event.target == editor.state.inputs[currentInputIndex];\r\n\r\n /** The list of tools which needs the default browser behaviour */\r\n var enableLineBreaks = editor.tools[tool].enableLineBreaks;\r\n\r\n /** This type of block creates when enter is pressed */\r\n var NEW_BLOCK_TYPE = editor.settings.initialBlockPlugin;\r\n\r\n /**\r\n * When toolbar is opened, select tool instead of making new paragraph\r\n */\r\n if ( isEnterPressedOnToolbar ) {\r\n\r\n event.preventDefault();\r\n\r\n editor.toolbar.toolbox.toolClicked(event);\r\n\r\n editor.toolbar.close();\r\n\r\n /**\r\n * Stop other listeners callback executions\r\n */\r\n event.stopPropagation();\r\n event.stopImmediatePropagation();\r\n\r\n return;\r\n\r\n }\r\n\r\n /**\r\n * Allow paragraph lineBreaks with shift enter\r\n * Or if shiftkey pressed and enter and enabledLineBreaks, the let new block creation\r\n */\r\n if ( event.shiftKey || enableLineBreaks ) {\r\n\r\n event.stopPropagation();\r\n event.stopImmediatePropagation();\r\n return;\r\n\r\n }\r\n\r\n var currentSelection = window.getSelection(),\r\n currentSelectedNode = currentSelection.anchorNode,\r\n caretAtTheEndOfText = editor.caret.position.atTheEnd(),\r\n isTextNodeHasParentBetweenContenteditable = false;\r\n\r\n /**\r\n * Allow making new

    in same block by SHIFT+ENTER and forbids to prevent default browser behaviour\r\n */\r\n if ( event.shiftKey && !enableLineBreaks ) {\r\n\r\n editor.callback.enterPressedOnBlock(editor.content.currentBlock, event);\r\n event.preventDefault();\r\n return;\r\n\r\n }\r\n\r\n /**\r\n * Workaround situation when caret at the Text node that has some wrapper Elements\r\n * Split block cant handle this.\r\n * We need to save default behavior\r\n */\r\n isTextNodeHasParentBetweenContenteditable = currentSelectedNode && currentSelectedNode.parentNode.contentEditable != 'true';\r\n\r\n /**\r\n * Split blocks when input has several nodes and caret placed in textNode\r\n */\r\n if (\r\n currentSelectedNode.nodeType == editor.core.nodeTypes.TEXT &&\r\n !isTextNodeHasParentBetweenContenteditable &&\r\n !caretAtTheEndOfText\r\n ) {\r\n\r\n event.preventDefault();\r\n\r\n editor.core.log('Splitting Text node...');\r\n\r\n editor.content.splitBlock(currentInputIndex);\r\n\r\n /** Show plus button when next input after split is empty*/\r\n if (!editor.state.inputs[currentInputIndex + 1].textContent.trim()) {\r\n\r\n editor.toolbar.showPlusButton();\r\n\r\n }\r\n\r\n } else {\r\n\r\n var islastNode = editor.content.isLastNode(currentSelectedNode);\r\n\r\n if ( islastNode && caretAtTheEndOfText ) {\r\n\r\n event.preventDefault();\r\n event.stopPropagation();\r\n event.stopImmediatePropagation();\r\n\r\n editor.core.log('ENTER clicked in last textNode. Create new BLOCK');\r\n\r\n editor.content.insertBlock({\r\n type: NEW_BLOCK_TYPE,\r\n block: editor.tools[NEW_BLOCK_TYPE].render()\r\n }, true);\r\n\r\n editor.toolbar.move();\r\n editor.toolbar.open();\r\n\r\n /** Show plus button with empty block */\r\n editor.toolbar.showPlusButton();\r\n\r\n }\r\n\r\n }\r\n\r\n /** get all inputs after new appending block */\r\n editor.ui.saveInputs();\r\n\r\n };\r\n\r\n /**\r\n * Escape behaviour\r\n * @param event\r\n * @private\r\n *\r\n * @description Closes toolbox and toolbar. Prevents default behaviour\r\n */\r\n var escapeKeyPressedOnRedactorsZone_ = function (event) {\r\n\r\n /** Close all toolbar */\r\n editor.toolbar.close();\r\n\r\n /** Close toolbox */\r\n editor.toolbar.toolbox.close();\r\n\r\n event.preventDefault();\r\n\r\n };\r\n\r\n /**\r\n * @param {Event} event\r\n * @private\r\n *\r\n * closes and moves toolbar\r\n */\r\n var arrowKeyPressed_ = function (event) {\r\n\r\n editor.content.workingNodeChanged();\r\n\r\n /* Closing toolbar */\r\n editor.toolbar.close();\r\n editor.toolbar.move();\r\n\r\n };\r\n\r\n /**\r\n * @private\r\n * @param {Event} event\r\n *\r\n * @description Closes all opened bars from toolbar.\r\n * If block is mark, clears highlightning\r\n */\r\n var defaultKeyPressedOnRedactorsZone_ = function () {\r\n\r\n editor.toolbar.close();\r\n\r\n if (!editor.toolbar.inline.actionsOpened) {\r\n\r\n editor.toolbar.inline.close();\r\n editor.content.clearMark();\r\n\r\n }\r\n\r\n };\r\n\r\n /**\r\n * Handler when clicked on redactors area\r\n *\r\n * @protected\r\n * @param event\r\n *\r\n * @description Detects clicked area. If it is first-level block area, marks as detected and\r\n * on next enter press will be inserted new block\r\n * Otherwise, save carets position (input index) and put caret to the editable zone.\r\n *\r\n * @see detectWhenClickedOnFirstLevelBlockArea_\r\n *\r\n */\r\n callbacks.redactorClicked = function (event) {\r\n\r\n detectWhenClickedOnFirstLevelBlockArea_();\r\n\r\n editor.content.workingNodeChanged(event.target);\r\n editor.ui.saveInputs();\r\n\r\n var selectedText = editor.toolbar.inline.getSelectionText(),\r\n firstLevelBlock;\r\n\r\n /** If selection range took off, then we hide inline toolbar */\r\n if (selectedText.length === 0) {\r\n\r\n editor.toolbar.inline.close();\r\n\r\n }\r\n\r\n /** Update current input index in memory when caret focused into existed input */\r\n if (event.target.contentEditable == 'true') {\r\n\r\n editor.caret.saveCurrentInputIndex();\r\n\r\n }\r\n\r\n if (editor.content.currentNode === null) {\r\n\r\n /**\r\n * If inputs in redactor does not exits, then we put input index 0 not -1\r\n */\r\n var indexOfLastInput = editor.state.inputs.length > 0 ? editor.state.inputs.length - 1 : 0;\r\n\r\n /** If we have any inputs */\r\n if (editor.state.inputs.length) {\r\n\r\n /** getting firstlevel parent of input */\r\n firstLevelBlock = editor.content.getFirstLevelBlock(editor.state.inputs[indexOfLastInput]);\r\n\r\n }\r\n\r\n /** If input is empty, then we set caret to the last input */\r\n if (editor.state.inputs.length && editor.state.inputs[indexOfLastInput].textContent === '' && firstLevelBlock.dataset.tool == editor.settings.initialBlockPlugin) {\r\n\r\n editor.caret.setToBlock(indexOfLastInput);\r\n\r\n } else {\r\n\r\n /** Create new input when caret clicked in redactors area */\r\n var NEW_BLOCK_TYPE = editor.settings.initialBlockPlugin;\r\n\r\n editor.content.insertBlock({\r\n type : NEW_BLOCK_TYPE,\r\n block : editor.tools[NEW_BLOCK_TYPE].render()\r\n });\r\n\r\n /** If there is no inputs except inserted */\r\n if (editor.state.inputs.length === 1) {\r\n\r\n editor.caret.setToBlock(indexOfLastInput);\r\n\r\n } else {\r\n\r\n /** Set caret to this appended input */\r\n editor.caret.setToNextBlock(indexOfLastInput);\r\n\r\n }\r\n\r\n }\r\n\r\n } else {\r\n\r\n /** Close all panels */\r\n editor.toolbar.settings.close();\r\n editor.toolbar.toolbox.close();\r\n\r\n }\r\n\r\n /**\r\n * Move toolbar and open\r\n */\r\n editor.toolbar.move();\r\n editor.toolbar.open();\r\n\r\n var inputIsEmpty = !editor.content.currentNode.textContent.trim(),\r\n currentNodeType = editor.content.currentNode.dataset.tool,\r\n isInitialType = currentNodeType == editor.settings.initialBlockPlugin;\r\n\r\n\r\n /** Hide plus buttons */\r\n editor.toolbar.hidePlusButton();\r\n\r\n if (!inputIsEmpty) {\r\n\r\n /** Mark current block */\r\n editor.content.markBlock();\r\n\r\n }\r\n\r\n if ( isInitialType && inputIsEmpty ) {\r\n\r\n /** Show plus button */\r\n editor.toolbar.showPlusButton();\r\n\r\n }\r\n\r\n\r\n };\r\n\r\n /**\r\n * This method allows to define, is caret in contenteditable element or not.\r\n *\r\n * @private\r\n *\r\n * @description Otherwise, if we get TEXT node from range container, that will means we have input index.\r\n * In this case we use default browsers behaviour (if plugin allows that) or overwritten action.\r\n * Therefore, to be sure that we've clicked first-level block area, we should have currentNode, which always\r\n * specifies to the first-level block. Other cases we just ignore.\r\n */\r\n var detectWhenClickedOnFirstLevelBlockArea_ = function () {\r\n\r\n var selection = window.getSelection(),\r\n anchorNode = selection.anchorNode,\r\n flag = false;\r\n\r\n if (selection.rangeCount === 0) {\r\n\r\n editor.content.editorAreaHightlighted = true;\r\n\r\n } else {\r\n\r\n if (!editor.core.isDomNode(anchorNode)) {\r\n\r\n anchorNode = anchorNode.parentNode;\r\n\r\n }\r\n\r\n /** Already founded, without loop */\r\n if (anchorNode.contentEditable == 'true') {\r\n\r\n flag = true;\r\n\r\n }\r\n\r\n while (anchorNode.contentEditable != 'true') {\r\n\r\n anchorNode = anchorNode.parentNode;\r\n\r\n if (anchorNode.contentEditable == 'true') {\r\n\r\n flag = true;\r\n\r\n }\r\n\r\n if (anchorNode == document.body) {\r\n\r\n break;\r\n\r\n }\r\n\r\n }\r\n\r\n /** If editable element founded, flag is \"TRUE\", Therefore we return \"FALSE\" */\r\n editor.content.editorAreaHightlighted = !flag;\r\n\r\n }\r\n\r\n };\r\n\r\n /**\r\n * Toolbar button click handler\r\n *\r\n * @param {Object} event - cursor to the button\r\n * @protected\r\n *\r\n * @description gets current tool and calls render method\r\n */\r\n callbacks.toolbarButtonClicked = function (event) {\r\n\r\n var button = this;\r\n\r\n editor.toolbar.current = button.dataset.type;\r\n\r\n editor.toolbar.toolbox.toolClicked(event);\r\n editor.toolbar.close();\r\n\r\n };\r\n\r\n /**\r\n * Show or Hide toolbox when plus button is clicked\r\n */\r\n callbacks.plusButtonClicked = function () {\r\n\r\n if (!editor.nodes.toolbox.classList.contains('opened')) {\r\n\r\n editor.toolbar.toolbox.open();\r\n\r\n } else {\r\n\r\n editor.toolbar.toolbox.close();\r\n\r\n }\r\n\r\n };\r\n\r\n /**\r\n * Block handlers for KeyDown events\r\n *\r\n * @protected\r\n * @param {Object} event\r\n *\r\n * Handles keydowns on block\r\n * @see blockRightOrDownArrowPressed_\r\n * @see backspacePressed_\r\n * @see blockLeftOrUpArrowPressed_\r\n */\r\n callbacks.blockKeydown = function (event) {\r\n\r\n let block = event.target; // event.target is input\r\n\r\n switch (event.keyCode) {\r\n\r\n case editor.core.keys.DOWN:\r\n case editor.core.keys.RIGHT:\r\n blockRightOrDownArrowPressed_(event);\r\n break;\r\n\r\n case editor.core.keys.BACKSPACE:\r\n backspacePressed_(block, event);\r\n break;\r\n\r\n case editor.core.keys.UP:\r\n case editor.core.keys.LEFT:\r\n blockLeftOrUpArrowPressed_(event);\r\n break;\r\n\r\n }\r\n\r\n };\r\n\r\n /**\r\n * RIGHT or DOWN keydowns on block\r\n *\r\n * @param {Object} event\r\n * @private\r\n *\r\n * @description watches the selection and gets closest editable element.\r\n * Uses method getDeepestTextNodeFromPosition to get the last node of next block\r\n * Sets caret if it is contenteditable\r\n */\r\n var blockRightOrDownArrowPressed_ = function (event) {\r\n\r\n var selection = window.getSelection(),\r\n inputs = editor.state.inputs,\r\n focusedNode = selection.anchorNode,\r\n focusedNodeHolder;\r\n\r\n /** Check for caret existance */\r\n if (!focusedNode) {\r\n\r\n return false;\r\n\r\n }\r\n\r\n /** Looking for closest (parent) contentEditable element of focused node */\r\n while (focusedNode.contentEditable != 'true') {\r\n\r\n focusedNodeHolder = focusedNode.parentNode;\r\n focusedNode = focusedNodeHolder;\r\n\r\n }\r\n\r\n /** Input index in DOM level */\r\n var editableElementIndex = 0;\r\n\r\n while (focusedNode != inputs[editableElementIndex]) {\r\n\r\n editableElementIndex ++;\r\n\r\n }\r\n\r\n /**\r\n * Founded contentEditable element doesn't have childs\r\n * Or maybe New created block\r\n */\r\n if (!focusedNode.textContent) {\r\n\r\n editor.caret.setToNextBlock(editableElementIndex);\r\n return;\r\n\r\n }\r\n\r\n /**\r\n * Do nothing when caret doesn not reaches the end of last child\r\n */\r\n var caretInLastChild = false,\r\n caretAtTheEndOfText = false;\r\n\r\n var lastChild,\r\n deepestTextnode;\r\n\r\n lastChild = focusedNode.childNodes[focusedNode.childNodes.length - 1 ];\r\n\r\n if (editor.core.isDomNode(lastChild)) {\r\n\r\n deepestTextnode = editor.content.getDeepestTextNodeFromPosition(lastChild, lastChild.childNodes.length);\r\n\r\n } else {\r\n\r\n deepestTextnode = lastChild;\r\n\r\n }\r\n\r\n caretInLastChild = selection.anchorNode == deepestTextnode;\r\n caretAtTheEndOfText = deepestTextnode.length == selection.anchorOffset;\r\n\r\n if ( !caretInLastChild || !caretAtTheEndOfText ) {\r\n\r\n editor.core.log('arrow [down|right] : caret does not reached the end');\r\n return false;\r\n\r\n }\r\n\r\n editor.caret.setToNextBlock(editableElementIndex);\r\n\r\n };\r\n\r\n /**\r\n * LEFT or UP keydowns on block\r\n *\r\n * @param {Object} event\r\n * @private\r\n *\r\n * watches the selection and gets closest editable element.\r\n * Uses method getDeepestTextNodeFromPosition to get the last node of previous block\r\n * Sets caret if it is contenteditable\r\n *\r\n */\r\n var blockLeftOrUpArrowPressed_ = function (event) {\r\n\r\n var selection = window.getSelection(),\r\n inputs = editor.state.inputs,\r\n focusedNode = selection.anchorNode,\r\n focusedNodeHolder;\r\n\r\n /** Check for caret existance */\r\n if (!focusedNode) {\r\n\r\n return false;\r\n\r\n }\r\n\r\n /**\r\n * LEFT or UP not at the beginning\r\n */\r\n if ( selection.anchorOffset !== 0) {\r\n\r\n return false;\r\n\r\n }\r\n\r\n /** Looking for parent contentEditable block */\r\n while (focusedNode.contentEditable != 'true') {\r\n\r\n focusedNodeHolder = focusedNode.parentNode;\r\n focusedNode = focusedNodeHolder;\r\n\r\n }\r\n\r\n /** Input index in DOM level */\r\n var editableElementIndex = 0;\r\n\r\n while (focusedNode != inputs[editableElementIndex]) {\r\n\r\n editableElementIndex ++;\r\n\r\n }\r\n\r\n /**\r\n * Do nothing if caret is not at the beginning of first child\r\n */\r\n var caretInFirstChild = false,\r\n caretAtTheBeginning = false;\r\n\r\n var firstChild,\r\n deepestTextnode;\r\n\r\n /**\r\n * Founded contentEditable element doesn't have childs\r\n * Or maybe New created block\r\n */\r\n if (!focusedNode.textContent) {\r\n\r\n editor.caret.setToPreviousBlock(editableElementIndex);\r\n return;\r\n\r\n }\r\n\r\n firstChild = focusedNode.childNodes[0];\r\n\r\n if (editor.core.isDomNode(firstChild)) {\r\n\r\n deepestTextnode = editor.content.getDeepestTextNodeFromPosition(firstChild, 0);\r\n\r\n } else {\r\n\r\n deepestTextnode = firstChild;\r\n\r\n }\r\n\r\n caretInFirstChild = selection.anchorNode == deepestTextnode;\r\n caretAtTheBeginning = selection.anchorOffset === 0;\r\n\r\n if ( caretInFirstChild && caretAtTheBeginning ) {\r\n\r\n editor.caret.setToPreviousBlock(editableElementIndex);\r\n\r\n }\r\n\r\n };\r\n\r\n /**\r\n * Handles backspace keydown\r\n *\r\n * @param {Element} block\r\n * @param {Object} event\r\n * @private\r\n *\r\n * @description if block is empty, delete the block and set caret to the previous block\r\n * If block is not empty, try to merge two blocks - current and previous\r\n * But it we try'n to remove first block, then we should set caret to the next block, not previous.\r\n * If we removed the last block, create new one\r\n */\r\n var backspacePressed_ = function (block, event) {\r\n\r\n var currentInputIndex = editor.caret.getCurrentInputIndex(),\r\n range,\r\n selectionLength,\r\n firstLevelBlocksCount;\r\n\r\n if (isNativeInput_(event.target)) {\r\n\r\n /** If input value is empty - remove block */\r\n if (event.target.value.trim() == '') {\r\n\r\n block.remove();\r\n\r\n } else {\r\n\r\n return;\r\n\r\n }\r\n\r\n }\r\n\r\n if (block.textContent.trim()) {\r\n\r\n range = editor.content.getRange();\r\n selectionLength = range.endOffset - range.startOffset;\r\n\r\n if (editor.caret.position.atStart() && !selectionLength && editor.state.inputs[currentInputIndex - 1]) {\r\n\r\n editor.content.mergeBlocks(currentInputIndex);\r\n\r\n } else {\r\n\r\n return;\r\n\r\n }\r\n\r\n }\r\n\r\n if (!selectionLength) {\r\n\r\n block.remove();\r\n\r\n }\r\n\r\n\r\n firstLevelBlocksCount = editor.nodes.redactor.childNodes.length;\r\n\r\n /**\r\n * If all blocks are removed\r\n */\r\n if (firstLevelBlocksCount === 0) {\r\n\r\n /** update currentNode variable */\r\n editor.content.currentNode = null;\r\n\r\n /** Inserting new empty initial block */\r\n editor.ui.addInitialBlock();\r\n\r\n /** Updating inputs state after deleting last block */\r\n editor.ui.saveInputs();\r\n\r\n /** Set to current appended block */\r\n window.setTimeout(function () {\r\n\r\n editor.caret.setToPreviousBlock(1);\r\n\r\n }, 10);\r\n\r\n } else {\r\n\r\n if (editor.caret.inputIndex !== 0) {\r\n\r\n /** Target block is not first */\r\n editor.caret.setToPreviousBlock(editor.caret.inputIndex);\r\n\r\n } else {\r\n\r\n /** If we try to delete first block */\r\n editor.caret.setToNextBlock(editor.caret.inputIndex);\r\n\r\n }\r\n\r\n }\r\n\r\n editor.toolbar.move();\r\n\r\n if (!editor.toolbar.opened) {\r\n\r\n editor.toolbar.open();\r\n\r\n }\r\n\r\n /** Updating inputs state */\r\n editor.ui.saveInputs();\r\n\r\n /** Prevent default browser behaviour */\r\n event.preventDefault();\r\n\r\n };\r\n\r\n /**\r\n * This method prevents default behaviour.\r\n *\r\n * @param {Object} event\r\n * @protected\r\n *\r\n * @description We get from clipboard pasted data, sanitize, make a fragment that contains of this sanitized nodes.\r\n * Firstly, we need to memorize the caret position. We can do that by getting the range of selection.\r\n * After all, we insert clear fragment into caret placed position. Then, we should move the caret to the last node\r\n */\r\n callbacks.blockPasteCallback = function (event) {\r\n\r\n /** If area is input or textarea then allow default behaviour */\r\n if ( isNativeInput_(event.target) ) {\r\n\r\n return;\r\n\r\n }\r\n\r\n /** Prevent default behaviour */\r\n event.preventDefault();\r\n\r\n var editableParent = editor.content.getEditableParent(event.target);\r\n\r\n /** Allow paste when event target placed in Editable element */\r\n if (!editableParent) {\r\n\r\n return;\r\n\r\n }\r\n\r\n /** get html pasted data - dirty data */\r\n var htmlData = event.clipboardData.getData('text/html'),\r\n plainData = event.clipboardData.getData('text/plain');\r\n\r\n\r\n /** Temporary DIV that is used to work with childs as arrays item */\r\n var div = editor.draw.node('DIV', '', {}),\r\n cleanData,\r\n wrappedData;\r\n\r\n /** Create fragment, that we paste to range after proccesing */\r\n cleanData = editor.sanitizer.clean(htmlData);\r\n\r\n\r\n /**\r\n * We wrap pasted text with

    tags to split it logically\r\n *\r\n * @type {string}\r\n */\r\n wrappedData = editor.content.wrapTextWithParagraphs(cleanData, plainData);\r\n div.innerHTML = wrappedData;\r\n\r\n var NEW_BLOCK_TYPE = editor.settings.initialBlockPlugin,\r\n currentBlockContent = editor.content.currentNode.firstChild.firstChild;\r\n\r\n /**\r\n * If there only one paragraph, just insert it\r\n */\r\n if (div.childNodes.length == 1) {\r\n\r\n editor.caret.insertNode(document.createTextNode(div.firstChild.innerHTML));\r\n return;\r\n\r\n }\r\n\r\n div.childNodes.forEach(function (paragraph, index) {\r\n\r\n\r\n /**\r\n * If there was no data in working node, replace it with first paragraph of pasted text\r\n */\r\n if (index == 0 && currentBlockContent.innerHTML.trim() === '') {\r\n\r\n editor.content.switchBlock(editor.content.currentNode, editor.tools[NEW_BLOCK_TYPE].render({\r\n text : paragraph.innerHTML\r\n }), NEW_BLOCK_TYPE);\r\n\r\n return;\r\n\r\n }\r\n\r\n editor.content.insertBlock({\r\n type : NEW_BLOCK_TYPE,\r\n block : editor.tools[NEW_BLOCK_TYPE].render({\r\n text : paragraph.innerHTML\r\n })\r\n });\r\n\r\n editor.caret.inputIndex++;\r\n\r\n });\r\n\r\n editor.caret.setToPreviousBlock(editor.caret.getCurrentInputIndex() + 1);\r\n\r\n\r\n };\r\n\r\n /**\r\n * used by UI module\r\n * Clicks on block settings button\r\n *\r\n * @param {Object} event\r\n * @protected\r\n * @description Opens toolbar settings\r\n */\r\n callbacks.showSettingsButtonClicked = function (event) {\r\n\r\n /**\r\n * Get type of current block\r\n * It uses to append settings from tool.settings property.\r\n * ...\r\n * Type is stored in data-type attribute on block\r\n */\r\n var currentToolType = editor.content.currentNode.dataset.tool;\r\n\r\n editor.toolbar.settings.toggle(currentToolType);\r\n\r\n /** Close toolbox when settings button is active */\r\n editor.toolbar.toolbox.close();\r\n editor.toolbar.settings.hideRemoveActions();\r\n\r\n };\r\n\r\n /**\r\n * Check block\r\n * @param target\r\n * @private\r\n *\r\n * @description Checks target is it native input\r\n */\r\n var isNativeInput_ = function (target) {\r\n\r\n var nativeInputAreas = ['INPUT', 'TEXTAREA'];\r\n\r\n return (nativeInputAreas.indexOf(target.tagName) != -1);\r\n\r\n };\r\n\r\n return callbacks;\r\n\r\n})({});\n\n\n// WEBPACK FOOTER //\n// ./modules/callbacks.js","/**\r\n * Codex Editor Draw module\r\n *\r\n * @author Codex Team\r\n * @version 1.0.\r\n */\r\n\r\nmodule.exports = (function (draw) {\r\n\r\n /**\r\n * Base editor wrapper\r\n */\r\n draw.wrapper = function () {\r\n\r\n var wrapper = document.createElement('div');\r\n\r\n wrapper.className += 'codex-editor';\r\n\r\n return wrapper;\r\n\r\n };\r\n\r\n /**\r\n * Content-editable holder\r\n */\r\n draw.redactor = function () {\r\n\r\n var redactor = document.createElement('div');\r\n\r\n redactor.className += 'ce-redactor';\r\n\r\n return redactor;\r\n\r\n };\r\n\r\n draw.ceBlock = function () {\r\n\r\n var block = document.createElement('DIV');\r\n\r\n block.className += 'ce_block';\r\n\r\n return block;\r\n\r\n };\r\n\r\n /**\r\n * Empty toolbar with toggler\r\n */\r\n draw.toolbar = function () {\r\n\r\n var bar = document.createElement('div');\r\n\r\n bar.className += 'ce-toolbar';\r\n\r\n return bar;\r\n\r\n };\r\n\r\n draw.toolbarContent = function () {\r\n\r\n var wrapper = document.createElement('DIV');\r\n\r\n wrapper.classList.add('ce-toolbar__content');\r\n\r\n return wrapper;\r\n\r\n };\r\n\r\n /**\r\n * Inline toolbar\r\n */\r\n draw.inlineToolbar = function () {\r\n\r\n var bar = document.createElement('DIV');\r\n\r\n bar.className += 'ce-toolbar-inline';\r\n\r\n return bar;\r\n\r\n };\r\n\r\n /**\r\n * Wrapper for inline toobar buttons\r\n */\r\n draw.inlineToolbarButtons = function () {\r\n\r\n var wrapper = document.createElement('DIV');\r\n\r\n wrapper.className += 'ce-toolbar-inline__buttons';\r\n\r\n return wrapper;\r\n\r\n };\r\n\r\n /**\r\n * For some actions\r\n */\r\n draw.inlineToolbarActions = function () {\r\n\r\n var wrapper = document.createElement('DIV');\r\n\r\n wrapper.className += 'ce-toolbar-inline__actions';\r\n\r\n return wrapper;\r\n\r\n };\r\n\r\n draw.inputForLink = function () {\r\n\r\n var input = document.createElement('INPUT');\r\n\r\n input.type = 'input';\r\n input.className += 'inputForLink';\r\n input.placeholder = 'Вставьте ссылку ...';\r\n input.setAttribute('form', 'defaultForm');\r\n\r\n input.setAttribute('autofocus', 'autofocus');\r\n\r\n return input;\r\n\r\n };\r\n\r\n /**\r\n * @todo Desc\r\n */\r\n draw.blockButtons = function () {\r\n\r\n var block = document.createElement('div');\r\n\r\n block.className += 'ce-toolbar__actions';\r\n\r\n return block;\r\n\r\n };\r\n\r\n /**\r\n * Block settings panel\r\n */\r\n draw.blockSettings = function () {\r\n\r\n var settings = document.createElement('div');\r\n\r\n settings.className += 'ce-settings';\r\n\r\n return settings;\r\n\r\n };\r\n\r\n draw.defaultSettings = function () {\r\n\r\n var div = document.createElement('div');\r\n\r\n div.classList.add('ce-settings_default');\r\n\r\n return div;\r\n\r\n };\r\n\r\n draw.pluginsSettings = function () {\r\n\r\n var div = document.createElement('div');\r\n\r\n div.classList.add('ce-settings_plugin');\r\n\r\n return div;\r\n\r\n };\r\n\r\n draw.plusButton = function () {\r\n\r\n var button = document.createElement('span');\r\n\r\n button.className = 'ce-toolbar__plus';\r\n // button.innerHTML = '';\r\n\r\n return button;\r\n\r\n };\r\n\r\n /**\r\n * Settings button in toolbar\r\n */\r\n draw.settingsButton = function () {\r\n\r\n var toggler = document.createElement('span');\r\n\r\n toggler.className = 'ce-toolbar__settings-btn';\r\n\r\n /** Toggler button*/\r\n toggler.innerHTML = '';\r\n\r\n return toggler;\r\n\r\n };\r\n\r\n /**\r\n * Redactor tools wrapper\r\n */\r\n\r\n draw.toolbox = function () {\r\n\r\n var wrapper = document.createElement('div');\r\n\r\n wrapper.className = 'ce-toolbar__tools';\r\n\r\n return wrapper;\r\n\r\n };\r\n\r\n /**\r\n * @protected\r\n *\r\n * Draws tool buttons for toolbox\r\n *\r\n * @param {String} type\r\n * @param {String} classname\r\n * @returns {Element}\r\n */\r\n draw.toolbarButton = function (type, classname) {\r\n\r\n var button = document.createElement('li'),\r\n toolIcon = document.createElement('i'),\r\n toolTitle = document.createElement('span');\r\n\r\n button.dataset.type = type;\r\n button.setAttribute('title', type);\r\n\r\n toolIcon.classList.add(classname);\r\n toolTitle.classList.add('ce_toolbar_tools--title');\r\n\r\n\r\n button.appendChild(toolIcon);\r\n button.appendChild(toolTitle);\r\n\r\n return button;\r\n\r\n };\r\n\r\n /**\r\n * @protected\r\n *\r\n * Draws tools for inline toolbar\r\n *\r\n * @param {String} type\r\n * @param {String} classname\r\n */\r\n draw.toolbarButtonInline = function (type, classname) {\r\n\r\n var button = document.createElement('BUTTON'),\r\n toolIcon = document.createElement('I');\r\n\r\n button.type = 'button';\r\n button.dataset.type = type;\r\n toolIcon.classList.add(classname);\r\n\r\n button.appendChild(toolIcon);\r\n\r\n return button;\r\n\r\n };\r\n\r\n /**\r\n * Redactor block\r\n */\r\n draw.block = function (tagName, content) {\r\n\r\n var node = document.createElement(tagName);\r\n\r\n node.innerHTML = content || '';\r\n\r\n return node;\r\n\r\n };\r\n\r\n /**\r\n * Creates Node with passed tagName and className\r\n * @param {string} tagName\r\n * @param {string} className\r\n * @param {object} properties - allow to assign properties\r\n */\r\n draw.node = function ( tagName, className, properties ) {\r\n\r\n var el = document.createElement( tagName );\r\n\r\n if ( className ) el.className = className;\r\n\r\n if ( properties ) {\r\n\r\n for (var name in properties) {\r\n\r\n el[name] = properties[name];\r\n\r\n }\r\n\r\n }\r\n\r\n return el;\r\n\r\n };\r\n\r\n /**\r\n * Unavailable plugin block\r\n */\r\n draw.unavailableBlock = function () {\r\n\r\n var wrapper = document.createElement('DIV');\r\n\r\n wrapper.classList.add('cdx-unavailable-block');\r\n\r\n return wrapper;\r\n\r\n };\r\n\r\n return draw;\r\n\r\n})({});\n\n\n// WEBPACK FOOTER //\n// ./modules/draw.js","/**\r\n * Codex Editor Caret Module\r\n *\r\n * @author Codex Team\r\n * @version 1.0\r\n */\r\n\r\nmodule.exports = (function (caret) {\r\n\r\n let editor = codex.editor;\r\n\r\n /**\r\n * @var {int} InputIndex - editable element in DOM\r\n */\r\n caret.inputIndex = null;\r\n\r\n /**\r\n * @var {int} offset - caret position in a text node.\r\n */\r\n caret.offset = null;\r\n\r\n /**\r\n * @var {int} focusedNodeIndex - we get index of child node from first-level block\r\n */\r\n caret.focusedNodeIndex = null;\r\n\r\n /**\r\n * Creates Document Range and sets caret to the element.\r\n * @protected\r\n * @uses caret.save — if you need to save caret position\r\n * @param {Element} el - Changed Node.\r\n */\r\n caret.set = function ( el, index, offset) {\r\n\r\n offset = offset || caret.offset || 0;\r\n index = index || caret.focusedNodeIndex || 0;\r\n\r\n var childs = el.childNodes,\r\n nodeToSet;\r\n\r\n if ( childs.length === 0 ) {\r\n\r\n nodeToSet = el;\r\n\r\n } else {\r\n\r\n nodeToSet = childs[index];\r\n\r\n }\r\n\r\n /** If Element is INPUT */\r\n if (el.contentEditable != 'true') {\r\n\r\n el.focus();\r\n return;\r\n\r\n }\r\n\r\n if (editor.core.isDomNode(nodeToSet)) {\r\n\r\n nodeToSet = editor.content.getDeepestTextNodeFromPosition(nodeToSet, nodeToSet.childNodes.length);\r\n\r\n }\r\n\r\n var range = document.createRange(),\r\n selection = window.getSelection();\r\n\r\n window.setTimeout(function () {\r\n\r\n range.setStart(nodeToSet, offset);\r\n range.setEnd(nodeToSet, offset);\r\n\r\n selection.removeAllRanges();\r\n selection.addRange(range);\r\n\r\n editor.caret.saveCurrentInputIndex();\r\n\r\n }, 20);\r\n\r\n };\r\n\r\n /**\r\n * @protected\r\n * Updates index of input and saves it in caret object\r\n */\r\n caret.saveCurrentInputIndex = function () {\r\n\r\n /** Index of Input that we paste sanitized content */\r\n var selection = window.getSelection(),\r\n inputs = editor.state.inputs,\r\n focusedNode = selection.anchorNode,\r\n focusedNodeHolder;\r\n\r\n if (!focusedNode) {\r\n\r\n return;\r\n\r\n }\r\n\r\n /** Looking for parent contentEditable block */\r\n while (focusedNode.contentEditable != 'true') {\r\n\r\n focusedNodeHolder = focusedNode.parentNode;\r\n focusedNode = focusedNodeHolder;\r\n\r\n }\r\n\r\n /** Input index in DOM level */\r\n var editableElementIndex = 0;\r\n\r\n while (focusedNode != inputs[editableElementIndex]) {\r\n\r\n editableElementIndex ++;\r\n\r\n }\r\n\r\n caret.inputIndex = editableElementIndex;\r\n\r\n };\r\n\r\n /**\r\n * Returns current input index (caret object)\r\n */\r\n caret.getCurrentInputIndex = function () {\r\n\r\n return caret.inputIndex;\r\n\r\n };\r\n\r\n /**\r\n * @param {int} index - index of first-level block after that we set caret into next input\r\n */\r\n caret.setToNextBlock = function (index) {\r\n\r\n var inputs = editor.state.inputs,\r\n nextInput = inputs[index + 1];\r\n\r\n if (!nextInput) {\r\n\r\n editor.core.log('We are reached the end');\r\n return;\r\n\r\n }\r\n\r\n /**\r\n * When new Block created or deleted content of input\r\n * We should add some text node to set caret\r\n */\r\n if (!nextInput.childNodes.length) {\r\n\r\n var emptyTextElement = document.createTextNode('');\r\n\r\n nextInput.appendChild(emptyTextElement);\r\n\r\n }\r\n\r\n editor.caret.inputIndex = index + 1;\r\n editor.caret.set(nextInput, 0, 0);\r\n editor.content.workingNodeChanged(nextInput);\r\n\r\n };\r\n\r\n /**\r\n * @param {int} index - index of target input.\r\n * Sets caret to input with this index\r\n */\r\n caret.setToBlock = function (index) {\r\n\r\n var inputs = editor.state.inputs,\r\n targetInput = inputs[index];\r\n\r\n if ( !targetInput ) {\r\n\r\n return;\r\n\r\n }\r\n\r\n /**\r\n * When new Block created or deleted content of input\r\n * We should add some text node to set caret\r\n */\r\n if (!targetInput.childNodes.length) {\r\n\r\n var emptyTextElement = document.createTextNode('');\r\n\r\n targetInput.appendChild(emptyTextElement);\r\n\r\n }\r\n\r\n editor.caret.inputIndex = index;\r\n editor.caret.set(targetInput, 0, 0);\r\n editor.content.workingNodeChanged(targetInput);\r\n\r\n };\r\n\r\n /**\r\n * @param {int} index - index of input\r\n */\r\n caret.setToPreviousBlock = function (index) {\r\n\r\n index = index || 0;\r\n\r\n var inputs = editor.state.inputs,\r\n previousInput = inputs[index - 1],\r\n lastChildNode,\r\n lengthOfLastChildNode,\r\n emptyTextElement;\r\n\r\n\r\n if (!previousInput) {\r\n\r\n editor.core.log('We are reached first node');\r\n return;\r\n\r\n }\r\n\r\n lastChildNode = editor.content.getDeepestTextNodeFromPosition(previousInput, previousInput.childNodes.length);\r\n lengthOfLastChildNode = lastChildNode.length;\r\n\r\n /**\r\n * When new Block created or deleted content of input\r\n * We should add some text node to set caret\r\n */\r\n if (!previousInput.childNodes.length) {\r\n\r\n emptyTextElement = document.createTextNode('');\r\n previousInput.appendChild(emptyTextElement);\r\n\r\n }\r\n editor.caret.inputIndex = index - 1;\r\n editor.caret.set(previousInput, previousInput.childNodes.length - 1, lengthOfLastChildNode);\r\n editor.content.workingNodeChanged(inputs[index - 1]);\r\n\r\n };\r\n\r\n caret.position = {\r\n\r\n atStart : function () {\r\n\r\n var selection = window.getSelection(),\r\n anchorOffset = selection.anchorOffset,\r\n anchorNode = selection.anchorNode,\r\n firstLevelBlock = editor.content.getFirstLevelBlock(anchorNode),\r\n pluginsRender = firstLevelBlock.childNodes[0];\r\n\r\n if (!editor.core.isDomNode(anchorNode)) {\r\n\r\n anchorNode = anchorNode.parentNode;\r\n\r\n }\r\n\r\n var isFirstNode = anchorNode === pluginsRender.childNodes[0],\r\n isOffsetZero = anchorOffset === 0;\r\n\r\n return isFirstNode && isOffsetZero;\r\n\r\n },\r\n\r\n atTheEnd : function () {\r\n\r\n var selection = window.getSelection(),\r\n anchorOffset = selection.anchorOffset,\r\n anchorNode = selection.anchorNode;\r\n\r\n /** Caret is at the end of input */\r\n return !anchorNode || !anchorNode.length || anchorOffset === anchorNode.length;\r\n\r\n }\r\n };\r\n\r\n\r\n /**\r\n * Inserts node at the caret location\r\n * @param node\r\n */\r\n caret.insertNode = function (node) {\r\n\r\n var selection, range;\r\n\r\n selection = window.getSelection();\r\n\r\n range = selection.getRangeAt(0);\r\n range.deleteContents();\r\n\r\n range.insertNode(node);\r\n\r\n range.setStartAfter(node);\r\n range.collapse(true);\r\n\r\n selection.removeAllRanges();\r\n selection.addRange(range);\r\n\r\n\r\n };\r\n\r\n return caret;\r\n\r\n})({});\n\n\n// WEBPACK FOOTER //\n// ./modules/caret.js","/**\r\n * Codex Editor Notification Module\r\n *\r\n * @author Codex Team\r\n * @version 1.0\r\n */\r\n\r\nmodule.exports = (function (notifications) {\r\n\r\n let editor = codex.editor;\r\n\r\n var queue = [];\r\n\r\n var addToQueue = function (settings) {\r\n\r\n queue.push(settings);\r\n\r\n var index = 0;\r\n\r\n while ( index < queue.length && queue.length > 5) {\r\n\r\n if (queue[index].type == 'confirm' || queue[index].type == 'prompt') {\r\n\r\n index++;\r\n continue;\r\n\r\n }\r\n\r\n queue[index].close();\r\n queue.splice(index, 1);\r\n\r\n }\r\n\r\n };\r\n\r\n notifications.createHolder = function () {\r\n\r\n var holder = editor.draw.node('DIV', 'cdx-notifications-block');\r\n\r\n editor.nodes.notifications = document.body.appendChild(holder);\r\n\r\n return holder;\r\n\r\n };\r\n\r\n\r\n /**\r\n * Error notificator. Shows block with message\r\n * @protected\r\n */\r\n notifications.errorThrown = function (errorMsg, event) {\r\n\r\n editor.notifications.notification({message: 'This action is not available currently', type: event.type});\r\n\r\n };\r\n\r\n /**\r\n *\r\n * Appends notification\r\n *\r\n * settings = {\r\n * type - notification type (reserved types: alert, confirm, prompt). Just add class 'cdx-notification-'+type\r\n * message - notification message\r\n * okMsg - confirm button text (default - 'Ok')\r\n * cancelBtn - cancel button text (default - 'Cancel'). Only for confirm and prompt types\r\n * confirm - function-handler for ok button click\r\n * cancel - function-handler for cancel button click. Only for confirm and prompt types\r\n * time - time (in seconds) after which notification will close (default - 10s)\r\n * }\r\n *\r\n * @param settings\r\n */\r\n notifications.notification = function (constructorSettings) {\r\n\r\n /** Private vars and methods */\r\n var notification = null,\r\n cancel = null,\r\n type = null,\r\n confirm = null,\r\n inputField = null;\r\n\r\n var confirmHandler = function () {\r\n\r\n close();\r\n\r\n if (typeof confirm !== 'function' ) {\r\n\r\n return;\r\n\r\n }\r\n\r\n if (type == 'prompt') {\r\n\r\n confirm(inputField.value);\r\n return;\r\n\r\n }\r\n\r\n confirm();\r\n\r\n };\r\n\r\n var cancelHandler = function () {\r\n\r\n close();\r\n\r\n if (typeof cancel !== 'function' ) {\r\n\r\n return;\r\n\r\n }\r\n\r\n cancel();\r\n\r\n };\r\n\r\n\r\n /** Public methods */\r\n function create(settings) {\r\n\r\n if (!(settings && settings.message)) {\r\n\r\n editor.core.log('Can\\'t create notification. Message is missed');\r\n return;\r\n\r\n }\r\n\r\n settings.type = settings.type || 'alert';\r\n settings.time = settings.time*1000 || 10000;\r\n\r\n var wrapper = editor.draw.node('DIV', 'cdx-notification'),\r\n message = editor.draw.node('DIV', 'cdx-notification__message'),\r\n input = editor.draw.node('INPUT', 'cdx-notification__input'),\r\n okBtn = editor.draw.node('SPAN', 'cdx-notification__ok-btn'),\r\n cancelBtn = editor.draw.node('SPAN', 'cdx-notification__cancel-btn');\r\n\r\n message.textContent = settings.message;\r\n okBtn.textContent = settings.okMsg || 'ОК';\r\n cancelBtn.textContent = settings.cancelMsg || 'Отмена';\r\n\r\n editor.listeners.add(okBtn, 'click', confirmHandler);\r\n editor.listeners.add(cancelBtn, 'click', cancelHandler);\r\n\r\n wrapper.appendChild(message);\r\n\r\n if (settings.type == 'prompt') {\r\n\r\n wrapper.appendChild(input);\r\n\r\n }\r\n\r\n wrapper.appendChild(okBtn);\r\n\r\n if (settings.type == 'prompt' || settings.type == 'confirm') {\r\n\r\n wrapper.appendChild(cancelBtn);\r\n\r\n }\r\n\r\n wrapper.classList.add('cdx-notification-' + settings.type);\r\n wrapper.dataset.type = settings.type;\r\n\r\n notification = wrapper;\r\n type = settings.type;\r\n confirm = settings.confirm;\r\n cancel = settings.cancel;\r\n inputField = input;\r\n\r\n if (settings.type != 'prompt' && settings.type != 'confirm') {\r\n\r\n window.setTimeout(close, settings.time);\r\n\r\n }\r\n\r\n };\r\n\r\n function send() {\r\n\r\n editor.nodes.notifications.appendChild(notification);\r\n inputField.focus();\r\n\r\n editor.nodes.notifications.classList.add('cdx-notification__notification-appending');\r\n\r\n window.setTimeout(function () {\r\n\r\n editor.nodes.notifications.classList.remove('cdx-notification__notification-appending');\r\n\r\n }, 100);\r\n\r\n addToQueue({type: type, close: close});\r\n\r\n };\r\n\r\n function close() {\r\n\r\n notification.remove();\r\n\r\n };\r\n\r\n\r\n if (constructorSettings) {\r\n\r\n create(constructorSettings);\r\n send();\r\n\r\n }\r\n\r\n return {\r\n create: create,\r\n send: send,\r\n close: close\r\n };\r\n\r\n };\r\n\r\n notifications.clear = function () {\r\n\r\n editor.nodes.notifications.innerHTML = '';\r\n queue = [];\r\n\r\n };\r\n\r\n return notifications;\r\n\r\n})({});\n\n\n// WEBPACK FOOTER //\n// ./modules/notifications.js","/**\r\n * Codex Editor Parser Module\r\n *\r\n * @author Codex Team\r\n * @version 1.1\r\n */\r\n\r\nmodule.exports = (function (parser) {\r\n\r\n let editor = codex.editor;\r\n\r\n /** inserting text */\r\n parser.insertPastedContent = function (blockType, tag) {\r\n\r\n editor.content.insertBlock({\r\n type : blockType.type,\r\n block : blockType.render({\r\n text : tag.innerHTML\r\n })\r\n });\r\n\r\n };\r\n\r\n /**\r\n * Check DOM node for display style: separated block or child-view\r\n */\r\n parser.isFirstLevelBlock = function (node) {\r\n\r\n return node.nodeType == editor.core.nodeTypes.TAG &&\r\n node.classList.contains(editor.ui.className.BLOCK_CLASSNAME);\r\n\r\n };\r\n\r\n return parser;\r\n\r\n})({});\r\n\n\n\n// WEBPACK FOOTER //\n// ./modules/parser.js","/**\r\n * Codex Sanitizer\r\n */\r\n\r\nmodule.exports = (function (sanitizer) {\r\n\r\n /** HTML Janitor library */\r\n let janitor = require('html-janitor');\r\n\r\n /** Codex Editor */\r\n let editor = codex.editor;\r\n\r\n sanitizer.prepare = function () {\r\n\r\n if (editor.settings.sanitizer && !editor.core.isEmpty(editor.settings.sanitizer)) {\r\n\r\n Config.CUSTOM = editor.settings.sanitizer;\r\n\r\n }\r\n\r\n };\r\n\r\n /**\r\n * Basic config\r\n */\r\n var Config = {\r\n\r\n /** User configuration */\r\n CUSTOM : null,\r\n\r\n BASIC : {\r\n\r\n tags: {\r\n p: {},\r\n a: {\r\n href: true,\r\n target: '_blank',\r\n rel: 'nofollow'\r\n },\r\n i: {},\r\n b: {},\r\n strong: {},\r\n em: {},\r\n span: {}\r\n }\r\n }\r\n };\r\n\r\n sanitizer.Config = Config;\r\n\r\n /**\r\n *\r\n * @param userCustomConfig\r\n * @returns {*}\r\n * @private\r\n *\r\n * @description If developer uses editor's API, then he can customize sane restrictions.\r\n * Or, sane config can be defined globally in editors initialization. That config will be used everywhere\r\n * At least, if there is no config overrides, that API uses BASIC Default configation\r\n */\r\n let init_ = function (userCustomConfig) {\r\n\r\n let configuration = userCustomConfig || Config.CUSTOM || Config.BASIC;\r\n\r\n return new janitor(configuration);\r\n\r\n };\r\n\r\n /**\r\n * Cleans string from unwanted tags\r\n * @protected\r\n * @param {String} dirtyString - taint string\r\n * @param {Object} customConfig - allowed tags\r\n */\r\n sanitizer.clean = function (dirtyString, customConfig) {\r\n\r\n let janitorInstance = init_(customConfig);\r\n\r\n return janitorInstance.clean(dirtyString);\r\n\r\n };\r\n\r\n return sanitizer;\r\n\r\n})({});\n\n\n// WEBPACK FOOTER //\n// ./modules/sanitizer.js","(function (root, factory) {\n if (typeof define === 'function' && define.amd) {\n define('html-janitor', factory);\n } else if (typeof exports === 'object') {\n module.exports = factory();\n } else {\n root.HTMLJanitor = factory();\n }\n}(this, function () {\n\n /**\n * @param {Object} config.tags Dictionary of allowed tags.\n * @param {boolean} config.keepNestedBlockElements Default false.\n */\n function HTMLJanitor(config) {\n\n var tagDefinitions = config['tags'];\n var tags = Object.keys(tagDefinitions);\n\n var validConfigValues = tags\n .map(function(k) { return typeof tagDefinitions[k]; })\n .every(function(type) { return type === 'object' || type === 'boolean' || type === 'function'; });\n\n if(!validConfigValues) {\n throw new Error(\"The configuration was invalid\");\n }\n\n this.config = config;\n }\n\n // TODO: not exhaustive?\n var blockElementNames = ['P', 'LI', 'TD', 'TH', 'DIV', 'H1', 'H2', 'H3', 'H4', 'H5', 'H6', 'PRE'];\n function isBlockElement(node) {\n return blockElementNames.indexOf(node.nodeName) !== -1;\n }\n\n var inlineElementNames = ['A', 'B', 'STRONG', 'I', 'EM', 'SUB', 'SUP', 'U', 'STRIKE'];\n function isInlineElement(node) {\n return inlineElementNames.indexOf(node.nodeName) !== -1;\n }\n\n HTMLJanitor.prototype.clean = function (html) {\n var sandbox = document.createElement('div');\n sandbox.innerHTML = html;\n\n this._sanitize(sandbox);\n\n return sandbox.innerHTML;\n };\n\n HTMLJanitor.prototype._sanitize = function (parentNode) {\n var treeWalker = createTreeWalker(parentNode);\n var node = treeWalker.firstChild();\n if (!node) { return; }\n\n do {\n // Ignore nodes that have already been sanitized\n if (node._sanitized) {\n continue;\n }\n\n if (node.nodeType === Node.TEXT_NODE) {\n // If this text node is just whitespace and the previous or next element\n // sibling is a block element, remove it\n // N.B.: This heuristic could change. Very specific to a bug with\n // `contenteditable` in Firefox: http://jsbin.com/EyuKase/1/edit?js,output\n // FIXME: make this an option?\n if (node.data.trim() === ''\n && ((node.previousElementSibling && isBlockElement(node.previousElementSibling))\n || (node.nextElementSibling && isBlockElement(node.nextElementSibling)))) {\n parentNode.removeChild(node);\n this._sanitize(parentNode);\n break;\n } else {\n continue;\n }\n }\n\n // Remove all comments\n if (node.nodeType === Node.COMMENT_NODE) {\n parentNode.removeChild(node);\n this._sanitize(parentNode);\n break;\n }\n\n var isInline = isInlineElement(node);\n var containsBlockElement;\n if (isInline) {\n containsBlockElement = Array.prototype.some.call(node.childNodes, isBlockElement);\n }\n\n // Block elements should not be nested (e.g.

  • ...); if\n // they are, we want to unwrap the inner block element.\n var isNotTopContainer = !! parentNode.parentNode;\n var isNestedBlockElement =\n isBlockElement(parentNode) &&\n isBlockElement(node) &&\n isNotTopContainer;\n\n var nodeName = node.nodeName.toLowerCase();\n\n var allowedAttrs = getAllowedAttrs(this.config, nodeName, node);\n\n var isInvalid = isInline && containsBlockElement;\n\n // Drop tag entirely according to the whitelist *and* if the markup\n // is invalid.\n if (isInvalid || shouldRejectNode(node, allowedAttrs)\n || (!this.config.keepNestedBlockElements && isNestedBlockElement)) {\n // Do not keep the inner text of SCRIPT/STYLE elements.\n if (! (node.nodeName === 'SCRIPT' || node.nodeName === 'STYLE')) {\n while (node.childNodes.length > 0) {\n parentNode.insertBefore(node.childNodes[0], node);\n }\n }\n parentNode.removeChild(node);\n\n this._sanitize(parentNode);\n break;\n }\n\n // Sanitize attributes\n for (var a = 0; a < node.attributes.length; a += 1) {\n var attr = node.attributes[a];\n\n if (shouldRejectAttr(attr, allowedAttrs, node)) {\n node.removeAttribute(attr.name);\n // Shift the array to continue looping.\n a = a - 1;\n }\n }\n\n // Sanitize children\n this._sanitize(node);\n\n // Mark node as sanitized so it's ignored in future runs\n node._sanitized = true;\n } while ((node = treeWalker.nextSibling()));\n };\n\n function createTreeWalker(node) {\n return document.createTreeWalker(node,\n NodeFilter.SHOW_TEXT | NodeFilter.SHOW_ELEMENT | NodeFilter.SHOW_COMMENT,\n null, false);\n }\n\n function getAllowedAttrs(config, nodeName, node){\n if (typeof config.tags[nodeName] === 'function') {\n return config.tags[nodeName](node);\n } else {\n return config.tags[nodeName];\n }\n }\n\n function shouldRejectNode(node, allowedAttrs){\n if (typeof allowedAttrs === 'undefined') {\n return true;\n } else if (typeof allowedAttrs === 'boolean') {\n return !allowedAttrs;\n }\n\n return false;\n }\n\n function shouldRejectAttr(attr, allowedAttrs, node){\n var attrName = attr.name.toLowerCase();\n\n if (allowedAttrs === true){\n return false;\n } else if (typeof allowedAttrs[attrName] === 'function'){\n return !allowedAttrs[attrName](attr.value, node);\n } else if (typeof allowedAttrs[attrName] === 'undefined'){\n return true;\n } else if (allowedAttrs[attrName] === false) {\n return true;\n } else if (typeof allowedAttrs[attrName] === 'string') {\n return (allowedAttrs[attrName] !== attr.value);\n }\n\n return false;\n }\n\n return HTMLJanitor;\n\n}));\n\n\n\n//////////////////\n// WEBPACK FOOTER\n// ./~/html-janitor/src/html-janitor.js\n// module id = 18\n// module chunks = 0","/**\r\n * Codex Editor Listeners module\r\n *\r\n * @author Codex Team\r\n * @version 1.0\r\n */\r\n\r\n/**\r\n * Module-decorator for event listeners assignment\r\n */\r\nmodule.exports = function (listeners) {\r\n\r\n var allListeners = [];\r\n\r\n /**\r\n * Search methods\r\n *\r\n * byElement, byType and byHandler returns array of suitable listeners\r\n * one and all takes element, eventType, and handler and returns first (all) suitable listener\r\n *\r\n */\r\n listeners.search = function () {\r\n\r\n var byElement = function (element, context) {\r\n\r\n var listenersOnElement = [];\r\n\r\n context = context || allListeners;\r\n\r\n for (var i = 0; i < context.length; i++) {\r\n\r\n var listener = context[i];\r\n\r\n if (listener.element === element) {\r\n\r\n listenersOnElement.push(listener);\r\n\r\n }\r\n\r\n }\r\n\r\n return listenersOnElement;\r\n\r\n };\r\n\r\n var byType = function (eventType, context) {\r\n\r\n var listenersWithType = [];\r\n\r\n context = context || allListeners;\r\n\r\n for (var i = 0; i < context.length; i++) {\r\n\r\n var listener = context[i];\r\n\r\n if (listener.type === eventType) {\r\n\r\n listenersWithType.push(listener);\r\n\r\n }\r\n\r\n }\r\n\r\n return listenersWithType;\r\n\r\n };\r\n\r\n var byHandler = function (handler, context) {\r\n\r\n var listenersWithHandler = [];\r\n\r\n context = context || allListeners;\r\n\r\n for (var i = 0; i < context.length; i++) {\r\n\r\n var listener = context[i];\r\n\r\n if (listener.handler === handler) {\r\n\r\n listenersWithHandler.push(listener);\r\n\r\n }\r\n\r\n }\r\n\r\n return listenersWithHandler;\r\n\r\n };\r\n\r\n var one = function (element, eventType, handler) {\r\n\r\n var result = allListeners;\r\n\r\n if (element)\r\n result = byElement(element, result);\r\n\r\n if (eventType)\r\n result = byType(eventType, result);\r\n\r\n if (handler)\r\n result = byHandler(handler, result);\r\n\r\n return result[0];\r\n\r\n };\r\n\r\n var all = function (element, eventType, handler) {\r\n\r\n var result = allListeners;\r\n\r\n if (element)\r\n result = byElement(element, result);\r\n\r\n if (eventType)\r\n result = byType(eventType, result);\r\n\r\n if (handler)\r\n result = byHandler(handler, result);\r\n\r\n return result;\r\n\r\n };\r\n\r\n return {\r\n byElement : byElement,\r\n byType : byType,\r\n byHandler : byHandler,\r\n one : one,\r\n all : all\r\n };\r\n\r\n }();\r\n\r\n listeners.add = function (element, eventType, handler, isCapture) {\r\n\r\n element.addEventListener(eventType, handler, isCapture);\r\n\r\n var data = {\r\n element: element,\r\n type: eventType,\r\n handler: handler\r\n };\r\n\r\n var alreadyAddedListener = listeners.search.one(element, eventType, handler);\r\n\r\n if (!alreadyAddedListener) {\r\n\r\n allListeners.push(data);\r\n\r\n }\r\n\r\n };\r\n\r\n listeners.remove = function (element, eventType, handler) {\r\n\r\n element.removeEventListener(eventType, handler);\r\n\r\n var existingListeners = listeners.search.all(element, eventType, handler);\r\n\r\n for (var i = 0; i < existingListeners.length; i++) {\r\n\r\n var index = allListeners.indexOf(existingListeners[i]);\r\n\r\n if (index > 0) {\r\n\r\n allListeners.splice(index, 1);\r\n\r\n }\r\n\r\n }\r\n\r\n };\r\n\r\n listeners.removeAll = function () {\r\n\r\n allListeners.map(function (current) {\r\n\r\n listeners.remove(current.element, current.type, current.handler);\r\n\r\n });\r\n\r\n };\r\n\r\n listeners.get = function (element, eventType, handler) {\r\n\r\n return listeners.search.all(element, eventType, handler);\r\n\r\n };\r\n\r\n return listeners;\r\n\r\n}({});\n\n\n// WEBPACK FOOTER //\n// ./modules/listeners.js","/**\r\n * Codex Editor Destroyer module\r\n *\r\n * @auhor Codex Team\r\n * @version 1.0\r\n */\r\n\r\nmodule.exports = function (destroyer) {\r\n\r\n let editor = codex.editor;\r\n\r\n destroyer.removeNodes = function () {\r\n\r\n editor.nodes.wrapper.remove();\r\n editor.nodes.notifications.remove();\r\n\r\n };\r\n\r\n destroyer.destroyPlugins = function () {\r\n\r\n for (var tool in editor.tools) {\r\n\r\n if (typeof editor.tools[tool].destroy === 'function') {\r\n\r\n editor.tools[tool].destroy();\r\n\r\n }\r\n\r\n }\r\n\r\n };\r\n\r\n destroyer.destroyScripts = function () {\r\n\r\n var scripts = document.getElementsByTagName('SCRIPT');\r\n\r\n for (var i = 0; i < scripts.length; i++) {\r\n\r\n if (scripts[i].id.indexOf(editor.scriptPrefix) + 1) {\r\n\r\n scripts[i].remove();\r\n i--;\r\n\r\n }\r\n\r\n }\r\n\r\n };\r\n\r\n\r\n /**\r\n * Delete editor data from webpage.\r\n * You should send settings argument with boolean flags:\r\n * @param settings.ui- remove redactor event listeners and DOM nodes\r\n * @param settings.scripts - remove redactor scripts from DOM\r\n * @param settings.plugins - remove plugin's objects\r\n * @param settings.core - remove editor core. You can remove core only if UI and scripts flags is true\r\n * }\r\n *\r\n */\r\n destroyer.destroy = function (settings) {\r\n\r\n if (!settings || typeof settings !== 'object') {\r\n\r\n return;\r\n\r\n }\r\n\r\n if (settings.ui) {\r\n\r\n destroyer.removeNodes();\r\n editor.listeners.removeAll();\r\n\r\n }\r\n\r\n if (settings.scripts) {\r\n\r\n destroyer.destroyScripts();\r\n\r\n }\r\n\r\n if (settings.plugins) {\r\n\r\n destroyer.destroyPlugins();\r\n\r\n }\r\n\r\n if (settings.ui && settings.scripts && settings.core) {\r\n\r\n delete codex.editor;\r\n\r\n }\r\n\r\n };\r\n\r\n return destroyer;\r\n\r\n}({});\n\n\n// WEBPACK FOOTER //\n// ./modules/destroyer.js","/**\r\n * Codex Editor Paste module\r\n *\r\n * @author Codex Team\r\n * @version 1.1.1\r\n */\r\n\r\nmodule.exports = function (paste) {\r\n\r\n let editor = codex.editor;\r\n\r\n var patterns = [];\r\n\r\n paste.prepare = function () {\r\n\r\n var tools = editor.tools;\r\n\r\n for (var tool in tools) {\r\n\r\n if (!tools[tool].renderOnPastePatterns || !Array.isArray(tools[tool].renderOnPastePatterns)) {\r\n\r\n continue;\r\n\r\n }\r\n\r\n tools[tool].renderOnPastePatterns.map(function (pattern) {\r\n\r\n\r\n patterns.push(pattern);\r\n\r\n });\r\n\r\n }\r\n\r\n return Promise.resolve();\r\n\r\n };\r\n\r\n /**\r\n * Saves data\r\n * @param event\r\n */\r\n paste.pasted = function (event) {\r\n\r\n var clipBoardData = event.clipboardData || window.clipboardData,\r\n content = clipBoardData.getData('Text');\r\n\r\n var result = analize(content);\r\n\r\n if (result) {\r\n\r\n event.preventDefault();\r\n event.stopImmediatePropagation();\r\n\r\n }\r\n\r\n return result;\r\n\r\n };\r\n\r\n /**\r\n * Analizes pated string and calls necessary method\r\n */\r\n\r\n var analize = function (string) {\r\n\r\n var result = false,\r\n content = editor.content.currentNode,\r\n plugin = content.dataset.tool;\r\n\r\n patterns.map( function (pattern) {\r\n\r\n var execArray = pattern.regex.exec(string),\r\n match = execArray && execArray[0];\r\n\r\n if ( match && match === string.trim()) {\r\n\r\n /** current block is not empty */\r\n if ( content.textContent.trim() && plugin == editor.settings.initialBlockPlugin ) {\r\n\r\n pasteToNewBlock_();\r\n\r\n }\r\n\r\n pattern.callback(string, pattern);\r\n result = true;\r\n\r\n }\r\n\r\n });\r\n\r\n return result;\r\n\r\n };\r\n\r\n var pasteToNewBlock_ = function () {\r\n\r\n /** Create new initial block */\r\n editor.content.insertBlock({\r\n\r\n type : editor.settings.initialBlockPlugin,\r\n block : editor.tools[editor.settings.initialBlockPlugin].render({\r\n text : ''\r\n })\r\n\r\n }, false);\r\n\r\n };\r\n\r\n return paste;\r\n\r\n}({});\n\n\n// WEBPACK FOOTER //\n// ./modules/paste.js"],"sourceRoot":""} \ No newline at end of file +{"version":3,"sources":["webpack:///codex-editor.js","webpack:///webpack/bootstrap ad1af9a0031475a14714","webpack:///./codex.js","webpack:///./modules/core.js","webpack:///./modules/tools.js","webpack:///./modules/ui.js","webpack:///./modules/transport.js","webpack:///./modules/renderer.js","webpack:///./modules/saver.js","webpack:///./modules/content.js","webpack:///./modules/toolbar/toolbar.js","webpack:///./modules/toolbar/settings.js","webpack:///./modules/toolbar/inline.js","webpack:///./modules/toolbar/toolbox.js","webpack:///./modules/callbacks.js","webpack:///./modules/draw.js","webpack:///./modules/caret.js","webpack:///./modules/notifications.js","webpack:///./modules/parser.js","webpack:///./modules/sanitizer.js","webpack:///./~/html-janitor/src/html-janitor.js","webpack:///./modules/listeners.js","webpack:///./modules/destroyer.js","webpack:///./modules/paste.js"],"names":["codex","modules","__webpack_require__","moduleId","installedModules","exports","module","id","loaded","call","m","c","p","editor","version","scriptPrefix","init","core","tools","ui","transport","renderer","saver","content","toolbar","callback","draw","caret","notifications","parser","sanitizer","listeners","destroyer","paste","settings","holderId","initialBlockPlugin","nodes","holder","wrapper","inlineToolbar","buttons","actions","toolbox","plusButton","showSettingsButton","showTrashButton","blockSettings","pluginSettings","defaultSettings","toolbarButtons","redactor","state","jsonOutput","blocks","inputs","start","userSettings","prepare","then","makeBlocksFromData","saveInputs","catch","error","log","_typeof","Symbol","iterator","obj","constructor","prototype","Promise","resolve","reject","data","hideToolbar","document","getElementById","undefined","Error","msg","type","arg","window","console","e","insertAfter","target","element","parentNode","insertBefore","nextSibling","nodeTypes","TAG","TEXT","COMMENT","keys","BACKSPACE","TAB","ENTER","SHIFT","CTRL","ALT","ESC","SPACE","LEFT","UP","DOWN","RIGHT","DELETE","META","isDomNode","el","nodeType","this","isEmpty","Object","length","ajax","url","encodedString","isFormData","prop","XMLHTTP","XMLHttpRequest","ActiveXObject","async","test","encodeURIComponent","withCredentials","responseContext","beforeSendResult","beforeSend","open","isFormData_","setRequestHeader","progress","upload","onprogress","bind","onreadystatechange","readyState","status","success","responseText","send","importScript","scriptPath","instanceName","script","createElement","defer","onload","onerror","src","head","appendChild","object","FormData","isNativeInput","nativeInputAreas","indexOf","tagName","resolve_","reject_","pluginsRequiresPreparation","allPlugins","pluginName","plugin","push","waitAllPluginsPreparation_","plugins","allPluginsProcessed__","reduce","previousValue","iteration","pluginIsReady__","callPluginsPrepareMethod_","available","loadingMessage","config","className","BLOCK_CLASSNAME","BLOCK_CONTENT","BLOCK_STRETCHED","BLOCK_HIGHLIGHTED","SETTINGS_ITEM","makeToolBar_","addTools_","makeInlineToolbar_","addInlineToolbarTools_","makeNotificationHolder_","bindEvents_","container","inlineToolbarButtons","inlineToolbarActions","blockButtons","makeToolbarSettings_","toolbarContent","makeToolbarContent_","settingsButton","makeRemoveBlockButton","pluginsSettings","createHolder","tool","toolName","toolButton","iconClassname","displayInToolbox","render","toolbarButton","bold","icon","command","italic","underline","link","name","toolbarButtonInline","setInlineToolbarButtonBehaviour","add","globalKeydown","redactorKeyDown","globalKeyup","redactorClicked","plusButtonClicked","showSettingsButtonClicked","button","toolbarButtonClicked","addBlockHandlers","block","blockKeydown","blockPasteCallback","inline","show","querySelectorAll","Array","map","current","addInitialBlock","initialBlock","initialBlockType","setAttribute","insertBlock","workingNodeChanged","event","toolClicked","currentRequest","input","arguments","node","fileSelected","clearInput","i","files","formData","multiple","append","selectAndUpload","args","accept","click","abort","items","appendBlocks","nodeSequence","index","appendNodeAtIndex","getNodeAsync","createBlockFromData","blockData","blocksList","position","toolData","unavailableBlock","innerHTML","dataset","inputPosition","stretched","isStretched","saveBlocks","html","childNodes","makeQueue","queue","getBlockData","makeFormDataFromBlocks","save","savedData","output","blockContent","pluginsContent","validate","result","currentNode","editorAreaHightlighted","sync","markBlock","classList","clearMark","remove","getFirstLevelBlock","body","contains","targetNode","replaceBlock","targetBlock","newBlock","replaceChild","needPlaceCaret","workingBlock","newBlockContent","blockType","composeNewBlock_","currentInputIndex","getCurrentInputIndex","editableElement","querySelector","emptyText","createTextNode","set","move","showPlusButton","setTimeout","setToNextBlock","switchBlock","blockToReplace","newBlockComposed","getDeepestTextNodeFromPosition","text","blockChilds","textContent","trim","removeChild","lookingFromStart","getRange","selection","getSelection","getRangeAt","splitBlock","inputIndex","textBeforeCaret","textNodeBeforeCaret","textAfterCaret","textNodeAfterCaret","anchorNode","anchorNodeText","caretOffset","anchorOffset","currentBlock","substring","previousChilds","nextChilds","reachedCurrent","child","previousChildsLength","nextChildsLength","newNode","NEW_BLOCK_TYPE","mergeBlocks","targetInputIndex","targetInput","currentInputContent","isLastNode","allChecked","allSiblingsEmpty_","sibling","wrapTextWithParagraphs","htmlData","plainData","split","join","paragraph","blockTyped","newWrapper","firstLevelBlocks","cloneNode","getEditableParent","contentEditable","defaultToolbarHeight","defaultOffset","opened","close","toggle","hidePlusButton","newYCoordinate","offsetTop","style","transform","Math","floor","hideRemoveActions","setting","toolType","makeSettings","settingsBlock","removeBlockWrapper","settingButton","actionWrapper","confirmAction","cancelAction","removeButtonClicked","confirmRemovingRequest","cancelRemovingRequest","action","showRemoveActions","firstLevelBlocksCount","buttonsOpened","actionsOpened","wrappersOffset","storedSelection","showInlineToolbar","selectedText","getSelectionText","showButtons","getWrappersOffset","newCoordinateX","newCoordinateY","coords","getSelectionCoords","offsetHeight","x","left","y","scrollY","top","closeButtons","closeAction","createLinkAction","defaultToolAction","forEach","hightlight","offset","getOffset","_x","_y","isNaN","offsetLeft","clientLeft","clientTop","offsetParent","range","sel","createRange","collapse","boundingLeft","boundingTop","rangeCount","cloneRange","getClientRects","rect","toString","showActions","inlineToolbarAnchorInputKeydown_","keyCode","editable","restoreSelection","setAnchor","value","preventDefault","stopImmediatePropagation","clearRange","isActive","isLinkActive","saveSelection","inputForLink","focus","dataType","execCommand","containerEl","preSelectionRange","selectNodeContents","setEnd","startContainer","startOffset","end","savedSel","charIndex","setStart","nextCharIndex","nodeStack","foundStart","stop","pop","removeAllRanges","addRange","queryCommandState","setButtonHighlighted","removeButtonsHighLight","tag","leaf","currentTool","barButtons","nextToolIndex","toolToSelect","visibleTool","appendCallback","UNREPLACEBLE_TOOLS","workingNode","setToBlock","callbacks","enterKeyPressed_","tabKeyPressedOnRedactorsZone_","enterKeyPressedOnRedactorsZone_","escapeKeyPressedOnRedactorsZone_","defaultKeyPressedOnRedactorsZone_","arrowKeyPressed_","nativeInputs","nativeInputsAreEmpty","textContentIsEmpty","blockIsEmpty","enterPressedOnBlock_","saveCurrentInputIndex","isEnterPressedOnToolbar","enableLineBreaks","stopPropagation","shiftKey","currentSelection","currentSelectedNode","caretAtTheEndOfText","atTheEnd","isTextNodeHasParentBetweenContenteditable","enterPressedOnBlock","islastNode","detectWhenClickedOnFirstLevelBlockArea_","firstLevelBlock","indexOfLastInput","inputIsEmpty","currentNodeType","isInitialType","flag","blockRightOrDownArrowPressed_","backspacePressed_","blockLeftOrUpArrowPressed_","focusedNodeHolder","focusedNode","editableElementIndex","lastChild","deepestTextnode","caretInLastChild","firstChild","caretInFirstChild","caretAtTheBeginning","setToPreviousBlock","selectionLength","endOffset","atStart","currentToolType","ceBlock","bar","placeholder","div","toggler","classname","toolIcon","toolTitle","properties","focusedNodeIndex","nodeToSet","childs","nextInput","emptyTextElement","lastChildNode","lengthOfLastChildNode","previousInput","pluginsRender","isFirstNode","isOffsetZero","insertNode","deleteContents","setStartAfter","addToQueue","splice","errorThrown","errorMsg","notification","message","constructorSettings","create","time","okBtn","cancelBtn","okMsg","cancelMsg","confirmHandler","cancelHandler","confirm","cancel","inputField","clear","insertPastedContent","isFirstLevelBlock","janitor","Config","CUSTOM","BASIC","tags","a","href","rel","b","strong","em","span","init_","userCustomConfig","configuration","clean","dirtyString","customConfig","janitorInstance","__WEBPACK_AMD_DEFINE_FACTORY__","__WEBPACK_AMD_DEFINE_RESULT__","root","factory","HTMLJanitor","tagDefinitions","validConfigValues","k","every","isBlockElement","blockElementNames","nodeName","isInlineElement","inlineElementNames","createTreeWalker","NodeFilter","SHOW_TEXT","SHOW_ELEMENT","SHOW_COMMENT","getAllowedAttrs","shouldRejectNode","allowedAttrs","shouldRejectAttr","attr","attrName","toLowerCase","sandbox","_sanitize","treeWalker","_sanitized","Node","TEXT_NODE","COMMENT_NODE","containsBlockElement","isInline","some","isNotTopContainer","isNestedBlockElement","isInvalid","keepNestedBlockElements","attributes","removeAttribute","previousElementSibling","nextElementSibling","allListeners","search","byElement","context","listenersOnElement","listener","byType","eventType","listenersWithType","byHandler","handler","listenersWithHandler","one","all","isCapture","addEventListener","alreadyAddedListener","removeEventListener","existingListeners","removeAll","get","removeNodes","destroyPlugins","destroy","destroyScripts","scripts","getElementsByTagName","patterns","renderOnPastePatterns","isArray","pattern","pasted","clipBoardData","clipboardData","getData","analize","string","execArray","regex","exec","match","pasteToNewBlock_","needsToHandlePasteEvent","cleanData","wrappedData","paragraphs","insertPastedParagraphs","editableParent","currentBlockContent"],"mappings":"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","file":"codex-editor.js","sourcesContent":["var codex = codex || {}; codex[\"editor\"] =\n/******/ (function(modules) { // webpackBootstrap\n/******/ \t// The module cache\n/******/ \tvar installedModules = {};\n/******/\n/******/ \t// The require function\n/******/ \tfunction __webpack_require__(moduleId) {\n/******/\n/******/ \t\t// Check if module is in cache\n/******/ \t\tif(installedModules[moduleId])\n/******/ \t\t\treturn installedModules[moduleId].exports;\n/******/\n/******/ \t\t// Create a new module (and put it into the cache)\n/******/ \t\tvar module = installedModules[moduleId] = {\n/******/ \t\t\texports: {},\n/******/ \t\t\tid: moduleId,\n/******/ \t\t\tloaded: false\n/******/ \t\t};\n/******/\n/******/ \t\t// Execute the module function\n/******/ \t\tmodules[moduleId].call(module.exports, module, module.exports, __webpack_require__);\n/******/\n/******/ \t\t// Flag the module as loaded\n/******/ \t\tmodule.loaded = true;\n/******/\n/******/ \t\t// Return the exports of the module\n/******/ \t\treturn module.exports;\n/******/ \t}\n/******/\n/******/\n/******/ \t// expose the modules object (__webpack_modules__)\n/******/ \t__webpack_require__.m = modules;\n/******/\n/******/ \t// expose the module cache\n/******/ \t__webpack_require__.c = installedModules;\n/******/\n/******/ \t// __webpack_public_path__\n/******/ \t__webpack_require__.p = \"\";\n/******/\n/******/ \t// Load entry module and return exports\n/******/ \treturn __webpack_require__(0);\n/******/ })\n/************************************************************************/\n/******/ ([\n/* 0 */\n/***/ function(module, exports, __webpack_require__) {\n\n\t'use strict';\n\t\n\t/**\r\n\t *\r\n\t * Codex Editor\r\n\t *\r\n\t * @author Codex Team\r\n\t */\n\t\n\tmodule.exports = function (editor) {\n\t\n\t 'use strict';\n\t\n\t editor.version = (\"1.6.4\");\n\t editor.scriptPrefix = 'cdx-script-';\n\t\n\t var init = function init() {\n\t\n\t editor.core = __webpack_require__(1);\n\t editor.tools = __webpack_require__(2);\n\t editor.ui = __webpack_require__(3);\n\t editor.transport = __webpack_require__(4);\n\t editor.renderer = __webpack_require__(5);\n\t editor.saver = __webpack_require__(6);\n\t editor.content = __webpack_require__(7);\n\t editor.toolbar = __webpack_require__(8);\n\t editor.callback = __webpack_require__(12);\n\t editor.draw = __webpack_require__(13);\n\t editor.caret = __webpack_require__(14);\n\t editor.notifications = __webpack_require__(15);\n\t editor.parser = __webpack_require__(16);\n\t editor.sanitizer = __webpack_require__(17);\n\t editor.listeners = __webpack_require__(19);\n\t editor.destroyer = __webpack_require__(20);\n\t editor.paste = __webpack_require__(21);\n\t };\n\t\n\t /**\r\n\t * @public\r\n\t * holds initial settings\r\n\t */\n\t editor.settings = {\n\t tools: ['paragraph', 'header', 'picture', 'list', 'quote', 'code', 'twitter', 'instagram', 'smile'],\n\t holderId: 'codex-editor',\n\t\n\t // Type of block showing on empty editor\n\t initialBlockPlugin: 'paragraph'\n\t };\n\t\n\t /**\r\n\t * public\r\n\t *\r\n\t * Static nodes\r\n\t */\n\t editor.nodes = {\n\t holder: null,\n\t wrapper: null,\n\t toolbar: null,\n\t inlineToolbar: {\n\t wrapper: null,\n\t buttons: null,\n\t actions: null\n\t },\n\t toolbox: null,\n\t notifications: null,\n\t plusButton: null,\n\t showSettingsButton: null,\n\t showTrashButton: null,\n\t blockSettings: null,\n\t pluginSettings: null,\n\t defaultSettings: null,\n\t toolbarButtons: {}, // { type : DomEl, ... }\n\t redactor: null\n\t };\n\t\n\t /**\r\n\t * @public\r\n\t *\r\n\t * Output state\r\n\t */\n\t editor.state = {\n\t jsonOutput: [],\n\t blocks: [],\n\t inputs: []\n\t };\n\t\n\t /**\r\n\t * @public\r\n\t * Editor plugins\r\n\t */\n\t editor.tools = {};\n\t\n\t /**\r\n\t * Initialization\r\n\t * @uses Promise cEditor.core.prepare\r\n\t * @param {Object} userSettings\r\n\t * @param {Array} userSettings.tools list of plugins\r\n\t * @param {String} userSettings.holderId Element's id to append editor\r\n\t *\r\n\t * Load user defined tools\r\n\t * Tools must contain this important objects :\r\n\t * @param {String} type - this is a type of plugin. It can be used as plugin name\r\n\t * @param {String} iconClassname - this a icon in toolbar\r\n\t * @param {Object} make - what should plugin do, when it is clicked\r\n\t * @param {Object} appendCallback - callback after clicking\r\n\t * @param {Element} settings - what settings does it have\r\n\t * @param {Object} render - plugin get JSON, and should return HTML\r\n\t * @param {Object} save - plugin gets HTML content, returns JSON\r\n\t * @param {Boolean} displayInToolbox - will be displayed in toolbox. Default value is TRUE\r\n\t * @param {Boolean} enableLineBreaks - inserts new block or break lines. Default value is FALSE\r\n\t *\r\n\t * @example\r\n\t * - type : 'header',\r\n\t * - iconClassname : 'ce-icon-header',\r\n\t * - make : headerTool.make,\r\n\t * - appendCallback : headerTool.appendCallback,\r\n\t * - settings : headerTool.makeSettings(),\r\n\t * - render : headerTool.render,\r\n\t * - save : headerTool.save,\r\n\t * - displayInToolbox : true,\r\n\t * - enableLineBreaks : false\r\n\t */\n\t editor.start = function (userSettings) {\n\t\n\t init();\n\t\n\t editor.core.prepare(userSettings)\n\t\n\t // If all ok, make UI, bind events and parse initial-content\n\t .then(editor.ui.prepare).then(editor.tools.prepare).then(editor.sanitizer.prepare).then(editor.paste.prepare).then(editor.transport.prepare).then(editor.renderer.makeBlocksFromData).then(editor.ui.saveInputs).catch(function (error) {\n\t\n\t editor.core.log('Initialization failed with error: %o', 'warn', error);\n\t });\n\t };\n\t\n\t return editor;\n\t}({});\n\n/***/ },\n/* 1 */\n/***/ function(module, exports) {\n\n\t'use strict';\n\t\n\tvar _typeof = typeof Symbol === \"function\" && typeof Symbol.iterator === \"symbol\" ? function (obj) { return typeof obj; } : function (obj) { return obj && typeof Symbol === \"function\" && obj.constructor === Symbol && obj !== Symbol.prototype ? \"symbol\" : typeof obj; };\n\t\n\t/**\r\n\t * Codex Editor Core\r\n\t *\r\n\t * @author Codex Team\r\n\t * @version 1.1.3\r\n\t */\n\t\n\tmodule.exports = function (core) {\n\t\n\t var editor = codex.editor;\n\t\n\t /**\r\n\t * @public\r\n\t *\r\n\t * Editor preparing method\r\n\t * @return Promise\r\n\t */\n\t core.prepare = function (userSettings) {\n\t\n\t return new Promise(function (resolve, reject) {\n\t\n\t if (userSettings) {\n\t\n\t editor.settings.tools = userSettings.tools || editor.settings.tools;\n\t }\n\t\n\t if (userSettings.data) {\n\t\n\t editor.state.blocks = userSettings.data;\n\t }\n\t\n\t if (userSettings.initialBlockPlugin) {\n\t\n\t editor.settings.initialBlockPlugin = userSettings.initialBlockPlugin;\n\t }\n\t\n\t if (userSettings.sanitizer) {\n\t\n\t editor.settings.sanitizer = userSettings.sanitizer;\n\t }\n\t\n\t editor.hideToolbar = userSettings.hideToolbar;\n\t\n\t editor.nodes.holder = document.getElementById(userSettings.holderId || editor.settings.holderId);\n\t\n\t if (_typeof(editor.nodes.holder) === undefined || editor.nodes.holder === null) {\n\t\n\t reject(Error(\"Holder wasn't found by ID: #\" + userSettings.holderId));\n\t } else {\n\t\n\t resolve();\n\t }\n\t });\n\t };\n\t\n\t /**\r\n\t * Logging method\r\n\t * @param type = ['log', 'info', 'warn']\r\n\t */\n\t core.log = function (msg, type, arg) {\n\t\n\t type = type || 'log';\n\t\n\t if (!arg) {\n\t\n\t arg = msg || 'undefined';\n\t msg = '[codex-editor]: %o';\n\t } else {\n\t\n\t msg = '[codex-editor]: ' + msg;\n\t }\n\t\n\t try {\n\t\n\t if ('console' in window && window.console[type]) {\n\t\n\t if (arg) window.console[type](msg, arg);else window.console[type](msg);\n\t }\n\t } catch (e) {}\n\t };\n\t\n\t /**\r\n\t * @protected\r\n\t *\r\n\t * Helper for insert one element after another\r\n\t */\n\t core.insertAfter = function (target, element) {\n\t\n\t target.parentNode.insertBefore(element, target.nextSibling);\n\t };\n\t\n\t /**\r\n\t * @const\r\n\t *\r\n\t * Readable DOM-node types map\r\n\t */\n\t core.nodeTypes = {\n\t TAG: 1,\n\t TEXT: 3,\n\t COMMENT: 8\n\t };\n\t\n\t /**\r\n\t * @const\r\n\t * Readable keys map\r\n\t */\n\t core.keys = { BACKSPACE: 8, TAB: 9, ENTER: 13, SHIFT: 16, CTRL: 17, ALT: 18, ESC: 27, SPACE: 32, LEFT: 37, UP: 38, DOWN: 40, RIGHT: 39, DELETE: 46, META: 91 };\n\t\n\t /**\r\n\t * @protected\r\n\t *\r\n\t * Check object for DOM node\r\n\t */\n\t core.isDomNode = function (el) {\n\t\n\t return el && (typeof el === 'undefined' ? 'undefined' : _typeof(el)) === 'object' && el.nodeType && el.nodeType == this.nodeTypes.TAG;\n\t };\n\t\n\t /**\r\n\t * Checks passed object for emptiness\r\n\t * @require ES5 - Object.keys\r\n\t * @param {object}\r\n\t */\n\t core.isEmpty = function (obj) {\n\t\n\t return Object.keys(obj).length === 0;\n\t };\n\t\n\t /**\r\n\t * Native Ajax\r\n\t * @param {String} settings.url - request URL\r\n\t * @param {function} settings.beforeSend - returned value will be passed as context to the Success, Error and Progress callbacks\r\n\t * @param {function} settings.success\r\n\t * @param {function} settings.progress\r\n\t */\n\t core.ajax = function (settings) {\n\t\n\t if (!settings || !settings.url) {\n\t\n\t return;\n\t }\n\t\n\t var XMLHTTP = window.XMLHttpRequest ? new XMLHttpRequest() : new ActiveXObject('Microsoft.XMLHTTP'),\n\t encodedString,\n\t isFormData,\n\t prop;\n\t\n\t settings.async = true;\n\t settings.type = settings.type || 'GET';\n\t settings.data = settings.data || '';\n\t settings['content-type'] = settings['content-type'] || 'application/json; charset=utf-8';\n\t\n\t if (settings.type == 'GET' && settings.data) {\n\t\n\t settings.url = /\\?/.test(settings.url) ? settings.url + '&' + settings.data : settings.url + '?' + settings.data;\n\t } else {\n\t\n\t encodedString = '';\n\t for (prop in settings.data) {\n\t\n\t encodedString += prop + '=' + encodeURIComponent(settings.data[prop]) + '&';\n\t }\n\t }\n\t\n\t if (settings.withCredentials) {\n\t\n\t XMLHTTP.withCredentials = true;\n\t }\n\t\n\t /**\r\n\t * Value returned in beforeSend funtion will be passed as context to the other response callbacks\r\n\t * If beforeSend returns false, AJAX will be blocked\r\n\t */\n\t var responseContext = void 0,\n\t beforeSendResult = void 0;\n\t\n\t if (typeof settings.beforeSend === 'function') {\n\t\n\t beforeSendResult = settings.beforeSend.call();\n\t\n\t if (beforeSendResult === false) {\n\t\n\t return;\n\t }\n\t }\n\t\n\t XMLHTTP.open(settings.type, settings.url, settings.async);\n\t\n\t /**\r\n\t * If we send FormData, we need no content-type header\r\n\t */\n\t isFormData = isFormData_(settings.data);\n\t\n\t if (!isFormData) {\n\t\n\t if (settings.type !== 'POST') {\n\t\n\t XMLHTTP.setRequestHeader('Content-type', settings['content-type']);\n\t } else {\n\t\n\t XMLHTTP.setRequestHeader('Content-type', 'application/x-www-form-urlencoded');\n\t }\n\t }\n\t\n\t XMLHTTP.setRequestHeader('X-Requested-With', 'XMLHttpRequest');\n\t\n\t responseContext = beforeSendResult || XMLHTTP;\n\t\n\t if (typeof settings.progress === 'function') {\n\t\n\t XMLHTTP.upload.onprogress = settings.progress.bind(responseContext);\n\t }\n\t\n\t XMLHTTP.onreadystatechange = function () {\n\t\n\t if (XMLHTTP.readyState === 4) {\n\t\n\t if (XMLHTTP.status === 200) {\n\t\n\t if (typeof settings.success === 'function') {\n\t\n\t settings.success.call(responseContext, XMLHTTP.responseText);\n\t }\n\t } else {\n\t\n\t if (typeof settings.error === 'function') {\n\t\n\t settings.error.call(responseContext, XMLHTTP.responseText, XMLHTTP.status);\n\t }\n\t }\n\t }\n\t };\n\t\n\t if (isFormData) {\n\t\n\t // Sending FormData\n\t XMLHTTP.send(settings.data);\n\t } else {\n\t\n\t // POST requests\n\t XMLHTTP.send(encodedString);\n\t }\n\t\n\t return XMLHTTP;\n\t };\n\t\n\t /**\r\n\t * Appends script to head of document\r\n\t * @return Promise\r\n\t */\n\t core.importScript = function (scriptPath, instanceName) {\n\t\n\t return new Promise(function (resolve, reject) {\n\t\n\t var script = void 0;\n\t\n\t /** Script is already loaded */\n\t if (!instanceName) {\n\t\n\t reject('Instance name is missed');\n\t } else if (document.getElementById(editor.scriptPrefix + instanceName)) {\n\t\n\t resolve(scriptPath);\n\t }\n\t\n\t script = document.createElement('SCRIPT');\n\t script.async = true;\n\t script.defer = true;\n\t script.id = editor.scriptPrefix + instanceName;\n\t\n\t script.onload = function () {\n\t\n\t resolve(scriptPath);\n\t };\n\t\n\t script.onerror = function () {\n\t\n\t reject(scriptPath);\n\t };\n\t\n\t script.src = scriptPath;\n\t document.head.appendChild(script);\n\t });\n\t };\n\t\n\t /**\r\n\t * Function for checking is it FormData object to send.\r\n\t * @param {Object} object to check\r\n\t * @return boolean\r\n\t */\n\t var isFormData_ = function isFormData_(object) {\n\t\n\t return object instanceof FormData;\n\t };\n\t\n\t /**\r\n\t * Check block\r\n\t * @param target\r\n\t * @description Checks target is it native input\r\n\t */\n\t core.isNativeInput = function (target) {\n\t\n\t var nativeInputAreas = ['INPUT', 'TEXTAREA'];\n\t\n\t return nativeInputAreas.indexOf(target.tagName) != -1;\n\t };\n\t\n\t return core;\n\t}({});\n\n/***/ },\n/* 2 */\n/***/ function(module, exports) {\n\n\t'use strict';\n\t\n\t/**\r\n\t* Module working with plugins\r\n\t*/\n\tmodule.exports = function () {\n\t\n\t var editor = codex.editor;\n\t\n\t /**\r\n\t * Initialize plugins before using\r\n\t * Ex. Load scripts or call some internal methods\r\n\t * @return Promise\r\n\t */\n\t function prepare() {\n\t\n\t return new Promise(function (resolve_, reject_) {\n\t\n\t Promise.resolve()\n\t\n\t /**\r\n\t * Compose a sequence of plugins that requires preparation\r\n\t */\n\t .then(function () {\n\t\n\t var pluginsRequiresPreparation = [],\n\t allPlugins = editor.tools;\n\t\n\t for (var pluginName in allPlugins) {\n\t\n\t var plugin = allPlugins[pluginName];\n\t\n\t if (plugin.prepare && typeof plugin.prepare != 'function' || !plugin.prepare) {\n\t\n\t continue;\n\t }\n\t\n\t pluginsRequiresPreparation.push(plugin);\n\t }\n\t\n\t /**\r\n\t * If no one passed plugins requires preparation, finish prepare() and go ahead\r\n\t */\n\t if (!pluginsRequiresPreparation.length) {\n\t\n\t resolve_();\n\t }\n\t\n\t return pluginsRequiresPreparation;\n\t })\n\t\n\t /** Wait plugins while they prepares */\n\t .then(waitAllPluginsPreparation_).then(function () {\n\t\n\t editor.core.log('Plugins loaded', 'info');\n\t resolve_();\n\t }).catch(function (error) {\n\t\n\t reject_(error);\n\t });\n\t });\n\t }\n\t\n\t /**\r\n\t * @param {array} plugins - list of tools that requires preparation\r\n\t * @return {Promise} resolved while all plugins will be ready or failed\r\n\t */\n\t function waitAllPluginsPreparation_(plugins) {\n\t\n\t /**\r\n\t * @calls allPluginsProcessed__ when all plugins prepared or failed\r\n\t */\n\t return new Promise(function (allPluginsProcessed__) {\n\t\n\t /**\r\n\t * pluck each element from queue\r\n\t * First, send resolved Promise as previous value\r\n\t * Each plugins \"prepare\" method returns a Promise, that's why\r\n\t * reduce current element will not be able to continue while can't get\r\n\t * a resolved Promise\r\n\t *\r\n\t * If last plugin is \"prepared\" then go to the next stage of initialization\r\n\t */\n\t plugins.reduce(function (previousValue, plugin, iteration) {\n\t\n\t return previousValue.then(function () {\n\t\n\t /**\r\n\t * Wait till plugins prepared\r\n\t * @calls pluginIsReady__ when plugin is ready or failed\r\n\t */\n\t return new Promise(function (pluginIsReady__) {\n\t\n\t callPluginsPrepareMethod_(plugin).then(pluginIsReady__).then(function () {\n\t\n\t plugin.available = true;\n\t }).catch(function (error) {\n\t\n\t editor.core.log('Plugin \\xAB' + plugin.type + '\\xBB was not loaded. Preparation failed because %o', 'warn', error);\n\t plugin.available = false;\n\t plugin.loadingMessage = error;\n\t\n\t /** Go ahead even some plugin has problems */\n\t pluginIsReady__();\n\t }).then(function () {\n\t\n\t /** If last plugin has problems then just ignore and continue */\n\t if (iteration == plugins.length - 1) {\n\t\n\t allPluginsProcessed__();\n\t }\n\t });\n\t });\n\t });\n\t }, Promise.resolve());\n\t });\n\t }\n\t\n\t var callPluginsPrepareMethod_ = function callPluginsPrepareMethod_(plugin) {\n\t\n\t return plugin.prepare(plugin.config || {});\n\t };\n\t\n\t return {\n\t prepare: prepare\n\t };\n\t}();\n\n/***/ },\n/* 3 */\n/***/ function(module, exports) {\n\n\t'use strict';\n\t\n\t/**\r\n\t * Codex Editor UI module\r\n\t *\r\n\t * @author Codex Team\r\n\t * @version 1.2.0\r\n\t */\n\t\n\tmodule.exports = function (ui) {\n\t\n\t var editor = codex.editor;\n\t\n\t /**\r\n\t * Basic editor classnames\r\n\t */\n\t ui.className = {\n\t\n\t /**\r\n\t * @const {string} BLOCK_CLASSNAME - redactor blocks name\r\n\t */\n\t BLOCK_CLASSNAME: 'ce-block',\n\t\n\t /**\r\n\t * @const {String} wrapper for plugins content\r\n\t */\n\t BLOCK_CONTENT: 'ce-block__content',\n\t\n\t /**\r\n\t * @const {String} BLOCK_STRETCHED - makes block stretched\r\n\t */\n\t BLOCK_STRETCHED: 'ce-block--stretched',\n\t\n\t /**\r\n\t * @const {String} BLOCK_HIGHLIGHTED - adds background\r\n\t */\n\t BLOCK_HIGHLIGHTED: 'ce-block--focused',\n\t\n\t /**\r\n\t * @const {String} - for all default settings\r\n\t */\n\t SETTINGS_ITEM: 'ce-settings__item'\n\t\n\t };\n\t\n\t /**\r\n\t * @protected\r\n\t *\r\n\t * Making main interface\r\n\t */\n\t ui.prepare = function () {\n\t\n\t return new Promise(function (resolve) {\n\t\n\t var wrapper = editor.draw.wrapper(),\n\t redactor = editor.draw.redactor(),\n\t toolbar = makeToolBar_();\n\t\n\t wrapper.appendChild(toolbar);\n\t wrapper.appendChild(redactor);\n\t\n\t /** Save created ui-elements to static nodes state */\n\t editor.nodes.wrapper = wrapper;\n\t editor.nodes.redactor = redactor;\n\t\n\t /** Append editor wrapper with redactor zone into holder */\n\t editor.nodes.holder.appendChild(wrapper);\n\t\n\t resolve();\n\t })\n\t\n\t /** Add toolbox tools */\n\t .then(addTools_)\n\t\n\t /** Make container for inline toolbar */\n\t .then(makeInlineToolbar_)\n\t\n\t /** Add inline toolbar tools */\n\t .then(addInlineToolbarTools_)\n\t\n\t /** Draw wrapper for notifications */\n\t .then(makeNotificationHolder_)\n\t\n\t /** Add eventlisteners to redactor elements */\n\t .then(bindEvents_).catch(function () {\n\t\n\t editor.core.log(\"Can't draw editor interface\");\n\t });\n\t };\n\t\n\t /**\r\n\t * @private\r\n\t * Draws inline toolbar zone\r\n\t */\n\t var makeInlineToolbar_ = function makeInlineToolbar_() {\n\t\n\t var container = editor.draw.inlineToolbar();\n\t\n\t /** Append to redactor new inline block */\n\t editor.nodes.inlineToolbar.wrapper = container;\n\t\n\t /** Draw toolbar buttons */\n\t editor.nodes.inlineToolbar.buttons = editor.draw.inlineToolbarButtons();\n\t\n\t /** Buttons action or settings */\n\t editor.nodes.inlineToolbar.actions = editor.draw.inlineToolbarActions();\n\t\n\t /** Append to inline toolbar buttons as part of it */\n\t editor.nodes.inlineToolbar.wrapper.appendChild(editor.nodes.inlineToolbar.buttons);\n\t editor.nodes.inlineToolbar.wrapper.appendChild(editor.nodes.inlineToolbar.actions);\n\t\n\t editor.nodes.wrapper.appendChild(editor.nodes.inlineToolbar.wrapper);\n\t };\n\t\n\t var makeToolBar_ = function makeToolBar_() {\n\t\n\t var toolbar = editor.draw.toolbar(),\n\t blockButtons = makeToolbarSettings_(),\n\t toolbarContent = makeToolbarContent_();\n\t\n\t /** Appending first-level block buttons */\n\t toolbar.appendChild(blockButtons);\n\t\n\t /** Append toolbarContent to toolbar */\n\t toolbar.appendChild(toolbarContent);\n\t\n\t /** Make toolbar global */\n\t editor.nodes.toolbar = toolbar;\n\t\n\t return toolbar;\n\t };\n\t\n\t var makeToolbarContent_ = function makeToolbarContent_() {\n\t\n\t var toolbarContent = editor.draw.toolbarContent(),\n\t toolbox = editor.draw.toolbox(),\n\t plusButton = editor.draw.plusButton();\n\t\n\t /** Append plus button */\n\t toolbarContent.appendChild(plusButton);\n\t\n\t /** Appending toolbar tools */\n\t toolbarContent.appendChild(toolbox);\n\t\n\t /** Make Toolbox and plusButton global */\n\t editor.nodes.toolbox = toolbox;\n\t editor.nodes.plusButton = plusButton;\n\t\n\t return toolbarContent;\n\t };\n\t\n\t var makeToolbarSettings_ = function makeToolbarSettings_() {\n\t\n\t var blockSettings = editor.draw.blockSettings(),\n\t blockButtons = editor.draw.blockButtons(),\n\t defaultSettings = editor.draw.defaultSettings(),\n\t showSettingsButton = editor.draw.settingsButton(),\n\t showTrashButton = editor.toolbar.settings.makeRemoveBlockButton(),\n\t pluginSettings = editor.draw.pluginsSettings();\n\t\n\t /** Add default and plugins settings */\n\t blockSettings.appendChild(pluginSettings);\n\t blockSettings.appendChild(defaultSettings);\n\t\n\t /**\r\n\t * Make blocks buttons\r\n\t * This block contains settings button and remove block button\r\n\t */\n\t blockButtons.appendChild(showSettingsButton);\n\t blockButtons.appendChild(showTrashButton);\n\t blockButtons.appendChild(blockSettings);\n\t\n\t /** Make BlockSettings, PluginSettings, DefaultSettings global */\n\t editor.nodes.blockSettings = blockSettings;\n\t editor.nodes.pluginSettings = pluginSettings;\n\t editor.nodes.defaultSettings = defaultSettings;\n\t editor.nodes.showSettingsButton = showSettingsButton;\n\t editor.nodes.showTrashButton = showTrashButton;\n\t\n\t return blockButtons;\n\t };\n\t\n\t /** Draw notifications holder */\n\t var makeNotificationHolder_ = function makeNotificationHolder_() {\n\t\n\t /** Append block with notifications to the document */\n\t editor.nodes.notifications = editor.notifications.createHolder();\n\t };\n\t\n\t /**\r\n\t * @private\r\n\t * Append tools passed in editor.tools\r\n\t */\n\t var addTools_ = function addTools_() {\n\t\n\t var tool, toolName, toolButton;\n\t\n\t for (toolName in editor.settings.tools) {\n\t\n\t tool = editor.settings.tools[toolName];\n\t\n\t editor.tools[toolName] = tool;\n\t\n\t if (!tool.iconClassname && tool.displayInToolbox) {\n\t\n\t editor.core.log('Toolbar icon classname missed. Tool %o skipped', 'warn', toolName);\n\t continue;\n\t }\n\t\n\t if (typeof tool.render != 'function') {\n\t\n\t editor.core.log('render method missed. Tool %o skipped', 'warn', toolName);\n\t continue;\n\t }\n\t\n\t if (!tool.displayInToolbox) {\n\t\n\t continue;\n\t } else {\n\t\n\t /** if tools is for toolbox */\n\t toolButton = editor.draw.toolbarButton(toolName, tool.iconClassname);\n\t\n\t editor.nodes.toolbox.appendChild(toolButton);\n\t\n\t editor.nodes.toolbarButtons[toolName] = toolButton;\n\t }\n\t }\n\t };\n\t\n\t var addInlineToolbarTools_ = function addInlineToolbarTools_() {\n\t\n\t var tools = {\n\t\n\t bold: {\n\t icon: 'ce-icon-bold',\n\t command: 'bold'\n\t },\n\t\n\t italic: {\n\t icon: 'ce-icon-italic',\n\t command: 'italic'\n\t },\n\t\n\t underline: {\n\t icon: 'ce-icon-underline',\n\t command: 'underline'\n\t },\n\t\n\t link: {\n\t icon: 'ce-icon-link',\n\t command: 'createLink'\n\t }\n\t };\n\t\n\t var toolButton, tool;\n\t\n\t for (var name in tools) {\n\t\n\t tool = tools[name];\n\t\n\t toolButton = editor.draw.toolbarButtonInline(name, tool.icon);\n\t\n\t editor.nodes.inlineToolbar.buttons.appendChild(toolButton);\n\t /**\r\n\t * Add callbacks to this buttons\r\n\t */\n\t editor.ui.setInlineToolbarButtonBehaviour(toolButton, tool.command);\n\t }\n\t };\n\t\n\t /**\r\n\t * @private\r\n\t * Bind editor UI events\r\n\t */\n\t var bindEvents_ = function bindEvents_() {\n\t\n\t editor.core.log('ui.bindEvents fired', 'info');\n\t\n\t // window.addEventListener('error', function (errorMsg, url, lineNumber) {\n\t // editor.notifications.errorThrown(errorMsg, event);\n\t // }, false );\n\t\n\t /** All keydowns on Document */\n\t editor.listeners.add(document, 'keydown', editor.callback.globalKeydown, false);\n\t\n\t /** All keydowns on Redactor zone */\n\t editor.listeners.add(editor.nodes.redactor, 'keydown', editor.callback.redactorKeyDown, false);\n\t\n\t /** All keydowns on Document */\n\t editor.listeners.add(document, 'keyup', editor.callback.globalKeyup, false);\n\t\n\t /**\r\n\t * Mouse click to radactor\r\n\t */\n\t editor.listeners.add(editor.nodes.redactor, 'click', editor.callback.redactorClicked, false);\n\t\n\t /**\r\n\t * Clicks to the Plus button\r\n\t */\n\t editor.listeners.add(editor.nodes.plusButton, 'click', editor.callback.plusButtonClicked, false);\n\t\n\t /**\r\n\t * Clicks to SETTINGS button in toolbar\r\n\t */\n\t editor.listeners.add(editor.nodes.showSettingsButton, 'click', editor.callback.showSettingsButtonClicked, false);\n\t\n\t /** Bind click listeners on toolbar buttons */\n\t for (var button in editor.nodes.toolbarButtons) {\n\t\n\t editor.listeners.add(editor.nodes.toolbarButtons[button], 'click', editor.callback.toolbarButtonClicked, false);\n\t }\n\t };\n\t\n\t ui.addBlockHandlers = function (block) {\n\t\n\t if (!block) return;\n\t\n\t /**\r\n\t * Block keydowns\r\n\t */\n\t editor.listeners.add(block, 'keydown', editor.callback.blockKeydown, false);\n\t\n\t /**\r\n\t * Pasting content from another source\r\n\t * We have two type of sanitization\r\n\t * First - uses deep-first search algorithm to get sub nodes,\r\n\t * sanitizes whole Block_content and replaces cleared nodes\r\n\t * This method is deprecated\r\n\t * Method is used in editor.callback.blockPaste(event)\r\n\t *\r\n\t * Secont - uses Mutation observer.\r\n\t * Observer \"observe\" DOM changes and send changings to callback.\r\n\t * Callback gets changed node, not whole Block_content.\r\n\t * Inserted or changed node, which we've gotten have been cleared and replaced with diry node\r\n\t *\r\n\t * Method is used in editor.callback.blockPasteViaSanitize(event)\r\n\t *\r\n\t * @uses html-janitor\r\n\t * @example editor.callback.blockPasteViaSanitize(event), the second method.\r\n\t *\r\n\t */\n\t editor.listeners.add(block, 'paste', editor.paste.blockPasteCallback, false);\n\t\n\t editor.listeners.add(block, 'mouseup', editor.toolbar.inline.show, false);\n\t };\n\t\n\t /** getting all contenteditable elements */\n\t ui.saveInputs = function () {\n\t\n\t var redactor = editor.nodes.redactor;\n\t\n\t editor.state.inputs = [];\n\t\n\t /** Save all inputs in global variable state */\n\t var inputs = redactor.querySelectorAll('[contenteditable], input, textarea');\n\t\n\t Array.prototype.map.call(inputs, function (current) {\n\t\n\t if (!current.type || current.type == 'text' || current.type == 'textarea') {\n\t\n\t editor.state.inputs.push(current);\n\t }\n\t });\n\t };\n\t\n\t /**\r\n\t * Adds first initial block on empty redactor\r\n\t */\n\t ui.addInitialBlock = function () {\n\t\n\t var initialBlockType = editor.settings.initialBlockPlugin,\n\t initialBlock;\n\t\n\t if (!editor.tools[initialBlockType]) {\n\t\n\t editor.core.log('Plugin %o was not implemented and can\\'t be used as initial block', 'warn', initialBlockType);\n\t return;\n\t }\n\t\n\t initialBlock = editor.tools[initialBlockType].render();\n\t\n\t initialBlock.setAttribute('data-placeholder', 'Расскажите свою историю...');\n\t\n\t editor.content.insertBlock({\n\t type: initialBlockType,\n\t block: initialBlock\n\t });\n\t\n\t editor.content.workingNodeChanged(initialBlock);\n\t };\n\t\n\t ui.setInlineToolbarButtonBehaviour = function (button, type) {\n\t\n\t editor.listeners.add(button, 'mousedown', function (event) {\n\t\n\t editor.toolbar.inline.toolClicked(event, type);\n\t }, false);\n\t };\n\t\n\t return ui;\n\t}({});\n\n/***/ },\n/* 4 */\n/***/ function(module, exports) {\n\n\t'use strict';\n\t\n\t/**\r\n\t *\r\n\t * Codex.Editor Transport Module\r\n\t *\r\n\t * @copyright 2017 Codex-Team\r\n\t * @version 1.2.0\r\n\t */\n\t\n\tmodule.exports = function (transport) {\n\t\n\t var editor = codex.editor;\n\t\n\t /**\r\n\t * @private {Object} current XmlHttpRequest instance\r\n\t */\n\t var currentRequest = null;\n\t\n\t /**\r\n\t * @type {null} | {DOMElement} input - keeps input element in memory\r\n\t */\n\t transport.input = null;\n\t\n\t /**\r\n\t * @property {Object} arguments - keep plugin settings and defined callbacks\r\n\t */\n\t transport.arguments = null;\n\t\n\t /**\r\n\t * Prepares input element where will be files\r\n\t */\n\t transport.prepare = function () {\n\t\n\t var input = editor.draw.node('INPUT', '', { type: 'file' });\n\t\n\t editor.listeners.add(input, 'change', editor.transport.fileSelected);\n\t editor.transport.input = input;\n\t };\n\t\n\t /** Clear input when files is uploaded */\n\t transport.clearInput = function () {\n\t\n\t /** Remove old input */\n\t transport.input = null;\n\t\n\t /** Prepare new one */\n\t transport.prepare();\n\t };\n\t\n\t /**\r\n\t * Callback for file selection\r\n\t * @param {Event} event\r\n\t */\n\t transport.fileSelected = function () {\n\t\n\t var input = this,\n\t i,\n\t files = input.files,\n\t formData = new FormData();\n\t\n\t if (editor.transport.arguments.multiple === true) {\n\t\n\t for (i = 0; i < files.length; i++) {\n\t\n\t formData.append('files[]', files[i], files[i].name);\n\t }\n\t } else {\n\t\n\t formData.append('files', files[0], files[0].name);\n\t }\n\t\n\t currentRequest = editor.core.ajax({\n\t type: 'POST',\n\t data: formData,\n\t url: editor.transport.arguments.url,\n\t beforeSend: editor.transport.arguments.beforeSend,\n\t success: editor.transport.arguments.success,\n\t error: editor.transport.arguments.error,\n\t progress: editor.transport.arguments.progress\n\t });\n\t\n\t /** Clear input */\n\t transport.clearInput();\n\t };\n\t\n\t /**\r\n\t * Use plugin callbacks\r\n\t * @protected\r\n\t *\r\n\t * @param {Object} args - can have :\r\n\t * @param {String} args.url - fetch URL\r\n\t * @param {Function} args.beforeSend - function calls before sending ajax\r\n\t * @param {Function} args.success - success callback\r\n\t * @param {Function} args.error - on error handler\r\n\t * @param {Function} args.progress - xhr onprogress handler\r\n\t * @param {Boolean} args.multiple - allow select several files\r\n\t * @param {String} args.accept - adds accept attribute\r\n\t */\n\t transport.selectAndUpload = function (args) {\n\t\n\t transport.arguments = args;\n\t\n\t if (args.multiple === true) {\n\t\n\t transport.input.setAttribute('multiple', 'multiple');\n\t }\n\t\n\t if (args.accept) {\n\t\n\t transport.input.setAttribute('accept', args.accept);\n\t }\n\t\n\t transport.input.click();\n\t };\n\t\n\t transport.abort = function () {\n\t\n\t currentRequest.abort();\n\t\n\t currentRequest = null;\n\t };\n\t\n\t return transport;\n\t}({});\n\n/***/ },\n/* 5 */\n/***/ function(module, exports) {\n\n\t'use strict';\n\t\n\t/**\r\n\t * Codex Editor Renderer Module\r\n\t *\r\n\t * @author Codex Team\r\n\t * @version 1.0\r\n\t */\n\t\n\tmodule.exports = function (renderer) {\n\t\n\t var editor = codex.editor;\n\t\n\t /**\r\n\t * Asyncronously parses input JSON to redactor blocks\r\n\t */\n\t renderer.makeBlocksFromData = function () {\n\t\n\t /**\r\n\t * If redactor is empty, add first paragraph to start writing\r\n\t */\n\t if (editor.core.isEmpty(editor.state.blocks) || !editor.state.blocks.items.length) {\n\t\n\t editor.ui.addInitialBlock();\n\t return;\n\t }\n\t\n\t Promise.resolve()\n\t\n\t /** First, get JSON from state */\n\t .then(function () {\n\t\n\t return editor.state.blocks;\n\t })\n\t\n\t /** Then, start to iterate they */\n\t .then(editor.renderer.appendBlocks)\n\t\n\t /** Write log if something goes wrong */\n\t .catch(function (error) {\n\t\n\t editor.core.log('Error while parsing JSON: %o', 'error', error);\n\t });\n\t };\n\t\n\t /**\r\n\t * Parses JSON to blocks\r\n\t * @param {object} data\r\n\t * @return Primise -> nodeList\r\n\t */\n\t renderer.appendBlocks = function (data) {\n\t\n\t var blocks = data.items;\n\t\n\t /**\r\n\t * Sequence of one-by-one blocks appending\r\n\t * Uses to save blocks order after async-handler\r\n\t */\n\t var nodeSequence = Promise.resolve();\n\t\n\t for (var index = 0; index < blocks.length; index++) {\n\t\n\t /** Add node to sequence at specified index */\n\t editor.renderer.appendNodeAtIndex(nodeSequence, blocks, index);\n\t }\n\t };\n\t\n\t /**\r\n\t * Append node at specified index\r\n\t */\n\t renderer.appendNodeAtIndex = function (nodeSequence, blocks, index) {\n\t\n\t /** We need to append node to sequence */\n\t nodeSequence\n\t\n\t /** first, get node async-aware */\n\t .then(function () {\n\t\n\t return editor.renderer.getNodeAsync(blocks, index);\n\t })\n\t\n\t /**\r\n\t * second, compose editor-block from JSON object\r\n\t */\n\t .then(editor.renderer.createBlockFromData)\n\t\n\t /**\r\n\t * now insert block to redactor\r\n\t */\n\t .then(function (blockData) {\n\t\n\t /**\r\n\t * blockData has 'block', 'type' and 'stretched' information\r\n\t */\n\t editor.content.insertBlock(blockData);\n\t\n\t /** Pass created block to next step */\n\t return blockData.block;\n\t })\n\t\n\t /** Log if something wrong with node */\n\t .catch(function (error) {\n\t\n\t editor.core.log('Node skipped while parsing because %o', 'error', error);\n\t });\n\t };\n\t\n\t /**\r\n\t * Asynchronously returns block data from blocksList by index\r\n\t * @return Promise to node\r\n\t */\n\t renderer.getNodeAsync = function (blocksList, index) {\n\t\n\t return Promise.resolve().then(function () {\n\t\n\t return {\n\t tool: blocksList[index],\n\t position: index\n\t };\n\t });\n\t };\n\t\n\t /**\r\n\t * Creates editor block by JSON-data\r\n\t *\r\n\t * @uses render method of each plugin\r\n\t *\r\n\t * @param {Object} toolData.tool\r\n\t * { header : {\r\n\t * text: '',\r\n\t * type: 'H3', ...\r\n\t * }\r\n\t * }\r\n\t * @param {Number} toolData.position - index in input-blocks array\r\n\t * @return {Object} with type and Element\r\n\t */\n\t renderer.createBlockFromData = function (toolData) {\n\t\n\t /** New parser */\n\t var block,\n\t tool = toolData.tool,\n\t pluginName = tool.type;\n\t\n\t /** Get first key of object that stores plugin name */\n\t // for (var pluginName in blockData) break;\n\t\n\t /** Check for plugin existance */\n\t if (!editor.tools[pluginName]) {\n\t\n\t throw Error('Plugin \\xAB' + pluginName + '\\xBB not found');\n\t }\n\t\n\t /** Check for plugin having render method */\n\t if (typeof editor.tools[pluginName].render != 'function') {\n\t\n\t throw Error('Plugin \\xAB' + pluginName + '\\xBB must have \\xABrender\\xBB method');\n\t }\n\t\n\t if (editor.tools[pluginName].available === false) {\n\t\n\t block = editor.draw.unavailableBlock();\n\t\n\t block.innerHTML = editor.tools[pluginName].loadingMessage;\n\t\n\t /**\r\n\t * Saver will extract data from initial block data by position in array\r\n\t */\n\t block.dataset.inputPosition = toolData.position;\n\t } else {\n\t\n\t /** New Parser */\n\t block = editor.tools[pluginName].render(tool.data);\n\t }\n\t\n\t /** is first-level block stretched */\n\t var stretched = editor.tools[pluginName].isStretched || false;\n\t\n\t /** Retrun type and block */\n\t return {\n\t type: pluginName,\n\t block: block,\n\t stretched: stretched\n\t };\n\t };\n\t\n\t return renderer;\n\t}({});\n\n/***/ },\n/* 6 */\n/***/ function(module, exports) {\n\n\t'use strict';\n\t\n\t/**\r\n\t * Codex Editor Saver\r\n\t *\r\n\t * @author Codex Team\r\n\t * @version 1.0.2\r\n\t */\n\t\n\tmodule.exports = function (saver) {\n\t\n\t var editor = codex.editor;\n\t\n\t /**\r\n\t * Saves blocks\r\n\t * @private\r\n\t */\n\t saver.saveBlocks = function () {\n\t\n\t /** Save html content of redactor to memory */\n\t editor.state.html = editor.nodes.redactor.innerHTML;\n\t\n\t /** Empty jsonOutput state */\n\t editor.state.jsonOutput = [];\n\t\n\t Promise.resolve().then(function () {\n\t\n\t return editor.nodes.redactor.childNodes;\n\t })\n\t /** Making a sequence from separate blocks */\n\t .then(editor.saver.makeQueue).then(function () {\n\t // editor.nodes.textarea.innerHTML = editor.state.html;\n\t }).catch(function (error) {\n\t\n\t editor.core.log(error);\n\t });\n\t };\n\t\n\t saver.makeQueue = function (blocks) {\n\t\n\t var queue = Promise.resolve();\n\t\n\t for (var index = 0; index < blocks.length; index++) {\n\t\n\t /** Add node to sequence at specified index */\n\t editor.saver.getBlockData(queue, blocks, index);\n\t }\n\t };\n\t\n\t /** Gets every block and makes From Data */\n\t saver.getBlockData = function (queue, blocks, index) {\n\t\n\t queue.then(function () {\n\t\n\t return editor.saver.getNodeAsync(blocks, index);\n\t }).then(editor.saver.makeFormDataFromBlocks);\n\t };\n\t\n\t /**\r\n\t * Asynchronously returns block data from blocksList by index\r\n\t * @return Promise to node\r\n\t */\n\t saver.getNodeAsync = function (blocksList, index) {\n\t\n\t return Promise.resolve().then(function () {\n\t\n\t return blocksList[index];\n\t });\n\t };\n\t\n\t saver.makeFormDataFromBlocks = function (block) {\n\t\n\t var pluginName = block.dataset.tool;\n\t\n\t /** Check for plugin existance */\n\t if (!editor.tools[pluginName]) {\n\t\n\t throw Error('Plugin \\xAB' + pluginName + '\\xBB not found');\n\t }\n\t\n\t /** Check for plugin having render method */\n\t if (typeof editor.tools[pluginName].save != 'function') {\n\t\n\t throw Error('Plugin \\xAB' + pluginName + '\\xBB must have save method');\n\t }\n\t\n\t /** Result saver */\n\t var blockContent = block.childNodes[0],\n\t pluginsContent = blockContent.childNodes[0],\n\t savedData,\n\t position,\n\t output;\n\t\n\t /** If plugin wasn't available then return data from cache */\n\t if (editor.tools[pluginName].available === false) {\n\t\n\t position = pluginsContent.dataset.inputPosition;\n\t savedData = codex.editor.state.blocks.items[position].data;\n\t } else {\n\t\n\t savedData = editor.tools[pluginName].save(pluginsContent);\n\t\n\t if (editor.tools[pluginName].validate) {\n\t\n\t var result = editor.tools[pluginName].validate(savedData);\n\t\n\t /**\r\n\t * Do not allow invalid data\r\n\t */\n\t if (!result) return;\n\t }\n\t }\n\t\n\t output = {\n\t type: pluginName,\n\t data: savedData\n\t };\n\t\n\t editor.state.jsonOutput.push(output);\n\t };\n\t\n\t return saver;\n\t}({});\n\n/***/ },\n/* 7 */\n/***/ function(module, exports) {\n\n\t'use strict';\n\t\n\t/**\r\n\t * Codex Editor Content Module\r\n\t * Works with DOM\r\n\t *\r\n\t * @module Codex Editor content module\r\n\t *\r\n\t * @author Codex Team\r\n\t * @version 1.3.13\r\n\t *\r\n\t * @description Module works with Elements that have been appended to the main DOM\r\n\t */\n\t\n\tmodule.exports = function (content) {\n\t\n\t var editor = codex.editor;\n\t\n\t /**\r\n\t * Links to current active block\r\n\t * @type {null | Element}\r\n\t */\n\t content.currentNode = null;\n\t\n\t /**\r\n\t * clicked in redactor area\r\n\t * @type {null | Boolean}\r\n\t */\n\t content.editorAreaHightlighted = null;\n\t\n\t /**\r\n\t * @deprecated\r\n\t * Synchronizes redactor with original textarea\r\n\t */\n\t content.sync = function () {\n\t\n\t editor.core.log('syncing...');\n\t\n\t /**\r\n\t * Save redactor content to editor.state\r\n\t */\n\t editor.state.html = editor.nodes.redactor.innerHTML;\n\t };\n\t\n\t /**\r\n\t * Appends background to the block\r\n\t *\r\n\t * @description add CSS class to highlight visually first-level block area\r\n\t */\n\t content.markBlock = function () {\n\t\n\t editor.content.currentNode.classList.add(editor.ui.className.BLOCK_HIGHLIGHTED);\n\t };\n\t\n\t /**\r\n\t * Clear background\r\n\t *\r\n\t * @description clears styles that highlights block\r\n\t */\n\t content.clearMark = function () {\n\t\n\t if (editor.content.currentNode) {\n\t\n\t editor.content.currentNode.classList.remove(editor.ui.className.BLOCK_HIGHLIGHTED);\n\t }\n\t };\n\t\n\t /**\r\n\t * Finds first-level block\r\n\t *\r\n\t * @param {Element} node - selected or clicked in redactors area node\r\n\t * @protected\r\n\t *\r\n\t * @description looks for first-level block.\r\n\t * gets parent while node is not first-level\r\n\t */\n\t content.getFirstLevelBlock = function (node) {\n\t\n\t if (!editor.core.isDomNode(node)) {\n\t\n\t node = node.parentNode;\n\t }\n\t\n\t if (node === editor.nodes.redactor || node === document.body) {\n\t\n\t return null;\n\t } else {\n\t\n\t while (!node.classList.contains(editor.ui.className.BLOCK_CLASSNAME)) {\n\t\n\t node = node.parentNode;\n\t }\n\t\n\t return node;\n\t }\n\t };\n\t\n\t /**\r\n\t * Trigger this event when working node changed\r\n\t * @param {Element} targetNode - first-level of this node will be current\r\n\t * @protected\r\n\t *\r\n\t * @description If targetNode is first-level then we set it as current else we look for parents to find first-level\r\n\t */\n\t content.workingNodeChanged = function (targetNode) {\n\t\n\t /** Clear background from previous marked block before we change */\n\t editor.content.clearMark();\n\t\n\t if (!targetNode) {\n\t\n\t return;\n\t }\n\t\n\t content.currentNode = content.getFirstLevelBlock(targetNode);\n\t };\n\t\n\t /**\r\n\t * Replaces one redactor block with another\r\n\t * @protected\r\n\t * @param {Element} targetBlock - block to replace. Mostly currentNode.\r\n\t * @param {Element} newBlock\r\n\t * @param {string} newBlockType - type of new block; we need to store it to data-attribute\r\n\t *\r\n\t * [!] Function does not saves old block content.\r\n\t * You can get it manually and pass with newBlock.innerHTML\r\n\t */\n\t content.replaceBlock = function (targetBlock, newBlock) {\n\t\n\t if (!targetBlock || !newBlock) {\n\t\n\t editor.core.log('replaceBlock: missed params');\n\t return;\n\t }\n\t\n\t /** If target-block is not a frist-level block, then we iterate parents to find it */\n\t while (!targetBlock.classList.contains(editor.ui.className.BLOCK_CLASSNAME)) {\n\t\n\t targetBlock = targetBlock.parentNode;\n\t }\n\t\n\t /** Replacing */\n\t editor.nodes.redactor.replaceChild(newBlock, targetBlock);\n\t\n\t /**\r\n\t * Set new node as current\r\n\t */\n\t editor.content.workingNodeChanged(newBlock);\n\t\n\t /**\r\n\t * Add block handlers\r\n\t */\n\t editor.ui.addBlockHandlers(newBlock);\n\t\n\t /**\r\n\t * Save changes\r\n\t */\n\t editor.ui.saveInputs();\n\t };\n\t\n\t /**\r\n\t * @protected\r\n\t *\r\n\t * Inserts new block to redactor\r\n\t * Wrapps block into a DIV with BLOCK_CLASSNAME class\r\n\t *\r\n\t * @param blockData {object}\r\n\t * @param blockData.block {Element} element with block content\r\n\t * @param blockData.type {string} block plugin\r\n\t * @param needPlaceCaret {bool} pass true to set caret in new block\r\n\t *\r\n\t */\n\t content.insertBlock = function (blockData, needPlaceCaret) {\n\t\n\t var workingBlock = editor.content.currentNode,\n\t newBlockContent = blockData.block,\n\t blockType = blockData.type,\n\t isStretched = blockData.stretched;\n\t\n\t var newBlock = composeNewBlock_(newBlockContent, blockType, isStretched);\n\t\n\t if (workingBlock) {\n\t\n\t editor.core.insertAfter(workingBlock, newBlock);\n\t } else {\n\t\n\t /**\r\n\t * If redactor is empty, append as first child\r\n\t */\n\t editor.nodes.redactor.appendChild(newBlock);\n\t }\n\t\n\t /**\r\n\t * Block handler\r\n\t */\n\t editor.ui.addBlockHandlers(newBlock);\n\t\n\t /**\r\n\t * Set new node as current\r\n\t */\n\t editor.content.workingNodeChanged(newBlock);\n\t\n\t /**\r\n\t * Save changes\r\n\t */\n\t editor.ui.saveInputs();\n\t\n\t if (needPlaceCaret) {\n\t\n\t /**\r\n\t * If we don't know input index then we set default value -1\r\n\t */\n\t var currentInputIndex = editor.caret.getCurrentInputIndex() || -1;\n\t\n\t if (currentInputIndex == -1) {\n\t\n\t var editableElement = newBlock.querySelector('[contenteditable]'),\n\t emptyText = document.createTextNode('');\n\t\n\t editableElement.appendChild(emptyText);\n\t editor.caret.set(editableElement, 0, 0);\n\t\n\t editor.toolbar.move();\n\t editor.toolbar.showPlusButton();\n\t } else {\n\t\n\t if (currentInputIndex === editor.state.inputs.length - 1) return;\n\t\n\t /** Timeout for browsers execution */\n\t window.setTimeout(function () {\n\t\n\t /** Setting to the new input */\n\t editor.caret.setToNextBlock(currentInputIndex);\n\t editor.toolbar.move();\n\t editor.toolbar.open();\n\t }, 10);\n\t }\n\t }\n\t\n\t /**\r\n\t * Block is inserted, wait for new click that defined focusing on editors area\r\n\t * @type {boolean}\r\n\t */\n\t content.editorAreaHightlighted = false;\n\t };\n\t\n\t /**\r\n\t * Replaces blocks with saving content\r\n\t * @protected\r\n\t * @param {Element} noteToReplace\r\n\t * @param {Element} newNode\r\n\t * @param {Element} blockType\r\n\t */\n\t content.switchBlock = function (blockToReplace, newBlock, tool) {\n\t\n\t tool = tool || editor.content.currentNode.dataset.tool;\n\t var newBlockComposed = composeNewBlock_(newBlock, tool);\n\t\n\t /** Replacing */\n\t editor.content.replaceBlock(blockToReplace, newBlockComposed);\n\t\n\t /** Save new Inputs when block is changed */\n\t editor.ui.saveInputs();\n\t };\n\t\n\t /**\r\n\t * Iterates between child noted and looking for #text node on deepest level\r\n\t * @protected\r\n\t *\r\n\t * @param {Element} block - node where find\r\n\t * @param {int} postiton - starting postion\r\n\t * Example: childNodex.length to find from the end\r\n\t * or 0 to find from the start\r\n\t * @return {Text} block\r\n\t * @uses DFS\r\n\t */\n\t content.getDeepestTextNodeFromPosition = function (block, position) {\n\t\n\t /**\r\n\t * Clear Block from empty and useless spaces with trim.\r\n\t * Such nodes we should remove\r\n\t */\n\t var blockChilds = block.childNodes,\n\t index,\n\t node,\n\t text;\n\t\n\t for (index = 0; index < blockChilds.length; index++) {\n\t\n\t node = blockChilds[index];\n\t\n\t if (node.nodeType == editor.core.nodeTypes.TEXT) {\n\t\n\t text = node.textContent.trim();\n\t\n\t /** Text is empty. We should remove this child from node before we start DFS\r\n\t * decrease the quantity of childs.\r\n\t */\n\t if (text === '') {\n\t\n\t block.removeChild(node);\n\t position--;\n\t }\n\t }\n\t }\n\t\n\t if (block.childNodes.length === 0) {\n\t\n\t return document.createTextNode('');\n\t }\n\t\n\t /** Setting default position when we deleted all empty nodes */\n\t if (position < 0) position = 1;\n\t\n\t var lookingFromStart = false;\n\t\n\t /** For looking from START */\n\t if (position === 0) {\n\t\n\t lookingFromStart = true;\n\t position = 1;\n\t }\n\t\n\t while (position) {\n\t\n\t /** initial verticle of node. */\n\t if (lookingFromStart) {\n\t\n\t block = block.childNodes[0];\n\t } else {\n\t\n\t block = block.childNodes[position - 1];\n\t }\n\t\n\t if (block.nodeType == editor.core.nodeTypes.TAG) {\n\t\n\t position = block.childNodes.length;\n\t } else if (block.nodeType == editor.core.nodeTypes.TEXT) {\n\t\n\t position = 0;\n\t }\n\t }\n\t\n\t return block;\n\t };\n\t\n\t /**\r\n\t * @private\r\n\t * @param {Element} block - current plugins render\r\n\t * @param {String} tool - plugins name\r\n\t * @param {Boolean} isStretched - make stretched block or not\r\n\t *\r\n\t * @description adds necessary information to wrap new created block by first-level holder\r\n\t */\n\t var composeNewBlock_ = function composeNewBlock_(block, tool, isStretched) {\n\t\n\t var newBlock = editor.draw.node('DIV', editor.ui.className.BLOCK_CLASSNAME, {}),\n\t blockContent = editor.draw.node('DIV', editor.ui.className.BLOCK_CONTENT, {});\n\t\n\t blockContent.appendChild(block);\n\t newBlock.appendChild(blockContent);\n\t\n\t if (isStretched) {\n\t\n\t blockContent.classList.add(editor.ui.className.BLOCK_STRETCHED);\n\t }\n\t\n\t newBlock.dataset.tool = tool;\n\t return newBlock;\n\t };\n\t\n\t /**\r\n\t * Returns Range object of current selection\r\n\t * @protected\r\n\t */\n\t content.getRange = function () {\n\t\n\t var selection = window.getSelection().getRangeAt(0);\n\t\n\t return selection;\n\t };\n\t\n\t /**\r\n\t * Divides block in two blocks (after and before caret)\r\n\t *\r\n\t * @protected\r\n\t * @param {int} inputIndex - target input index\r\n\t *\r\n\t * @description splits current input content to the separate blocks\r\n\t * When enter is pressed among the words, that text will be splited.\r\n\t */\n\t content.splitBlock = function (inputIndex) {\n\t\n\t var selection = window.getSelection(),\n\t anchorNode = selection.anchorNode,\n\t anchorNodeText = anchorNode.textContent,\n\t caretOffset = selection.anchorOffset,\n\t textBeforeCaret,\n\t textNodeBeforeCaret,\n\t textAfterCaret,\n\t textNodeAfterCaret;\n\t\n\t var currentBlock = editor.content.currentNode.querySelector('[contentEditable]');\n\t\n\t textBeforeCaret = anchorNodeText.substring(0, caretOffset);\n\t textAfterCaret = anchorNodeText.substring(caretOffset);\n\t\n\t textNodeBeforeCaret = document.createTextNode(textBeforeCaret);\n\t\n\t if (textAfterCaret) {\n\t\n\t textNodeAfterCaret = document.createTextNode(textAfterCaret);\n\t }\n\t\n\t var previousChilds = [],\n\t nextChilds = [],\n\t reachedCurrent = false;\n\t\n\t if (textNodeAfterCaret) {\n\t\n\t nextChilds.push(textNodeAfterCaret);\n\t }\n\t\n\t for (var i = 0, child; !!(child = currentBlock.childNodes[i]); i++) {\n\t\n\t if (child != anchorNode) {\n\t\n\t if (!reachedCurrent) {\n\t\n\t previousChilds.push(child);\n\t } else {\n\t\n\t nextChilds.push(child);\n\t }\n\t } else {\n\t\n\t reachedCurrent = true;\n\t }\n\t }\n\t\n\t /** Clear current input */\n\t editor.state.inputs[inputIndex].innerHTML = '';\n\t\n\t /**\r\n\t * Append all childs founded before anchorNode\r\n\t */\n\t var previousChildsLength = previousChilds.length;\n\t\n\t for (i = 0; i < previousChildsLength; i++) {\n\t\n\t editor.state.inputs[inputIndex].appendChild(previousChilds[i]);\n\t }\n\t\n\t editor.state.inputs[inputIndex].appendChild(textNodeBeforeCaret);\n\t\n\t /**\r\n\t * Append text node which is after caret\r\n\t */\n\t var nextChildsLength = nextChilds.length,\n\t newNode = document.createElement('div');\n\t\n\t for (i = 0; i < nextChildsLength; i++) {\n\t\n\t newNode.appendChild(nextChilds[i]);\n\t }\n\t\n\t newNode = newNode.innerHTML;\n\t\n\t /** This type of block creates when enter is pressed */\n\t var NEW_BLOCK_TYPE = editor.settings.initialBlockPlugin;\n\t\n\t /**\r\n\t * Make new paragraph with text after caret\r\n\t */\n\t editor.content.insertBlock({\n\t type: NEW_BLOCK_TYPE,\n\t block: editor.tools[NEW_BLOCK_TYPE].render({\n\t text: newNode\n\t })\n\t }, true);\n\t };\n\t\n\t /**\r\n\t * Merges two blocks — current and target\r\n\t * If target index is not exist, then previous will be as target\r\n\t *\r\n\t * @protected\r\n\t * @param {int} currentInputIndex\r\n\t * @param {int} targetInputIndex\r\n\t *\r\n\t * @description gets two inputs indexes and merges into one\r\n\t */\n\t content.mergeBlocks = function (currentInputIndex, targetInputIndex) {\n\t\n\t /** If current input index is zero, then prevent method execution */\n\t if (currentInputIndex === 0) {\n\t\n\t return;\n\t }\n\t\n\t var targetInput,\n\t currentInputContent = editor.state.inputs[currentInputIndex].innerHTML;\n\t\n\t if (!targetInputIndex) {\n\t\n\t targetInput = editor.state.inputs[currentInputIndex - 1];\n\t } else {\n\t\n\t targetInput = editor.state.inputs[targetInputIndex];\n\t }\n\t\n\t targetInput.innerHTML += currentInputContent;\n\t };\n\t\n\t /**\r\n\t * Iterates all right siblings and parents, which has right siblings\r\n\t * while it does not reached the first-level block\r\n\t *\r\n\t * @param {Element} node\r\n\t * @return {boolean}\r\n\t */\n\t content.isLastNode = function (node) {\n\t\n\t // console.log('погнали перебор родителей');\n\t\n\t var allChecked = false;\n\t\n\t while (!allChecked) {\n\t\n\t // console.log('Смотрим на %o', node);\n\t // console.log('Проверим, пустые ли соседи справа');\n\t\n\t if (!allSiblingsEmpty_(node)) {\n\t\n\t // console.log('Есть непустые соседи. Узел не последний. Выходим.');\n\t return false;\n\t }\n\t\n\t node = node.parentNode;\n\t\n\t /**\r\n\t * Проверяем родителей до тех пор, пока не найдем блок первого уровня\r\n\t */\n\t if (node.classList.contains(editor.ui.className.BLOCK_CONTENT)) {\n\t\n\t allChecked = true;\n\t }\n\t }\n\t\n\t return true;\n\t };\n\t\n\t /**\r\n\t * Checks if all element right siblings is empty\r\n\t * @param node\r\n\t */\n\t var allSiblingsEmpty_ = function allSiblingsEmpty_(node) {\n\t\n\t /**\r\n\t * Нужно убедиться, что после пустого соседа ничего нет\r\n\t */\n\t var sibling = node.nextSibling;\n\t\n\t while (sibling) {\n\t\n\t if (sibling.textContent.length) {\n\t\n\t return false;\n\t }\n\t\n\t sibling = sibling.nextSibling;\n\t }\n\t\n\t return true;\n\t };\n\t\n\t /**\r\n\t * @public\r\n\t *\r\n\t * @param {string} htmlData - html content as string\r\n\t * @param {string} plainData - plain text\r\n\t * @return {string} - html content as string\r\n\t */\n\t content.wrapTextWithParagraphs = function (htmlData, plainData) {\n\t\n\t if (!htmlData) {\n\t\n\t return '

    ' + plainData.split('\\n\\n').join('

    ') + '

    ';\n\t }\n\t\n\t var wrapper = document.createElement('DIV'),\n\t newWrapper = document.createElement('DIV'),\n\t i,\n\t paragraph,\n\t firstLevelBlocks = ['DIV', 'P'],\n\t blockTyped,\n\t node;\n\t\n\t /**\r\n\t * Make HTML Element to Wrap Text\r\n\t * It allows us to work with input data as HTML content\r\n\t */\n\t wrapper.innerHTML = htmlData;\n\t paragraph = document.createElement('P');\n\t\n\t for (i = 0; i < wrapper.childNodes.length; i++) {\n\t\n\t node = wrapper.childNodes[i];\n\t\n\t blockTyped = firstLevelBlocks.indexOf(node.tagName) != -1;\n\t\n\t /**\r\n\t * If node is first-levet\r\n\t * we add this node to our new wrapper\r\n\t */\n\t if (blockTyped) {\n\t\n\t /**\r\n\t * If we had splitted inline nodes to paragraph before\r\n\t */\n\t if (paragraph.childNodes.length) {\n\t\n\t newWrapper.appendChild(paragraph.cloneNode(true));\n\t\n\t /** empty paragraph */\n\t paragraph = null;\n\t paragraph = document.createElement('P');\n\t }\n\t\n\t newWrapper.appendChild(node.cloneNode(true));\n\t } else {\n\t\n\t /** Collect all inline nodes to one as paragraph */\n\t paragraph.appendChild(node.cloneNode(true));\n\t\n\t /** if node is last we should append this node to paragraph and paragraph to new wrapper */\n\t if (i == wrapper.childNodes.length - 1) {\n\t\n\t newWrapper.appendChild(paragraph.cloneNode(true));\n\t }\n\t }\n\t }\n\t\n\t return newWrapper.innerHTML;\n\t };\n\t\n\t /**\r\n\t * Finds closest Contenteditable parent from Element\r\n\t * @param {Element} node element looking from\r\n\t * @return {Element} node contenteditable\r\n\t */\n\t content.getEditableParent = function (node) {\n\t\n\t while (node && node.contentEditable != 'true') {\n\t\n\t node = node.parentNode;\n\t }\n\t\n\t return node;\n\t };\n\t\n\t return content;\n\t}({});\n\n/***/ },\n/* 8 */\n/***/ function(module, exports, __webpack_require__) {\n\n\t'use strict';\n\t\n\t/**\r\n\t * Codex Editor toolbar module\r\n\t *\r\n\t * Contains:\r\n\t * - Inline toolbox\r\n\t * - Toolbox within plus button\r\n\t * - Settings section\r\n\t *\r\n\t * @author Codex Team\r\n\t * @version 1.0\r\n\t */\n\t\n\tmodule.exports = function (toolbar) {\n\t\n\t var editor = codex.editor;\n\t\n\t toolbar.settings = __webpack_require__(9);\n\t toolbar.inline = __webpack_require__(10);\n\t toolbar.toolbox = __webpack_require__(11);\n\t\n\t /**\r\n\t * Margin between focused node and toolbar\r\n\t */\n\t toolbar.defaultToolbarHeight = 49;\n\t\n\t toolbar.defaultOffset = 34;\n\t\n\t toolbar.opened = false;\n\t\n\t toolbar.current = null;\n\t\n\t /**\r\n\t * @protected\r\n\t */\n\t toolbar.open = function () {\n\t\n\t if (editor.hideToolbar) {\n\t\n\t return;\n\t }\n\t\n\t editor.nodes.toolbar.classList.add('opened');\n\t this.opened = true;\n\t };\n\t\n\t /**\r\n\t * @protected\r\n\t */\n\t toolbar.close = function () {\n\t\n\t editor.nodes.toolbar.classList.remove('opened');\n\t\n\t toolbar.opened = false;\n\t toolbar.current = null;\n\t\n\t for (var button in editor.nodes.toolbarButtons) {\n\t\n\t editor.nodes.toolbarButtons[button].classList.remove('selected');\n\t }\n\t\n\t /** Close toolbox when toolbar is not displayed */\n\t editor.toolbar.toolbox.close();\n\t editor.toolbar.settings.close();\n\t };\n\t\n\t toolbar.toggle = function () {\n\t\n\t if (!this.opened) {\n\t\n\t this.open();\n\t } else {\n\t\n\t this.close();\n\t }\n\t };\n\t\n\t toolbar.hidePlusButton = function () {\n\t\n\t editor.nodes.plusButton.classList.add('hide');\n\t };\n\t\n\t toolbar.showPlusButton = function () {\n\t\n\t editor.nodes.plusButton.classList.remove('hide');\n\t };\n\t\n\t /**\r\n\t * Moving toolbar to the specified node\r\n\t */\n\t toolbar.move = function () {\n\t\n\t /** Close Toolbox when we move toolbar */\n\t editor.toolbar.toolbox.close();\n\t\n\t if (!editor.content.currentNode) {\n\t\n\t return;\n\t }\n\t\n\t var newYCoordinate = editor.content.currentNode.offsetTop - editor.toolbar.defaultToolbarHeight / 2 + editor.toolbar.defaultOffset;\n\t\n\t editor.nodes.toolbar.style.transform = 'translate3D(0, ' + Math.floor(newYCoordinate) + 'px, 0)';\n\t\n\t /** Close trash actions */\n\t editor.toolbar.settings.hideRemoveActions();\n\t };\n\t\n\t return toolbar;\n\t}({});\n\n/***/ },\n/* 9 */\n/***/ function(module, exports) {\n\n\t'use strict';\n\t\n\t/**\r\n\t * Toolbar settings\r\n\t *\r\n\t * @version 1.0.4\r\n\t */\n\t\n\tmodule.exports = function (settings) {\n\t\n\t var editor = codex.editor;\n\t\n\t settings.opened = false;\n\t\n\t settings.setting = null;\n\t settings.actions = null;\n\t\n\t /**\r\n\t * Append and open settings\r\n\t */\n\t settings.open = function (toolType) {\n\t\n\t /**\r\n\t * Append settings content\r\n\t * It's stored in tool.settings\r\n\t */\n\t if (!editor.tools[toolType] || !editor.tools[toolType].makeSettings) {\n\t\n\t editor.core.log('Plugin \\xAB' + toolType + '\\xBB has no settings', 'warn');\n\t editor.nodes.pluginSettings.innerHTML = '\\u041F\\u043B\\u0430\\u0433\\u0438\\u043D \\xAB' + toolType + '\\xBB \\u043D\\u0435 \\u0438\\u043C\\u0435\\u0435\\u0442 \\u043D\\u0430\\u0441\\u0442\\u0440\\u043E\\u0435\\u043A';\n\t } else {\n\t\n\t /**\r\n\t * Draw settings block\r\n\t */\n\t var settingsBlock = editor.tools[toolType].makeSettings();\n\t\n\t editor.nodes.pluginSettings.appendChild(settingsBlock);\n\t }\n\t\n\t /** Open settings block */\n\t editor.nodes.blockSettings.classList.add('opened');\n\t this.opened = true;\n\t };\n\t\n\t /**\r\n\t * Close and clear settings\r\n\t */\n\t settings.close = function () {\n\t\n\t editor.nodes.blockSettings.classList.remove('opened');\n\t editor.nodes.pluginSettings.innerHTML = '';\n\t\n\t this.opened = false;\n\t };\n\t\n\t /**\r\n\t * @param {string} toolType - plugin type\r\n\t */\n\t settings.toggle = function (toolType) {\n\t\n\t if (!this.opened) {\n\t\n\t this.open(toolType);\n\t } else {\n\t\n\t this.close();\n\t }\n\t };\n\t\n\t /**\r\n\t * Here we will draw buttons and add listeners to components\r\n\t */\n\t settings.makeRemoveBlockButton = function () {\n\t\n\t var removeBlockWrapper = editor.draw.node('SPAN', 'ce-toolbar__remove-btn', {}),\n\t settingButton = editor.draw.node('SPAN', 'ce-toolbar__remove-setting', { innerHTML: '' }),\n\t actionWrapper = editor.draw.node('DIV', 'ce-toolbar__remove-confirmation', {}),\n\t confirmAction = editor.draw.node('DIV', 'ce-toolbar__remove-confirm', { textContent: 'Удалить блок' }),\n\t cancelAction = editor.draw.node('DIV', 'ce-toolbar__remove-cancel', { textContent: 'Отмена' });\n\t\n\t editor.listeners.add(settingButton, 'click', editor.toolbar.settings.removeButtonClicked, false);\n\t\n\t editor.listeners.add(confirmAction, 'click', editor.toolbar.settings.confirmRemovingRequest, false);\n\t\n\t editor.listeners.add(cancelAction, 'click', editor.toolbar.settings.cancelRemovingRequest, false);\n\t\n\t actionWrapper.appendChild(confirmAction);\n\t actionWrapper.appendChild(cancelAction);\n\t\n\t removeBlockWrapper.appendChild(settingButton);\n\t removeBlockWrapper.appendChild(actionWrapper);\n\t\n\t /** Save setting */\n\t editor.toolbar.settings.setting = settingButton;\n\t editor.toolbar.settings.actions = actionWrapper;\n\t\n\t return removeBlockWrapper;\n\t };\n\t\n\t settings.removeButtonClicked = function () {\n\t\n\t var action = editor.toolbar.settings.actions;\n\t\n\t if (action.classList.contains('opened')) {\n\t\n\t editor.toolbar.settings.hideRemoveActions();\n\t } else {\n\t\n\t editor.toolbar.settings.showRemoveActions();\n\t }\n\t\n\t editor.toolbar.toolbox.close();\n\t editor.toolbar.settings.close();\n\t };\n\t\n\t settings.cancelRemovingRequest = function () {\n\t\n\t editor.toolbar.settings.actions.classList.remove('opened');\n\t };\n\t\n\t settings.confirmRemovingRequest = function () {\n\t\n\t var currentBlock = editor.content.currentNode,\n\t firstLevelBlocksCount;\n\t\n\t currentBlock.remove();\n\t\n\t firstLevelBlocksCount = editor.nodes.redactor.childNodes.length;\n\t\n\t /**\r\n\t * If all blocks are removed\r\n\t */\n\t if (firstLevelBlocksCount === 0) {\n\t\n\t /** update currentNode variable */\n\t editor.content.currentNode = null;\n\t\n\t /** Inserting new empty initial block */\n\t editor.ui.addInitialBlock();\n\t }\n\t\n\t editor.ui.saveInputs();\n\t\n\t editor.toolbar.close();\n\t };\n\t\n\t settings.showRemoveActions = function () {\n\t\n\t editor.toolbar.settings.actions.classList.add('opened');\n\t };\n\t\n\t settings.hideRemoveActions = function () {\n\t\n\t editor.toolbar.settings.actions.classList.remove('opened');\n\t };\n\t\n\t return settings;\n\t}({});\n\n/***/ },\n/* 10 */\n/***/ function(module, exports) {\n\n\t'use strict';\n\t\n\t/**\r\n\t * Inline toolbar\r\n\t *\r\n\t * Contains from tools:\r\n\t * Bold, Italic, Underline and Anchor\r\n\t *\r\n\t * @author Codex Team\r\n\t * @version 1.0\r\n\t */\n\t\n\tmodule.exports = function (inline) {\n\t\n\t var editor = codex.editor;\n\t\n\t inline.buttonsOpened = null;\n\t inline.actionsOpened = null;\n\t inline.wrappersOffset = null;\n\t\n\t /**\r\n\t * saving selection that need for execCommand for styling\r\n\t *\r\n\t */\n\t inline.storedSelection = null;\n\t\n\t /**\r\n\t * @protected\r\n\t *\r\n\t * Open inline toobar\r\n\t */\n\t inline.show = function () {\n\t\n\t var currentNode = editor.content.currentNode,\n\t tool = currentNode.dataset.tool,\n\t plugin;\n\t\n\t /**\r\n\t * tool allowed to open inline toolbar\r\n\t */\n\t plugin = editor.tools[tool];\n\t\n\t if (!plugin.showInlineToolbar) return;\n\t\n\t var selectedText = inline.getSelectionText(),\n\t toolbar = editor.nodes.inlineToolbar.wrapper;\n\t\n\t if (selectedText.length > 0) {\n\t\n\t /** Move toolbar and open */\n\t editor.toolbar.inline.move();\n\t\n\t /** Open inline toolbar */\n\t toolbar.classList.add('opened');\n\t\n\t /** show buttons of inline toolbar */\n\t editor.toolbar.inline.showButtons();\n\t }\n\t };\n\t\n\t /**\r\n\t * @protected\r\n\t *\r\n\t * Closes inline toolbar\r\n\t */\n\t inline.close = function () {\n\t\n\t var toolbar = editor.nodes.inlineToolbar.wrapper;\n\t\n\t toolbar.classList.remove('opened');\n\t };\n\t\n\t /**\r\n\t * @private\r\n\t *\r\n\t * Moving toolbar\r\n\t */\n\t inline.move = function () {\n\t\n\t if (!this.wrappersOffset) {\n\t\n\t this.wrappersOffset = this.getWrappersOffset();\n\t }\n\t\n\t var coords = this.getSelectionCoords(),\n\t defaultOffset = 0,\n\t toolbar = editor.nodes.inlineToolbar.wrapper,\n\t newCoordinateX,\n\t newCoordinateY;\n\t\n\t if (toolbar.offsetHeight === 0) {\n\t\n\t defaultOffset = 40;\n\t }\n\t\n\t newCoordinateX = coords.x - this.wrappersOffset.left;\n\t newCoordinateY = coords.y + window.scrollY - this.wrappersOffset.top - defaultOffset - toolbar.offsetHeight;\n\t\n\t toolbar.style.transform = 'translate3D(' + Math.floor(newCoordinateX) + 'px, ' + Math.floor(newCoordinateY) + 'px, 0)';\n\t\n\t /** Close everything */\n\t editor.toolbar.inline.closeButtons();\n\t editor.toolbar.inline.closeAction();\n\t };\n\t\n\t /**\r\n\t * @private\r\n\t *\r\n\t * Tool Clicked\r\n\t */\n\t\n\t inline.toolClicked = function (event, type) {\n\t\n\t /**\r\n\t * For simple tools we use default browser function\r\n\t * For more complicated tools, we should write our own behavior\r\n\t */\n\t switch (type) {\n\t case 'createLink':\n\t editor.toolbar.inline.createLinkAction(event, type);break;\n\t default:\n\t editor.toolbar.inline.defaultToolAction(type);break;\n\t }\n\t\n\t /**\r\n\t * highlight buttons\r\n\t * after making some action\r\n\t */\n\t editor.nodes.inlineToolbar.buttons.childNodes.forEach(editor.toolbar.inline.hightlight);\n\t };\n\t\n\t /**\r\n\t * @private\r\n\t *\r\n\t * Saving wrappers offset in DOM\r\n\t */\n\t inline.getWrappersOffset = function () {\n\t\n\t var wrapper = editor.nodes.wrapper,\n\t offset = this.getOffset(wrapper);\n\t\n\t this.wrappersOffset = offset;\n\t return offset;\n\t };\n\t\n\t /**\r\n\t * @private\r\n\t *\r\n\t * Calculates offset of DOM element\r\n\t *\r\n\t * @param el\r\n\t * @returns {{top: number, left: number}}\r\n\t */\n\t inline.getOffset = function (el) {\n\t\n\t var _x = 0;\n\t var _y = 0;\n\t\n\t while (el && !isNaN(el.offsetLeft) && !isNaN(el.offsetTop)) {\n\t\n\t _x += el.offsetLeft + el.clientLeft;\n\t _y += el.offsetTop + el.clientTop;\n\t el = el.offsetParent;\n\t }\n\t return { top: _y, left: _x };\n\t };\n\t\n\t /**\r\n\t * @private\r\n\t *\r\n\t * Calculates position of selected text\r\n\t * @returns {{x: number, y: number}}\r\n\t */\n\t inline.getSelectionCoords = function () {\n\t\n\t var sel = document.selection,\n\t range;\n\t var x = 0,\n\t y = 0;\n\t\n\t if (sel) {\n\t\n\t if (sel.type != 'Control') {\n\t\n\t range = sel.createRange();\n\t range.collapse(true);\n\t x = range.boundingLeft;\n\t y = range.boundingTop;\n\t }\n\t } else if (window.getSelection) {\n\t\n\t sel = window.getSelection();\n\t\n\t if (sel.rangeCount) {\n\t\n\t range = sel.getRangeAt(0).cloneRange();\n\t if (range.getClientRects) {\n\t\n\t range.collapse(true);\n\t var rect = range.getClientRects()[0];\n\t\n\t if (!rect) {\n\t\n\t return;\n\t }\n\t\n\t x = rect.left;\n\t y = rect.top;\n\t }\n\t }\n\t }\n\t return { x: x, y: y };\n\t };\n\t\n\t /**\r\n\t * @private\r\n\t *\r\n\t * Returns selected text as String\r\n\t * @returns {string}\r\n\t */\n\t inline.getSelectionText = function () {\n\t\n\t var selectedText = '';\n\t\n\t // all modern browsers and IE9+\n\t if (window.getSelection) {\n\t\n\t selectedText = window.getSelection().toString();\n\t }\n\t\n\t return selectedText;\n\t };\n\t\n\t /** Opens buttons block */\n\t inline.showButtons = function () {\n\t\n\t var buttons = editor.nodes.inlineToolbar.buttons;\n\t\n\t buttons.classList.add('opened');\n\t\n\t editor.toolbar.inline.buttonsOpened = true;\n\t\n\t /** highlight buttons */\n\t editor.nodes.inlineToolbar.buttons.childNodes.forEach(editor.toolbar.inline.hightlight);\n\t };\n\t\n\t /** Makes buttons disappear */\n\t inline.closeButtons = function () {\n\t\n\t var buttons = editor.nodes.inlineToolbar.buttons;\n\t\n\t buttons.classList.remove('opened');\n\t\n\t editor.toolbar.inline.buttonsOpened = false;\n\t };\n\t\n\t /** Open buttons defined action if exist */\n\t inline.showActions = function () {\n\t\n\t var action = editor.nodes.inlineToolbar.actions;\n\t\n\t action.classList.add('opened');\n\t\n\t editor.toolbar.inline.actionsOpened = true;\n\t };\n\t\n\t /** Close actions block */\n\t inline.closeAction = function () {\n\t\n\t var action = editor.nodes.inlineToolbar.actions;\n\t\n\t action.innerHTML = '';\n\t action.classList.remove('opened');\n\t editor.toolbar.inline.actionsOpened = false;\n\t };\n\t\n\t /**\r\n\t * Callback for keydowns in inline toolbar \"Insert link...\" input\r\n\t */\n\t var inlineToolbarAnchorInputKeydown_ = function inlineToolbarAnchorInputKeydown_(event) {\n\t\n\t if (event.keyCode != editor.core.keys.ENTER) {\n\t\n\t return;\n\t }\n\t\n\t var editable = editor.content.currentNode,\n\t storedSelection = editor.toolbar.inline.storedSelection;\n\t\n\t editor.toolbar.inline.restoreSelection(editable, storedSelection);\n\t editor.toolbar.inline.setAnchor(this.value);\n\t\n\t /**\r\n\t * Preventing events that will be able to happen\r\n\t */\n\t event.preventDefault();\n\t event.stopImmediatePropagation();\n\t\n\t editor.toolbar.inline.clearRange();\n\t };\n\t\n\t /** Action for link creation or for setting anchor */\n\t inline.createLinkAction = function (event) {\n\t\n\t var isActive = this.isLinkActive();\n\t\n\t var editable = editor.content.currentNode,\n\t storedSelection = editor.toolbar.inline.saveSelection(editable);\n\t\n\t /** Save globally selection */\n\t editor.toolbar.inline.storedSelection = storedSelection;\n\t\n\t if (isActive) {\n\t\n\t /**\r\n\t * Changing stored selection. if we want to remove anchor from word\r\n\t * we should remove anchor from whole word, not only selected part.\r\n\t * The solution is than we get the length of current link\r\n\t * Change start position to - end of selection minus length of anchor\r\n\t */\n\t editor.toolbar.inline.restoreSelection(editable, storedSelection);\n\t\n\t editor.toolbar.inline.defaultToolAction('unlink');\n\t } else {\n\t\n\t /** Create input and close buttons */\n\t var action = editor.draw.inputForLink();\n\t\n\t editor.nodes.inlineToolbar.actions.appendChild(action);\n\t\n\t editor.toolbar.inline.closeButtons();\n\t editor.toolbar.inline.showActions();\n\t\n\t /**\r\n\t * focus to input\r\n\t * Solution: https://developer.mozilla.org/ru/docs/Web/API/HTMLElement/focus\r\n\t * Prevents event after showing input and when we need to focus an input which is in unexisted form\r\n\t */\n\t action.focus();\n\t event.preventDefault();\n\t\n\t /** Callback to link action */\n\t editor.listeners.add(action, 'keydown', inlineToolbarAnchorInputKeydown_, false);\n\t }\n\t };\n\t\n\t inline.isLinkActive = function () {\n\t\n\t var isActive = false;\n\t\n\t editor.nodes.inlineToolbar.buttons.childNodes.forEach(function (tool) {\n\t\n\t var dataType = tool.dataset.type;\n\t\n\t if (dataType == 'link' && tool.classList.contains('hightlighted')) {\n\t\n\t isActive = true;\n\t }\n\t });\n\t\n\t return isActive;\n\t };\n\t\n\t /** default action behavior of tool */\n\t inline.defaultToolAction = function (type) {\n\t\n\t document.execCommand(type, false, null);\n\t };\n\t\n\t /**\r\n\t * @private\r\n\t *\r\n\t * Sets URL\r\n\t *\r\n\t * @param {String} url - URL\r\n\t */\n\t inline.setAnchor = function (url) {\n\t\n\t document.execCommand('createLink', false, url);\n\t\n\t /** Close after URL inserting */\n\t editor.toolbar.inline.closeAction();\n\t };\n\t\n\t /**\r\n\t * @private\r\n\t *\r\n\t * Saves selection\r\n\t */\n\t inline.saveSelection = function (containerEl) {\n\t\n\t var range = window.getSelection().getRangeAt(0),\n\t preSelectionRange = range.cloneRange(),\n\t start;\n\t\n\t preSelectionRange.selectNodeContents(containerEl);\n\t preSelectionRange.setEnd(range.startContainer, range.startOffset);\n\t\n\t start = preSelectionRange.toString().length;\n\t\n\t return {\n\t start: start,\n\t end: start + range.toString().length\n\t };\n\t };\n\t\n\t /**\r\n\t * @private\r\n\t *\r\n\t * Sets to previous selection (Range)\r\n\t *\r\n\t * @param {Element} containerEl - editable element where we restore range\r\n\t * @param {Object} savedSel - range basic information to restore\r\n\t */\n\t inline.restoreSelection = function (containerEl, savedSel) {\n\t\n\t var range = document.createRange(),\n\t charIndex = 0;\n\t\n\t range.setStart(containerEl, 0);\n\t range.collapse(true);\n\t\n\t var nodeStack = [containerEl],\n\t node,\n\t foundStart = false,\n\t stop = false,\n\t nextCharIndex;\n\t\n\t while (!stop && (node = nodeStack.pop())) {\n\t\n\t if (node.nodeType == 3) {\n\t\n\t nextCharIndex = charIndex + node.length;\n\t\n\t if (!foundStart && savedSel.start >= charIndex && savedSel.start <= nextCharIndex) {\n\t\n\t range.setStart(node, savedSel.start - charIndex);\n\t foundStart = true;\n\t }\n\t if (foundStart && savedSel.end >= charIndex && savedSel.end <= nextCharIndex) {\n\t\n\t range.setEnd(node, savedSel.end - charIndex);\n\t stop = true;\n\t }\n\t charIndex = nextCharIndex;\n\t } else {\n\t\n\t var i = node.childNodes.length;\n\t\n\t while (i--) {\n\t\n\t nodeStack.push(node.childNodes[i]);\n\t }\n\t }\n\t }\n\t\n\t var sel = window.getSelection();\n\t\n\t sel.removeAllRanges();\n\t sel.addRange(range);\n\t };\n\t\n\t /**\r\n\t * @private\r\n\t *\r\n\t * Removes all ranges from window selection\r\n\t */\n\t inline.clearRange = function () {\n\t\n\t var selection = window.getSelection();\n\t\n\t selection.removeAllRanges();\n\t };\n\t\n\t /**\r\n\t * @private\r\n\t *\r\n\t * sets or removes hightlight\r\n\t */\n\t inline.hightlight = function (tool) {\n\t\n\t var dataType = tool.dataset.type;\n\t\n\t if (document.queryCommandState(dataType)) {\n\t\n\t editor.toolbar.inline.setButtonHighlighted(tool);\n\t } else {\n\t\n\t editor.toolbar.inline.removeButtonsHighLight(tool);\n\t }\n\t\n\t /**\r\n\t *\r\n\t * hightlight for anchors\r\n\t */\n\t var selection = window.getSelection(),\n\t tag = selection.anchorNode.parentNode;\n\t\n\t if (tag.tagName == 'A' && dataType == 'link') {\n\t\n\t editor.toolbar.inline.setButtonHighlighted(tool);\n\t }\n\t };\n\t\n\t /**\r\n\t * @private\r\n\t *\r\n\t * Mark button if text is already executed\r\n\t */\n\t inline.setButtonHighlighted = function (button) {\n\t\n\t button.classList.add('hightlighted');\n\t\n\t /** At link tool we also change icon */\n\t if (button.dataset.type == 'link') {\n\t\n\t var icon = button.childNodes[0];\n\t\n\t icon.classList.remove('ce-icon-link');\n\t icon.classList.add('ce-icon-unlink');\n\t }\n\t };\n\t\n\t /**\r\n\t * @private\r\n\t *\r\n\t * Removes hightlight\r\n\t */\n\t inline.removeButtonsHighLight = function (button) {\n\t\n\t button.classList.remove('hightlighted');\n\t\n\t /** At link tool we also change icon */\n\t if (button.dataset.type == 'link') {\n\t\n\t var icon = button.childNodes[0];\n\t\n\t icon.classList.remove('ce-icon-unlink');\n\t icon.classList.add('ce-icon-link');\n\t }\n\t };\n\t\n\t return inline;\n\t}({});\n\n/***/ },\n/* 11 */\n/***/ function(module, exports) {\n\n\t'use strict';\n\t\n\t/**\r\n\t * Codex Editor toolbox\r\n\t *\r\n\t * All tools be able to appended here\r\n\t *\r\n\t * @author Codex Team\r\n\t * @version 1.0\r\n\t */\n\t\n\tmodule.exports = function (toolbox) {\n\t\n\t var editor = codex.editor;\n\t\n\t toolbox.opened = false;\n\t\n\t /** Shows toolbox */\n\t toolbox.open = function () {\n\t\n\t /** Close setting if toolbox is opened */\n\t if (editor.toolbar.settings.opened) {\n\t\n\t editor.toolbar.settings.close();\n\t }\n\t\n\t /** display toolbox */\n\t editor.nodes.toolbox.classList.add('opened');\n\t\n\t /** Animate plus button */\n\t editor.nodes.plusButton.classList.add('clicked');\n\t\n\t /** toolbox state */\n\t editor.toolbar.toolbox.opened = true;\n\t };\n\t\n\t /** Closes toolbox */\n\t toolbox.close = function () {\n\t\n\t /** Makes toolbox disapear */\n\t editor.nodes.toolbox.classList.remove('opened');\n\t\n\t /** Rotate plus button */\n\t editor.nodes.plusButton.classList.remove('clicked');\n\t\n\t /** toolbox state */\n\t editor.toolbar.toolbox.opened = false;\n\t\n\t editor.toolbar.current = null;\n\t };\n\t\n\t toolbox.leaf = function () {\n\t\n\t var currentTool = editor.toolbar.current,\n\t tools = Object.keys(editor.tools),\n\t barButtons = editor.nodes.toolbarButtons,\n\t nextToolIndex = 0,\n\t toolToSelect = void 0,\n\t visibleTool = void 0,\n\t tool = void 0;\n\t\n\t if (!currentTool) {\n\t\n\t /** Get first tool from object*/\n\t for (tool in editor.tools) {\n\t\n\t if (editor.tools[tool].displayInToolbox) {\n\t\n\t break;\n\t }\n\t\n\t nextToolIndex++;\n\t }\n\t } else {\n\t\n\t nextToolIndex = (tools.indexOf(currentTool) + 1) % tools.length;\n\t visibleTool = tools[nextToolIndex];\n\t\n\t while (!editor.tools[visibleTool].displayInToolbox) {\n\t\n\t nextToolIndex = (nextToolIndex + 1) % tools.length;\n\t visibleTool = tools[nextToolIndex];\n\t }\n\t }\n\t\n\t toolToSelect = tools[nextToolIndex];\n\t\n\t for (var button in barButtons) {\n\t\n\t barButtons[button].classList.remove('selected');\n\t }\n\t\n\t barButtons[toolToSelect].classList.add('selected');\n\t editor.toolbar.current = toolToSelect;\n\t };\n\t\n\t /**\r\n\t * Transforming selected node type into selected toolbar element type\r\n\t * @param {event} event\r\n\t */\n\t toolbox.toolClicked = function (event) {\n\t\n\t /**\r\n\t * UNREPLACEBLE_TOOLS this types of tools are forbidden to replace even they are empty\r\n\t */\n\t var UNREPLACEBLE_TOOLS = ['image', 'link', 'list', 'instagram', 'twitter', 'embed'],\n\t tool = editor.tools[editor.toolbar.current],\n\t workingNode = editor.content.currentNode,\n\t currentInputIndex = editor.caret.inputIndex,\n\t newBlockContent,\n\t appendCallback,\n\t blockData;\n\t\n\t /** Make block from plugin */\n\t newBlockContent = tool.render();\n\t\n\t /** information about block */\n\t blockData = {\n\t block: newBlockContent,\n\t type: tool.type,\n\t stretched: false\n\t };\n\t\n\t if (workingNode && UNREPLACEBLE_TOOLS.indexOf(workingNode.dataset.tool) === -1 && workingNode.textContent.trim() === '') {\n\t\n\t /** Replace current block */\n\t editor.content.switchBlock(workingNode, newBlockContent, tool.type);\n\t } else {\n\t\n\t /** Insert new Block from plugin */\n\t editor.content.insertBlock(blockData);\n\t\n\t /** increase input index */\n\t currentInputIndex++;\n\t }\n\t\n\t /** Fire tool append callback */\n\t appendCallback = tool.appendCallback;\n\t\n\t if (appendCallback && typeof appendCallback == 'function') {\n\t\n\t appendCallback.call(event);\n\t }\n\t\n\t window.setTimeout(function () {\n\t\n\t /** Set caret to current block */\n\t editor.caret.setToBlock(currentInputIndex);\n\t }, 10);\n\t\n\t /**\r\n\t * Changing current Node\r\n\t */\n\t editor.content.workingNodeChanged();\n\t\n\t /**\r\n\t * Move toolbar when node is changed\r\n\t */\n\t editor.toolbar.move();\n\t };\n\t\n\t return toolbox;\n\t}({});\n\n/***/ },\n/* 12 */\n/***/ function(module, exports) {\n\n\t'use strict';\n\t\n\t/**\r\n\t * @module Codex Editor Callbacks module\r\n\t * @description Module works with editor added Elements\r\n\t *\r\n\t * @author Codex Team\r\n\t * @version 1.4.0\r\n\t */\n\t\n\tmodule.exports = function (callbacks) {\n\t\n\t var editor = codex.editor;\n\t\n\t /**\r\n\t * used by UI module\r\n\t * @description Routes all keydowns on document\r\n\t * @param {Object} event\r\n\t */\n\t callbacks.globalKeydown = function (event) {\n\t\n\t switch (event.keyCode) {\n\t case editor.core.keys.ENTER:\n\t enterKeyPressed_(event);break;\n\t }\n\t };\n\t\n\t /**\r\n\t * used by UI module\r\n\t * @description Routes all keydowns on redactors area\r\n\t * @param {Object} event\r\n\t */\n\t callbacks.redactorKeyDown = function (event) {\n\t\n\t switch (event.keyCode) {\n\t case editor.core.keys.TAB:\n\t tabKeyPressedOnRedactorsZone_(event);break;\n\t case editor.core.keys.ENTER:\n\t enterKeyPressedOnRedactorsZone_(event);break;\n\t case editor.core.keys.ESC:\n\t escapeKeyPressedOnRedactorsZone_(event);break;\n\t default:\n\t defaultKeyPressedOnRedactorsZone_(event);break;\n\t }\n\t };\n\t\n\t /**\r\n\t * used by UI module\r\n\t * @description Routes all keyup events\r\n\t * @param {Object} event\r\n\t */\n\t callbacks.globalKeyup = function (event) {\n\t\n\t switch (event.keyCode) {\n\t case editor.core.keys.UP:\n\t case editor.core.keys.LEFT:\n\t case editor.core.keys.RIGHT:\n\t case editor.core.keys.DOWN:\n\t arrowKeyPressed_(event);break;\n\t }\n\t };\n\t\n\t /**\r\n\t * @param {Object} event\r\n\t * @private\r\n\t *\r\n\t * Handles behaviour when tab pressed\r\n\t * @description if Content is empty show toolbox (if it is closed) or leaf tools\r\n\t * uses Toolbars toolbox module to handle the situation\r\n\t */\n\t var tabKeyPressedOnRedactorsZone_ = function tabKeyPressedOnRedactorsZone_(event) {\n\t\n\t /**\r\n\t * Wait for solution. Would like to know the behaviour\r\n\t * @todo Add spaces\r\n\t */\n\t event.preventDefault();\n\t\n\t var nativeInputs = editor.content.currentNode.querySelectorAll('textarea, input'),\n\t nativeInputsAreEmpty = true,\n\t textContentIsEmpty = !editor.content.currentNode.textContent.trim();\n\t\n\t Array.prototype.map.call(nativeInputs, function (input) {\n\t\n\t if (input.type == 'textarea' || input.type == 'text') {\n\t\n\t nativeInputsAreEmpty = nativeInputsAreEmpty && !input.value.trim();\n\t }\n\t });\n\t\n\t var blockIsEmpty = textContentIsEmpty && nativeInputsAreEmpty;\n\t\n\t if (!blockIsEmpty) {\n\t\n\t return;\n\t }\n\t\n\t if (!editor.toolbar.opened) {\n\t\n\t editor.toolbar.open();\n\t }\n\t\n\t if (editor.toolbar.opened && !editor.toolbar.toolbox.opened) {\n\t\n\t editor.toolbar.toolbox.open();\n\t } else {\n\t\n\t editor.toolbar.toolbox.leaf();\n\t }\n\t };\n\t\n\t /**\r\n\t * Handles global EnterKey Press\r\n\t * @see enterPressedOnBlock_\r\n\t * @param {Object} event\r\n\t */\n\t var enterKeyPressed_ = function enterKeyPressed_() {\n\t\n\t if (editor.content.editorAreaHightlighted) {\n\t\n\t /**\r\n\t * it means that we lose input index, saved index before is not correct\r\n\t * therefore we need to set caret when we insert new block\r\n\t */\n\t editor.caret.inputIndex = -1;\n\t\n\t enterPressedOnBlock_();\n\t }\n\t };\n\t\n\t /**\r\n\t * Callback for enter key pressing in first-level block area\r\n\t *\r\n\t * @param {Event} event\r\n\t * @private\r\n\t *\r\n\t * @description Inserts new block with initial type from settings\r\n\t */\n\t var enterPressedOnBlock_ = function enterPressedOnBlock_() {\n\t\n\t var NEW_BLOCK_TYPE = editor.settings.initialBlockPlugin;\n\t\n\t editor.content.insertBlock({\n\t type: NEW_BLOCK_TYPE,\n\t block: editor.tools[NEW_BLOCK_TYPE].render()\n\t }, true);\n\t\n\t editor.toolbar.move();\n\t editor.toolbar.open();\n\t };\n\t\n\t /**\r\n\t * ENTER key handler\r\n\t *\r\n\t * @param {Object} event\r\n\t * @private\r\n\t *\r\n\t * @description Makes new block with initial type from settings\r\n\t */\n\t var enterKeyPressedOnRedactorsZone_ = function enterKeyPressedOnRedactorsZone_(event) {\n\t\n\t if (event.target.contentEditable == 'true') {\n\t\n\t /** Update input index */\n\t editor.caret.saveCurrentInputIndex();\n\t }\n\t\n\t var currentInputIndex = editor.caret.getCurrentInputIndex() || 0,\n\t workingNode = editor.content.currentNode,\n\t tool = workingNode.dataset.tool,\n\t isEnterPressedOnToolbar = editor.toolbar.opened && editor.toolbar.current && event.target == editor.state.inputs[currentInputIndex];\n\t\n\t /** The list of tools which needs the default browser behaviour */\n\t var enableLineBreaks = editor.tools[tool].enableLineBreaks;\n\t\n\t /** This type of block creates when enter is pressed */\n\t var NEW_BLOCK_TYPE = editor.settings.initialBlockPlugin;\n\t\n\t /**\r\n\t * When toolbar is opened, select tool instead of making new paragraph\r\n\t */\n\t if (isEnterPressedOnToolbar) {\n\t\n\t event.preventDefault();\n\t\n\t editor.toolbar.toolbox.toolClicked(event);\n\t\n\t editor.toolbar.close();\n\t\n\t /**\r\n\t * Stop other listeners callback executions\r\n\t */\n\t event.stopPropagation();\n\t event.stopImmediatePropagation();\n\t\n\t return;\n\t }\n\t\n\t /**\r\n\t * Allow paragraph lineBreaks with shift enter\r\n\t * Or if shiftkey pressed and enter and enabledLineBreaks, the let new block creation\r\n\t */\n\t if (event.shiftKey || enableLineBreaks) {\n\t\n\t event.stopPropagation();\n\t event.stopImmediatePropagation();\n\t return;\n\t }\n\t\n\t var currentSelection = window.getSelection(),\n\t currentSelectedNode = currentSelection.anchorNode,\n\t caretAtTheEndOfText = editor.caret.position.atTheEnd(),\n\t isTextNodeHasParentBetweenContenteditable = false;\n\t\n\t /**\r\n\t * Allow making new

    in same block by SHIFT+ENTER and forbids to prevent default browser behaviour\r\n\t */\n\t if (event.shiftKey && !enableLineBreaks) {\n\t\n\t editor.callback.enterPressedOnBlock(editor.content.currentBlock, event);\n\t event.preventDefault();\n\t return;\n\t }\n\t\n\t /**\r\n\t * Workaround situation when caret at the Text node that has some wrapper Elements\r\n\t * Split block cant handle this.\r\n\t * We need to save default behavior\r\n\t */\n\t isTextNodeHasParentBetweenContenteditable = currentSelectedNode && currentSelectedNode.parentNode.contentEditable != 'true';\n\t\n\t /**\r\n\t * Split blocks when input has several nodes and caret placed in textNode\r\n\t */\n\t if (currentSelectedNode.nodeType == editor.core.nodeTypes.TEXT && !isTextNodeHasParentBetweenContenteditable && !caretAtTheEndOfText) {\n\t\n\t event.preventDefault();\n\t\n\t editor.core.log('Splitting Text node...');\n\t\n\t editor.content.splitBlock(currentInputIndex);\n\t\n\t /** Show plus button when next input after split is empty*/\n\t if (!editor.state.inputs[currentInputIndex + 1].textContent.trim()) {\n\t\n\t editor.toolbar.showPlusButton();\n\t }\n\t } else {\n\t\n\t var islastNode = editor.content.isLastNode(currentSelectedNode);\n\t\n\t if (islastNode && caretAtTheEndOfText) {\n\t\n\t event.preventDefault();\n\t event.stopPropagation();\n\t event.stopImmediatePropagation();\n\t\n\t editor.core.log('ENTER clicked in last textNode. Create new BLOCK');\n\t\n\t editor.content.insertBlock({\n\t type: NEW_BLOCK_TYPE,\n\t block: editor.tools[NEW_BLOCK_TYPE].render()\n\t }, true);\n\t\n\t editor.toolbar.move();\n\t editor.toolbar.open();\n\t\n\t /** Show plus button with empty block */\n\t editor.toolbar.showPlusButton();\n\t }\n\t }\n\t\n\t /** get all inputs after new appending block */\n\t editor.ui.saveInputs();\n\t };\n\t\n\t /**\r\n\t * Escape behaviour\r\n\t * @param event\r\n\t * @private\r\n\t *\r\n\t * @description Closes toolbox and toolbar. Prevents default behaviour\r\n\t */\n\t var escapeKeyPressedOnRedactorsZone_ = function escapeKeyPressedOnRedactorsZone_(event) {\n\t\n\t /** Close all toolbar */\n\t editor.toolbar.close();\n\t\n\t /** Close toolbox */\n\t editor.toolbar.toolbox.close();\n\t\n\t event.preventDefault();\n\t };\n\t\n\t /**\r\n\t * @param {Event} event\r\n\t * @private\r\n\t *\r\n\t * closes and moves toolbar\r\n\t */\n\t var arrowKeyPressed_ = function arrowKeyPressed_(event) {\n\t\n\t editor.content.workingNodeChanged();\n\t\n\t /* Closing toolbar */\n\t editor.toolbar.close();\n\t editor.toolbar.move();\n\t };\n\t\n\t /**\r\n\t * @private\r\n\t * @param {Event} event\r\n\t *\r\n\t * @description Closes all opened bars from toolbar.\r\n\t * If block is mark, clears highlightning\r\n\t */\n\t var defaultKeyPressedOnRedactorsZone_ = function defaultKeyPressedOnRedactorsZone_() {\n\t\n\t editor.toolbar.close();\n\t\n\t if (!editor.toolbar.inline.actionsOpened) {\n\t\n\t editor.toolbar.inline.close();\n\t editor.content.clearMark();\n\t }\n\t };\n\t\n\t /**\r\n\t * Handler when clicked on redactors area\r\n\t *\r\n\t * @protected\r\n\t * @param event\r\n\t *\r\n\t * @description Detects clicked area. If it is first-level block area, marks as detected and\r\n\t * on next enter press will be inserted new block\r\n\t * Otherwise, save carets position (input index) and put caret to the editable zone.\r\n\t *\r\n\t * @see detectWhenClickedOnFirstLevelBlockArea_\r\n\t *\r\n\t */\n\t callbacks.redactorClicked = function (event) {\n\t\n\t detectWhenClickedOnFirstLevelBlockArea_();\n\t\n\t editor.content.workingNodeChanged(event.target);\n\t editor.ui.saveInputs();\n\t\n\t var selectedText = editor.toolbar.inline.getSelectionText(),\n\t firstLevelBlock;\n\t\n\t /** If selection range took off, then we hide inline toolbar */\n\t if (selectedText.length === 0) {\n\t\n\t editor.toolbar.inline.close();\n\t }\n\t\n\t /** Update current input index in memory when caret focused into existed input */\n\t if (event.target.contentEditable == 'true') {\n\t\n\t editor.caret.saveCurrentInputIndex();\n\t }\n\t\n\t if (editor.content.currentNode === null) {\n\t\n\t /**\r\n\t * If inputs in redactor does not exits, then we put input index 0 not -1\r\n\t */\n\t var indexOfLastInput = editor.state.inputs.length > 0 ? editor.state.inputs.length - 1 : 0;\n\t\n\t /** If we have any inputs */\n\t if (editor.state.inputs.length) {\n\t\n\t /** getting firstlevel parent of input */\n\t firstLevelBlock = editor.content.getFirstLevelBlock(editor.state.inputs[indexOfLastInput]);\n\t }\n\t\n\t /** If input is empty, then we set caret to the last input */\n\t if (editor.state.inputs.length && editor.state.inputs[indexOfLastInput].textContent === '' && firstLevelBlock.dataset.tool == editor.settings.initialBlockPlugin) {\n\t\n\t editor.caret.setToBlock(indexOfLastInput);\n\t } else {\n\t\n\t /** Create new input when caret clicked in redactors area */\n\t var NEW_BLOCK_TYPE = editor.settings.initialBlockPlugin;\n\t\n\t editor.content.insertBlock({\n\t type: NEW_BLOCK_TYPE,\n\t block: editor.tools[NEW_BLOCK_TYPE].render()\n\t });\n\t\n\t /** If there is no inputs except inserted */\n\t if (editor.state.inputs.length === 1) {\n\t\n\t editor.caret.setToBlock(indexOfLastInput);\n\t } else {\n\t\n\t /** Set caret to this appended input */\n\t editor.caret.setToNextBlock(indexOfLastInput);\n\t }\n\t }\n\t } else {\n\t\n\t /** Close all panels */\n\t editor.toolbar.settings.close();\n\t editor.toolbar.toolbox.close();\n\t }\n\t\n\t /**\r\n\t * Move toolbar and open\r\n\t */\n\t editor.toolbar.move();\n\t editor.toolbar.open();\n\t\n\t var inputIsEmpty = !editor.content.currentNode.textContent.trim(),\n\t currentNodeType = editor.content.currentNode.dataset.tool,\n\t isInitialType = currentNodeType == editor.settings.initialBlockPlugin;\n\t\n\t /** Hide plus buttons */\n\t editor.toolbar.hidePlusButton();\n\t\n\t if (!inputIsEmpty) {\n\t\n\t /** Mark current block */\n\t editor.content.markBlock();\n\t }\n\t\n\t if (isInitialType && inputIsEmpty) {\n\t\n\t /** Show plus button */\n\t editor.toolbar.showPlusButton();\n\t }\n\t };\n\t\n\t /**\r\n\t * This method allows to define, is caret in contenteditable element or not.\r\n\t *\r\n\t * @private\r\n\t *\r\n\t * @description Otherwise, if we get TEXT node from range container, that will means we have input index.\r\n\t * In this case we use default browsers behaviour (if plugin allows that) or overwritten action.\r\n\t * Therefore, to be sure that we've clicked first-level block area, we should have currentNode, which always\r\n\t * specifies to the first-level block. Other cases we just ignore.\r\n\t */\n\t var detectWhenClickedOnFirstLevelBlockArea_ = function detectWhenClickedOnFirstLevelBlockArea_() {\n\t\n\t var selection = window.getSelection(),\n\t anchorNode = selection.anchorNode,\n\t flag = false;\n\t\n\t if (selection.rangeCount === 0) {\n\t\n\t editor.content.editorAreaHightlighted = true;\n\t } else {\n\t\n\t if (!editor.core.isDomNode(anchorNode)) {\n\t\n\t anchorNode = anchorNode.parentNode;\n\t }\n\t\n\t /** Already founded, without loop */\n\t if (anchorNode.contentEditable == 'true') {\n\t\n\t flag = true;\n\t }\n\t\n\t while (anchorNode.contentEditable != 'true') {\n\t\n\t anchorNode = anchorNode.parentNode;\n\t\n\t if (anchorNode.contentEditable == 'true') {\n\t\n\t flag = true;\n\t }\n\t\n\t if (anchorNode == document.body) {\n\t\n\t break;\n\t }\n\t }\n\t\n\t /** If editable element founded, flag is \"TRUE\", Therefore we return \"FALSE\" */\n\t editor.content.editorAreaHightlighted = !flag;\n\t }\n\t };\n\t\n\t /**\r\n\t * Toolbar button click handler\r\n\t *\r\n\t * @param {Object} event - cursor to the button\r\n\t * @protected\r\n\t *\r\n\t * @description gets current tool and calls render method\r\n\t */\n\t callbacks.toolbarButtonClicked = function (event) {\n\t\n\t var button = this;\n\t\n\t editor.toolbar.current = button.dataset.type;\n\t\n\t editor.toolbar.toolbox.toolClicked(event);\n\t editor.toolbar.close();\n\t };\n\t\n\t /**\r\n\t * Show or Hide toolbox when plus button is clicked\r\n\t */\n\t callbacks.plusButtonClicked = function () {\n\t\n\t if (!editor.nodes.toolbox.classList.contains('opened')) {\n\t\n\t editor.toolbar.toolbox.open();\n\t } else {\n\t\n\t editor.toolbar.toolbox.close();\n\t }\n\t };\n\t\n\t /**\r\n\t * Block handlers for KeyDown events\r\n\t *\r\n\t * @protected\r\n\t * @param {Object} event\r\n\t *\r\n\t * Handles keydowns on block\r\n\t * @see blockRightOrDownArrowPressed_\r\n\t * @see backspacePressed_\r\n\t * @see blockLeftOrUpArrowPressed_\r\n\t */\n\t callbacks.blockKeydown = function (event) {\n\t\n\t var block = event.target; // event.target is input\n\t\n\t switch (event.keyCode) {\n\t\n\t case editor.core.keys.DOWN:\n\t case editor.core.keys.RIGHT:\n\t blockRightOrDownArrowPressed_(event);\n\t break;\n\t\n\t case editor.core.keys.BACKSPACE:\n\t backspacePressed_(block, event);\n\t break;\n\t\n\t case editor.core.keys.UP:\n\t case editor.core.keys.LEFT:\n\t blockLeftOrUpArrowPressed_(event);\n\t break;\n\t\n\t }\n\t };\n\t\n\t /**\r\n\t * RIGHT or DOWN keydowns on block\r\n\t *\r\n\t * @param {Object} event\r\n\t * @private\r\n\t *\r\n\t * @description watches the selection and gets closest editable element.\r\n\t * Uses method getDeepestTextNodeFromPosition to get the last node of next block\r\n\t * Sets caret if it is contenteditable\r\n\t */\n\t var blockRightOrDownArrowPressed_ = function blockRightOrDownArrowPressed_(event) {\n\t\n\t var selection = window.getSelection(),\n\t inputs = editor.state.inputs,\n\t focusedNode = selection.anchorNode,\n\t focusedNodeHolder;\n\t\n\t /** Check for caret existance */\n\t if (!focusedNode) {\n\t\n\t return false;\n\t }\n\t\n\t /** Looking for closest (parent) contentEditable element of focused node */\n\t while (focusedNode.contentEditable != 'true') {\n\t\n\t focusedNodeHolder = focusedNode.parentNode;\n\t focusedNode = focusedNodeHolder;\n\t }\n\t\n\t /** Input index in DOM level */\n\t var editableElementIndex = 0;\n\t\n\t while (focusedNode != inputs[editableElementIndex]) {\n\t\n\t editableElementIndex++;\n\t }\n\t\n\t /**\r\n\t * Founded contentEditable element doesn't have childs\r\n\t * Or maybe New created block\r\n\t */\n\t if (!focusedNode.textContent) {\n\t\n\t editor.caret.setToNextBlock(editableElementIndex);\n\t return;\n\t }\n\t\n\t /**\r\n\t * Do nothing when caret doesn not reaches the end of last child\r\n\t */\n\t var caretInLastChild = false,\n\t caretAtTheEndOfText = false;\n\t\n\t var lastChild, deepestTextnode;\n\t\n\t lastChild = focusedNode.childNodes[focusedNode.childNodes.length - 1];\n\t\n\t if (editor.core.isDomNode(lastChild)) {\n\t\n\t deepestTextnode = editor.content.getDeepestTextNodeFromPosition(lastChild, lastChild.childNodes.length);\n\t } else {\n\t\n\t deepestTextnode = lastChild;\n\t }\n\t\n\t caretInLastChild = selection.anchorNode == deepestTextnode;\n\t caretAtTheEndOfText = deepestTextnode.length == selection.anchorOffset;\n\t\n\t if (!caretInLastChild || !caretAtTheEndOfText) {\n\t\n\t editor.core.log('arrow [down|right] : caret does not reached the end');\n\t return false;\n\t }\n\t\n\t editor.caret.setToNextBlock(editableElementIndex);\n\t };\n\t\n\t /**\r\n\t * LEFT or UP keydowns on block\r\n\t *\r\n\t * @param {Object} event\r\n\t * @private\r\n\t *\r\n\t * watches the selection and gets closest editable element.\r\n\t * Uses method getDeepestTextNodeFromPosition to get the last node of previous block\r\n\t * Sets caret if it is contenteditable\r\n\t *\r\n\t */\n\t var blockLeftOrUpArrowPressed_ = function blockLeftOrUpArrowPressed_(event) {\n\t\n\t var selection = window.getSelection(),\n\t inputs = editor.state.inputs,\n\t focusedNode = selection.anchorNode,\n\t focusedNodeHolder;\n\t\n\t /** Check for caret existance */\n\t if (!focusedNode) {\n\t\n\t return false;\n\t }\n\t\n\t /**\r\n\t * LEFT or UP not at the beginning\r\n\t */\n\t if (selection.anchorOffset !== 0) {\n\t\n\t return false;\n\t }\n\t\n\t /** Looking for parent contentEditable block */\n\t while (focusedNode.contentEditable != 'true') {\n\t\n\t focusedNodeHolder = focusedNode.parentNode;\n\t focusedNode = focusedNodeHolder;\n\t }\n\t\n\t /** Input index in DOM level */\n\t var editableElementIndex = 0;\n\t\n\t while (focusedNode != inputs[editableElementIndex]) {\n\t\n\t editableElementIndex++;\n\t }\n\t\n\t /**\r\n\t * Do nothing if caret is not at the beginning of first child\r\n\t */\n\t var caretInFirstChild = false,\n\t caretAtTheBeginning = false;\n\t\n\t var firstChild, deepestTextnode;\n\t\n\t /**\r\n\t * Founded contentEditable element doesn't have childs\r\n\t * Or maybe New created block\r\n\t */\n\t if (!focusedNode.textContent) {\n\t\n\t editor.caret.setToPreviousBlock(editableElementIndex);\n\t return;\n\t }\n\t\n\t firstChild = focusedNode.childNodes[0];\n\t\n\t if (editor.core.isDomNode(firstChild)) {\n\t\n\t deepestTextnode = editor.content.getDeepestTextNodeFromPosition(firstChild, 0);\n\t } else {\n\t\n\t deepestTextnode = firstChild;\n\t }\n\t\n\t caretInFirstChild = selection.anchorNode == deepestTextnode;\n\t caretAtTheBeginning = selection.anchorOffset === 0;\n\t\n\t if (caretInFirstChild && caretAtTheBeginning) {\n\t\n\t editor.caret.setToPreviousBlock(editableElementIndex);\n\t }\n\t };\n\t\n\t /**\r\n\t * Handles backspace keydown\r\n\t *\r\n\t * @param {Element} block\r\n\t * @param {Object} event\r\n\t * @private\r\n\t *\r\n\t * @description if block is empty, delete the block and set caret to the previous block\r\n\t * If block is not empty, try to merge two blocks - current and previous\r\n\t * But it we try'n to remove first block, then we should set caret to the next block, not previous.\r\n\t * If we removed the last block, create new one\r\n\t */\n\t var backspacePressed_ = function backspacePressed_(block, event) {\n\t\n\t var currentInputIndex = editor.caret.getCurrentInputIndex(),\n\t range,\n\t selectionLength,\n\t firstLevelBlocksCount;\n\t\n\t if (editor.core.isNativeInput(event.target)) {\n\t\n\t /** If input value is empty - remove block */\n\t if (event.target.value.trim() == '') {\n\t\n\t block.remove();\n\t } else {\n\t\n\t return;\n\t }\n\t }\n\t\n\t if (block.textContent.trim()) {\n\t\n\t range = editor.content.getRange();\n\t selectionLength = range.endOffset - range.startOffset;\n\t\n\t if (editor.caret.position.atStart() && !selectionLength && editor.state.inputs[currentInputIndex - 1]) {\n\t\n\t editor.content.mergeBlocks(currentInputIndex);\n\t } else {\n\t\n\t return;\n\t }\n\t }\n\t\n\t if (!selectionLength) {\n\t\n\t block.remove();\n\t }\n\t\n\t firstLevelBlocksCount = editor.nodes.redactor.childNodes.length;\n\t\n\t /**\r\n\t * If all blocks are removed\r\n\t */\n\t if (firstLevelBlocksCount === 0) {\n\t\n\t /** update currentNode variable */\n\t editor.content.currentNode = null;\n\t\n\t /** Inserting new empty initial block */\n\t editor.ui.addInitialBlock();\n\t\n\t /** Updating inputs state after deleting last block */\n\t editor.ui.saveInputs();\n\t\n\t /** Set to current appended block */\n\t window.setTimeout(function () {\n\t\n\t editor.caret.setToPreviousBlock(1);\n\t }, 10);\n\t } else {\n\t\n\t if (editor.caret.inputIndex !== 0) {\n\t\n\t /** Target block is not first */\n\t editor.caret.setToPreviousBlock(editor.caret.inputIndex);\n\t } else {\n\t\n\t /** If we try to delete first block */\n\t editor.caret.setToNextBlock(editor.caret.inputIndex);\n\t }\n\t }\n\t\n\t editor.toolbar.move();\n\t\n\t if (!editor.toolbar.opened) {\n\t\n\t editor.toolbar.open();\n\t }\n\t\n\t /** Updating inputs state */\n\t editor.ui.saveInputs();\n\t\n\t /** Prevent default browser behaviour */\n\t event.preventDefault();\n\t };\n\t\n\t /**\r\n\t * used by UI module\r\n\t * Clicks on block settings button\r\n\t *\r\n\t * @param {Object} event\r\n\t * @protected\r\n\t * @description Opens toolbar settings\r\n\t */\n\t callbacks.showSettingsButtonClicked = function (event) {\n\t\n\t /**\r\n\t * Get type of current block\r\n\t * It uses to append settings from tool.settings property.\r\n\t * ...\r\n\t * Type is stored in data-type attribute on block\r\n\t */\n\t var currentToolType = editor.content.currentNode.dataset.tool;\n\t\n\t editor.toolbar.settings.toggle(currentToolType);\n\t\n\t /** Close toolbox when settings button is active */\n\t editor.toolbar.toolbox.close();\n\t editor.toolbar.settings.hideRemoveActions();\n\t };\n\t\n\t return callbacks;\n\t}({});\n\n/***/ },\n/* 13 */\n/***/ function(module, exports) {\n\n\t'use strict';\n\t\n\t/**\r\n\t * Codex Editor Draw module\r\n\t *\r\n\t * @author Codex Team\r\n\t * @version 1.0.\r\n\t */\n\t\n\tmodule.exports = function (draw) {\n\t\n\t /**\r\n\t * Base editor wrapper\r\n\t */\n\t draw.wrapper = function () {\n\t\n\t var wrapper = document.createElement('div');\n\t\n\t wrapper.className += 'codex-editor';\n\t\n\t return wrapper;\n\t };\n\t\n\t /**\r\n\t * Content-editable holder\r\n\t */\n\t draw.redactor = function () {\n\t\n\t var redactor = document.createElement('div');\n\t\n\t redactor.className += 'ce-redactor';\n\t\n\t return redactor;\n\t };\n\t\n\t draw.ceBlock = function () {\n\t\n\t var block = document.createElement('DIV');\n\t\n\t block.className += 'ce_block';\n\t\n\t return block;\n\t };\n\t\n\t /**\r\n\t * Empty toolbar with toggler\r\n\t */\n\t draw.toolbar = function () {\n\t\n\t var bar = document.createElement('div');\n\t\n\t bar.className += 'ce-toolbar';\n\t\n\t return bar;\n\t };\n\t\n\t draw.toolbarContent = function () {\n\t\n\t var wrapper = document.createElement('DIV');\n\t\n\t wrapper.classList.add('ce-toolbar__content');\n\t\n\t return wrapper;\n\t };\n\t\n\t /**\r\n\t * Inline toolbar\r\n\t */\n\t draw.inlineToolbar = function () {\n\t\n\t var bar = document.createElement('DIV');\n\t\n\t bar.className += 'ce-toolbar-inline';\n\t\n\t return bar;\n\t };\n\t\n\t /**\r\n\t * Wrapper for inline toobar buttons\r\n\t */\n\t draw.inlineToolbarButtons = function () {\n\t\n\t var wrapper = document.createElement('DIV');\n\t\n\t wrapper.className += 'ce-toolbar-inline__buttons';\n\t\n\t return wrapper;\n\t };\n\t\n\t /**\r\n\t * For some actions\r\n\t */\n\t draw.inlineToolbarActions = function () {\n\t\n\t var wrapper = document.createElement('DIV');\n\t\n\t wrapper.className += 'ce-toolbar-inline__actions';\n\t\n\t return wrapper;\n\t };\n\t\n\t draw.inputForLink = function () {\n\t\n\t var input = document.createElement('INPUT');\n\t\n\t input.type = 'input';\n\t input.className += 'inputForLink';\n\t input.placeholder = 'Вставьте ссылку ...';\n\t input.setAttribute('form', 'defaultForm');\n\t\n\t input.setAttribute('autofocus', 'autofocus');\n\t\n\t return input;\n\t };\n\t\n\t /**\r\n\t * @todo Desc\r\n\t */\n\t draw.blockButtons = function () {\n\t\n\t var block = document.createElement('div');\n\t\n\t block.className += 'ce-toolbar__actions';\n\t\n\t return block;\n\t };\n\t\n\t /**\r\n\t * Block settings panel\r\n\t */\n\t draw.blockSettings = function () {\n\t\n\t var settings = document.createElement('div');\n\t\n\t settings.className += 'ce-settings';\n\t\n\t return settings;\n\t };\n\t\n\t draw.defaultSettings = function () {\n\t\n\t var div = document.createElement('div');\n\t\n\t div.classList.add('ce-settings_default');\n\t\n\t return div;\n\t };\n\t\n\t draw.pluginsSettings = function () {\n\t\n\t var div = document.createElement('div');\n\t\n\t div.classList.add('ce-settings_plugin');\n\t\n\t return div;\n\t };\n\t\n\t draw.plusButton = function () {\n\t\n\t var button = document.createElement('span');\n\t\n\t button.className = 'ce-toolbar__plus';\n\t // button.innerHTML = '';\n\t\n\t return button;\n\t };\n\t\n\t /**\r\n\t * Settings button in toolbar\r\n\t */\n\t draw.settingsButton = function () {\n\t\n\t var toggler = document.createElement('span');\n\t\n\t toggler.className = 'ce-toolbar__settings-btn';\n\t\n\t /** Toggler button*/\n\t toggler.innerHTML = '';\n\t\n\t return toggler;\n\t };\n\t\n\t /**\r\n\t * Redactor tools wrapper\r\n\t */\n\t\n\t draw.toolbox = function () {\n\t\n\t var wrapper = document.createElement('div');\n\t\n\t wrapper.className = 'ce-toolbar__tools';\n\t\n\t return wrapper;\n\t };\n\t\n\t /**\r\n\t * @protected\r\n\t *\r\n\t * Draws tool buttons for toolbox\r\n\t *\r\n\t * @param {String} type\r\n\t * @param {String} classname\r\n\t * @returns {Element}\r\n\t */\n\t draw.toolbarButton = function (type, classname) {\n\t\n\t var button = document.createElement('li'),\n\t toolIcon = document.createElement('i'),\n\t toolTitle = document.createElement('span');\n\t\n\t button.dataset.type = type;\n\t button.setAttribute('title', type);\n\t\n\t toolIcon.classList.add(classname);\n\t toolTitle.classList.add('ce_toolbar_tools--title');\n\t\n\t button.appendChild(toolIcon);\n\t button.appendChild(toolTitle);\n\t\n\t return button;\n\t };\n\t\n\t /**\r\n\t * @protected\r\n\t *\r\n\t * Draws tools for inline toolbar\r\n\t *\r\n\t * @param {String} type\r\n\t * @param {String} classname\r\n\t */\n\t draw.toolbarButtonInline = function (type, classname) {\n\t\n\t var button = document.createElement('BUTTON'),\n\t toolIcon = document.createElement('I');\n\t\n\t button.type = 'button';\n\t button.dataset.type = type;\n\t toolIcon.classList.add(classname);\n\t\n\t button.appendChild(toolIcon);\n\t\n\t return button;\n\t };\n\t\n\t /**\r\n\t * Redactor block\r\n\t */\n\t draw.block = function (tagName, content) {\n\t\n\t var node = document.createElement(tagName);\n\t\n\t node.innerHTML = content || '';\n\t\n\t return node;\n\t };\n\t\n\t /**\r\n\t * Creates Node with passed tagName and className\r\n\t * @param {string} tagName\r\n\t * @param {string} className\r\n\t * @param {object} properties - allow to assign properties\r\n\t */\n\t draw.node = function (tagName, className, properties) {\n\t\n\t var el = document.createElement(tagName);\n\t\n\t if (className) el.className = className;\n\t\n\t if (properties) {\n\t\n\t for (var name in properties) {\n\t\n\t el[name] = properties[name];\n\t }\n\t }\n\t\n\t return el;\n\t };\n\t\n\t /**\r\n\t * Unavailable plugin block\r\n\t */\n\t draw.unavailableBlock = function () {\n\t\n\t var wrapper = document.createElement('DIV');\n\t\n\t wrapper.classList.add('cdx-unavailable-block');\n\t\n\t return wrapper;\n\t };\n\t\n\t return draw;\n\t}({});\n\n/***/ },\n/* 14 */\n/***/ function(module, exports) {\n\n\t'use strict';\n\t\n\t/**\r\n\t * Codex Editor Caret Module\r\n\t *\r\n\t * @author Codex Team\r\n\t * @version 1.0\r\n\t */\n\t\n\tmodule.exports = function (caret) {\n\t\n\t var editor = codex.editor;\n\t\n\t /**\r\n\t * @var {int} InputIndex - editable element in DOM\r\n\t */\n\t caret.inputIndex = null;\n\t\n\t /**\r\n\t * @var {int} offset - caret position in a text node.\r\n\t */\n\t caret.offset = null;\n\t\n\t /**\r\n\t * @var {int} focusedNodeIndex - we get index of child node from first-level block\r\n\t */\n\t caret.focusedNodeIndex = null;\n\t\n\t /**\r\n\t * Creates Document Range and sets caret to the element.\r\n\t * @protected\r\n\t * @uses caret.save — if you need to save caret position\r\n\t * @param {Element} el - Changed Node.\r\n\t */\n\t caret.set = function (el, index, offset) {\n\t\n\t offset = offset || caret.offset || 0;\n\t index = index || caret.focusedNodeIndex || 0;\n\t\n\t var childs = el.childNodes,\n\t nodeToSet;\n\t\n\t if (childs.length === 0) {\n\t\n\t nodeToSet = el;\n\t } else {\n\t\n\t nodeToSet = childs[index];\n\t }\n\t\n\t /** If Element is INPUT */\n\t if (el.contentEditable != 'true') {\n\t\n\t el.focus();\n\t return;\n\t }\n\t\n\t if (editor.core.isDomNode(nodeToSet)) {\n\t\n\t nodeToSet = editor.content.getDeepestTextNodeFromPosition(nodeToSet, nodeToSet.childNodes.length);\n\t }\n\t\n\t var range = document.createRange(),\n\t selection = window.getSelection();\n\t\n\t window.setTimeout(function () {\n\t\n\t range.setStart(nodeToSet, offset);\n\t range.setEnd(nodeToSet, offset);\n\t\n\t selection.removeAllRanges();\n\t selection.addRange(range);\n\t\n\t editor.caret.saveCurrentInputIndex();\n\t }, 20);\n\t };\n\t\n\t /**\r\n\t * @protected\r\n\t * Updates index of input and saves it in caret object\r\n\t */\n\t caret.saveCurrentInputIndex = function () {\n\t\n\t /** Index of Input that we paste sanitized content */\n\t var selection = window.getSelection(),\n\t inputs = editor.state.inputs,\n\t focusedNode = selection.anchorNode,\n\t focusedNodeHolder;\n\t\n\t if (!focusedNode) {\n\t\n\t return;\n\t }\n\t\n\t /** Looking for parent contentEditable block */\n\t while (focusedNode.contentEditable != 'true') {\n\t\n\t focusedNodeHolder = focusedNode.parentNode;\n\t focusedNode = focusedNodeHolder;\n\t }\n\t\n\t /** Input index in DOM level */\n\t var editableElementIndex = 0;\n\t\n\t while (focusedNode != inputs[editableElementIndex]) {\n\t\n\t editableElementIndex++;\n\t }\n\t\n\t caret.inputIndex = editableElementIndex;\n\t };\n\t\n\t /**\r\n\t * Returns current input index (caret object)\r\n\t */\n\t caret.getCurrentInputIndex = function () {\n\t\n\t return caret.inputIndex;\n\t };\n\t\n\t /**\r\n\t * @param {int} index - index of first-level block after that we set caret into next input\r\n\t */\n\t caret.setToNextBlock = function (index) {\n\t\n\t var inputs = editor.state.inputs,\n\t nextInput = inputs[index + 1];\n\t\n\t if (!nextInput) {\n\t\n\t editor.core.log('We are reached the end');\n\t return;\n\t }\n\t\n\t /**\r\n\t * When new Block created or deleted content of input\r\n\t * We should add some text node to set caret\r\n\t */\n\t if (!nextInput.childNodes.length) {\n\t\n\t var emptyTextElement = document.createTextNode('');\n\t\n\t nextInput.appendChild(emptyTextElement);\n\t }\n\t\n\t editor.caret.inputIndex = index + 1;\n\t editor.caret.set(nextInput, 0, 0);\n\t editor.content.workingNodeChanged(nextInput);\n\t };\n\t\n\t /**\r\n\t * @param {int} index - index of target input.\r\n\t * Sets caret to input with this index\r\n\t */\n\t caret.setToBlock = function (index) {\n\t\n\t var inputs = editor.state.inputs,\n\t targetInput = inputs[index];\n\t\n\t if (!targetInput) {\n\t\n\t return;\n\t }\n\t\n\t /**\r\n\t * When new Block created or deleted content of input\r\n\t * We should add some text node to set caret\r\n\t */\n\t if (!targetInput.childNodes.length) {\n\t\n\t var emptyTextElement = document.createTextNode('');\n\t\n\t targetInput.appendChild(emptyTextElement);\n\t }\n\t\n\t editor.caret.inputIndex = index;\n\t editor.caret.set(targetInput, 0, 0);\n\t editor.content.workingNodeChanged(targetInput);\n\t };\n\t\n\t /**\r\n\t * @param {int} index - index of input\r\n\t */\n\t caret.setToPreviousBlock = function (index) {\n\t\n\t index = index || 0;\n\t\n\t var inputs = editor.state.inputs,\n\t previousInput = inputs[index - 1],\n\t lastChildNode,\n\t lengthOfLastChildNode,\n\t emptyTextElement;\n\t\n\t if (!previousInput) {\n\t\n\t editor.core.log('We are reached first node');\n\t return;\n\t }\n\t\n\t lastChildNode = editor.content.getDeepestTextNodeFromPosition(previousInput, previousInput.childNodes.length);\n\t lengthOfLastChildNode = lastChildNode.length;\n\t\n\t /**\r\n\t * When new Block created or deleted content of input\r\n\t * We should add some text node to set caret\r\n\t */\n\t if (!previousInput.childNodes.length) {\n\t\n\t emptyTextElement = document.createTextNode('');\n\t previousInput.appendChild(emptyTextElement);\n\t }\n\t editor.caret.inputIndex = index - 1;\n\t editor.caret.set(previousInput, previousInput.childNodes.length - 1, lengthOfLastChildNode);\n\t editor.content.workingNodeChanged(inputs[index - 1]);\n\t };\n\t\n\t caret.position = {\n\t\n\t atStart: function atStart() {\n\t\n\t var selection = window.getSelection(),\n\t anchorOffset = selection.anchorOffset,\n\t anchorNode = selection.anchorNode,\n\t firstLevelBlock = editor.content.getFirstLevelBlock(anchorNode),\n\t pluginsRender = firstLevelBlock.childNodes[0];\n\t\n\t if (!editor.core.isDomNode(anchorNode)) {\n\t\n\t anchorNode = anchorNode.parentNode;\n\t }\n\t\n\t var isFirstNode = anchorNode === pluginsRender.childNodes[0],\n\t isOffsetZero = anchorOffset === 0;\n\t\n\t return isFirstNode && isOffsetZero;\n\t },\n\t\n\t atTheEnd: function atTheEnd() {\n\t\n\t var selection = window.getSelection(),\n\t anchorOffset = selection.anchorOffset,\n\t anchorNode = selection.anchorNode;\n\t\n\t /** Caret is at the end of input */\n\t return !anchorNode || !anchorNode.length || anchorOffset === anchorNode.length;\n\t }\n\t };\n\t\n\t /**\r\n\t * Inserts node at the caret location\r\n\t * @param node\r\n\t */\n\t caret.insertNode = function (node) {\n\t\n\t var selection, range;\n\t\n\t selection = window.getSelection();\n\t\n\t range = selection.getRangeAt(0);\n\t range.deleteContents();\n\t\n\t range.insertNode(node);\n\t\n\t range.setStartAfter(node);\n\t range.collapse(true);\n\t\n\t selection.removeAllRanges();\n\t selection.addRange(range);\n\t };\n\t\n\t return caret;\n\t}({});\n\n/***/ },\n/* 15 */\n/***/ function(module, exports) {\n\n\t'use strict';\n\t\n\t/**\r\n\t * Codex Editor Notification Module\r\n\t *\r\n\t * @author Codex Team\r\n\t * @version 1.0\r\n\t */\n\t\n\tmodule.exports = function (notifications) {\n\t\n\t var editor = codex.editor;\n\t\n\t var queue = [];\n\t\n\t var addToQueue = function addToQueue(settings) {\n\t\n\t queue.push(settings);\n\t\n\t var index = 0;\n\t\n\t while (index < queue.length && queue.length > 5) {\n\t\n\t if (queue[index].type == 'confirm' || queue[index].type == 'prompt') {\n\t\n\t index++;\n\t continue;\n\t }\n\t\n\t queue[index].close();\n\t queue.splice(index, 1);\n\t }\n\t };\n\t\n\t notifications.createHolder = function () {\n\t\n\t var holder = editor.draw.node('DIV', 'cdx-notifications-block');\n\t\n\t editor.nodes.notifications = document.body.appendChild(holder);\n\t\n\t return holder;\n\t };\n\t\n\t /**\r\n\t * Error notificator. Shows block with message\r\n\t * @protected\r\n\t */\n\t notifications.errorThrown = function (errorMsg, event) {\n\t\n\t editor.notifications.notification({ message: 'This action is not available currently', type: event.type });\n\t };\n\t\n\t /**\r\n\t *\r\n\t * Appends notification\r\n\t *\r\n\t * settings = {\r\n\t * type - notification type (reserved types: alert, confirm, prompt). Just add class 'cdx-notification-'+type\r\n\t * message - notification message\r\n\t * okMsg - confirm button text (default - 'Ok')\r\n\t * cancelBtn - cancel button text (default - 'Cancel'). Only for confirm and prompt types\r\n\t * confirm - function-handler for ok button click\r\n\t * cancel - function-handler for cancel button click. Only for confirm and prompt types\r\n\t * time - time (in seconds) after which notification will close (default - 10s)\r\n\t * }\r\n\t *\r\n\t * @param settings\r\n\t */\n\t notifications.notification = function (constructorSettings) {\n\t\n\t /** Private vars and methods */\n\t var notification = null,\n\t cancel = null,\n\t type = null,\n\t confirm = null,\n\t inputField = null;\n\t\n\t var confirmHandler = function confirmHandler() {\n\t\n\t close();\n\t\n\t if (typeof confirm !== 'function') {\n\t\n\t return;\n\t }\n\t\n\t if (type == 'prompt') {\n\t\n\t confirm(inputField.value);\n\t return;\n\t }\n\t\n\t confirm();\n\t };\n\t\n\t var cancelHandler = function cancelHandler() {\n\t\n\t close();\n\t\n\t if (typeof cancel !== 'function') {\n\t\n\t return;\n\t }\n\t\n\t cancel();\n\t };\n\t\n\t /** Public methods */\n\t function create(settings) {\n\t\n\t if (!(settings && settings.message)) {\n\t\n\t editor.core.log('Can\\'t create notification. Message is missed');\n\t return;\n\t }\n\t\n\t settings.type = settings.type || 'alert';\n\t settings.time = settings.time * 1000 || 10000;\n\t\n\t var wrapper = editor.draw.node('DIV', 'cdx-notification'),\n\t message = editor.draw.node('DIV', 'cdx-notification__message'),\n\t input = editor.draw.node('INPUT', 'cdx-notification__input'),\n\t okBtn = editor.draw.node('SPAN', 'cdx-notification__ok-btn'),\n\t cancelBtn = editor.draw.node('SPAN', 'cdx-notification__cancel-btn');\n\t\n\t message.textContent = settings.message;\n\t okBtn.textContent = settings.okMsg || 'ОК';\n\t cancelBtn.textContent = settings.cancelMsg || 'Отмена';\n\t\n\t editor.listeners.add(okBtn, 'click', confirmHandler);\n\t editor.listeners.add(cancelBtn, 'click', cancelHandler);\n\t\n\t wrapper.appendChild(message);\n\t\n\t if (settings.type == 'prompt') {\n\t\n\t wrapper.appendChild(input);\n\t }\n\t\n\t wrapper.appendChild(okBtn);\n\t\n\t if (settings.type == 'prompt' || settings.type == 'confirm') {\n\t\n\t wrapper.appendChild(cancelBtn);\n\t }\n\t\n\t wrapper.classList.add('cdx-notification-' + settings.type);\n\t wrapper.dataset.type = settings.type;\n\t\n\t notification = wrapper;\n\t type = settings.type;\n\t confirm = settings.confirm;\n\t cancel = settings.cancel;\n\t inputField = input;\n\t\n\t if (settings.type != 'prompt' && settings.type != 'confirm') {\n\t\n\t window.setTimeout(close, settings.time);\n\t }\n\t };\n\t\n\t function send() {\n\t\n\t editor.nodes.notifications.appendChild(notification);\n\t inputField.focus();\n\t\n\t editor.nodes.notifications.classList.add('cdx-notification__notification-appending');\n\t\n\t window.setTimeout(function () {\n\t\n\t editor.nodes.notifications.classList.remove('cdx-notification__notification-appending');\n\t }, 100);\n\t\n\t addToQueue({ type: type, close: close });\n\t };\n\t\n\t function close() {\n\t\n\t notification.remove();\n\t };\n\t\n\t if (constructorSettings) {\n\t\n\t create(constructorSettings);\n\t send();\n\t }\n\t\n\t return {\n\t create: create,\n\t send: send,\n\t close: close\n\t };\n\t };\n\t\n\t notifications.clear = function () {\n\t\n\t editor.nodes.notifications.innerHTML = '';\n\t queue = [];\n\t };\n\t\n\t return notifications;\n\t}({});\n\n/***/ },\n/* 16 */\n/***/ function(module, exports) {\n\n\t\"use strict\";\n\t\n\t/**\r\n\t * Codex Editor Parser Module\r\n\t *\r\n\t * @author Codex Team\r\n\t * @version 1.1\r\n\t */\n\t\n\tmodule.exports = function (parser) {\n\t\n\t var editor = codex.editor;\n\t\n\t /** inserting text */\n\t parser.insertPastedContent = function (blockType, tag) {\n\t\n\t editor.content.insertBlock({\n\t type: blockType.type,\n\t block: blockType.render({\n\t text: tag.innerHTML\n\t })\n\t });\n\t };\n\t\n\t /**\r\n\t * Check DOM node for display style: separated block or child-view\r\n\t */\n\t parser.isFirstLevelBlock = function (node) {\n\t\n\t return node.nodeType == editor.core.nodeTypes.TAG && node.classList.contains(editor.ui.className.BLOCK_CLASSNAME);\n\t };\n\t\n\t return parser;\n\t}({});\n\n/***/ },\n/* 17 */\n/***/ function(module, exports, __webpack_require__) {\n\n\t'use strict';\n\t\n\t/**\r\n\t * Codex Sanitizer\r\n\t */\n\t\n\tmodule.exports = function (sanitizer) {\n\t\n\t /** HTML Janitor library */\n\t var janitor = __webpack_require__(18);\n\t\n\t /** Codex Editor */\n\t var editor = codex.editor;\n\t\n\t sanitizer.prepare = function () {\n\t\n\t if (editor.settings.sanitizer && !editor.core.isEmpty(editor.settings.sanitizer)) {\n\t\n\t Config.CUSTOM = editor.settings.sanitizer;\n\t }\n\t };\n\t\n\t /**\r\n\t * Basic config\r\n\t */\n\t var Config = {\n\t\n\t /** User configuration */\n\t CUSTOM: null,\n\t\n\t BASIC: {\n\t\n\t tags: {\n\t p: {},\n\t a: {\n\t href: true,\n\t target: '_blank',\n\t rel: 'nofollow'\n\t },\n\t i: {},\n\t b: {},\n\t strong: {},\n\t em: {},\n\t span: {}\n\t }\n\t }\n\t };\n\t\n\t sanitizer.Config = Config;\n\t\n\t /**\r\n\t *\r\n\t * @param userCustomConfig\r\n\t * @returns {*}\r\n\t * @private\r\n\t *\r\n\t * @description If developer uses editor's API, then he can customize sane restrictions.\r\n\t * Or, sane config can be defined globally in editors initialization. That config will be used everywhere\r\n\t * At least, if there is no config overrides, that API uses BASIC Default configation\r\n\t */\n\t var init_ = function init_(userCustomConfig) {\n\t\n\t var configuration = userCustomConfig || Config.CUSTOM || Config.BASIC;\n\t\n\t return new janitor(configuration);\n\t };\n\t\n\t /**\r\n\t * Cleans string from unwanted tags\r\n\t * @protected\r\n\t * @param {String} dirtyString - taint string\r\n\t * @param {Object} customConfig - allowed tags\r\n\t */\n\t sanitizer.clean = function (dirtyString, customConfig) {\n\t\n\t var janitorInstance = init_(customConfig);\n\t\n\t return janitorInstance.clean(dirtyString);\n\t };\n\t\n\t return sanitizer;\n\t}({});\n\n/***/ },\n/* 18 */\n/***/ function(module, exports, __webpack_require__) {\n\n\tvar __WEBPACK_AMD_DEFINE_FACTORY__, __WEBPACK_AMD_DEFINE_RESULT__;(function (root, factory) {\n\t if (true) {\n\t !(__WEBPACK_AMD_DEFINE_FACTORY__ = (factory), __WEBPACK_AMD_DEFINE_RESULT__ = (typeof __WEBPACK_AMD_DEFINE_FACTORY__ === 'function' ? (__WEBPACK_AMD_DEFINE_FACTORY__.call(exports, __webpack_require__, exports, module)) : __WEBPACK_AMD_DEFINE_FACTORY__), __WEBPACK_AMD_DEFINE_RESULT__ !== undefined && (module.exports = __WEBPACK_AMD_DEFINE_RESULT__));\n\t } else if (typeof exports === 'object') {\n\t module.exports = factory();\n\t } else {\n\t root.HTMLJanitor = factory();\n\t }\n\t}(this, function () {\n\t\n\t /**\n\t * @param {Object} config.tags Dictionary of allowed tags.\n\t * @param {boolean} config.keepNestedBlockElements Default false.\n\t */\n\t function HTMLJanitor(config) {\n\t\n\t var tagDefinitions = config['tags'];\n\t var tags = Object.keys(tagDefinitions);\n\t\n\t var validConfigValues = tags\n\t .map(function(k) { return typeof tagDefinitions[k]; })\n\t .every(function(type) { return type === 'object' || type === 'boolean' || type === 'function'; });\n\t\n\t if(!validConfigValues) {\n\t throw new Error(\"The configuration was invalid\");\n\t }\n\t\n\t this.config = config;\n\t }\n\t\n\t // TODO: not exhaustive?\n\t var blockElementNames = ['P', 'LI', 'TD', 'TH', 'DIV', 'H1', 'H2', 'H3', 'H4', 'H5', 'H6', 'PRE'];\n\t function isBlockElement(node) {\n\t return blockElementNames.indexOf(node.nodeName) !== -1;\n\t }\n\t\n\t var inlineElementNames = ['A', 'B', 'STRONG', 'I', 'EM', 'SUB', 'SUP', 'U', 'STRIKE'];\n\t function isInlineElement(node) {\n\t return inlineElementNames.indexOf(node.nodeName) !== -1;\n\t }\n\t\n\t HTMLJanitor.prototype.clean = function (html) {\n\t var sandbox = document.createElement('div');\n\t sandbox.innerHTML = html;\n\t\n\t this._sanitize(sandbox);\n\t\n\t return sandbox.innerHTML;\n\t };\n\t\n\t HTMLJanitor.prototype._sanitize = function (parentNode) {\n\t var treeWalker = createTreeWalker(parentNode);\n\t var node = treeWalker.firstChild();\n\t if (!node) { return; }\n\t\n\t do {\n\t // Ignore nodes that have already been sanitized\n\t if (node._sanitized) {\n\t continue;\n\t }\n\t\n\t if (node.nodeType === Node.TEXT_NODE) {\n\t // If this text node is just whitespace and the previous or next element\n\t // sibling is a block element, remove it\n\t // N.B.: This heuristic could change. Very specific to a bug with\n\t // `contenteditable` in Firefox: http://jsbin.com/EyuKase/1/edit?js,output\n\t // FIXME: make this an option?\n\t if (node.data.trim() === ''\n\t && ((node.previousElementSibling && isBlockElement(node.previousElementSibling))\n\t || (node.nextElementSibling && isBlockElement(node.nextElementSibling)))) {\n\t parentNode.removeChild(node);\n\t this._sanitize(parentNode);\n\t break;\n\t } else {\n\t continue;\n\t }\n\t }\n\t\n\t // Remove all comments\n\t if (node.nodeType === Node.COMMENT_NODE) {\n\t parentNode.removeChild(node);\n\t this._sanitize(parentNode);\n\t break;\n\t }\n\t\n\t var isInline = isInlineElement(node);\n\t var containsBlockElement;\n\t if (isInline) {\n\t containsBlockElement = Array.prototype.some.call(node.childNodes, isBlockElement);\n\t }\n\t\n\t // Block elements should not be nested (e.g.

  • ...); if\n\t // they are, we want to unwrap the inner block element.\n\t var isNotTopContainer = !! parentNode.parentNode;\n\t var isNestedBlockElement =\n\t isBlockElement(parentNode) &&\n\t isBlockElement(node) &&\n\t isNotTopContainer;\n\t\n\t var nodeName = node.nodeName.toLowerCase();\n\t\n\t var allowedAttrs = getAllowedAttrs(this.config, nodeName, node);\n\t\n\t var isInvalid = isInline && containsBlockElement;\n\t\n\t // Drop tag entirely according to the whitelist *and* if the markup\n\t // is invalid.\n\t if (isInvalid || shouldRejectNode(node, allowedAttrs)\n\t || (!this.config.keepNestedBlockElements && isNestedBlockElement)) {\n\t // Do not keep the inner text of SCRIPT/STYLE elements.\n\t if (! (node.nodeName === 'SCRIPT' || node.nodeName === 'STYLE')) {\n\t while (node.childNodes.length > 0) {\n\t parentNode.insertBefore(node.childNodes[0], node);\n\t }\n\t }\n\t parentNode.removeChild(node);\n\t\n\t this._sanitize(parentNode);\n\t break;\n\t }\n\t\n\t // Sanitize attributes\n\t for (var a = 0; a < node.attributes.length; a += 1) {\n\t var attr = node.attributes[a];\n\t\n\t if (shouldRejectAttr(attr, allowedAttrs, node)) {\n\t node.removeAttribute(attr.name);\n\t // Shift the array to continue looping.\n\t a = a - 1;\n\t }\n\t }\n\t\n\t // Sanitize children\n\t this._sanitize(node);\n\t\n\t // Mark node as sanitized so it's ignored in future runs\n\t node._sanitized = true;\n\t } while ((node = treeWalker.nextSibling()));\n\t };\n\t\n\t function createTreeWalker(node) {\n\t return document.createTreeWalker(node,\n\t NodeFilter.SHOW_TEXT | NodeFilter.SHOW_ELEMENT | NodeFilter.SHOW_COMMENT,\n\t null, false);\n\t }\n\t\n\t function getAllowedAttrs(config, nodeName, node){\n\t if (typeof config.tags[nodeName] === 'function') {\n\t return config.tags[nodeName](node);\n\t } else {\n\t return config.tags[nodeName];\n\t }\n\t }\n\t\n\t function shouldRejectNode(node, allowedAttrs){\n\t if (typeof allowedAttrs === 'undefined') {\n\t return true;\n\t } else if (typeof allowedAttrs === 'boolean') {\n\t return !allowedAttrs;\n\t }\n\t\n\t return false;\n\t }\n\t\n\t function shouldRejectAttr(attr, allowedAttrs, node){\n\t var attrName = attr.name.toLowerCase();\n\t\n\t if (allowedAttrs === true){\n\t return false;\n\t } else if (typeof allowedAttrs[attrName] === 'function'){\n\t return !allowedAttrs[attrName](attr.value, node);\n\t } else if (typeof allowedAttrs[attrName] === 'undefined'){\n\t return true;\n\t } else if (allowedAttrs[attrName] === false) {\n\t return true;\n\t } else if (typeof allowedAttrs[attrName] === 'string') {\n\t return (allowedAttrs[attrName] !== attr.value);\n\t }\n\t\n\t return false;\n\t }\n\t\n\t return HTMLJanitor;\n\t\n\t}));\n\n\n/***/ },\n/* 19 */\n/***/ function(module, exports) {\n\n\t\"use strict\";\n\t\n\t/**\r\n\t * Codex Editor Listeners module\r\n\t *\r\n\t * @author Codex Team\r\n\t * @version 1.0\r\n\t */\n\t\n\t/**\r\n\t * Module-decorator for event listeners assignment\r\n\t */\n\tmodule.exports = function (listeners) {\n\t\n\t var allListeners = [];\n\t\n\t /**\r\n\t * Search methods\r\n\t *\r\n\t * byElement, byType and byHandler returns array of suitable listeners\r\n\t * one and all takes element, eventType, and handler and returns first (all) suitable listener\r\n\t *\r\n\t */\n\t listeners.search = function () {\n\t\n\t var byElement = function byElement(element, context) {\n\t\n\t var listenersOnElement = [];\n\t\n\t context = context || allListeners;\n\t\n\t for (var i = 0; i < context.length; i++) {\n\t\n\t var listener = context[i];\n\t\n\t if (listener.element === element) {\n\t\n\t listenersOnElement.push(listener);\n\t }\n\t }\n\t\n\t return listenersOnElement;\n\t };\n\t\n\t var byType = function byType(eventType, context) {\n\t\n\t var listenersWithType = [];\n\t\n\t context = context || allListeners;\n\t\n\t for (var i = 0; i < context.length; i++) {\n\t\n\t var listener = context[i];\n\t\n\t if (listener.type === eventType) {\n\t\n\t listenersWithType.push(listener);\n\t }\n\t }\n\t\n\t return listenersWithType;\n\t };\n\t\n\t var byHandler = function byHandler(handler, context) {\n\t\n\t var listenersWithHandler = [];\n\t\n\t context = context || allListeners;\n\t\n\t for (var i = 0; i < context.length; i++) {\n\t\n\t var listener = context[i];\n\t\n\t if (listener.handler === handler) {\n\t\n\t listenersWithHandler.push(listener);\n\t }\n\t }\n\t\n\t return listenersWithHandler;\n\t };\n\t\n\t var one = function one(element, eventType, handler) {\n\t\n\t var result = allListeners;\n\t\n\t if (element) result = byElement(element, result);\n\t\n\t if (eventType) result = byType(eventType, result);\n\t\n\t if (handler) result = byHandler(handler, result);\n\t\n\t return result[0];\n\t };\n\t\n\t var all = function all(element, eventType, handler) {\n\t\n\t var result = allListeners;\n\t\n\t if (element) result = byElement(element, result);\n\t\n\t if (eventType) result = byType(eventType, result);\n\t\n\t if (handler) result = byHandler(handler, result);\n\t\n\t return result;\n\t };\n\t\n\t return {\n\t byElement: byElement,\n\t byType: byType,\n\t byHandler: byHandler,\n\t one: one,\n\t all: all\n\t };\n\t }();\n\t\n\t listeners.add = function (element, eventType, handler, isCapture) {\n\t\n\t element.addEventListener(eventType, handler, isCapture);\n\t\n\t var data = {\n\t element: element,\n\t type: eventType,\n\t handler: handler\n\t };\n\t\n\t var alreadyAddedListener = listeners.search.one(element, eventType, handler);\n\t\n\t if (!alreadyAddedListener) {\n\t\n\t allListeners.push(data);\n\t }\n\t };\n\t\n\t listeners.remove = function (element, eventType, handler) {\n\t\n\t element.removeEventListener(eventType, handler);\n\t\n\t var existingListeners = listeners.search.all(element, eventType, handler);\n\t\n\t for (var i = 0; i < existingListeners.length; i++) {\n\t\n\t var index = allListeners.indexOf(existingListeners[i]);\n\t\n\t if (index > 0) {\n\t\n\t allListeners.splice(index, 1);\n\t }\n\t }\n\t };\n\t\n\t listeners.removeAll = function () {\n\t\n\t allListeners.map(function (current) {\n\t\n\t listeners.remove(current.element, current.type, current.handler);\n\t });\n\t };\n\t\n\t listeners.get = function (element, eventType, handler) {\n\t\n\t return listeners.search.all(element, eventType, handler);\n\t };\n\t\n\t return listeners;\n\t}({});\n\n/***/ },\n/* 20 */\n/***/ function(module, exports) {\n\n\t'use strict';\n\t\n\tvar _typeof = typeof Symbol === \"function\" && typeof Symbol.iterator === \"symbol\" ? function (obj) { return typeof obj; } : function (obj) { return obj && typeof Symbol === \"function\" && obj.constructor === Symbol && obj !== Symbol.prototype ? \"symbol\" : typeof obj; };\n\t\n\t/**\r\n\t * Codex Editor Destroyer module\r\n\t *\r\n\t * @auhor Codex Team\r\n\t * @version 1.0\r\n\t */\n\t\n\tmodule.exports = function (destroyer) {\n\t\n\t var editor = codex.editor;\n\t\n\t destroyer.removeNodes = function () {\n\t\n\t editor.nodes.wrapper.remove();\n\t editor.nodes.notifications.remove();\n\t };\n\t\n\t destroyer.destroyPlugins = function () {\n\t\n\t for (var tool in editor.tools) {\n\t\n\t if (typeof editor.tools[tool].destroy === 'function') {\n\t\n\t editor.tools[tool].destroy();\n\t }\n\t }\n\t };\n\t\n\t destroyer.destroyScripts = function () {\n\t\n\t var scripts = document.getElementsByTagName('SCRIPT');\n\t\n\t for (var i = 0; i < scripts.length; i++) {\n\t\n\t if (scripts[i].id.indexOf(editor.scriptPrefix) + 1) {\n\t\n\t scripts[i].remove();\n\t i--;\n\t }\n\t }\n\t };\n\t\n\t /**\r\n\t * Delete editor data from webpage.\r\n\t * You should send settings argument with boolean flags:\r\n\t * @param settings.ui- remove redactor event listeners and DOM nodes\r\n\t * @param settings.scripts - remove redactor scripts from DOM\r\n\t * @param settings.plugins - remove plugin's objects\r\n\t * @param settings.core - remove editor core. You can remove core only if UI and scripts flags is true\r\n\t * }\r\n\t *\r\n\t */\n\t destroyer.destroy = function (settings) {\n\t\n\t if (!settings || (typeof settings === 'undefined' ? 'undefined' : _typeof(settings)) !== 'object') {\n\t\n\t return;\n\t }\n\t\n\t if (settings.ui) {\n\t\n\t destroyer.removeNodes();\n\t editor.listeners.removeAll();\n\t }\n\t\n\t if (settings.scripts) {\n\t\n\t destroyer.destroyScripts();\n\t }\n\t\n\t if (settings.plugins) {\n\t\n\t destroyer.destroyPlugins();\n\t }\n\t\n\t if (settings.ui && settings.scripts && settings.core) {\n\t\n\t delete codex.editor;\n\t }\n\t };\n\t\n\t return destroyer;\n\t}({});\n\n/***/ },\n/* 21 */\n/***/ function(module, exports) {\n\n\t'use strict';\n\t\n\t/**\r\n\t * Codex Editor Paste module\r\n\t *\r\n\t * @author Codex Team\r\n\t * @version 1.1.1\r\n\t */\n\t\n\tmodule.exports = function (paste) {\n\t\n\t var editor = codex.editor;\n\t\n\t var patterns = [];\n\t\n\t paste.prepare = function () {\n\t\n\t var tools = editor.tools;\n\t\n\t for (var tool in tools) {\n\t\n\t if (!tools[tool].renderOnPastePatterns || !Array.isArray(tools[tool].renderOnPastePatterns)) {\n\t\n\t continue;\n\t }\n\t\n\t tools[tool].renderOnPastePatterns.map(function (pattern) {\n\t\n\t patterns.push(pattern);\n\t });\n\t }\n\t\n\t return Promise.resolve();\n\t };\n\t\n\t /**\r\n\t * Saves data\r\n\t * @param event\r\n\t */\n\t paste.pasted = function (event) {\n\t\n\t var clipBoardData = event.clipboardData || window.clipboardData,\n\t content = clipBoardData.getData('Text');\n\t\n\t var result = analize(content);\n\t\n\t if (result) {\n\t\n\t event.preventDefault();\n\t event.stopImmediatePropagation();\n\t }\n\t\n\t return result;\n\t };\n\t\n\t /**\r\n\t * Analizes pated string and calls necessary method\r\n\t */\n\t\n\t var analize = function analize(string) {\n\t\n\t var result = false,\n\t content = editor.content.currentNode,\n\t plugin = content.dataset.tool;\n\t\n\t patterns.map(function (pattern) {\n\t\n\t var execArray = pattern.regex.exec(string),\n\t match = execArray && execArray[0];\n\t\n\t if (match && match === string.trim()) {\n\t\n\t /** current block is not empty */\n\t if (content.textContent.trim() && plugin == editor.settings.initialBlockPlugin) {\n\t\n\t pasteToNewBlock_();\n\t }\n\t\n\t pattern.callback(string, pattern);\n\t result = true;\n\t }\n\t });\n\t\n\t return result;\n\t };\n\t\n\t var pasteToNewBlock_ = function pasteToNewBlock_() {\n\t\n\t /** Create new initial block */\n\t editor.content.insertBlock({\n\t\n\t type: editor.settings.initialBlockPlugin,\n\t block: editor.tools[editor.settings.initialBlockPlugin].render({\n\t text: ''\n\t })\n\t\n\t }, false);\n\t };\n\t\n\t /**\r\n\t * This method prevents default behaviour.\r\n\t *\r\n\t * @param {Object} event\r\n\t * @protected\r\n\t *\r\n\t * @description We get from clipboard pasted data, sanitize, make a fragment that contains of this sanitized nodes.\r\n\t * Firstly, we need to memorize the caret position. We can do that by getting the range of selection.\r\n\t * After all, we insert clear fragment into caret placed position. Then, we should move the caret to the last node\r\n\t */\n\t paste.blockPasteCallback = function (event) {\n\t\n\t if (!needsToHandlePasteEvent(event.target)) {\n\t\n\t return;\n\t }\n\t\n\t /** get html pasted data - dirty data */\n\t var htmlData = event.clipboardData.getData('text/html'),\n\t plainData = event.clipboardData.getData('text/plain');\n\t\n\t /** Temporary DIV that is used to work with text's paragraphs as DOM-elements*/\n\t var paragraphs = editor.draw.node('DIV', '', {}),\n\t cleanData,\n\t wrappedData;\n\t\n\t /** Create fragment, that we paste to range after proccesing */\n\t cleanData = editor.sanitizer.clean(htmlData);\n\t\n\t /**\r\n\t * We wrap pasted text with

    tags to split it logically\r\n\t * @type {string}\r\n\t */\n\t wrappedData = editor.content.wrapTextWithParagraphs(cleanData, plainData);\n\t paragraphs.innerHTML = wrappedData;\n\t\n\t /**\r\n\t * If there only one paragraph, lets user agent paste it\r\n\t */\n\t if (paragraphs.childNodes.length == 1) {\n\t\n\t return;\n\t }\n\t\n\t /** Prevent default behaviour */\n\t event.preventDefault();\n\t\n\t insertPastedParagraphs(paragraphs.childNodes);\n\t };\n\t\n\t /**\r\n\t * Checks if we should handle paste event on block\r\n\t * @param block\r\n\t *\r\n\t * @return {boolean}\r\n\t */\n\t var needsToHandlePasteEvent = function needsToHandlePasteEvent(block) {\n\t\n\t /** If area is input or textarea then allow default behaviour */\n\t if (editor.core.isNativeInput(block)) {\n\t\n\t return false;\n\t }\n\t\n\t var editableParent = editor.content.getEditableParent(block);\n\t\n\t /** Allow paste when event target placed in Editable element */\n\t if (!editableParent) {\n\t\n\t return false;\n\t }\n\t\n\t return true;\n\t };\n\t\n\t var insertPastedParagraphs = function insertPastedParagraphs(paragraphs) {\n\t\n\t var NEW_BLOCK_TYPE = editor.settings.initialBlockPlugin,\n\t currentBlockContent = editor.content.currentNode.firstChild.firstChild;\n\t\n\t paragraphs.forEach(function (paragraph, index) {\n\t\n\t /**\r\n\t * If there was no data in working node, replace it with first paragraph of pasted text\r\n\t */\n\t if (index == 0 && currentBlockContent.innerHTML.trim() === '') {\n\t\n\t editor.content.switchBlock(editor.content.currentNode, editor.tools[NEW_BLOCK_TYPE].render({\n\t text: paragraph.innerHTML\n\t }), NEW_BLOCK_TYPE);\n\t\n\t return;\n\t }\n\t\n\t editor.content.insertBlock({\n\t type: NEW_BLOCK_TYPE,\n\t block: editor.tools[NEW_BLOCK_TYPE].render({\n\t text: paragraph.innerHTML\n\t })\n\t });\n\t\n\t editor.caret.inputIndex++;\n\t });\n\t\n\t editor.caret.setToPreviousBlock(editor.caret.getCurrentInputIndex() + 1);\n\t };\n\t\n\t return paste;\n\t}({});\n\n/***/ }\n/******/ ]);\n\n\n// WEBPACK FOOTER //\n// codex-editor.js"," \t// The module cache\n \tvar installedModules = {};\n\n \t// The require function\n \tfunction __webpack_require__(moduleId) {\n\n \t\t// Check if module is in cache\n \t\tif(installedModules[moduleId])\n \t\t\treturn installedModules[moduleId].exports;\n\n \t\t// Create a new module (and put it into the cache)\n \t\tvar module = installedModules[moduleId] = {\n \t\t\texports: {},\n \t\t\tid: moduleId,\n \t\t\tloaded: false\n \t\t};\n\n \t\t// Execute the module function\n \t\tmodules[moduleId].call(module.exports, module, module.exports, __webpack_require__);\n\n \t\t// Flag the module as loaded\n \t\tmodule.loaded = true;\n\n \t\t// Return the exports of the module\n \t\treturn module.exports;\n \t}\n\n\n \t// expose the modules object (__webpack_modules__)\n \t__webpack_require__.m = modules;\n\n \t// expose the module cache\n \t__webpack_require__.c = installedModules;\n\n \t// __webpack_public_path__\n \t__webpack_require__.p = \"\";\n\n \t// Load entry module and return exports\n \treturn __webpack_require__(0);\n\n\n\n// WEBPACK FOOTER //\n// webpack/bootstrap ad1af9a0031475a14714","/**\r\n *\r\n * Codex Editor\r\n *\r\n * @author Codex Team\r\n */\r\n\r\nmodule.exports = (function (editor) {\r\n\r\n 'use strict';\r\n\r\n editor.version = VERSION;\r\n editor.scriptPrefix = 'cdx-script-';\r\n\r\n var init = function () {\r\n\r\n editor.core = require('./modules/core');\r\n editor.tools = require('./modules/tools');\r\n editor.ui = require('./modules/ui');\r\n editor.transport = require('./modules/transport');\r\n editor.renderer = require('./modules/renderer');\r\n editor.saver = require('./modules/saver');\r\n editor.content = require('./modules/content');\r\n editor.toolbar = require('./modules/toolbar/toolbar');\r\n editor.callback = require('./modules/callbacks');\r\n editor.draw = require('./modules/draw');\r\n editor.caret = require('./modules/caret');\r\n editor.notifications = require('./modules/notifications');\r\n editor.parser = require('./modules/parser');\r\n editor.sanitizer = require('./modules/sanitizer');\r\n editor.listeners = require('./modules/listeners');\r\n editor.destroyer = require('./modules/destroyer');\r\n editor.paste = require('./modules/paste');\r\n\r\n };\r\n\r\n /**\r\n * @public\r\n * holds initial settings\r\n */\r\n editor.settings = {\r\n tools : ['paragraph', 'header', 'picture', 'list', 'quote', 'code', 'twitter', 'instagram', 'smile'],\r\n holderId : 'codex-editor',\r\n\r\n // Type of block showing on empty editor\r\n initialBlockPlugin: 'paragraph'\r\n };\r\n\r\n /**\r\n * public\r\n *\r\n * Static nodes\r\n */\r\n editor.nodes = {\r\n holder : null,\r\n wrapper : null,\r\n toolbar : null,\r\n inlineToolbar : {\r\n wrapper : null,\r\n buttons : null,\r\n actions : null\r\n },\r\n toolbox : null,\r\n notifications : null,\r\n plusButton : null,\r\n showSettingsButton: null,\r\n showTrashButton : null,\r\n blockSettings : null,\r\n pluginSettings : null,\r\n defaultSettings : null,\r\n toolbarButtons : {}, // { type : DomEl, ... }\r\n redactor : null\r\n };\r\n\r\n /**\r\n * @public\r\n *\r\n * Output state\r\n */\r\n editor.state = {\r\n jsonOutput : [],\r\n blocks : [],\r\n inputs : []\r\n };\r\n\r\n /**\r\n * @public\r\n * Editor plugins\r\n */\r\n editor.tools = {};\r\n\r\n /**\r\n * Initialization\r\n * @uses Promise cEditor.core.prepare\r\n * @param {Object} userSettings\r\n * @param {Array} userSettings.tools list of plugins\r\n * @param {String} userSettings.holderId Element's id to append editor\r\n *\r\n * Load user defined tools\r\n * Tools must contain this important objects :\r\n * @param {String} type - this is a type of plugin. It can be used as plugin name\r\n * @param {String} iconClassname - this a icon in toolbar\r\n * @param {Object} make - what should plugin do, when it is clicked\r\n * @param {Object} appendCallback - callback after clicking\r\n * @param {Element} settings - what settings does it have\r\n * @param {Object} render - plugin get JSON, and should return HTML\r\n * @param {Object} save - plugin gets HTML content, returns JSON\r\n * @param {Boolean} displayInToolbox - will be displayed in toolbox. Default value is TRUE\r\n * @param {Boolean} enableLineBreaks - inserts new block or break lines. Default value is FALSE\r\n *\r\n * @example\r\n * - type : 'header',\r\n * - iconClassname : 'ce-icon-header',\r\n * - make : headerTool.make,\r\n * - appendCallback : headerTool.appendCallback,\r\n * - settings : headerTool.makeSettings(),\r\n * - render : headerTool.render,\r\n * - save : headerTool.save,\r\n * - displayInToolbox : true,\r\n * - enableLineBreaks : false\r\n */\r\n editor.start = function (userSettings) {\r\n\r\n init();\r\n\r\n editor.core.prepare(userSettings)\r\n\r\n // If all ok, make UI, bind events and parse initial-content\r\n .then(editor.ui.prepare)\r\n .then(editor.tools.prepare)\r\n .then(editor.sanitizer.prepare)\r\n .then(editor.paste.prepare)\r\n .then(editor.transport.prepare)\r\n .then(editor.renderer.makeBlocksFromData)\r\n .then(editor.ui.saveInputs)\r\n .catch(function (error) {\r\n\r\n editor.core.log('Initialization failed with error: %o', 'warn', error);\r\n\r\n });\r\n\r\n };\r\n\r\n return editor;\r\n\r\n})({});\r\n\n\n\n// WEBPACK FOOTER //\n// ./codex.js","/**\r\n * Codex Editor Core\r\n *\r\n * @author Codex Team\r\n * @version 1.1.3\r\n */\r\n\r\nmodule.exports = (function (core) {\r\n\r\n let editor = codex.editor;\r\n\r\n /**\r\n * @public\r\n *\r\n * Editor preparing method\r\n * @return Promise\r\n */\r\n core.prepare = function (userSettings) {\r\n\r\n return new Promise(function (resolve, reject) {\r\n\r\n if ( userSettings ) {\r\n\r\n editor.settings.tools = userSettings.tools || editor.settings.tools;\r\n\r\n }\r\n\r\n if (userSettings.data) {\r\n\r\n editor.state.blocks = userSettings.data;\r\n\r\n }\r\n\r\n if (userSettings.initialBlockPlugin) {\r\n\r\n editor.settings.initialBlockPlugin = userSettings.initialBlockPlugin;\r\n\r\n }\r\n\r\n if (userSettings.sanitizer) {\r\n\r\n editor.settings.sanitizer = userSettings.sanitizer;\r\n\r\n }\r\n\r\n editor.hideToolbar = userSettings.hideToolbar;\r\n\r\n editor.nodes.holder = document.getElementById(userSettings.holderId || editor.settings.holderId);\r\n\r\n if (typeof editor.nodes.holder === undefined || editor.nodes.holder === null) {\r\n\r\n reject(Error(\"Holder wasn't found by ID: #\" + userSettings.holderId));\r\n\r\n } else {\r\n\r\n resolve();\r\n\r\n }\r\n\r\n });\r\n\r\n };\r\n\r\n /**\r\n * Logging method\r\n * @param type = ['log', 'info', 'warn']\r\n */\r\n core.log = function (msg, type, arg) {\r\n\r\n type = type || 'log';\r\n\r\n if (!arg) {\r\n\r\n arg = msg || 'undefined';\r\n msg = '[codex-editor]: %o';\r\n\r\n } else {\r\n\r\n msg = '[codex-editor]: ' + msg;\r\n\r\n }\r\n\r\n try{\r\n\r\n if ( 'console' in window && window.console[ type ] ) {\r\n\r\n if ( arg ) window.console[ type ]( msg, arg );\r\n else window.console[ type ]( msg );\r\n\r\n }\r\n\r\n }catch(e) {}\r\n\r\n };\r\n\r\n /**\r\n * @protected\r\n *\r\n * Helper for insert one element after another\r\n */\r\n core.insertAfter = function (target, element) {\r\n\r\n target.parentNode.insertBefore(element, target.nextSibling);\r\n\r\n };\r\n\r\n /**\r\n * @const\r\n *\r\n * Readable DOM-node types map\r\n */\r\n core.nodeTypes = {\r\n TAG : 1,\r\n TEXT : 3,\r\n COMMENT : 8\r\n };\r\n\r\n /**\r\n * @const\r\n * Readable keys map\r\n */\r\n core.keys = { BACKSPACE: 8, TAB: 9, ENTER: 13, SHIFT: 16, CTRL: 17, ALT: 18, ESC: 27, SPACE: 32, LEFT: 37, UP: 38, DOWN: 40, RIGHT: 39, DELETE: 46, META: 91 };\r\n\r\n /**\r\n * @protected\r\n *\r\n * Check object for DOM node\r\n */\r\n core.isDomNode = function (el) {\r\n\r\n return el && typeof el === 'object' && el.nodeType && el.nodeType == this.nodeTypes.TAG;\r\n\r\n };\r\n\r\n /**\r\n * Checks passed object for emptiness\r\n * @require ES5 - Object.keys\r\n * @param {object}\r\n */\r\n core.isEmpty = function ( obj ) {\r\n\r\n return Object.keys(obj).length === 0;\r\n\r\n };\r\n\r\n /**\r\n * Native Ajax\r\n * @param {String} settings.url - request URL\r\n * @param {function} settings.beforeSend - returned value will be passed as context to the Success, Error and Progress callbacks\r\n * @param {function} settings.success\r\n * @param {function} settings.progress\r\n */\r\n core.ajax = function (settings) {\r\n\r\n if (!settings || !settings.url) {\r\n\r\n return;\r\n\r\n }\r\n\r\n var XMLHTTP = window.XMLHttpRequest ? new XMLHttpRequest() : new ActiveXObject('Microsoft.XMLHTTP'),\r\n encodedString,\r\n isFormData,\r\n prop;\r\n\r\n\r\n settings.async = true;\r\n settings.type = settings.type || 'GET';\r\n settings.data = settings.data || '';\r\n settings['content-type'] = settings['content-type'] || 'application/json; charset=utf-8';\r\n\r\n if (settings.type == 'GET' && settings.data) {\r\n\r\n settings.url = /\\?/.test(settings.url) ? settings.url + '&' + settings.data : settings.url + '?' + settings.data;\r\n\r\n } else {\r\n\r\n encodedString = '';\r\n for(prop in settings.data) {\r\n\r\n encodedString += (prop + '=' + encodeURIComponent(settings.data[prop]) + '&');\r\n\r\n }\r\n\r\n }\r\n\r\n if (settings.withCredentials) {\r\n\r\n XMLHTTP.withCredentials = true;\r\n\r\n }\r\n\r\n /**\r\n * Value returned in beforeSend funtion will be passed as context to the other response callbacks\r\n * If beforeSend returns false, AJAX will be blocked\r\n */\r\n let responseContext,\r\n beforeSendResult;\r\n\r\n if (typeof settings.beforeSend === 'function') {\r\n\r\n beforeSendResult = settings.beforeSend.call();\r\n\r\n if (beforeSendResult === false) {\r\n\r\n return;\r\n\r\n }\r\n\r\n }\r\n\r\n XMLHTTP.open( settings.type, settings.url, settings.async );\r\n\r\n /**\r\n * If we send FormData, we need no content-type header\r\n */\r\n isFormData = isFormData_(settings.data);\r\n\r\n if (!isFormData) {\r\n\r\n if (settings.type !== 'POST') {\r\n\r\n XMLHTTP.setRequestHeader('Content-type', settings['content-type']);\r\n\r\n } else {\r\n\r\n XMLHTTP.setRequestHeader('Content-type', 'application/x-www-form-urlencoded');\r\n\r\n }\r\n\r\n }\r\n\r\n XMLHTTP.setRequestHeader('X-Requested-With', 'XMLHttpRequest');\r\n\r\n responseContext = beforeSendResult || XMLHTTP;\r\n\r\n if (typeof settings.progress === 'function') {\r\n\r\n XMLHTTP.upload.onprogress = settings.progress.bind(responseContext);\r\n\r\n }\r\n\r\n XMLHTTP.onreadystatechange = function () {\r\n\r\n if (XMLHTTP.readyState === 4) {\r\n\r\n if (XMLHTTP.status === 200) {\r\n\r\n if (typeof settings.success === 'function') {\r\n\r\n settings.success.call(responseContext, XMLHTTP.responseText);\r\n\r\n }\r\n\r\n } else {\r\n\r\n if (typeof settings.error === 'function') {\r\n\r\n settings.error.call(responseContext, XMLHTTP.responseText, XMLHTTP.status);\r\n\r\n }\r\n\r\n }\r\n\r\n }\r\n\r\n };\r\n\r\n if (isFormData) {\r\n\r\n // Sending FormData\r\n XMLHTTP.send(settings.data);\r\n\r\n } else {\r\n\r\n // POST requests\r\n XMLHTTP.send(encodedString);\r\n\r\n }\r\n\r\n return XMLHTTP;\r\n\r\n };\r\n\r\n /**\r\n * Appends script to head of document\r\n * @return Promise\r\n */\r\n core.importScript = function (scriptPath, instanceName) {\r\n\r\n return new Promise(function (resolve, reject) {\r\n\r\n let script;\r\n\r\n /** Script is already loaded */\r\n if ( !instanceName ) {\r\n\r\n reject('Instance name is missed');\r\n\r\n } else if ( document.getElementById(editor.scriptPrefix + instanceName) ) {\r\n\r\n resolve(scriptPath);\r\n\r\n }\r\n\r\n script = document.createElement('SCRIPT');\r\n script.async = true;\r\n script.defer = true;\r\n script.id = editor.scriptPrefix + instanceName;\r\n\r\n script.onload = function () {\r\n\r\n resolve(scriptPath);\r\n\r\n };\r\n\r\n script.onerror = function () {\r\n\r\n reject(scriptPath);\r\n\r\n };\r\n\r\n script.src = scriptPath;\r\n document.head.appendChild(script);\r\n\r\n });\r\n\r\n };\r\n\r\n /**\r\n * Function for checking is it FormData object to send.\r\n * @param {Object} object to check\r\n * @return boolean\r\n */\r\n var isFormData_ = function (object) {\r\n\r\n return object instanceof FormData;\r\n\r\n };\r\n\r\n /**\r\n * Check block\r\n * @param target\r\n * @description Checks target is it native input\r\n */\r\n core.isNativeInput = function (target) {\r\n\r\n var nativeInputAreas = ['INPUT', 'TEXTAREA'];\r\n\r\n return (nativeInputAreas.indexOf(target.tagName) != -1);\r\n\r\n };\r\n\r\n\r\n return core;\r\n\r\n})({});\r\n\n\n\n// WEBPACK FOOTER //\n// ./modules/core.js","/**\r\n* Module working with plugins\r\n*/\r\nmodule.exports = (function () {\r\n\r\n let editor = codex.editor;\r\n\r\n /**\r\n * Initialize plugins before using\r\n * Ex. Load scripts or call some internal methods\r\n * @return Promise\r\n */\r\n function prepare() {\r\n\r\n return new Promise(function (resolve_, reject_) {\r\n\r\n Promise.resolve()\r\n\r\n /**\r\n * Compose a sequence of plugins that requires preparation\r\n */\r\n .then(function () {\r\n\r\n let pluginsRequiresPreparation = [],\r\n allPlugins = editor.tools;\r\n\r\n for ( let pluginName in allPlugins ) {\r\n\r\n let plugin = allPlugins[pluginName];\r\n\r\n if (plugin.prepare && typeof plugin.prepare != 'function' || !plugin.prepare) {\r\n\r\n continue;\r\n\r\n }\r\n\r\n pluginsRequiresPreparation.push(plugin);\r\n\r\n }\r\n\r\n /**\r\n * If no one passed plugins requires preparation, finish prepare() and go ahead\r\n */\r\n if (!pluginsRequiresPreparation.length) {\r\n\r\n resolve_();\r\n\r\n }\r\n\r\n return pluginsRequiresPreparation;\r\n\r\n })\r\n\r\n /** Wait plugins while they prepares */\r\n .then(waitAllPluginsPreparation_)\r\n\r\n .then(function () {\r\n\r\n editor.core.log('Plugins loaded', 'info');\r\n resolve_();\r\n\r\n }).catch(function (error) {\r\n\r\n reject_(error);\r\n\r\n });\r\n\r\n });\r\n\r\n }\r\n\r\n /**\r\n * @param {array} plugins - list of tools that requires preparation\r\n * @return {Promise} resolved while all plugins will be ready or failed\r\n */\r\n function waitAllPluginsPreparation_(plugins) {\r\n\r\n /**\r\n * @calls allPluginsProcessed__ when all plugins prepared or failed\r\n */\r\n return new Promise (function (allPluginsProcessed__) {\r\n\r\n /**\r\n * pluck each element from queue\r\n * First, send resolved Promise as previous value\r\n * Each plugins \"prepare\" method returns a Promise, that's why\r\n * reduce current element will not be able to continue while can't get\r\n * a resolved Promise\r\n *\r\n * If last plugin is \"prepared\" then go to the next stage of initialization\r\n */\r\n plugins.reduce(function (previousValue, plugin, iteration) {\r\n\r\n return previousValue.then(function () {\r\n\r\n /**\r\n * Wait till plugins prepared\r\n * @calls pluginIsReady__ when plugin is ready or failed\r\n */\r\n return new Promise ( function (pluginIsReady__) {\r\n\r\n callPluginsPrepareMethod_( plugin )\r\n\r\n .then( pluginIsReady__ )\r\n .then( function () {\r\n\r\n plugin.available = true;\r\n\r\n })\r\n\r\n .catch(function (error) {\r\n\r\n editor.core.log(`Plugin «${plugin.type}» was not loaded. Preparation failed because %o`, 'warn', error);\r\n plugin.available = false;\r\n plugin.loadingMessage = error;\r\n\r\n /** Go ahead even some plugin has problems */\r\n pluginIsReady__();\r\n\r\n })\r\n\r\n .then(function () {\r\n\r\n /** If last plugin has problems then just ignore and continue */\r\n if (iteration == plugins.length - 1) {\r\n\r\n allPluginsProcessed__();\r\n\r\n }\r\n\r\n });\r\n\r\n });\r\n\r\n });\r\n\r\n }, Promise.resolve() );\r\n\r\n });\r\n\r\n }\r\n\r\n var callPluginsPrepareMethod_ = function (plugin) {\r\n\r\n return plugin.prepare( plugin.config || {} );\r\n\r\n };\r\n\r\n return {\r\n prepare: prepare\r\n };\r\n\r\n}());\n\n\n// WEBPACK FOOTER //\n// ./modules/tools.js","/**\r\n * Codex Editor UI module\r\n *\r\n * @author Codex Team\r\n * @version 1.2.0\r\n */\r\n\r\nmodule.exports = (function (ui) {\r\n\r\n let editor = codex.editor;\r\n\r\n /**\r\n * Basic editor classnames\r\n */\r\n ui.className = {\r\n\r\n /**\r\n * @const {string} BLOCK_CLASSNAME - redactor blocks name\r\n */\r\n BLOCK_CLASSNAME : 'ce-block',\r\n\r\n /**\r\n * @const {String} wrapper for plugins content\r\n */\r\n BLOCK_CONTENT : 'ce-block__content',\r\n\r\n /**\r\n * @const {String} BLOCK_STRETCHED - makes block stretched\r\n */\r\n BLOCK_STRETCHED : 'ce-block--stretched',\r\n\r\n /**\r\n * @const {String} BLOCK_HIGHLIGHTED - adds background\r\n */\r\n BLOCK_HIGHLIGHTED : 'ce-block--focused',\r\n\r\n /**\r\n * @const {String} - for all default settings\r\n */\r\n SETTINGS_ITEM : 'ce-settings__item'\r\n\r\n };\r\n\r\n /**\r\n * @protected\r\n *\r\n * Making main interface\r\n */\r\n ui.prepare = function () {\r\n\r\n return new Promise(function (resolve) {\r\n\r\n let wrapper = editor.draw.wrapper(),\r\n redactor = editor.draw.redactor(),\r\n toolbar = makeToolBar_();\r\n\r\n wrapper.appendChild(toolbar);\r\n wrapper.appendChild(redactor);\r\n\r\n /** Save created ui-elements to static nodes state */\r\n editor.nodes.wrapper = wrapper;\r\n editor.nodes.redactor = redactor;\r\n\r\n /** Append editor wrapper with redactor zone into holder */\r\n editor.nodes.holder.appendChild(wrapper);\r\n\r\n resolve();\r\n\r\n })\r\n\r\n /** Add toolbox tools */\r\n .then(addTools_)\r\n\r\n /** Make container for inline toolbar */\r\n .then(makeInlineToolbar_)\r\n\r\n /** Add inline toolbar tools */\r\n .then(addInlineToolbarTools_)\r\n\r\n /** Draw wrapper for notifications */\r\n .then(makeNotificationHolder_)\r\n\r\n /** Add eventlisteners to redactor elements */\r\n .then(bindEvents_)\r\n\r\n .catch( function () {\r\n\r\n editor.core.log(\"Can't draw editor interface\");\r\n\r\n });\r\n\r\n };\r\n\r\n /**\r\n * @private\r\n * Draws inline toolbar zone\r\n */\r\n var makeInlineToolbar_ = function () {\r\n\r\n var container = editor.draw.inlineToolbar();\r\n\r\n /** Append to redactor new inline block */\r\n editor.nodes.inlineToolbar.wrapper = container;\r\n\r\n /** Draw toolbar buttons */\r\n editor.nodes.inlineToolbar.buttons = editor.draw.inlineToolbarButtons();\r\n\r\n /** Buttons action or settings */\r\n editor.nodes.inlineToolbar.actions = editor.draw.inlineToolbarActions();\r\n\r\n /** Append to inline toolbar buttons as part of it */\r\n editor.nodes.inlineToolbar.wrapper.appendChild(editor.nodes.inlineToolbar.buttons);\r\n editor.nodes.inlineToolbar.wrapper.appendChild(editor.nodes.inlineToolbar.actions);\r\n\r\n editor.nodes.wrapper.appendChild(editor.nodes.inlineToolbar.wrapper);\r\n\r\n };\r\n\r\n var makeToolBar_ = function () {\r\n\r\n let toolbar = editor.draw.toolbar(),\r\n blockButtons = makeToolbarSettings_(),\r\n toolbarContent = makeToolbarContent_();\r\n\r\n /** Appending first-level block buttons */\r\n toolbar.appendChild(blockButtons);\r\n\r\n /** Append toolbarContent to toolbar */\r\n toolbar.appendChild(toolbarContent);\r\n\r\n /** Make toolbar global */\r\n editor.nodes.toolbar = toolbar;\r\n\r\n return toolbar;\r\n\r\n };\r\n\r\n var makeToolbarContent_ = function () {\r\n\r\n let toolbarContent = editor.draw.toolbarContent(),\r\n toolbox = editor.draw.toolbox(),\r\n plusButton = editor.draw.plusButton();\r\n\r\n /** Append plus button */\r\n toolbarContent.appendChild(plusButton);\r\n\r\n /** Appending toolbar tools */\r\n toolbarContent.appendChild(toolbox);\r\n\r\n /** Make Toolbox and plusButton global */\r\n editor.nodes.toolbox = toolbox;\r\n editor.nodes.plusButton = plusButton;\r\n\r\n return toolbarContent;\r\n\r\n };\r\n\r\n var makeToolbarSettings_ = function () {\r\n\r\n let blockSettings = editor.draw.blockSettings(),\r\n blockButtons = editor.draw.blockButtons(),\r\n defaultSettings = editor.draw.defaultSettings(),\r\n showSettingsButton = editor.draw.settingsButton(),\r\n showTrashButton = editor.toolbar.settings.makeRemoveBlockButton(),\r\n pluginSettings = editor.draw.pluginsSettings();\r\n\r\n /** Add default and plugins settings */\r\n blockSettings.appendChild(pluginSettings);\r\n blockSettings.appendChild(defaultSettings);\r\n\r\n /**\r\n * Make blocks buttons\r\n * This block contains settings button and remove block button\r\n */\r\n blockButtons.appendChild(showSettingsButton);\r\n blockButtons.appendChild(showTrashButton);\r\n blockButtons.appendChild(blockSettings);\r\n\r\n /** Make BlockSettings, PluginSettings, DefaultSettings global */\r\n editor.nodes.blockSettings = blockSettings;\r\n editor.nodes.pluginSettings = pluginSettings;\r\n editor.nodes.defaultSettings = defaultSettings;\r\n editor.nodes.showSettingsButton = showSettingsButton;\r\n editor.nodes.showTrashButton = showTrashButton;\r\n\r\n return blockButtons;\r\n\r\n };\r\n\r\n /** Draw notifications holder */\r\n var makeNotificationHolder_ = function () {\r\n\r\n /** Append block with notifications to the document */\r\n editor.nodes.notifications = editor.notifications.createHolder();\r\n\r\n };\r\n\r\n /**\r\n * @private\r\n * Append tools passed in editor.tools\r\n */\r\n var addTools_ = function () {\r\n\r\n var tool,\r\n toolName,\r\n toolButton;\r\n\r\n for ( toolName in editor.settings.tools ) {\r\n\r\n tool = editor.settings.tools[toolName];\r\n\r\n editor.tools[toolName] = tool;\r\n\r\n if (!tool.iconClassname && tool.displayInToolbox) {\r\n\r\n editor.core.log('Toolbar icon classname missed. Tool %o skipped', 'warn', toolName);\r\n continue;\r\n\r\n }\r\n\r\n if (typeof tool.render != 'function') {\r\n\r\n editor.core.log('render method missed. Tool %o skipped', 'warn', toolName);\r\n continue;\r\n\r\n }\r\n\r\n if (!tool.displayInToolbox) {\r\n\r\n continue;\r\n\r\n } else {\r\n\r\n /** if tools is for toolbox */\r\n toolButton = editor.draw.toolbarButton(toolName, tool.iconClassname);\r\n\r\n editor.nodes.toolbox.appendChild(toolButton);\r\n\r\n editor.nodes.toolbarButtons[toolName] = toolButton;\r\n\r\n }\r\n\r\n }\r\n\r\n };\r\n\r\n var addInlineToolbarTools_ = function () {\r\n\r\n var tools = {\r\n\r\n bold: {\r\n icon : 'ce-icon-bold',\r\n command : 'bold'\r\n },\r\n\r\n italic: {\r\n icon : 'ce-icon-italic',\r\n command : 'italic'\r\n },\r\n\r\n underline: {\r\n icon : 'ce-icon-underline',\r\n command : 'underline'\r\n },\r\n\r\n link: {\r\n icon : 'ce-icon-link',\r\n command : 'createLink'\r\n }\r\n };\r\n\r\n var toolButton,\r\n tool;\r\n\r\n for(var name in tools) {\r\n\r\n tool = tools[name];\r\n\r\n toolButton = editor.draw.toolbarButtonInline(name, tool.icon);\r\n\r\n editor.nodes.inlineToolbar.buttons.appendChild(toolButton);\r\n /**\r\n * Add callbacks to this buttons\r\n */\r\n editor.ui.setInlineToolbarButtonBehaviour(toolButton, tool.command);\r\n\r\n }\r\n\r\n };\r\n\r\n /**\r\n * @private\r\n * Bind editor UI events\r\n */\r\n var bindEvents_ = function () {\r\n\r\n editor.core.log('ui.bindEvents fired', 'info');\r\n\r\n // window.addEventListener('error', function (errorMsg, url, lineNumber) {\r\n // editor.notifications.errorThrown(errorMsg, event);\r\n // }, false );\r\n\r\n /** All keydowns on Document */\r\n editor.listeners.add(document, 'keydown', editor.callback.globalKeydown, false);\r\n\r\n /** All keydowns on Redactor zone */\r\n editor.listeners.add(editor.nodes.redactor, 'keydown', editor.callback.redactorKeyDown, false);\r\n\r\n /** All keydowns on Document */\r\n editor.listeners.add(document, 'keyup', editor.callback.globalKeyup, false );\r\n\r\n /**\r\n * Mouse click to radactor\r\n */\r\n editor.listeners.add(editor.nodes.redactor, 'click', editor.callback.redactorClicked, false );\r\n\r\n /**\r\n * Clicks to the Plus button\r\n */\r\n editor.listeners.add(editor.nodes.plusButton, 'click', editor.callback.plusButtonClicked, false);\r\n\r\n /**\r\n * Clicks to SETTINGS button in toolbar\r\n */\r\n editor.listeners.add(editor.nodes.showSettingsButton, 'click', editor.callback.showSettingsButtonClicked, false );\r\n\r\n /** Bind click listeners on toolbar buttons */\r\n for (var button in editor.nodes.toolbarButtons) {\r\n\r\n editor.listeners.add(editor.nodes.toolbarButtons[button], 'click', editor.callback.toolbarButtonClicked, false);\r\n\r\n }\r\n\r\n };\r\n\r\n ui.addBlockHandlers = function (block) {\r\n\r\n if (!block) return;\r\n\r\n /**\r\n * Block keydowns\r\n */\r\n editor.listeners.add(block, 'keydown', editor.callback.blockKeydown, false);\r\n\r\n /**\r\n * Pasting content from another source\r\n * We have two type of sanitization\r\n * First - uses deep-first search algorithm to get sub nodes,\r\n * sanitizes whole Block_content and replaces cleared nodes\r\n * This method is deprecated\r\n * Method is used in editor.callback.blockPaste(event)\r\n *\r\n * Secont - uses Mutation observer.\r\n * Observer \"observe\" DOM changes and send changings to callback.\r\n * Callback gets changed node, not whole Block_content.\r\n * Inserted or changed node, which we've gotten have been cleared and replaced with diry node\r\n *\r\n * Method is used in editor.callback.blockPasteViaSanitize(event)\r\n *\r\n * @uses html-janitor\r\n * @example editor.callback.blockPasteViaSanitize(event), the second method.\r\n *\r\n */\r\n editor.listeners.add(block, 'paste', editor.paste.blockPasteCallback, false);\r\n\r\n editor.listeners.add(block, 'mouseup', editor.toolbar.inline.show, false);\r\n\r\n };\r\n\r\n /** getting all contenteditable elements */\r\n ui.saveInputs = function () {\r\n\r\n var redactor = editor.nodes.redactor;\r\n\r\n editor.state.inputs = [];\r\n\r\n /** Save all inputs in global variable state */\r\n var inputs = redactor.querySelectorAll('[contenteditable], input, textarea');\r\n\r\n Array.prototype.map.call(inputs, function (current) {\r\n\r\n if (!current.type || current.type == 'text' || current.type == 'textarea') {\r\n\r\n editor.state.inputs.push(current);\r\n\r\n }\r\n\r\n });\r\n\r\n };\r\n\r\n /**\r\n * Adds first initial block on empty redactor\r\n */\r\n ui.addInitialBlock = function () {\r\n\r\n var initialBlockType = editor.settings.initialBlockPlugin,\r\n initialBlock;\r\n\r\n if ( !editor.tools[initialBlockType] ) {\r\n\r\n editor.core.log('Plugin %o was not implemented and can\\'t be used as initial block', 'warn', initialBlockType);\r\n return;\r\n\r\n }\r\n\r\n initialBlock = editor.tools[initialBlockType].render();\r\n\r\n initialBlock.setAttribute('data-placeholder', 'Расскажите свою историю...');\r\n\r\n editor.content.insertBlock({\r\n type : initialBlockType,\r\n block : initialBlock\r\n });\r\n\r\n editor.content.workingNodeChanged(initialBlock);\r\n\r\n };\r\n\r\n ui.setInlineToolbarButtonBehaviour = function (button, type) {\r\n\r\n editor.listeners.add(button, 'mousedown', function (event) {\r\n\r\n editor.toolbar.inline.toolClicked(event, type);\r\n\r\n }, false);\r\n\r\n };\r\n\r\n return ui;\r\n\r\n})({});\r\n\n\n\n// WEBPACK FOOTER //\n// ./modules/ui.js","/**\r\n *\r\n * Codex.Editor Transport Module\r\n *\r\n * @copyright 2017 Codex-Team\r\n * @version 1.2.0\r\n */\r\n\r\nmodule.exports = (function (transport) {\r\n\r\n let editor = codex.editor;\r\n\r\n\r\n /**\r\n * @private {Object} current XmlHttpRequest instance\r\n */\r\n var currentRequest = null;\r\n\r\n\r\n /**\r\n * @type {null} | {DOMElement} input - keeps input element in memory\r\n */\r\n transport.input = null;\r\n\r\n /**\r\n * @property {Object} arguments - keep plugin settings and defined callbacks\r\n */\r\n transport.arguments = null;\r\n\r\n /**\r\n * Prepares input element where will be files\r\n */\r\n transport.prepare = function () {\r\n\r\n let input = editor.draw.node( 'INPUT', '', { type : 'file' } );\r\n\r\n editor.listeners.add(input, 'change', editor.transport.fileSelected);\r\n editor.transport.input = input;\r\n\r\n };\r\n\r\n /** Clear input when files is uploaded */\r\n transport.clearInput = function () {\r\n\r\n /** Remove old input */\r\n transport.input = null;\r\n\r\n /** Prepare new one */\r\n transport.prepare();\r\n\r\n };\r\n\r\n /**\r\n * Callback for file selection\r\n * @param {Event} event\r\n */\r\n transport.fileSelected = function () {\r\n\r\n var input = this,\r\n i,\r\n files = input.files,\r\n formData = new FormData();\r\n\r\n if (editor.transport.arguments.multiple === true) {\r\n\r\n for ( i = 0; i < files.length; i++) {\r\n\r\n formData.append('files[]', files[i], files[i].name);\r\n\r\n }\r\n\r\n } else {\r\n\r\n formData.append('files', files[0], files[0].name);\r\n\r\n }\r\n\r\n currentRequest = editor.core.ajax({\r\n type : 'POST',\r\n data : formData,\r\n url : editor.transport.arguments.url,\r\n beforeSend : editor.transport.arguments.beforeSend,\r\n success : editor.transport.arguments.success,\r\n error : editor.transport.arguments.error,\r\n progress : editor.transport.arguments.progress\r\n });\r\n\r\n /** Clear input */\r\n transport.clearInput();\r\n\r\n };\r\n\r\n /**\r\n * Use plugin callbacks\r\n * @protected\r\n *\r\n * @param {Object} args - can have :\r\n * @param {String} args.url - fetch URL\r\n * @param {Function} args.beforeSend - function calls before sending ajax\r\n * @param {Function} args.success - success callback\r\n * @param {Function} args.error - on error handler\r\n * @param {Function} args.progress - xhr onprogress handler\r\n * @param {Boolean} args.multiple - allow select several files\r\n * @param {String} args.accept - adds accept attribute\r\n */\r\n transport.selectAndUpload = function (args) {\r\n\r\n transport.arguments = args;\r\n\r\n if ( args.multiple === true) {\r\n\r\n transport.input.setAttribute('multiple', 'multiple');\r\n\r\n }\r\n\r\n if ( args.accept ) {\r\n\r\n transport.input.setAttribute('accept', args.accept);\r\n\r\n }\r\n\r\n transport.input.click();\r\n\r\n };\r\n\r\n transport.abort = function () {\r\n\r\n currentRequest.abort();\r\n\r\n currentRequest = null;\r\n\r\n };\r\n\r\n return transport;\r\n\r\n})({});\n\n\n// WEBPACK FOOTER //\n// ./modules/transport.js","/**\r\n * Codex Editor Renderer Module\r\n *\r\n * @author Codex Team\r\n * @version 1.0\r\n */\r\n\r\nmodule.exports = (function (renderer) {\r\n\r\n let editor = codex.editor;\r\n\r\n /**\r\n * Asyncronously parses input JSON to redactor blocks\r\n */\r\n renderer.makeBlocksFromData = function () {\r\n\r\n /**\r\n * If redactor is empty, add first paragraph to start writing\r\n */\r\n if (editor.core.isEmpty(editor.state.blocks) || !editor.state.blocks.items.length) {\r\n\r\n editor.ui.addInitialBlock();\r\n return;\r\n\r\n }\r\n\r\n Promise.resolve()\r\n\r\n /** First, get JSON from state */\r\n .then(function () {\r\n\r\n return editor.state.blocks;\r\n\r\n })\r\n\r\n /** Then, start to iterate they */\r\n .then(editor.renderer.appendBlocks)\r\n\r\n /** Write log if something goes wrong */\r\n .catch(function (error) {\r\n\r\n editor.core.log('Error while parsing JSON: %o', 'error', error);\r\n\r\n });\r\n\r\n };\r\n\r\n /**\r\n * Parses JSON to blocks\r\n * @param {object} data\r\n * @return Primise -> nodeList\r\n */\r\n renderer.appendBlocks = function (data) {\r\n\r\n var blocks = data.items;\r\n\r\n /**\r\n * Sequence of one-by-one blocks appending\r\n * Uses to save blocks order after async-handler\r\n */\r\n var nodeSequence = Promise.resolve();\r\n\r\n for (var index = 0; index < blocks.length ; index++ ) {\r\n\r\n /** Add node to sequence at specified index */\r\n editor.renderer.appendNodeAtIndex(nodeSequence, blocks, index);\r\n\r\n }\r\n\r\n };\r\n\r\n /**\r\n * Append node at specified index\r\n */\r\n renderer.appendNodeAtIndex = function (nodeSequence, blocks, index) {\r\n\r\n /** We need to append node to sequence */\r\n nodeSequence\r\n\r\n /** first, get node async-aware */\r\n .then(function () {\r\n\r\n return editor.renderer.getNodeAsync(blocks, index);\r\n\r\n })\r\n\r\n /**\r\n * second, compose editor-block from JSON object\r\n */\r\n .then(editor.renderer.createBlockFromData)\r\n\r\n /**\r\n * now insert block to redactor\r\n */\r\n .then(function (blockData) {\r\n\r\n /**\r\n * blockData has 'block', 'type' and 'stretched' information\r\n */\r\n editor.content.insertBlock(blockData);\r\n\r\n /** Pass created block to next step */\r\n return blockData.block;\r\n\r\n })\r\n\r\n /** Log if something wrong with node */\r\n .catch(function (error) {\r\n\r\n editor.core.log('Node skipped while parsing because %o', 'error', error);\r\n\r\n });\r\n\r\n };\r\n\r\n /**\r\n * Asynchronously returns block data from blocksList by index\r\n * @return Promise to node\r\n */\r\n renderer.getNodeAsync = function (blocksList, index) {\r\n\r\n return Promise.resolve().then(function () {\r\n\r\n return {\r\n tool : blocksList[index],\r\n position : index\r\n };\r\n\r\n });\r\n\r\n };\r\n\r\n /**\r\n * Creates editor block by JSON-data\r\n *\r\n * @uses render method of each plugin\r\n *\r\n * @param {Object} toolData.tool\r\n * { header : {\r\n * text: '',\r\n * type: 'H3', ...\r\n * }\r\n * }\r\n * @param {Number} toolData.position - index in input-blocks array\r\n * @return {Object} with type and Element\r\n */\r\n renderer.createBlockFromData = function ( toolData ) {\r\n\r\n /** New parser */\r\n var block,\r\n tool = toolData.tool,\r\n pluginName = tool.type;\r\n\r\n /** Get first key of object that stores plugin name */\r\n // for (var pluginName in blockData) break;\r\n\r\n /** Check for plugin existance */\r\n if (!editor.tools[pluginName]) {\r\n\r\n throw Error(`Plugin «${pluginName}» not found`);\r\n\r\n }\r\n\r\n /** Check for plugin having render method */\r\n if (typeof editor.tools[pluginName].render != 'function') {\r\n\r\n throw Error(`Plugin «${pluginName}» must have «render» method`);\r\n\r\n }\r\n\r\n if ( editor.tools[pluginName].available === false ) {\r\n\r\n block = editor.draw.unavailableBlock();\r\n\r\n block.innerHTML = editor.tools[pluginName].loadingMessage;\r\n\r\n /**\r\n * Saver will extract data from initial block data by position in array\r\n */\r\n block.dataset.inputPosition = toolData.position;\r\n\r\n } else {\r\n\r\n /** New Parser */\r\n block = editor.tools[pluginName].render(tool.data);\r\n\r\n }\r\n\r\n /** is first-level block stretched */\r\n var stretched = editor.tools[pluginName].isStretched || false;\r\n\r\n /** Retrun type and block */\r\n return {\r\n type : pluginName,\r\n block : block,\r\n stretched : stretched\r\n };\r\n\r\n };\r\n\r\n return renderer;\r\n\r\n})({});\n\n\n// WEBPACK FOOTER //\n// ./modules/renderer.js","/**\r\n * Codex Editor Saver\r\n *\r\n * @author Codex Team\r\n * @version 1.0.2\r\n */\r\n\r\nmodule.exports = (function (saver) {\r\n\r\n let editor = codex.editor;\r\n\r\n /**\r\n * Saves blocks\r\n * @private\r\n */\r\n saver.saveBlocks = function () {\r\n\r\n /** Save html content of redactor to memory */\r\n editor.state.html = editor.nodes.redactor.innerHTML;\r\n\r\n /** Empty jsonOutput state */\r\n editor.state.jsonOutput = [];\r\n\r\n Promise.resolve()\r\n\r\n .then(function () {\r\n\r\n return editor.nodes.redactor.childNodes;\r\n\r\n })\r\n /** Making a sequence from separate blocks */\r\n .then(editor.saver.makeQueue)\r\n\r\n .then(function () {\r\n // editor.nodes.textarea.innerHTML = editor.state.html;\r\n })\r\n\r\n .catch( function (error) {\r\n\r\n editor.core.log(error);\r\n\r\n });\r\n\r\n };\r\n\r\n saver.makeQueue = function (blocks) {\r\n\r\n var queue = Promise.resolve();\r\n\r\n for(var index = 0; index < blocks.length; index++) {\r\n\r\n /** Add node to sequence at specified index */\r\n editor.saver.getBlockData(queue, blocks, index);\r\n\r\n }\r\n\r\n };\r\n\r\n /** Gets every block and makes From Data */\r\n saver.getBlockData = function (queue, blocks, index) {\r\n\r\n queue.then(function () {\r\n\r\n return editor.saver.getNodeAsync(blocks, index);\r\n\r\n })\r\n\r\n .then(editor.saver.makeFormDataFromBlocks);\r\n\r\n };\r\n\r\n\r\n /**\r\n * Asynchronously returns block data from blocksList by index\r\n * @return Promise to node\r\n */\r\n saver.getNodeAsync = function (blocksList, index) {\r\n\r\n return Promise.resolve().then(function () {\r\n\r\n return blocksList[index];\r\n\r\n });\r\n\r\n };\r\n\r\n saver.makeFormDataFromBlocks = function (block) {\r\n\r\n var pluginName = block.dataset.tool;\r\n\r\n /** Check for plugin existance */\r\n if (!editor.tools[pluginName]) {\r\n\r\n throw Error(`Plugin «${pluginName}» not found`);\r\n\r\n }\r\n\r\n /** Check for plugin having render method */\r\n if (typeof editor.tools[pluginName].save != 'function') {\r\n\r\n throw Error(`Plugin «${pluginName}» must have save method`);\r\n\r\n }\r\n\r\n /** Result saver */\r\n var blockContent = block.childNodes[0],\r\n pluginsContent = blockContent.childNodes[0],\r\n savedData,\r\n position,\r\n output;\r\n\r\n /** If plugin wasn't available then return data from cache */\r\n if ( editor.tools[pluginName].available === false ) {\r\n\r\n position = pluginsContent.dataset.inputPosition;\r\n savedData = codex.editor.state.blocks.items[position].data;\r\n\r\n } else {\r\n\r\n savedData = editor.tools[pluginName].save(pluginsContent);\r\n\r\n if (editor.tools[pluginName].validate) {\r\n\r\n var result = editor.tools[pluginName].validate(savedData);\r\n\r\n /**\r\n * Do not allow invalid data\r\n */\r\n if (!result)\r\n return;\r\n\r\n }\r\n\r\n }\r\n\r\n output = {\r\n type : pluginName,\r\n data : savedData\r\n };\r\n\r\n editor.state.jsonOutput.push(output);\r\n\r\n };\r\n\r\n return saver;\r\n\r\n})({});\n\n\n// WEBPACK FOOTER //\n// ./modules/saver.js","/**\r\n * Codex Editor Content Module\r\n * Works with DOM\r\n *\r\n * @module Codex Editor content module\r\n *\r\n * @author Codex Team\r\n * @version 1.3.13\r\n *\r\n * @description Module works with Elements that have been appended to the main DOM\r\n */\r\n\r\nmodule.exports = (function (content) {\r\n\r\n let editor = codex.editor;\r\n\r\n /**\r\n * Links to current active block\r\n * @type {null | Element}\r\n */\r\n content.currentNode = null;\r\n\r\n /**\r\n * clicked in redactor area\r\n * @type {null | Boolean}\r\n */\r\n content.editorAreaHightlighted = null;\r\n\r\n /**\r\n * @deprecated\r\n * Synchronizes redactor with original textarea\r\n */\r\n content.sync = function () {\r\n\r\n editor.core.log('syncing...');\r\n\r\n /**\r\n * Save redactor content to editor.state\r\n */\r\n editor.state.html = editor.nodes.redactor.innerHTML;\r\n\r\n };\r\n\r\n /**\r\n * Appends background to the block\r\n *\r\n * @description add CSS class to highlight visually first-level block area\r\n */\r\n content.markBlock = function () {\r\n\r\n editor.content.currentNode.classList.add(editor.ui.className.BLOCK_HIGHLIGHTED);\r\n\r\n };\r\n\r\n /**\r\n * Clear background\r\n *\r\n * @description clears styles that highlights block\r\n */\r\n content.clearMark = function () {\r\n\r\n if (editor.content.currentNode) {\r\n\r\n editor.content.currentNode.classList.remove(editor.ui.className.BLOCK_HIGHLIGHTED);\r\n\r\n }\r\n\r\n };\r\n\r\n /**\r\n * Finds first-level block\r\n *\r\n * @param {Element} node - selected or clicked in redactors area node\r\n * @protected\r\n *\r\n * @description looks for first-level block.\r\n * gets parent while node is not first-level\r\n */\r\n content.getFirstLevelBlock = function (node) {\r\n\r\n if (!editor.core.isDomNode(node)) {\r\n\r\n node = node.parentNode;\r\n\r\n }\r\n\r\n if (node === editor.nodes.redactor || node === document.body) {\r\n\r\n return null;\r\n\r\n } else {\r\n\r\n while(!node.classList.contains(editor.ui.className.BLOCK_CLASSNAME)) {\r\n\r\n node = node.parentNode;\r\n\r\n }\r\n\r\n return node;\r\n\r\n }\r\n\r\n };\r\n\r\n /**\r\n * Trigger this event when working node changed\r\n * @param {Element} targetNode - first-level of this node will be current\r\n * @protected\r\n *\r\n * @description If targetNode is first-level then we set it as current else we look for parents to find first-level\r\n */\r\n content.workingNodeChanged = function (targetNode) {\r\n\r\n /** Clear background from previous marked block before we change */\r\n editor.content.clearMark();\r\n\r\n if (!targetNode) {\r\n\r\n return;\r\n\r\n }\r\n\r\n content.currentNode = content.getFirstLevelBlock(targetNode);\r\n\r\n };\r\n\r\n /**\r\n * Replaces one redactor block with another\r\n * @protected\r\n * @param {Element} targetBlock - block to replace. Mostly currentNode.\r\n * @param {Element} newBlock\r\n * @param {string} newBlockType - type of new block; we need to store it to data-attribute\r\n *\r\n * [!] Function does not saves old block content.\r\n * You can get it manually and pass with newBlock.innerHTML\r\n */\r\n content.replaceBlock = function (targetBlock, newBlock) {\r\n\r\n if (!targetBlock || !newBlock) {\r\n\r\n editor.core.log('replaceBlock: missed params');\r\n return;\r\n\r\n }\r\n\r\n /** If target-block is not a frist-level block, then we iterate parents to find it */\r\n while(!targetBlock.classList.contains(editor.ui.className.BLOCK_CLASSNAME)) {\r\n\r\n targetBlock = targetBlock.parentNode;\r\n\r\n }\r\n\r\n /** Replacing */\r\n editor.nodes.redactor.replaceChild(newBlock, targetBlock);\r\n\r\n /**\r\n * Set new node as current\r\n */\r\n editor.content.workingNodeChanged(newBlock);\r\n\r\n /**\r\n * Add block handlers\r\n */\r\n editor.ui.addBlockHandlers(newBlock);\r\n\r\n /**\r\n * Save changes\r\n */\r\n editor.ui.saveInputs();\r\n\r\n };\r\n\r\n /**\r\n * @protected\r\n *\r\n * Inserts new block to redactor\r\n * Wrapps block into a DIV with BLOCK_CLASSNAME class\r\n *\r\n * @param blockData {object}\r\n * @param blockData.block {Element} element with block content\r\n * @param blockData.type {string} block plugin\r\n * @param needPlaceCaret {bool} pass true to set caret in new block\r\n *\r\n */\r\n content.insertBlock = function ( blockData, needPlaceCaret ) {\r\n\r\n var workingBlock = editor.content.currentNode,\r\n newBlockContent = blockData.block,\r\n blockType = blockData.type,\r\n isStretched = blockData.stretched;\r\n\r\n var newBlock = composeNewBlock_(newBlockContent, blockType, isStretched);\r\n\r\n if (workingBlock) {\r\n\r\n editor.core.insertAfter(workingBlock, newBlock);\r\n\r\n } else {\r\n\r\n /**\r\n * If redactor is empty, append as first child\r\n */\r\n editor.nodes.redactor.appendChild(newBlock);\r\n\r\n }\r\n\r\n /**\r\n * Block handler\r\n */\r\n editor.ui.addBlockHandlers(newBlock);\r\n\r\n /**\r\n * Set new node as current\r\n */\r\n editor.content.workingNodeChanged(newBlock);\r\n\r\n /**\r\n * Save changes\r\n */\r\n editor.ui.saveInputs();\r\n\r\n\r\n if ( needPlaceCaret ) {\r\n\r\n /**\r\n * If we don't know input index then we set default value -1\r\n */\r\n var currentInputIndex = editor.caret.getCurrentInputIndex() || -1;\r\n\r\n\r\n if (currentInputIndex == -1) {\r\n\r\n\r\n var editableElement = newBlock.querySelector('[contenteditable]'),\r\n emptyText = document.createTextNode('');\r\n\r\n editableElement.appendChild(emptyText);\r\n editor.caret.set(editableElement, 0, 0);\r\n\r\n editor.toolbar.move();\r\n editor.toolbar.showPlusButton();\r\n\r\n\r\n } else {\r\n\r\n if (currentInputIndex === editor.state.inputs.length - 1)\r\n return;\r\n\r\n /** Timeout for browsers execution */\r\n window.setTimeout(function () {\r\n\r\n /** Setting to the new input */\r\n editor.caret.setToNextBlock(currentInputIndex);\r\n editor.toolbar.move();\r\n editor.toolbar.open();\r\n\r\n }, 10);\r\n\r\n }\r\n\r\n }\r\n\r\n /**\r\n * Block is inserted, wait for new click that defined focusing on editors area\r\n * @type {boolean}\r\n */\r\n content.editorAreaHightlighted = false;\r\n\r\n };\r\n\r\n /**\r\n * Replaces blocks with saving content\r\n * @protected\r\n * @param {Element} noteToReplace\r\n * @param {Element} newNode\r\n * @param {Element} blockType\r\n */\r\n content.switchBlock = function (blockToReplace, newBlock, tool) {\r\n\r\n tool = tool || editor.content.currentNode.dataset.tool;\r\n var newBlockComposed = composeNewBlock_(newBlock, tool);\r\n\r\n /** Replacing */\r\n editor.content.replaceBlock(blockToReplace, newBlockComposed);\r\n\r\n /** Save new Inputs when block is changed */\r\n editor.ui.saveInputs();\r\n\r\n };\r\n\r\n /**\r\n * Iterates between child noted and looking for #text node on deepest level\r\n * @protected\r\n *\r\n * @param {Element} block - node where find\r\n * @param {int} postiton - starting postion\r\n * Example: childNodex.length to find from the end\r\n * or 0 to find from the start\r\n * @return {Text} block\r\n * @uses DFS\r\n */\r\n content.getDeepestTextNodeFromPosition = function (block, position) {\r\n\r\n /**\r\n * Clear Block from empty and useless spaces with trim.\r\n * Such nodes we should remove\r\n */\r\n var blockChilds = block.childNodes,\r\n index,\r\n node,\r\n text;\r\n\r\n for(index = 0; index < blockChilds.length; index++) {\r\n\r\n node = blockChilds[index];\r\n\r\n if (node.nodeType == editor.core.nodeTypes.TEXT) {\r\n\r\n text = node.textContent.trim();\r\n\r\n /** Text is empty. We should remove this child from node before we start DFS\r\n * decrease the quantity of childs.\r\n */\r\n if (text === '') {\r\n\r\n block.removeChild(node);\r\n position--;\r\n\r\n }\r\n\r\n }\r\n\r\n }\r\n\r\n if (block.childNodes.length === 0) {\r\n\r\n return document.createTextNode('');\r\n\r\n }\r\n\r\n /** Setting default position when we deleted all empty nodes */\r\n if ( position < 0 )\r\n position = 1;\r\n\r\n var lookingFromStart = false;\r\n\r\n /** For looking from START */\r\n if (position === 0) {\r\n\r\n lookingFromStart = true;\r\n position = 1;\r\n\r\n }\r\n\r\n while ( position ) {\r\n\r\n /** initial verticle of node. */\r\n if ( lookingFromStart ) {\r\n\r\n block = block.childNodes[0];\r\n\r\n } else {\r\n\r\n block = block.childNodes[position - 1];\r\n\r\n }\r\n\r\n if ( block.nodeType == editor.core.nodeTypes.TAG ) {\r\n\r\n position = block.childNodes.length;\r\n\r\n } else if (block.nodeType == editor.core.nodeTypes.TEXT ) {\r\n\r\n position = 0;\r\n\r\n }\r\n\r\n }\r\n\r\n return block;\r\n\r\n };\r\n\r\n /**\r\n * @private\r\n * @param {Element} block - current plugins render\r\n * @param {String} tool - plugins name\r\n * @param {Boolean} isStretched - make stretched block or not\r\n *\r\n * @description adds necessary information to wrap new created block by first-level holder\r\n */\r\n var composeNewBlock_ = function (block, tool, isStretched) {\r\n\r\n var newBlock = editor.draw.node('DIV', editor.ui.className.BLOCK_CLASSNAME, {}),\r\n blockContent = editor.draw.node('DIV', editor.ui.className.BLOCK_CONTENT, {});\r\n\r\n blockContent.appendChild(block);\r\n newBlock.appendChild(blockContent);\r\n\r\n if (isStretched) {\r\n\r\n blockContent.classList.add(editor.ui.className.BLOCK_STRETCHED);\r\n\r\n }\r\n\r\n newBlock.dataset.tool = tool;\r\n return newBlock;\r\n\r\n };\r\n\r\n /**\r\n * Returns Range object of current selection\r\n * @protected\r\n */\r\n content.getRange = function () {\r\n\r\n var selection = window.getSelection().getRangeAt(0);\r\n\r\n return selection;\r\n\r\n };\r\n\r\n /**\r\n * Divides block in two blocks (after and before caret)\r\n *\r\n * @protected\r\n * @param {int} inputIndex - target input index\r\n *\r\n * @description splits current input content to the separate blocks\r\n * When enter is pressed among the words, that text will be splited.\r\n */\r\n content.splitBlock = function (inputIndex) {\r\n\r\n var selection = window.getSelection(),\r\n anchorNode = selection.anchorNode,\r\n anchorNodeText = anchorNode.textContent,\r\n caretOffset = selection.anchorOffset,\r\n textBeforeCaret,\r\n textNodeBeforeCaret,\r\n textAfterCaret,\r\n textNodeAfterCaret;\r\n\r\n var currentBlock = editor.content.currentNode.querySelector('[contentEditable]');\r\n\r\n\r\n textBeforeCaret = anchorNodeText.substring(0, caretOffset);\r\n textAfterCaret = anchorNodeText.substring(caretOffset);\r\n\r\n textNodeBeforeCaret = document.createTextNode(textBeforeCaret);\r\n\r\n if (textAfterCaret) {\r\n\r\n textNodeAfterCaret = document.createTextNode(textAfterCaret);\r\n\r\n }\r\n\r\n var previousChilds = [],\r\n nextChilds = [],\r\n reachedCurrent = false;\r\n\r\n if (textNodeAfterCaret) {\r\n\r\n nextChilds.push(textNodeAfterCaret);\r\n\r\n }\r\n\r\n for ( var i = 0, child; !!(child = currentBlock.childNodes[i]); i++) {\r\n\r\n if ( child != anchorNode ) {\r\n\r\n if ( !reachedCurrent ) {\r\n\r\n previousChilds.push(child);\r\n\r\n } else {\r\n\r\n nextChilds.push(child);\r\n\r\n }\r\n\r\n } else {\r\n\r\n reachedCurrent = true;\r\n\r\n }\r\n\r\n }\r\n\r\n /** Clear current input */\r\n editor.state.inputs[inputIndex].innerHTML = '';\r\n\r\n /**\r\n * Append all childs founded before anchorNode\r\n */\r\n var previousChildsLength = previousChilds.length;\r\n\r\n for(i = 0; i < previousChildsLength; i++) {\r\n\r\n editor.state.inputs[inputIndex].appendChild(previousChilds[i]);\r\n\r\n }\r\n\r\n editor.state.inputs[inputIndex].appendChild(textNodeBeforeCaret);\r\n\r\n /**\r\n * Append text node which is after caret\r\n */\r\n var nextChildsLength = nextChilds.length,\r\n newNode = document.createElement('div');\r\n\r\n for(i = 0; i < nextChildsLength; i++) {\r\n\r\n newNode.appendChild(nextChilds[i]);\r\n\r\n }\r\n\r\n newNode = newNode.innerHTML;\r\n\r\n /** This type of block creates when enter is pressed */\r\n var NEW_BLOCK_TYPE = editor.settings.initialBlockPlugin;\r\n\r\n /**\r\n * Make new paragraph with text after caret\r\n */\r\n editor.content.insertBlock({\r\n type : NEW_BLOCK_TYPE,\r\n block : editor.tools[NEW_BLOCK_TYPE].render({\r\n text : newNode\r\n })\r\n }, true );\r\n\r\n };\r\n\r\n /**\r\n * Merges two blocks — current and target\r\n * If target index is not exist, then previous will be as target\r\n *\r\n * @protected\r\n * @param {int} currentInputIndex\r\n * @param {int} targetInputIndex\r\n *\r\n * @description gets two inputs indexes and merges into one\r\n */\r\n content.mergeBlocks = function (currentInputIndex, targetInputIndex) {\r\n\r\n /** If current input index is zero, then prevent method execution */\r\n if (currentInputIndex === 0) {\r\n\r\n return;\r\n\r\n }\r\n\r\n var targetInput,\r\n currentInputContent = editor.state.inputs[currentInputIndex].innerHTML;\r\n\r\n if (!targetInputIndex) {\r\n\r\n targetInput = editor.state.inputs[currentInputIndex - 1];\r\n\r\n } else {\r\n\r\n targetInput = editor.state.inputs[targetInputIndex];\r\n\r\n }\r\n\r\n targetInput.innerHTML += currentInputContent;\r\n\r\n };\r\n\r\n /**\r\n * Iterates all right siblings and parents, which has right siblings\r\n * while it does not reached the first-level block\r\n *\r\n * @param {Element} node\r\n * @return {boolean}\r\n */\r\n content.isLastNode = function (node) {\r\n\r\n // console.log('погнали перебор родителей');\r\n\r\n var allChecked = false;\r\n\r\n while ( !allChecked ) {\r\n\r\n // console.log('Смотрим на %o', node);\r\n // console.log('Проверим, пустые ли соседи справа');\r\n\r\n if ( !allSiblingsEmpty_(node) ) {\r\n\r\n // console.log('Есть непустые соседи. Узел не последний. Выходим.');\r\n return false;\r\n\r\n }\r\n\r\n node = node.parentNode;\r\n\r\n /**\r\n * Проверяем родителей до тех пор, пока не найдем блок первого уровня\r\n */\r\n if ( node.classList.contains(editor.ui.className.BLOCK_CONTENT) ) {\r\n\r\n allChecked = true;\r\n\r\n }\r\n\r\n }\r\n\r\n return true;\r\n\r\n };\r\n\r\n /**\r\n * Checks if all element right siblings is empty\r\n * @param node\r\n */\r\n var allSiblingsEmpty_ = function (node) {\r\n\r\n /**\r\n * Нужно убедиться, что после пустого соседа ничего нет\r\n */\r\n var sibling = node.nextSibling;\r\n\r\n while ( sibling ) {\r\n\r\n if (sibling.textContent.length) {\r\n\r\n return false;\r\n\r\n }\r\n\r\n sibling = sibling.nextSibling;\r\n\r\n }\r\n\r\n return true;\r\n\r\n };\r\n\r\n /**\r\n * @public\r\n *\r\n * @param {string} htmlData - html content as string\r\n * @param {string} plainData - plain text\r\n * @return {string} - html content as string\r\n */\r\n content.wrapTextWithParagraphs = function (htmlData, plainData) {\r\n\r\n if (!htmlData) {\r\n\r\n return '

    ' + plainData.split('\\n\\n').join('

    ') + '

    ';\r\n\r\n }\r\n\r\n var wrapper = document.createElement('DIV'),\r\n newWrapper = document.createElement('DIV'),\r\n i,\r\n paragraph,\r\n firstLevelBlocks = ['DIV', 'P'],\r\n blockTyped,\r\n node;\r\n\r\n /**\r\n * Make HTML Element to Wrap Text\r\n * It allows us to work with input data as HTML content\r\n */\r\n wrapper.innerHTML = htmlData;\r\n paragraph = document.createElement('P');\r\n\r\n for (i = 0; i < wrapper.childNodes.length; i++) {\r\n\r\n node = wrapper.childNodes[i];\r\n\r\n blockTyped = firstLevelBlocks.indexOf(node.tagName) != -1;\r\n\r\n /**\r\n * If node is first-levet\r\n * we add this node to our new wrapper\r\n */\r\n if ( blockTyped ) {\r\n\r\n /**\r\n * If we had splitted inline nodes to paragraph before\r\n */\r\n if ( paragraph.childNodes.length ) {\r\n\r\n newWrapper.appendChild(paragraph.cloneNode(true));\r\n\r\n /** empty paragraph */\r\n paragraph = null;\r\n paragraph = document.createElement('P');\r\n\r\n }\r\n\r\n newWrapper.appendChild(node.cloneNode(true));\r\n\r\n } else {\r\n\r\n /** Collect all inline nodes to one as paragraph */\r\n paragraph.appendChild(node.cloneNode(true));\r\n\r\n /** if node is last we should append this node to paragraph and paragraph to new wrapper */\r\n if ( i == wrapper.childNodes.length - 1 ) {\r\n\r\n newWrapper.appendChild(paragraph.cloneNode(true));\r\n\r\n }\r\n\r\n }\r\n\r\n }\r\n\r\n return newWrapper.innerHTML;\r\n\r\n };\r\n\r\n /**\r\n * Finds closest Contenteditable parent from Element\r\n * @param {Element} node element looking from\r\n * @return {Element} node contenteditable\r\n */\r\n content.getEditableParent = function (node) {\r\n\r\n while (node && node.contentEditable != 'true') {\r\n\r\n node = node.parentNode;\r\n\r\n }\r\n\r\n return node;\r\n\r\n };\r\n\r\n return content;\r\n\r\n})({});\n\n\n// WEBPACK FOOTER //\n// ./modules/content.js","/**\r\n * Codex Editor toolbar module\r\n *\r\n * Contains:\r\n * - Inline toolbox\r\n * - Toolbox within plus button\r\n * - Settings section\r\n *\r\n * @author Codex Team\r\n * @version 1.0\r\n */\r\n\r\nmodule.exports = (function (toolbar) {\r\n\r\n let editor = codex.editor;\r\n\r\n toolbar.settings = require('./settings');\r\n toolbar.inline = require('./inline');\r\n toolbar.toolbox = require('./toolbox');\r\n\r\n /**\r\n * Margin between focused node and toolbar\r\n */\r\n toolbar.defaultToolbarHeight = 49;\r\n\r\n toolbar.defaultOffset = 34;\r\n\r\n toolbar.opened = false;\r\n\r\n toolbar.current = null;\r\n\r\n /**\r\n * @protected\r\n */\r\n toolbar.open = function () {\r\n\r\n if (editor.hideToolbar) {\r\n\r\n return;\r\n\r\n }\r\n\r\n editor.nodes.toolbar.classList.add('opened');\r\n this.opened = true;\r\n\r\n };\r\n\r\n /**\r\n * @protected\r\n */\r\n toolbar.close = function () {\r\n\r\n editor.nodes.toolbar.classList.remove('opened');\r\n\r\n toolbar.opened = false;\r\n toolbar.current = null;\r\n\r\n for (var button in editor.nodes.toolbarButtons) {\r\n\r\n editor.nodes.toolbarButtons[button].classList.remove('selected');\r\n\r\n }\r\n\r\n /** Close toolbox when toolbar is not displayed */\r\n editor.toolbar.toolbox.close();\r\n editor.toolbar.settings.close();\r\n\r\n };\r\n\r\n toolbar.toggle = function () {\r\n\r\n if ( !this.opened ) {\r\n\r\n this.open();\r\n\r\n } else {\r\n\r\n this.close();\r\n\r\n }\r\n\r\n };\r\n\r\n toolbar.hidePlusButton = function () {\r\n\r\n editor.nodes.plusButton.classList.add('hide');\r\n\r\n };\r\n\r\n toolbar.showPlusButton = function () {\r\n\r\n editor.nodes.plusButton.classList.remove('hide');\r\n\r\n };\r\n\r\n /**\r\n * Moving toolbar to the specified node\r\n */\r\n toolbar.move = function () {\r\n\r\n /** Close Toolbox when we move toolbar */\r\n editor.toolbar.toolbox.close();\r\n\r\n if (!editor.content.currentNode) {\r\n\r\n return;\r\n\r\n }\r\n\r\n var newYCoordinate = editor.content.currentNode.offsetTop - (editor.toolbar.defaultToolbarHeight / 2) + editor.toolbar.defaultOffset;\r\n\r\n editor.nodes.toolbar.style.transform = `translate3D(0, ${Math.floor(newYCoordinate)}px, 0)`;\r\n\r\n /** Close trash actions */\r\n editor.toolbar.settings.hideRemoveActions();\r\n\r\n };\r\n\r\n return toolbar;\r\n\r\n})({});\n\n\n// WEBPACK FOOTER //\n// ./modules/toolbar/toolbar.js","/**\r\n * Toolbar settings\r\n *\r\n * @version 1.0.4\r\n */\r\n\r\nmodule.exports = (function (settings) {\r\n\r\n let editor = codex.editor;\r\n\r\n settings.opened = false;\r\n\r\n settings.setting = null;\r\n settings.actions = null;\r\n\r\n /**\r\n * Append and open settings\r\n */\r\n settings.open = function (toolType) {\r\n\r\n /**\r\n * Append settings content\r\n * It's stored in tool.settings\r\n */\r\n if (!editor.tools[toolType] || !editor.tools[toolType].makeSettings ) {\r\n\r\n editor.core.log(`Plugin «${toolType}» has no settings`, 'warn');\r\n editor.nodes.pluginSettings.innerHTML = `Плагин «${toolType}» не имеет настроек`;\r\n\r\n } else {\r\n\r\n /**\r\n * Draw settings block\r\n */\r\n var settingsBlock = editor.tools[toolType].makeSettings();\r\n\r\n editor.nodes.pluginSettings.appendChild(settingsBlock);\r\n\r\n }\r\n\r\n /** Open settings block */\r\n editor.nodes.blockSettings.classList.add('opened');\r\n this.opened = true;\r\n\r\n };\r\n\r\n /**\r\n * Close and clear settings\r\n */\r\n settings.close = function () {\r\n\r\n editor.nodes.blockSettings.classList.remove('opened');\r\n editor.nodes.pluginSettings.innerHTML = '';\r\n\r\n this.opened = false;\r\n\r\n };\r\n\r\n /**\r\n * @param {string} toolType - plugin type\r\n */\r\n settings.toggle = function ( toolType ) {\r\n\r\n if ( !this.opened ) {\r\n\r\n this.open(toolType);\r\n\r\n } else {\r\n\r\n this.close();\r\n\r\n }\r\n\r\n };\r\n\r\n /**\r\n * Here we will draw buttons and add listeners to components\r\n */\r\n settings.makeRemoveBlockButton = function () {\r\n\r\n var removeBlockWrapper = editor.draw.node('SPAN', 'ce-toolbar__remove-btn', {}),\r\n settingButton = editor.draw.node('SPAN', 'ce-toolbar__remove-setting', { innerHTML : '' }),\r\n actionWrapper = editor.draw.node('DIV', 'ce-toolbar__remove-confirmation', {}),\r\n confirmAction = editor.draw.node('DIV', 'ce-toolbar__remove-confirm', { textContent : 'Удалить блок' }),\r\n cancelAction = editor.draw.node('DIV', 'ce-toolbar__remove-cancel', { textContent : 'Отмена' });\r\n\r\n editor.listeners.add(settingButton, 'click', editor.toolbar.settings.removeButtonClicked, false);\r\n\r\n editor.listeners.add(confirmAction, 'click', editor.toolbar.settings.confirmRemovingRequest, false);\r\n\r\n editor.listeners.add(cancelAction, 'click', editor.toolbar.settings.cancelRemovingRequest, false);\r\n\r\n actionWrapper.appendChild(confirmAction);\r\n actionWrapper.appendChild(cancelAction);\r\n\r\n removeBlockWrapper.appendChild(settingButton);\r\n removeBlockWrapper.appendChild(actionWrapper);\r\n\r\n /** Save setting */\r\n editor.toolbar.settings.setting = settingButton;\r\n editor.toolbar.settings.actions = actionWrapper;\r\n\r\n return removeBlockWrapper;\r\n\r\n };\r\n\r\n settings.removeButtonClicked = function () {\r\n\r\n var action = editor.toolbar.settings.actions;\r\n\r\n if (action.classList.contains('opened')) {\r\n\r\n editor.toolbar.settings.hideRemoveActions();\r\n\r\n } else {\r\n\r\n editor.toolbar.settings.showRemoveActions();\r\n\r\n }\r\n\r\n editor.toolbar.toolbox.close();\r\n editor.toolbar.settings.close();\r\n\r\n };\r\n\r\n settings.cancelRemovingRequest = function () {\r\n\r\n editor.toolbar.settings.actions.classList.remove('opened');\r\n\r\n };\r\n\r\n settings.confirmRemovingRequest = function () {\r\n\r\n var currentBlock = editor.content.currentNode,\r\n firstLevelBlocksCount;\r\n\r\n currentBlock.remove();\r\n\r\n firstLevelBlocksCount = editor.nodes.redactor.childNodes.length;\r\n\r\n /**\r\n * If all blocks are removed\r\n */\r\n if (firstLevelBlocksCount === 0) {\r\n\r\n /** update currentNode variable */\r\n editor.content.currentNode = null;\r\n\r\n /** Inserting new empty initial block */\r\n editor.ui.addInitialBlock();\r\n\r\n }\r\n\r\n editor.ui.saveInputs();\r\n\r\n editor.toolbar.close();\r\n\r\n };\r\n\r\n settings.showRemoveActions = function () {\r\n\r\n editor.toolbar.settings.actions.classList.add('opened');\r\n\r\n };\r\n\r\n settings.hideRemoveActions = function () {\r\n\r\n editor.toolbar.settings.actions.classList.remove('opened');\r\n\r\n };\r\n\r\n return settings;\r\n\r\n})({});\n\n\n// WEBPACK FOOTER //\n// ./modules/toolbar/settings.js","/**\r\n * Inline toolbar\r\n *\r\n * Contains from tools:\r\n * Bold, Italic, Underline and Anchor\r\n *\r\n * @author Codex Team\r\n * @version 1.0\r\n */\r\n\r\nmodule.exports = (function (inline) {\r\n\r\n let editor = codex.editor;\r\n\r\n inline.buttonsOpened = null;\r\n inline.actionsOpened = null;\r\n inline.wrappersOffset = null;\r\n\r\n /**\r\n * saving selection that need for execCommand for styling\r\n *\r\n */\r\n inline.storedSelection = null;\r\n\r\n /**\r\n * @protected\r\n *\r\n * Open inline toobar\r\n */\r\n inline.show = function () {\r\n\r\n var currentNode = editor.content.currentNode,\r\n tool = currentNode.dataset.tool,\r\n plugin;\r\n\r\n /**\r\n * tool allowed to open inline toolbar\r\n */\r\n plugin = editor.tools[tool];\r\n\r\n if (!plugin.showInlineToolbar)\r\n return;\r\n\r\n var selectedText = inline.getSelectionText(),\r\n toolbar = editor.nodes.inlineToolbar.wrapper;\r\n\r\n if (selectedText.length > 0) {\r\n\r\n /** Move toolbar and open */\r\n editor.toolbar.inline.move();\r\n\r\n /** Open inline toolbar */\r\n toolbar.classList.add('opened');\r\n\r\n /** show buttons of inline toolbar */\r\n editor.toolbar.inline.showButtons();\r\n\r\n }\r\n\r\n };\r\n\r\n /**\r\n * @protected\r\n *\r\n * Closes inline toolbar\r\n */\r\n inline.close = function () {\r\n\r\n var toolbar = editor.nodes.inlineToolbar.wrapper;\r\n\r\n toolbar.classList.remove('opened');\r\n\r\n };\r\n\r\n /**\r\n * @private\r\n *\r\n * Moving toolbar\r\n */\r\n inline.move = function () {\r\n\r\n if (!this.wrappersOffset) {\r\n\r\n this.wrappersOffset = this.getWrappersOffset();\r\n\r\n }\r\n\r\n var coords = this.getSelectionCoords(),\r\n defaultOffset = 0,\r\n toolbar = editor.nodes.inlineToolbar.wrapper,\r\n newCoordinateX,\r\n newCoordinateY;\r\n\r\n if (toolbar.offsetHeight === 0) {\r\n\r\n defaultOffset = 40;\r\n\r\n }\r\n\r\n newCoordinateX = coords.x - this.wrappersOffset.left;\r\n newCoordinateY = coords.y + window.scrollY - this.wrappersOffset.top - defaultOffset - toolbar.offsetHeight;\r\n\r\n toolbar.style.transform = `translate3D(${Math.floor(newCoordinateX)}px, ${Math.floor(newCoordinateY)}px, 0)`;\r\n\r\n /** Close everything */\r\n editor.toolbar.inline.closeButtons();\r\n editor.toolbar.inline.closeAction();\r\n\r\n };\r\n\r\n /**\r\n * @private\r\n *\r\n * Tool Clicked\r\n */\r\n\r\n inline.toolClicked = function (event, type) {\r\n\r\n /**\r\n * For simple tools we use default browser function\r\n * For more complicated tools, we should write our own behavior\r\n */\r\n switch (type) {\r\n case 'createLink' : editor.toolbar.inline.createLinkAction(event, type); break;\r\n default : editor.toolbar.inline.defaultToolAction(type); break;\r\n }\r\n\r\n /**\r\n * highlight buttons\r\n * after making some action\r\n */\r\n editor.nodes.inlineToolbar.buttons.childNodes.forEach(editor.toolbar.inline.hightlight);\r\n\r\n };\r\n\r\n /**\r\n * @private\r\n *\r\n * Saving wrappers offset in DOM\r\n */\r\n inline.getWrappersOffset = function () {\r\n\r\n var wrapper = editor.nodes.wrapper,\r\n offset = this.getOffset(wrapper);\r\n\r\n this.wrappersOffset = offset;\r\n return offset;\r\n\r\n };\r\n\r\n /**\r\n * @private\r\n *\r\n * Calculates offset of DOM element\r\n *\r\n * @param el\r\n * @returns {{top: number, left: number}}\r\n */\r\n inline.getOffset = function ( el ) {\r\n\r\n var _x = 0;\r\n var _y = 0;\r\n\r\n while( el && !isNaN( el.offsetLeft ) && !isNaN( el.offsetTop ) ) {\r\n\r\n _x += (el.offsetLeft + el.clientLeft);\r\n _y += (el.offsetTop + el.clientTop);\r\n el = el.offsetParent;\r\n\r\n }\r\n return { top: _y, left: _x };\r\n\r\n };\r\n\r\n /**\r\n * @private\r\n *\r\n * Calculates position of selected text\r\n * @returns {{x: number, y: number}}\r\n */\r\n inline.getSelectionCoords = function () {\r\n\r\n var sel = document.selection, range;\r\n var x = 0, y = 0;\r\n\r\n if (sel) {\r\n\r\n if (sel.type != 'Control') {\r\n\r\n range = sel.createRange();\r\n range.collapse(true);\r\n x = range.boundingLeft;\r\n y = range.boundingTop;\r\n\r\n }\r\n\r\n } else if (window.getSelection) {\r\n\r\n sel = window.getSelection();\r\n\r\n if (sel.rangeCount) {\r\n\r\n range = sel.getRangeAt(0).cloneRange();\r\n if (range.getClientRects) {\r\n\r\n range.collapse(true);\r\n var rect = range.getClientRects()[0];\r\n\r\n if (!rect) {\r\n\r\n return;\r\n\r\n }\r\n\r\n x = rect.left;\r\n y = rect.top;\r\n\r\n }\r\n\r\n }\r\n\r\n }\r\n return { x: x, y: y };\r\n\r\n };\r\n\r\n /**\r\n * @private\r\n *\r\n * Returns selected text as String\r\n * @returns {string}\r\n */\r\n inline.getSelectionText = function () {\r\n\r\n var selectedText = '';\r\n\r\n // all modern browsers and IE9+\r\n if (window.getSelection) {\r\n\r\n selectedText = window.getSelection().toString();\r\n\r\n }\r\n\r\n return selectedText;\r\n\r\n };\r\n\r\n /** Opens buttons block */\r\n inline.showButtons = function () {\r\n\r\n var buttons = editor.nodes.inlineToolbar.buttons;\r\n\r\n buttons.classList.add('opened');\r\n\r\n editor.toolbar.inline.buttonsOpened = true;\r\n\r\n /** highlight buttons */\r\n editor.nodes.inlineToolbar.buttons.childNodes.forEach(editor.toolbar.inline.hightlight);\r\n\r\n };\r\n\r\n /** Makes buttons disappear */\r\n inline.closeButtons = function () {\r\n\r\n var buttons = editor.nodes.inlineToolbar.buttons;\r\n\r\n buttons.classList.remove('opened');\r\n\r\n editor.toolbar.inline.buttonsOpened = false;\r\n\r\n };\r\n\r\n /** Open buttons defined action if exist */\r\n inline.showActions = function () {\r\n\r\n var action = editor.nodes.inlineToolbar.actions;\r\n\r\n action.classList.add('opened');\r\n\r\n editor.toolbar.inline.actionsOpened = true;\r\n\r\n };\r\n\r\n /** Close actions block */\r\n inline.closeAction = function () {\r\n\r\n var action = editor.nodes.inlineToolbar.actions;\r\n\r\n action.innerHTML = '';\r\n action.classList.remove('opened');\r\n editor.toolbar.inline.actionsOpened = false;\r\n\r\n };\r\n\r\n\r\n /**\r\n * Callback for keydowns in inline toolbar \"Insert link...\" input\r\n */\r\n let inlineToolbarAnchorInputKeydown_ = function (event) {\r\n\r\n if (event.keyCode != editor.core.keys.ENTER) {\r\n\r\n return;\r\n\r\n }\r\n\r\n let editable = editor.content.currentNode,\r\n storedSelection = editor.toolbar.inline.storedSelection;\r\n\r\n editor.toolbar.inline.restoreSelection(editable, storedSelection);\r\n editor.toolbar.inline.setAnchor(this.value);\r\n\r\n /**\r\n * Preventing events that will be able to happen\r\n */\r\n event.preventDefault();\r\n event.stopImmediatePropagation();\r\n\r\n editor.toolbar.inline.clearRange();\r\n\r\n };\r\n\r\n /** Action for link creation or for setting anchor */\r\n inline.createLinkAction = function (event) {\r\n\r\n var isActive = this.isLinkActive();\r\n\r\n var editable = editor.content.currentNode,\r\n storedSelection = editor.toolbar.inline.saveSelection(editable);\r\n\r\n /** Save globally selection */\r\n editor.toolbar.inline.storedSelection = storedSelection;\r\n\r\n if (isActive) {\r\n\r\n\r\n /**\r\n * Changing stored selection. if we want to remove anchor from word\r\n * we should remove anchor from whole word, not only selected part.\r\n * The solution is than we get the length of current link\r\n * Change start position to - end of selection minus length of anchor\r\n */\r\n editor.toolbar.inline.restoreSelection(editable, storedSelection);\r\n\r\n editor.toolbar.inline.defaultToolAction('unlink');\r\n\r\n } else {\r\n\r\n /** Create input and close buttons */\r\n var action = editor.draw.inputForLink();\r\n\r\n editor.nodes.inlineToolbar.actions.appendChild(action);\r\n\r\n editor.toolbar.inline.closeButtons();\r\n editor.toolbar.inline.showActions();\r\n\r\n /**\r\n * focus to input\r\n * Solution: https://developer.mozilla.org/ru/docs/Web/API/HTMLElement/focus\r\n * Prevents event after showing input and when we need to focus an input which is in unexisted form\r\n */\r\n action.focus();\r\n event.preventDefault();\r\n\r\n /** Callback to link action */\r\n editor.listeners.add(action, 'keydown', inlineToolbarAnchorInputKeydown_, false);\r\n\r\n }\r\n\r\n };\r\n\r\n inline.isLinkActive = function () {\r\n\r\n var isActive = false;\r\n\r\n editor.nodes.inlineToolbar.buttons.childNodes.forEach(function (tool) {\r\n\r\n var dataType = tool.dataset.type;\r\n\r\n if (dataType == 'link' && tool.classList.contains('hightlighted')) {\r\n\r\n isActive = true;\r\n\r\n }\r\n\r\n });\r\n\r\n return isActive;\r\n\r\n };\r\n\r\n /** default action behavior of tool */\r\n inline.defaultToolAction = function (type) {\r\n\r\n document.execCommand(type, false, null);\r\n\r\n };\r\n\r\n /**\r\n * @private\r\n *\r\n * Sets URL\r\n *\r\n * @param {String} url - URL\r\n */\r\n inline.setAnchor = function (url) {\r\n\r\n document.execCommand('createLink', false, url);\r\n\r\n /** Close after URL inserting */\r\n editor.toolbar.inline.closeAction();\r\n\r\n };\r\n\r\n /**\r\n * @private\r\n *\r\n * Saves selection\r\n */\r\n inline.saveSelection = function (containerEl) {\r\n\r\n var range = window.getSelection().getRangeAt(0),\r\n preSelectionRange = range.cloneRange(),\r\n start;\r\n\r\n preSelectionRange.selectNodeContents(containerEl);\r\n preSelectionRange.setEnd(range.startContainer, range.startOffset);\r\n\r\n start = preSelectionRange.toString().length;\r\n\r\n return {\r\n start: start,\r\n end: start + range.toString().length\r\n };\r\n\r\n };\r\n\r\n /**\r\n * @private\r\n *\r\n * Sets to previous selection (Range)\r\n *\r\n * @param {Element} containerEl - editable element where we restore range\r\n * @param {Object} savedSel - range basic information to restore\r\n */\r\n inline.restoreSelection = function (containerEl, savedSel) {\r\n\r\n var range = document.createRange(),\r\n charIndex = 0;\r\n\r\n range.setStart(containerEl, 0);\r\n range.collapse(true);\r\n\r\n var nodeStack = [ containerEl ],\r\n node,\r\n foundStart = false,\r\n stop = false,\r\n nextCharIndex;\r\n\r\n while (!stop && (node = nodeStack.pop())) {\r\n\r\n if (node.nodeType == 3) {\r\n\r\n nextCharIndex = charIndex + node.length;\r\n\r\n if (!foundStart && savedSel.start >= charIndex && savedSel.start <= nextCharIndex) {\r\n\r\n range.setStart(node, savedSel.start - charIndex);\r\n foundStart = true;\r\n\r\n }\r\n if (foundStart && savedSel.end >= charIndex && savedSel.end <= nextCharIndex) {\r\n\r\n range.setEnd(node, savedSel.end - charIndex);\r\n stop = true;\r\n\r\n }\r\n charIndex = nextCharIndex;\r\n\r\n } else {\r\n\r\n var i = node.childNodes.length;\r\n\r\n while (i--) {\r\n\r\n nodeStack.push(node.childNodes[i]);\r\n\r\n }\r\n\r\n }\r\n\r\n }\r\n\r\n var sel = window.getSelection();\r\n\r\n sel.removeAllRanges();\r\n sel.addRange(range);\r\n\r\n };\r\n\r\n /**\r\n * @private\r\n *\r\n * Removes all ranges from window selection\r\n */\r\n inline.clearRange = function () {\r\n\r\n var selection = window.getSelection();\r\n\r\n selection.removeAllRanges();\r\n\r\n };\r\n\r\n /**\r\n * @private\r\n *\r\n * sets or removes hightlight\r\n */\r\n inline.hightlight = function (tool) {\r\n\r\n var dataType = tool.dataset.type;\r\n\r\n if (document.queryCommandState(dataType)) {\r\n\r\n editor.toolbar.inline.setButtonHighlighted(tool);\r\n\r\n } else {\r\n\r\n editor.toolbar.inline.removeButtonsHighLight(tool);\r\n\r\n }\r\n\r\n /**\r\n *\r\n * hightlight for anchors\r\n */\r\n var selection = window.getSelection(),\r\n tag = selection.anchorNode.parentNode;\r\n\r\n if (tag.tagName == 'A' && dataType == 'link') {\r\n\r\n editor.toolbar.inline.setButtonHighlighted(tool);\r\n\r\n }\r\n\r\n };\r\n\r\n /**\r\n * @private\r\n *\r\n * Mark button if text is already executed\r\n */\r\n inline.setButtonHighlighted = function (button) {\r\n\r\n button.classList.add('hightlighted');\r\n\r\n /** At link tool we also change icon */\r\n if (button.dataset.type == 'link') {\r\n\r\n var icon = button.childNodes[0];\r\n\r\n icon.classList.remove('ce-icon-link');\r\n icon.classList.add('ce-icon-unlink');\r\n\r\n }\r\n\r\n };\r\n\r\n /**\r\n * @private\r\n *\r\n * Removes hightlight\r\n */\r\n inline.removeButtonsHighLight = function (button) {\r\n\r\n button.classList.remove('hightlighted');\r\n\r\n /** At link tool we also change icon */\r\n if (button.dataset.type == 'link') {\r\n\r\n var icon = button.childNodes[0];\r\n\r\n icon.classList.remove('ce-icon-unlink');\r\n icon.classList.add('ce-icon-link');\r\n\r\n }\r\n\r\n };\r\n\r\n\r\n return inline;\r\n\r\n})({});\n\n\n// WEBPACK FOOTER //\n// ./modules/toolbar/inline.js","/**\r\n * Codex Editor toolbox\r\n *\r\n * All tools be able to appended here\r\n *\r\n * @author Codex Team\r\n * @version 1.0\r\n */\r\n\r\nmodule.exports = (function (toolbox) {\r\n\r\n let editor = codex.editor;\r\n\r\n toolbox.opened = false;\r\n\r\n /** Shows toolbox */\r\n toolbox.open = function () {\r\n\r\n /** Close setting if toolbox is opened */\r\n if (editor.toolbar.settings.opened) {\r\n\r\n editor.toolbar.settings.close();\r\n\r\n }\r\n\r\n /** display toolbox */\r\n editor.nodes.toolbox.classList.add('opened');\r\n\r\n /** Animate plus button */\r\n editor.nodes.plusButton.classList.add('clicked');\r\n\r\n /** toolbox state */\r\n editor.toolbar.toolbox.opened = true;\r\n\r\n };\r\n\r\n /** Closes toolbox */\r\n toolbox.close = function () {\r\n\r\n /** Makes toolbox disapear */\r\n editor.nodes.toolbox.classList.remove('opened');\r\n\r\n /** Rotate plus button */\r\n editor.nodes.plusButton.classList.remove('clicked');\r\n\r\n /** toolbox state */\r\n editor.toolbar.toolbox.opened = false;\r\n\r\n editor.toolbar.current = null;\r\n\r\n };\r\n\r\n toolbox.leaf = function () {\r\n\r\n let currentTool = editor.toolbar.current,\r\n tools = Object.keys(editor.tools),\r\n barButtons = editor.nodes.toolbarButtons,\r\n nextToolIndex = 0,\r\n toolToSelect,\r\n visibleTool,\r\n tool;\r\n\r\n if ( !currentTool ) {\r\n\r\n /** Get first tool from object*/\r\n for(tool in editor.tools) {\r\n\r\n if (editor.tools[tool].displayInToolbox) {\r\n\r\n break;\r\n\r\n }\r\n\r\n nextToolIndex ++;\r\n\r\n }\r\n\r\n } else {\r\n\r\n nextToolIndex = (tools.indexOf(currentTool) + 1) % tools.length;\r\n visibleTool = tools[nextToolIndex];\r\n\r\n while (!editor.tools[visibleTool].displayInToolbox) {\r\n\r\n nextToolIndex = (nextToolIndex + 1) % tools.length;\r\n visibleTool = tools[nextToolIndex];\r\n\r\n }\r\n\r\n }\r\n\r\n toolToSelect = tools[nextToolIndex];\r\n\r\n for ( var button in barButtons ) {\r\n\r\n barButtons[button].classList.remove('selected');\r\n\r\n }\r\n\r\n barButtons[toolToSelect].classList.add('selected');\r\n editor.toolbar.current = toolToSelect;\r\n\r\n };\r\n\r\n /**\r\n * Transforming selected node type into selected toolbar element type\r\n * @param {event} event\r\n */\r\n toolbox.toolClicked = function (event) {\r\n\r\n /**\r\n * UNREPLACEBLE_TOOLS this types of tools are forbidden to replace even they are empty\r\n */\r\n var UNREPLACEBLE_TOOLS = ['image', 'link', 'list', 'instagram', 'twitter', 'embed'],\r\n tool = editor.tools[editor.toolbar.current],\r\n workingNode = editor.content.currentNode,\r\n currentInputIndex = editor.caret.inputIndex,\r\n newBlockContent,\r\n appendCallback,\r\n blockData;\r\n\r\n /** Make block from plugin */\r\n newBlockContent = tool.render();\r\n\r\n /** information about block */\r\n blockData = {\r\n block : newBlockContent,\r\n type : tool.type,\r\n stretched : false\r\n };\r\n\r\n if (\r\n workingNode &&\r\n UNREPLACEBLE_TOOLS.indexOf(workingNode.dataset.tool) === -1 &&\r\n workingNode.textContent.trim() === ''\r\n ) {\r\n\r\n /** Replace current block */\r\n editor.content.switchBlock(workingNode, newBlockContent, tool.type);\r\n\r\n } else {\r\n\r\n /** Insert new Block from plugin */\r\n editor.content.insertBlock(blockData);\r\n\r\n /** increase input index */\r\n currentInputIndex++;\r\n\r\n }\r\n\r\n /** Fire tool append callback */\r\n appendCallback = tool.appendCallback;\r\n\r\n if (appendCallback && typeof appendCallback == 'function') {\r\n\r\n appendCallback.call(event);\r\n\r\n }\r\n\r\n window.setTimeout(function () {\r\n\r\n /** Set caret to current block */\r\n editor.caret.setToBlock(currentInputIndex);\r\n\r\n }, 10);\r\n\r\n\r\n /**\r\n * Changing current Node\r\n */\r\n editor.content.workingNodeChanged();\r\n\r\n /**\r\n * Move toolbar when node is changed\r\n */\r\n editor.toolbar.move();\r\n\r\n };\r\n\r\n return toolbox;\r\n\r\n})({});\n\n\n// WEBPACK FOOTER //\n// ./modules/toolbar/toolbox.js","/**\r\n * @module Codex Editor Callbacks module\r\n * @description Module works with editor added Elements\r\n *\r\n * @author Codex Team\r\n * @version 1.4.0\r\n */\r\n\r\nmodule.exports = (function (callbacks) {\r\n\r\n let editor = codex.editor;\r\n\r\n /**\r\n * used by UI module\r\n * @description Routes all keydowns on document\r\n * @param {Object} event\r\n */\r\n callbacks.globalKeydown = function (event) {\r\n\r\n switch (event.keyCode) {\r\n case editor.core.keys.ENTER : enterKeyPressed_(event); break;\r\n }\r\n\r\n };\r\n\r\n /**\r\n * used by UI module\r\n * @description Routes all keydowns on redactors area\r\n * @param {Object} event\r\n */\r\n callbacks.redactorKeyDown = function (event) {\r\n\r\n switch (event.keyCode) {\r\n case editor.core.keys.TAB : tabKeyPressedOnRedactorsZone_(event); break;\r\n case editor.core.keys.ENTER : enterKeyPressedOnRedactorsZone_(event); break;\r\n case editor.core.keys.ESC : escapeKeyPressedOnRedactorsZone_(event); break;\r\n default : defaultKeyPressedOnRedactorsZone_(event); break;\r\n }\r\n\r\n };\r\n\r\n /**\r\n * used by UI module\r\n * @description Routes all keyup events\r\n * @param {Object} event\r\n */\r\n callbacks.globalKeyup = function (event) {\r\n\r\n switch (event.keyCode) {\r\n case editor.core.keys.UP :\r\n case editor.core.keys.LEFT :\r\n case editor.core.keys.RIGHT :\r\n case editor.core.keys.DOWN : arrowKeyPressed_(event); break;\r\n }\r\n\r\n };\r\n\r\n /**\r\n * @param {Object} event\r\n * @private\r\n *\r\n * Handles behaviour when tab pressed\r\n * @description if Content is empty show toolbox (if it is closed) or leaf tools\r\n * uses Toolbars toolbox module to handle the situation\r\n */\r\n var tabKeyPressedOnRedactorsZone_ = function (event) {\r\n\r\n /**\r\n * Wait for solution. Would like to know the behaviour\r\n * @todo Add spaces\r\n */\r\n event.preventDefault();\r\n\r\n var nativeInputs = editor.content.currentNode.querySelectorAll('textarea, input'),\r\n nativeInputsAreEmpty = true,\r\n textContentIsEmpty = !editor.content.currentNode.textContent.trim();\r\n\r\n Array.prototype.map.call(nativeInputs, function (input) {\r\n\r\n if (input.type == 'textarea' || input.type == 'text') {\r\n\r\n nativeInputsAreEmpty = nativeInputsAreEmpty && !input.value.trim();\r\n\r\n }\r\n\r\n });\r\n\r\n var blockIsEmpty = textContentIsEmpty && nativeInputsAreEmpty;\r\n\r\n if (!blockIsEmpty) {\r\n\r\n return;\r\n\r\n }\r\n\r\n if ( !editor.toolbar.opened ) {\r\n\r\n editor.toolbar.open();\r\n\r\n }\r\n\r\n if (editor.toolbar.opened && !editor.toolbar.toolbox.opened) {\r\n\r\n editor.toolbar.toolbox.open();\r\n\r\n } else {\r\n\r\n editor.toolbar.toolbox.leaf();\r\n\r\n }\r\n\r\n };\r\n\r\n /**\r\n * Handles global EnterKey Press\r\n * @see enterPressedOnBlock_\r\n * @param {Object} event\r\n */\r\n var enterKeyPressed_ = function () {\r\n\r\n if (editor.content.editorAreaHightlighted) {\r\n\r\n /**\r\n * it means that we lose input index, saved index before is not correct\r\n * therefore we need to set caret when we insert new block\r\n */\r\n editor.caret.inputIndex = -1;\r\n\r\n enterPressedOnBlock_();\r\n\r\n }\r\n\r\n };\r\n\r\n /**\r\n * Callback for enter key pressing in first-level block area\r\n *\r\n * @param {Event} event\r\n * @private\r\n *\r\n * @description Inserts new block with initial type from settings\r\n */\r\n var enterPressedOnBlock_ = function () {\r\n\r\n var NEW_BLOCK_TYPE = editor.settings.initialBlockPlugin;\r\n\r\n editor.content.insertBlock({\r\n type : NEW_BLOCK_TYPE,\r\n block : editor.tools[NEW_BLOCK_TYPE].render()\r\n }, true );\r\n\r\n editor.toolbar.move();\r\n editor.toolbar.open();\r\n\r\n };\r\n\r\n\r\n /**\r\n * ENTER key handler\r\n *\r\n * @param {Object} event\r\n * @private\r\n *\r\n * @description Makes new block with initial type from settings\r\n */\r\n var enterKeyPressedOnRedactorsZone_ = function (event) {\r\n\r\n if (event.target.contentEditable == 'true') {\r\n\r\n /** Update input index */\r\n editor.caret.saveCurrentInputIndex();\r\n\r\n }\r\n\r\n var currentInputIndex = editor.caret.getCurrentInputIndex() || 0,\r\n workingNode = editor.content.currentNode,\r\n tool = workingNode.dataset.tool,\r\n isEnterPressedOnToolbar = editor.toolbar.opened &&\r\n editor.toolbar.current &&\r\n event.target == editor.state.inputs[currentInputIndex];\r\n\r\n /** The list of tools which needs the default browser behaviour */\r\n var enableLineBreaks = editor.tools[tool].enableLineBreaks;\r\n\r\n /** This type of block creates when enter is pressed */\r\n var NEW_BLOCK_TYPE = editor.settings.initialBlockPlugin;\r\n\r\n /**\r\n * When toolbar is opened, select tool instead of making new paragraph\r\n */\r\n if ( isEnterPressedOnToolbar ) {\r\n\r\n event.preventDefault();\r\n\r\n editor.toolbar.toolbox.toolClicked(event);\r\n\r\n editor.toolbar.close();\r\n\r\n /**\r\n * Stop other listeners callback executions\r\n */\r\n event.stopPropagation();\r\n event.stopImmediatePropagation();\r\n\r\n return;\r\n\r\n }\r\n\r\n /**\r\n * Allow paragraph lineBreaks with shift enter\r\n * Or if shiftkey pressed and enter and enabledLineBreaks, the let new block creation\r\n */\r\n if ( event.shiftKey || enableLineBreaks ) {\r\n\r\n event.stopPropagation();\r\n event.stopImmediatePropagation();\r\n return;\r\n\r\n }\r\n\r\n var currentSelection = window.getSelection(),\r\n currentSelectedNode = currentSelection.anchorNode,\r\n caretAtTheEndOfText = editor.caret.position.atTheEnd(),\r\n isTextNodeHasParentBetweenContenteditable = false;\r\n\r\n /**\r\n * Allow making new

    in same block by SHIFT+ENTER and forbids to prevent default browser behaviour\r\n */\r\n if ( event.shiftKey && !enableLineBreaks ) {\r\n\r\n editor.callback.enterPressedOnBlock(editor.content.currentBlock, event);\r\n event.preventDefault();\r\n return;\r\n\r\n }\r\n\r\n /**\r\n * Workaround situation when caret at the Text node that has some wrapper Elements\r\n * Split block cant handle this.\r\n * We need to save default behavior\r\n */\r\n isTextNodeHasParentBetweenContenteditable = currentSelectedNode && currentSelectedNode.parentNode.contentEditable != 'true';\r\n\r\n /**\r\n * Split blocks when input has several nodes and caret placed in textNode\r\n */\r\n if (\r\n currentSelectedNode.nodeType == editor.core.nodeTypes.TEXT &&\r\n !isTextNodeHasParentBetweenContenteditable &&\r\n !caretAtTheEndOfText\r\n ) {\r\n\r\n event.preventDefault();\r\n\r\n editor.core.log('Splitting Text node...');\r\n\r\n editor.content.splitBlock(currentInputIndex);\r\n\r\n /** Show plus button when next input after split is empty*/\r\n if (!editor.state.inputs[currentInputIndex + 1].textContent.trim()) {\r\n\r\n editor.toolbar.showPlusButton();\r\n\r\n }\r\n\r\n } else {\r\n\r\n var islastNode = editor.content.isLastNode(currentSelectedNode);\r\n\r\n if ( islastNode && caretAtTheEndOfText ) {\r\n\r\n event.preventDefault();\r\n event.stopPropagation();\r\n event.stopImmediatePropagation();\r\n\r\n editor.core.log('ENTER clicked in last textNode. Create new BLOCK');\r\n\r\n editor.content.insertBlock({\r\n type: NEW_BLOCK_TYPE,\r\n block: editor.tools[NEW_BLOCK_TYPE].render()\r\n }, true);\r\n\r\n editor.toolbar.move();\r\n editor.toolbar.open();\r\n\r\n /** Show plus button with empty block */\r\n editor.toolbar.showPlusButton();\r\n\r\n }\r\n\r\n }\r\n\r\n /** get all inputs after new appending block */\r\n editor.ui.saveInputs();\r\n\r\n };\r\n\r\n /**\r\n * Escape behaviour\r\n * @param event\r\n * @private\r\n *\r\n * @description Closes toolbox and toolbar. Prevents default behaviour\r\n */\r\n var escapeKeyPressedOnRedactorsZone_ = function (event) {\r\n\r\n /** Close all toolbar */\r\n editor.toolbar.close();\r\n\r\n /** Close toolbox */\r\n editor.toolbar.toolbox.close();\r\n\r\n event.preventDefault();\r\n\r\n };\r\n\r\n /**\r\n * @param {Event} event\r\n * @private\r\n *\r\n * closes and moves toolbar\r\n */\r\n var arrowKeyPressed_ = function (event) {\r\n\r\n editor.content.workingNodeChanged();\r\n\r\n /* Closing toolbar */\r\n editor.toolbar.close();\r\n editor.toolbar.move();\r\n\r\n };\r\n\r\n /**\r\n * @private\r\n * @param {Event} event\r\n *\r\n * @description Closes all opened bars from toolbar.\r\n * If block is mark, clears highlightning\r\n */\r\n var defaultKeyPressedOnRedactorsZone_ = function () {\r\n\r\n editor.toolbar.close();\r\n\r\n if (!editor.toolbar.inline.actionsOpened) {\r\n\r\n editor.toolbar.inline.close();\r\n editor.content.clearMark();\r\n\r\n }\r\n\r\n };\r\n\r\n /**\r\n * Handler when clicked on redactors area\r\n *\r\n * @protected\r\n * @param event\r\n *\r\n * @description Detects clicked area. If it is first-level block area, marks as detected and\r\n * on next enter press will be inserted new block\r\n * Otherwise, save carets position (input index) and put caret to the editable zone.\r\n *\r\n * @see detectWhenClickedOnFirstLevelBlockArea_\r\n *\r\n */\r\n callbacks.redactorClicked = function (event) {\r\n\r\n detectWhenClickedOnFirstLevelBlockArea_();\r\n\r\n editor.content.workingNodeChanged(event.target);\r\n editor.ui.saveInputs();\r\n\r\n var selectedText = editor.toolbar.inline.getSelectionText(),\r\n firstLevelBlock;\r\n\r\n /** If selection range took off, then we hide inline toolbar */\r\n if (selectedText.length === 0) {\r\n\r\n editor.toolbar.inline.close();\r\n\r\n }\r\n\r\n /** Update current input index in memory when caret focused into existed input */\r\n if (event.target.contentEditable == 'true') {\r\n\r\n editor.caret.saveCurrentInputIndex();\r\n\r\n }\r\n\r\n if (editor.content.currentNode === null) {\r\n\r\n /**\r\n * If inputs in redactor does not exits, then we put input index 0 not -1\r\n */\r\n var indexOfLastInput = editor.state.inputs.length > 0 ? editor.state.inputs.length - 1 : 0;\r\n\r\n /** If we have any inputs */\r\n if (editor.state.inputs.length) {\r\n\r\n /** getting firstlevel parent of input */\r\n firstLevelBlock = editor.content.getFirstLevelBlock(editor.state.inputs[indexOfLastInput]);\r\n\r\n }\r\n\r\n /** If input is empty, then we set caret to the last input */\r\n if (editor.state.inputs.length && editor.state.inputs[indexOfLastInput].textContent === '' && firstLevelBlock.dataset.tool == editor.settings.initialBlockPlugin) {\r\n\r\n editor.caret.setToBlock(indexOfLastInput);\r\n\r\n } else {\r\n\r\n /** Create new input when caret clicked in redactors area */\r\n var NEW_BLOCK_TYPE = editor.settings.initialBlockPlugin;\r\n\r\n editor.content.insertBlock({\r\n type : NEW_BLOCK_TYPE,\r\n block : editor.tools[NEW_BLOCK_TYPE].render()\r\n });\r\n\r\n /** If there is no inputs except inserted */\r\n if (editor.state.inputs.length === 1) {\r\n\r\n editor.caret.setToBlock(indexOfLastInput);\r\n\r\n } else {\r\n\r\n /** Set caret to this appended input */\r\n editor.caret.setToNextBlock(indexOfLastInput);\r\n\r\n }\r\n\r\n }\r\n\r\n } else {\r\n\r\n /** Close all panels */\r\n editor.toolbar.settings.close();\r\n editor.toolbar.toolbox.close();\r\n\r\n }\r\n\r\n /**\r\n * Move toolbar and open\r\n */\r\n editor.toolbar.move();\r\n editor.toolbar.open();\r\n\r\n var inputIsEmpty = !editor.content.currentNode.textContent.trim(),\r\n currentNodeType = editor.content.currentNode.dataset.tool,\r\n isInitialType = currentNodeType == editor.settings.initialBlockPlugin;\r\n\r\n\r\n /** Hide plus buttons */\r\n editor.toolbar.hidePlusButton();\r\n\r\n if (!inputIsEmpty) {\r\n\r\n /** Mark current block */\r\n editor.content.markBlock();\r\n\r\n }\r\n\r\n if ( isInitialType && inputIsEmpty ) {\r\n\r\n /** Show plus button */\r\n editor.toolbar.showPlusButton();\r\n\r\n }\r\n\r\n\r\n };\r\n\r\n /**\r\n * This method allows to define, is caret in contenteditable element or not.\r\n *\r\n * @private\r\n *\r\n * @description Otherwise, if we get TEXT node from range container, that will means we have input index.\r\n * In this case we use default browsers behaviour (if plugin allows that) or overwritten action.\r\n * Therefore, to be sure that we've clicked first-level block area, we should have currentNode, which always\r\n * specifies to the first-level block. Other cases we just ignore.\r\n */\r\n var detectWhenClickedOnFirstLevelBlockArea_ = function () {\r\n\r\n var selection = window.getSelection(),\r\n anchorNode = selection.anchorNode,\r\n flag = false;\r\n\r\n if (selection.rangeCount === 0) {\r\n\r\n editor.content.editorAreaHightlighted = true;\r\n\r\n } else {\r\n\r\n if (!editor.core.isDomNode(anchorNode)) {\r\n\r\n anchorNode = anchorNode.parentNode;\r\n\r\n }\r\n\r\n /** Already founded, without loop */\r\n if (anchorNode.contentEditable == 'true') {\r\n\r\n flag = true;\r\n\r\n }\r\n\r\n while (anchorNode.contentEditable != 'true') {\r\n\r\n anchorNode = anchorNode.parentNode;\r\n\r\n if (anchorNode.contentEditable == 'true') {\r\n\r\n flag = true;\r\n\r\n }\r\n\r\n if (anchorNode == document.body) {\r\n\r\n break;\r\n\r\n }\r\n\r\n }\r\n\r\n /** If editable element founded, flag is \"TRUE\", Therefore we return \"FALSE\" */\r\n editor.content.editorAreaHightlighted = !flag;\r\n\r\n }\r\n\r\n };\r\n\r\n /**\r\n * Toolbar button click handler\r\n *\r\n * @param {Object} event - cursor to the button\r\n * @protected\r\n *\r\n * @description gets current tool and calls render method\r\n */\r\n callbacks.toolbarButtonClicked = function (event) {\r\n\r\n var button = this;\r\n\r\n editor.toolbar.current = button.dataset.type;\r\n\r\n editor.toolbar.toolbox.toolClicked(event);\r\n editor.toolbar.close();\r\n\r\n };\r\n\r\n /**\r\n * Show or Hide toolbox when plus button is clicked\r\n */\r\n callbacks.plusButtonClicked = function () {\r\n\r\n if (!editor.nodes.toolbox.classList.contains('opened')) {\r\n\r\n editor.toolbar.toolbox.open();\r\n\r\n } else {\r\n\r\n editor.toolbar.toolbox.close();\r\n\r\n }\r\n\r\n };\r\n\r\n /**\r\n * Block handlers for KeyDown events\r\n *\r\n * @protected\r\n * @param {Object} event\r\n *\r\n * Handles keydowns on block\r\n * @see blockRightOrDownArrowPressed_\r\n * @see backspacePressed_\r\n * @see blockLeftOrUpArrowPressed_\r\n */\r\n callbacks.blockKeydown = function (event) {\r\n\r\n let block = event.target; // event.target is input\r\n\r\n switch (event.keyCode) {\r\n\r\n case editor.core.keys.DOWN:\r\n case editor.core.keys.RIGHT:\r\n blockRightOrDownArrowPressed_(event);\r\n break;\r\n\r\n case editor.core.keys.BACKSPACE:\r\n backspacePressed_(block, event);\r\n break;\r\n\r\n case editor.core.keys.UP:\r\n case editor.core.keys.LEFT:\r\n blockLeftOrUpArrowPressed_(event);\r\n break;\r\n\r\n }\r\n\r\n };\r\n\r\n /**\r\n * RIGHT or DOWN keydowns on block\r\n *\r\n * @param {Object} event\r\n * @private\r\n *\r\n * @description watches the selection and gets closest editable element.\r\n * Uses method getDeepestTextNodeFromPosition to get the last node of next block\r\n * Sets caret if it is contenteditable\r\n */\r\n var blockRightOrDownArrowPressed_ = function (event) {\r\n\r\n var selection = window.getSelection(),\r\n inputs = editor.state.inputs,\r\n focusedNode = selection.anchorNode,\r\n focusedNodeHolder;\r\n\r\n /** Check for caret existance */\r\n if (!focusedNode) {\r\n\r\n return false;\r\n\r\n }\r\n\r\n /** Looking for closest (parent) contentEditable element of focused node */\r\n while (focusedNode.contentEditable != 'true') {\r\n\r\n focusedNodeHolder = focusedNode.parentNode;\r\n focusedNode = focusedNodeHolder;\r\n\r\n }\r\n\r\n /** Input index in DOM level */\r\n var editableElementIndex = 0;\r\n\r\n while (focusedNode != inputs[editableElementIndex]) {\r\n\r\n editableElementIndex ++;\r\n\r\n }\r\n\r\n /**\r\n * Founded contentEditable element doesn't have childs\r\n * Or maybe New created block\r\n */\r\n if (!focusedNode.textContent) {\r\n\r\n editor.caret.setToNextBlock(editableElementIndex);\r\n return;\r\n\r\n }\r\n\r\n /**\r\n * Do nothing when caret doesn not reaches the end of last child\r\n */\r\n var caretInLastChild = false,\r\n caretAtTheEndOfText = false;\r\n\r\n var lastChild,\r\n deepestTextnode;\r\n\r\n lastChild = focusedNode.childNodes[focusedNode.childNodes.length - 1 ];\r\n\r\n if (editor.core.isDomNode(lastChild)) {\r\n\r\n deepestTextnode = editor.content.getDeepestTextNodeFromPosition(lastChild, lastChild.childNodes.length);\r\n\r\n } else {\r\n\r\n deepestTextnode = lastChild;\r\n\r\n }\r\n\r\n caretInLastChild = selection.anchorNode == deepestTextnode;\r\n caretAtTheEndOfText = deepestTextnode.length == selection.anchorOffset;\r\n\r\n if ( !caretInLastChild || !caretAtTheEndOfText ) {\r\n\r\n editor.core.log('arrow [down|right] : caret does not reached the end');\r\n return false;\r\n\r\n }\r\n\r\n editor.caret.setToNextBlock(editableElementIndex);\r\n\r\n };\r\n\r\n /**\r\n * LEFT or UP keydowns on block\r\n *\r\n * @param {Object} event\r\n * @private\r\n *\r\n * watches the selection and gets closest editable element.\r\n * Uses method getDeepestTextNodeFromPosition to get the last node of previous block\r\n * Sets caret if it is contenteditable\r\n *\r\n */\r\n var blockLeftOrUpArrowPressed_ = function (event) {\r\n\r\n var selection = window.getSelection(),\r\n inputs = editor.state.inputs,\r\n focusedNode = selection.anchorNode,\r\n focusedNodeHolder;\r\n\r\n /** Check for caret existance */\r\n if (!focusedNode) {\r\n\r\n return false;\r\n\r\n }\r\n\r\n /**\r\n * LEFT or UP not at the beginning\r\n */\r\n if ( selection.anchorOffset !== 0) {\r\n\r\n return false;\r\n\r\n }\r\n\r\n /** Looking for parent contentEditable block */\r\n while (focusedNode.contentEditable != 'true') {\r\n\r\n focusedNodeHolder = focusedNode.parentNode;\r\n focusedNode = focusedNodeHolder;\r\n\r\n }\r\n\r\n /** Input index in DOM level */\r\n var editableElementIndex = 0;\r\n\r\n while (focusedNode != inputs[editableElementIndex]) {\r\n\r\n editableElementIndex ++;\r\n\r\n }\r\n\r\n /**\r\n * Do nothing if caret is not at the beginning of first child\r\n */\r\n var caretInFirstChild = false,\r\n caretAtTheBeginning = false;\r\n\r\n var firstChild,\r\n deepestTextnode;\r\n\r\n /**\r\n * Founded contentEditable element doesn't have childs\r\n * Or maybe New created block\r\n */\r\n if (!focusedNode.textContent) {\r\n\r\n editor.caret.setToPreviousBlock(editableElementIndex);\r\n return;\r\n\r\n }\r\n\r\n firstChild = focusedNode.childNodes[0];\r\n\r\n if (editor.core.isDomNode(firstChild)) {\r\n\r\n deepestTextnode = editor.content.getDeepestTextNodeFromPosition(firstChild, 0);\r\n\r\n } else {\r\n\r\n deepestTextnode = firstChild;\r\n\r\n }\r\n\r\n caretInFirstChild = selection.anchorNode == deepestTextnode;\r\n caretAtTheBeginning = selection.anchorOffset === 0;\r\n\r\n if ( caretInFirstChild && caretAtTheBeginning ) {\r\n\r\n editor.caret.setToPreviousBlock(editableElementIndex);\r\n\r\n }\r\n\r\n };\r\n\r\n /**\r\n * Handles backspace keydown\r\n *\r\n * @param {Element} block\r\n * @param {Object} event\r\n * @private\r\n *\r\n * @description if block is empty, delete the block and set caret to the previous block\r\n * If block is not empty, try to merge two blocks - current and previous\r\n * But it we try'n to remove first block, then we should set caret to the next block, not previous.\r\n * If we removed the last block, create new one\r\n */\r\n var backspacePressed_ = function (block, event) {\r\n\r\n var currentInputIndex = editor.caret.getCurrentInputIndex(),\r\n range,\r\n selectionLength,\r\n firstLevelBlocksCount;\r\n\r\n if (editor.core.isNativeInput(event.target)) {\r\n\r\n /** If input value is empty - remove block */\r\n if (event.target.value.trim() == '') {\r\n\r\n block.remove();\r\n\r\n } else {\r\n\r\n return;\r\n\r\n }\r\n\r\n }\r\n\r\n if (block.textContent.trim()) {\r\n\r\n range = editor.content.getRange();\r\n selectionLength = range.endOffset - range.startOffset;\r\n\r\n if (editor.caret.position.atStart() && !selectionLength && editor.state.inputs[currentInputIndex - 1]) {\r\n\r\n editor.content.mergeBlocks(currentInputIndex);\r\n\r\n } else {\r\n\r\n return;\r\n\r\n }\r\n\r\n }\r\n\r\n if (!selectionLength) {\r\n\r\n block.remove();\r\n\r\n }\r\n\r\n\r\n firstLevelBlocksCount = editor.nodes.redactor.childNodes.length;\r\n\r\n /**\r\n * If all blocks are removed\r\n */\r\n if (firstLevelBlocksCount === 0) {\r\n\r\n /** update currentNode variable */\r\n editor.content.currentNode = null;\r\n\r\n /** Inserting new empty initial block */\r\n editor.ui.addInitialBlock();\r\n\r\n /** Updating inputs state after deleting last block */\r\n editor.ui.saveInputs();\r\n\r\n /** Set to current appended block */\r\n window.setTimeout(function () {\r\n\r\n editor.caret.setToPreviousBlock(1);\r\n\r\n }, 10);\r\n\r\n } else {\r\n\r\n if (editor.caret.inputIndex !== 0) {\r\n\r\n /** Target block is not first */\r\n editor.caret.setToPreviousBlock(editor.caret.inputIndex);\r\n\r\n } else {\r\n\r\n /** If we try to delete first block */\r\n editor.caret.setToNextBlock(editor.caret.inputIndex);\r\n\r\n }\r\n\r\n }\r\n\r\n editor.toolbar.move();\r\n\r\n if (!editor.toolbar.opened) {\r\n\r\n editor.toolbar.open();\r\n\r\n }\r\n\r\n /** Updating inputs state */\r\n editor.ui.saveInputs();\r\n\r\n /** Prevent default browser behaviour */\r\n event.preventDefault();\r\n\r\n };\r\n\r\n /**\r\n * used by UI module\r\n * Clicks on block settings button\r\n *\r\n * @param {Object} event\r\n * @protected\r\n * @description Opens toolbar settings\r\n */\r\n callbacks.showSettingsButtonClicked = function (event) {\r\n\r\n /**\r\n * Get type of current block\r\n * It uses to append settings from tool.settings property.\r\n * ...\r\n * Type is stored in data-type attribute on block\r\n */\r\n var currentToolType = editor.content.currentNode.dataset.tool;\r\n\r\n editor.toolbar.settings.toggle(currentToolType);\r\n\r\n /** Close toolbox when settings button is active */\r\n editor.toolbar.toolbox.close();\r\n editor.toolbar.settings.hideRemoveActions();\r\n\r\n };\r\n\r\n return callbacks;\r\n\r\n})({});\n\n\n// WEBPACK FOOTER //\n// ./modules/callbacks.js","/**\r\n * Codex Editor Draw module\r\n *\r\n * @author Codex Team\r\n * @version 1.0.\r\n */\r\n\r\nmodule.exports = (function (draw) {\r\n\r\n /**\r\n * Base editor wrapper\r\n */\r\n draw.wrapper = function () {\r\n\r\n var wrapper = document.createElement('div');\r\n\r\n wrapper.className += 'codex-editor';\r\n\r\n return wrapper;\r\n\r\n };\r\n\r\n /**\r\n * Content-editable holder\r\n */\r\n draw.redactor = function () {\r\n\r\n var redactor = document.createElement('div');\r\n\r\n redactor.className += 'ce-redactor';\r\n\r\n return redactor;\r\n\r\n };\r\n\r\n draw.ceBlock = function () {\r\n\r\n var block = document.createElement('DIV');\r\n\r\n block.className += 'ce_block';\r\n\r\n return block;\r\n\r\n };\r\n\r\n /**\r\n * Empty toolbar with toggler\r\n */\r\n draw.toolbar = function () {\r\n\r\n var bar = document.createElement('div');\r\n\r\n bar.className += 'ce-toolbar';\r\n\r\n return bar;\r\n\r\n };\r\n\r\n draw.toolbarContent = function () {\r\n\r\n var wrapper = document.createElement('DIV');\r\n\r\n wrapper.classList.add('ce-toolbar__content');\r\n\r\n return wrapper;\r\n\r\n };\r\n\r\n /**\r\n * Inline toolbar\r\n */\r\n draw.inlineToolbar = function () {\r\n\r\n var bar = document.createElement('DIV');\r\n\r\n bar.className += 'ce-toolbar-inline';\r\n\r\n return bar;\r\n\r\n };\r\n\r\n /**\r\n * Wrapper for inline toobar buttons\r\n */\r\n draw.inlineToolbarButtons = function () {\r\n\r\n var wrapper = document.createElement('DIV');\r\n\r\n wrapper.className += 'ce-toolbar-inline__buttons';\r\n\r\n return wrapper;\r\n\r\n };\r\n\r\n /**\r\n * For some actions\r\n */\r\n draw.inlineToolbarActions = function () {\r\n\r\n var wrapper = document.createElement('DIV');\r\n\r\n wrapper.className += 'ce-toolbar-inline__actions';\r\n\r\n return wrapper;\r\n\r\n };\r\n\r\n draw.inputForLink = function () {\r\n\r\n var input = document.createElement('INPUT');\r\n\r\n input.type = 'input';\r\n input.className += 'inputForLink';\r\n input.placeholder = 'Вставьте ссылку ...';\r\n input.setAttribute('form', 'defaultForm');\r\n\r\n input.setAttribute('autofocus', 'autofocus');\r\n\r\n return input;\r\n\r\n };\r\n\r\n /**\r\n * @todo Desc\r\n */\r\n draw.blockButtons = function () {\r\n\r\n var block = document.createElement('div');\r\n\r\n block.className += 'ce-toolbar__actions';\r\n\r\n return block;\r\n\r\n };\r\n\r\n /**\r\n * Block settings panel\r\n */\r\n draw.blockSettings = function () {\r\n\r\n var settings = document.createElement('div');\r\n\r\n settings.className += 'ce-settings';\r\n\r\n return settings;\r\n\r\n };\r\n\r\n draw.defaultSettings = function () {\r\n\r\n var div = document.createElement('div');\r\n\r\n div.classList.add('ce-settings_default');\r\n\r\n return div;\r\n\r\n };\r\n\r\n draw.pluginsSettings = function () {\r\n\r\n var div = document.createElement('div');\r\n\r\n div.classList.add('ce-settings_plugin');\r\n\r\n return div;\r\n\r\n };\r\n\r\n draw.plusButton = function () {\r\n\r\n var button = document.createElement('span');\r\n\r\n button.className = 'ce-toolbar__plus';\r\n // button.innerHTML = '';\r\n\r\n return button;\r\n\r\n };\r\n\r\n /**\r\n * Settings button in toolbar\r\n */\r\n draw.settingsButton = function () {\r\n\r\n var toggler = document.createElement('span');\r\n\r\n toggler.className = 'ce-toolbar__settings-btn';\r\n\r\n /** Toggler button*/\r\n toggler.innerHTML = '';\r\n\r\n return toggler;\r\n\r\n };\r\n\r\n /**\r\n * Redactor tools wrapper\r\n */\r\n\r\n draw.toolbox = function () {\r\n\r\n var wrapper = document.createElement('div');\r\n\r\n wrapper.className = 'ce-toolbar__tools';\r\n\r\n return wrapper;\r\n\r\n };\r\n\r\n /**\r\n * @protected\r\n *\r\n * Draws tool buttons for toolbox\r\n *\r\n * @param {String} type\r\n * @param {String} classname\r\n * @returns {Element}\r\n */\r\n draw.toolbarButton = function (type, classname) {\r\n\r\n var button = document.createElement('li'),\r\n toolIcon = document.createElement('i'),\r\n toolTitle = document.createElement('span');\r\n\r\n button.dataset.type = type;\r\n button.setAttribute('title', type);\r\n\r\n toolIcon.classList.add(classname);\r\n toolTitle.classList.add('ce_toolbar_tools--title');\r\n\r\n\r\n button.appendChild(toolIcon);\r\n button.appendChild(toolTitle);\r\n\r\n return button;\r\n\r\n };\r\n\r\n /**\r\n * @protected\r\n *\r\n * Draws tools for inline toolbar\r\n *\r\n * @param {String} type\r\n * @param {String} classname\r\n */\r\n draw.toolbarButtonInline = function (type, classname) {\r\n\r\n var button = document.createElement('BUTTON'),\r\n toolIcon = document.createElement('I');\r\n\r\n button.type = 'button';\r\n button.dataset.type = type;\r\n toolIcon.classList.add(classname);\r\n\r\n button.appendChild(toolIcon);\r\n\r\n return button;\r\n\r\n };\r\n\r\n /**\r\n * Redactor block\r\n */\r\n draw.block = function (tagName, content) {\r\n\r\n var node = document.createElement(tagName);\r\n\r\n node.innerHTML = content || '';\r\n\r\n return node;\r\n\r\n };\r\n\r\n /**\r\n * Creates Node with passed tagName and className\r\n * @param {string} tagName\r\n * @param {string} className\r\n * @param {object} properties - allow to assign properties\r\n */\r\n draw.node = function ( tagName, className, properties ) {\r\n\r\n var el = document.createElement( tagName );\r\n\r\n if ( className ) el.className = className;\r\n\r\n if ( properties ) {\r\n\r\n for (var name in properties) {\r\n\r\n el[name] = properties[name];\r\n\r\n }\r\n\r\n }\r\n\r\n return el;\r\n\r\n };\r\n\r\n /**\r\n * Unavailable plugin block\r\n */\r\n draw.unavailableBlock = function () {\r\n\r\n var wrapper = document.createElement('DIV');\r\n\r\n wrapper.classList.add('cdx-unavailable-block');\r\n\r\n return wrapper;\r\n\r\n };\r\n\r\n return draw;\r\n\r\n})({});\n\n\n// WEBPACK FOOTER //\n// ./modules/draw.js","/**\r\n * Codex Editor Caret Module\r\n *\r\n * @author Codex Team\r\n * @version 1.0\r\n */\r\n\r\nmodule.exports = (function (caret) {\r\n\r\n let editor = codex.editor;\r\n\r\n /**\r\n * @var {int} InputIndex - editable element in DOM\r\n */\r\n caret.inputIndex = null;\r\n\r\n /**\r\n * @var {int} offset - caret position in a text node.\r\n */\r\n caret.offset = null;\r\n\r\n /**\r\n * @var {int} focusedNodeIndex - we get index of child node from first-level block\r\n */\r\n caret.focusedNodeIndex = null;\r\n\r\n /**\r\n * Creates Document Range and sets caret to the element.\r\n * @protected\r\n * @uses caret.save — if you need to save caret position\r\n * @param {Element} el - Changed Node.\r\n */\r\n caret.set = function ( el, index, offset) {\r\n\r\n offset = offset || caret.offset || 0;\r\n index = index || caret.focusedNodeIndex || 0;\r\n\r\n var childs = el.childNodes,\r\n nodeToSet;\r\n\r\n if ( childs.length === 0 ) {\r\n\r\n nodeToSet = el;\r\n\r\n } else {\r\n\r\n nodeToSet = childs[index];\r\n\r\n }\r\n\r\n /** If Element is INPUT */\r\n if (el.contentEditable != 'true') {\r\n\r\n el.focus();\r\n return;\r\n\r\n }\r\n\r\n if (editor.core.isDomNode(nodeToSet)) {\r\n\r\n nodeToSet = editor.content.getDeepestTextNodeFromPosition(nodeToSet, nodeToSet.childNodes.length);\r\n\r\n }\r\n\r\n var range = document.createRange(),\r\n selection = window.getSelection();\r\n\r\n window.setTimeout(function () {\r\n\r\n range.setStart(nodeToSet, offset);\r\n range.setEnd(nodeToSet, offset);\r\n\r\n selection.removeAllRanges();\r\n selection.addRange(range);\r\n\r\n editor.caret.saveCurrentInputIndex();\r\n\r\n }, 20);\r\n\r\n };\r\n\r\n /**\r\n * @protected\r\n * Updates index of input and saves it in caret object\r\n */\r\n caret.saveCurrentInputIndex = function () {\r\n\r\n /** Index of Input that we paste sanitized content */\r\n var selection = window.getSelection(),\r\n inputs = editor.state.inputs,\r\n focusedNode = selection.anchorNode,\r\n focusedNodeHolder;\r\n\r\n if (!focusedNode) {\r\n\r\n return;\r\n\r\n }\r\n\r\n /** Looking for parent contentEditable block */\r\n while (focusedNode.contentEditable != 'true') {\r\n\r\n focusedNodeHolder = focusedNode.parentNode;\r\n focusedNode = focusedNodeHolder;\r\n\r\n }\r\n\r\n /** Input index in DOM level */\r\n var editableElementIndex = 0;\r\n\r\n while (focusedNode != inputs[editableElementIndex]) {\r\n\r\n editableElementIndex ++;\r\n\r\n }\r\n\r\n caret.inputIndex = editableElementIndex;\r\n\r\n };\r\n\r\n /**\r\n * Returns current input index (caret object)\r\n */\r\n caret.getCurrentInputIndex = function () {\r\n\r\n return caret.inputIndex;\r\n\r\n };\r\n\r\n /**\r\n * @param {int} index - index of first-level block after that we set caret into next input\r\n */\r\n caret.setToNextBlock = function (index) {\r\n\r\n var inputs = editor.state.inputs,\r\n nextInput = inputs[index + 1];\r\n\r\n if (!nextInput) {\r\n\r\n editor.core.log('We are reached the end');\r\n return;\r\n\r\n }\r\n\r\n /**\r\n * When new Block created or deleted content of input\r\n * We should add some text node to set caret\r\n */\r\n if (!nextInput.childNodes.length) {\r\n\r\n var emptyTextElement = document.createTextNode('');\r\n\r\n nextInput.appendChild(emptyTextElement);\r\n\r\n }\r\n\r\n editor.caret.inputIndex = index + 1;\r\n editor.caret.set(nextInput, 0, 0);\r\n editor.content.workingNodeChanged(nextInput);\r\n\r\n };\r\n\r\n /**\r\n * @param {int} index - index of target input.\r\n * Sets caret to input with this index\r\n */\r\n caret.setToBlock = function (index) {\r\n\r\n var inputs = editor.state.inputs,\r\n targetInput = inputs[index];\r\n\r\n if ( !targetInput ) {\r\n\r\n return;\r\n\r\n }\r\n\r\n /**\r\n * When new Block created or deleted content of input\r\n * We should add some text node to set caret\r\n */\r\n if (!targetInput.childNodes.length) {\r\n\r\n var emptyTextElement = document.createTextNode('');\r\n\r\n targetInput.appendChild(emptyTextElement);\r\n\r\n }\r\n\r\n editor.caret.inputIndex = index;\r\n editor.caret.set(targetInput, 0, 0);\r\n editor.content.workingNodeChanged(targetInput);\r\n\r\n };\r\n\r\n /**\r\n * @param {int} index - index of input\r\n */\r\n caret.setToPreviousBlock = function (index) {\r\n\r\n index = index || 0;\r\n\r\n var inputs = editor.state.inputs,\r\n previousInput = inputs[index - 1],\r\n lastChildNode,\r\n lengthOfLastChildNode,\r\n emptyTextElement;\r\n\r\n\r\n if (!previousInput) {\r\n\r\n editor.core.log('We are reached first node');\r\n return;\r\n\r\n }\r\n\r\n lastChildNode = editor.content.getDeepestTextNodeFromPosition(previousInput, previousInput.childNodes.length);\r\n lengthOfLastChildNode = lastChildNode.length;\r\n\r\n /**\r\n * When new Block created or deleted content of input\r\n * We should add some text node to set caret\r\n */\r\n if (!previousInput.childNodes.length) {\r\n\r\n emptyTextElement = document.createTextNode('');\r\n previousInput.appendChild(emptyTextElement);\r\n\r\n }\r\n editor.caret.inputIndex = index - 1;\r\n editor.caret.set(previousInput, previousInput.childNodes.length - 1, lengthOfLastChildNode);\r\n editor.content.workingNodeChanged(inputs[index - 1]);\r\n\r\n };\r\n\r\n caret.position = {\r\n\r\n atStart : function () {\r\n\r\n var selection = window.getSelection(),\r\n anchorOffset = selection.anchorOffset,\r\n anchorNode = selection.anchorNode,\r\n firstLevelBlock = editor.content.getFirstLevelBlock(anchorNode),\r\n pluginsRender = firstLevelBlock.childNodes[0];\r\n\r\n if (!editor.core.isDomNode(anchorNode)) {\r\n\r\n anchorNode = anchorNode.parentNode;\r\n\r\n }\r\n\r\n var isFirstNode = anchorNode === pluginsRender.childNodes[0],\r\n isOffsetZero = anchorOffset === 0;\r\n\r\n return isFirstNode && isOffsetZero;\r\n\r\n },\r\n\r\n atTheEnd : function () {\r\n\r\n var selection = window.getSelection(),\r\n anchorOffset = selection.anchorOffset,\r\n anchorNode = selection.anchorNode;\r\n\r\n /** Caret is at the end of input */\r\n return !anchorNode || !anchorNode.length || anchorOffset === anchorNode.length;\r\n\r\n }\r\n };\r\n\r\n\r\n /**\r\n * Inserts node at the caret location\r\n * @param node\r\n */\r\n caret.insertNode = function (node) {\r\n\r\n var selection, range;\r\n\r\n selection = window.getSelection();\r\n\r\n range = selection.getRangeAt(0);\r\n range.deleteContents();\r\n\r\n range.insertNode(node);\r\n\r\n range.setStartAfter(node);\r\n range.collapse(true);\r\n\r\n selection.removeAllRanges();\r\n selection.addRange(range);\r\n\r\n\r\n };\r\n\r\n return caret;\r\n\r\n})({});\n\n\n// WEBPACK FOOTER //\n// ./modules/caret.js","/**\r\n * Codex Editor Notification Module\r\n *\r\n * @author Codex Team\r\n * @version 1.0\r\n */\r\n\r\nmodule.exports = (function (notifications) {\r\n\r\n let editor = codex.editor;\r\n\r\n var queue = [];\r\n\r\n var addToQueue = function (settings) {\r\n\r\n queue.push(settings);\r\n\r\n var index = 0;\r\n\r\n while ( index < queue.length && queue.length > 5) {\r\n\r\n if (queue[index].type == 'confirm' || queue[index].type == 'prompt') {\r\n\r\n index++;\r\n continue;\r\n\r\n }\r\n\r\n queue[index].close();\r\n queue.splice(index, 1);\r\n\r\n }\r\n\r\n };\r\n\r\n notifications.createHolder = function () {\r\n\r\n var holder = editor.draw.node('DIV', 'cdx-notifications-block');\r\n\r\n editor.nodes.notifications = document.body.appendChild(holder);\r\n\r\n return holder;\r\n\r\n };\r\n\r\n\r\n /**\r\n * Error notificator. Shows block with message\r\n * @protected\r\n */\r\n notifications.errorThrown = function (errorMsg, event) {\r\n\r\n editor.notifications.notification({message: 'This action is not available currently', type: event.type});\r\n\r\n };\r\n\r\n /**\r\n *\r\n * Appends notification\r\n *\r\n * settings = {\r\n * type - notification type (reserved types: alert, confirm, prompt). Just add class 'cdx-notification-'+type\r\n * message - notification message\r\n * okMsg - confirm button text (default - 'Ok')\r\n * cancelBtn - cancel button text (default - 'Cancel'). Only for confirm and prompt types\r\n * confirm - function-handler for ok button click\r\n * cancel - function-handler for cancel button click. Only for confirm and prompt types\r\n * time - time (in seconds) after which notification will close (default - 10s)\r\n * }\r\n *\r\n * @param settings\r\n */\r\n notifications.notification = function (constructorSettings) {\r\n\r\n /** Private vars and methods */\r\n var notification = null,\r\n cancel = null,\r\n type = null,\r\n confirm = null,\r\n inputField = null;\r\n\r\n var confirmHandler = function () {\r\n\r\n close();\r\n\r\n if (typeof confirm !== 'function' ) {\r\n\r\n return;\r\n\r\n }\r\n\r\n if (type == 'prompt') {\r\n\r\n confirm(inputField.value);\r\n return;\r\n\r\n }\r\n\r\n confirm();\r\n\r\n };\r\n\r\n var cancelHandler = function () {\r\n\r\n close();\r\n\r\n if (typeof cancel !== 'function' ) {\r\n\r\n return;\r\n\r\n }\r\n\r\n cancel();\r\n\r\n };\r\n\r\n\r\n /** Public methods */\r\n function create(settings) {\r\n\r\n if (!(settings && settings.message)) {\r\n\r\n editor.core.log('Can\\'t create notification. Message is missed');\r\n return;\r\n\r\n }\r\n\r\n settings.type = settings.type || 'alert';\r\n settings.time = settings.time*1000 || 10000;\r\n\r\n var wrapper = editor.draw.node('DIV', 'cdx-notification'),\r\n message = editor.draw.node('DIV', 'cdx-notification__message'),\r\n input = editor.draw.node('INPUT', 'cdx-notification__input'),\r\n okBtn = editor.draw.node('SPAN', 'cdx-notification__ok-btn'),\r\n cancelBtn = editor.draw.node('SPAN', 'cdx-notification__cancel-btn');\r\n\r\n message.textContent = settings.message;\r\n okBtn.textContent = settings.okMsg || 'ОК';\r\n cancelBtn.textContent = settings.cancelMsg || 'Отмена';\r\n\r\n editor.listeners.add(okBtn, 'click', confirmHandler);\r\n editor.listeners.add(cancelBtn, 'click', cancelHandler);\r\n\r\n wrapper.appendChild(message);\r\n\r\n if (settings.type == 'prompt') {\r\n\r\n wrapper.appendChild(input);\r\n\r\n }\r\n\r\n wrapper.appendChild(okBtn);\r\n\r\n if (settings.type == 'prompt' || settings.type == 'confirm') {\r\n\r\n wrapper.appendChild(cancelBtn);\r\n\r\n }\r\n\r\n wrapper.classList.add('cdx-notification-' + settings.type);\r\n wrapper.dataset.type = settings.type;\r\n\r\n notification = wrapper;\r\n type = settings.type;\r\n confirm = settings.confirm;\r\n cancel = settings.cancel;\r\n inputField = input;\r\n\r\n if (settings.type != 'prompt' && settings.type != 'confirm') {\r\n\r\n window.setTimeout(close, settings.time);\r\n\r\n }\r\n\r\n };\r\n\r\n function send() {\r\n\r\n editor.nodes.notifications.appendChild(notification);\r\n inputField.focus();\r\n\r\n editor.nodes.notifications.classList.add('cdx-notification__notification-appending');\r\n\r\n window.setTimeout(function () {\r\n\r\n editor.nodes.notifications.classList.remove('cdx-notification__notification-appending');\r\n\r\n }, 100);\r\n\r\n addToQueue({type: type, close: close});\r\n\r\n };\r\n\r\n function close() {\r\n\r\n notification.remove();\r\n\r\n };\r\n\r\n\r\n if (constructorSettings) {\r\n\r\n create(constructorSettings);\r\n send();\r\n\r\n }\r\n\r\n return {\r\n create: create,\r\n send: send,\r\n close: close\r\n };\r\n\r\n };\r\n\r\n notifications.clear = function () {\r\n\r\n editor.nodes.notifications.innerHTML = '';\r\n queue = [];\r\n\r\n };\r\n\r\n return notifications;\r\n\r\n})({});\n\n\n// WEBPACK FOOTER //\n// ./modules/notifications.js","/**\r\n * Codex Editor Parser Module\r\n *\r\n * @author Codex Team\r\n * @version 1.1\r\n */\r\n\r\nmodule.exports = (function (parser) {\r\n\r\n let editor = codex.editor;\r\n\r\n /** inserting text */\r\n parser.insertPastedContent = function (blockType, tag) {\r\n\r\n editor.content.insertBlock({\r\n type : blockType.type,\r\n block : blockType.render({\r\n text : tag.innerHTML\r\n })\r\n });\r\n\r\n };\r\n\r\n /**\r\n * Check DOM node for display style: separated block or child-view\r\n */\r\n parser.isFirstLevelBlock = function (node) {\r\n\r\n return node.nodeType == editor.core.nodeTypes.TAG &&\r\n node.classList.contains(editor.ui.className.BLOCK_CLASSNAME);\r\n\r\n };\r\n\r\n return parser;\r\n\r\n})({});\r\n\n\n\n// WEBPACK FOOTER //\n// ./modules/parser.js","/**\r\n * Codex Sanitizer\r\n */\r\n\r\nmodule.exports = (function (sanitizer) {\r\n\r\n /** HTML Janitor library */\r\n let janitor = require('html-janitor');\r\n\r\n /** Codex Editor */\r\n let editor = codex.editor;\r\n\r\n sanitizer.prepare = function () {\r\n\r\n if (editor.settings.sanitizer && !editor.core.isEmpty(editor.settings.sanitizer)) {\r\n\r\n Config.CUSTOM = editor.settings.sanitizer;\r\n\r\n }\r\n\r\n };\r\n\r\n /**\r\n * Basic config\r\n */\r\n var Config = {\r\n\r\n /** User configuration */\r\n CUSTOM : null,\r\n\r\n BASIC : {\r\n\r\n tags: {\r\n p: {},\r\n a: {\r\n href: true,\r\n target: '_blank',\r\n rel: 'nofollow'\r\n },\r\n i: {},\r\n b: {},\r\n strong: {},\r\n em: {},\r\n span: {}\r\n }\r\n }\r\n };\r\n\r\n sanitizer.Config = Config;\r\n\r\n /**\r\n *\r\n * @param userCustomConfig\r\n * @returns {*}\r\n * @private\r\n *\r\n * @description If developer uses editor's API, then he can customize sane restrictions.\r\n * Or, sane config can be defined globally in editors initialization. That config will be used everywhere\r\n * At least, if there is no config overrides, that API uses BASIC Default configation\r\n */\r\n let init_ = function (userCustomConfig) {\r\n\r\n let configuration = userCustomConfig || Config.CUSTOM || Config.BASIC;\r\n\r\n return new janitor(configuration);\r\n\r\n };\r\n\r\n /**\r\n * Cleans string from unwanted tags\r\n * @protected\r\n * @param {String} dirtyString - taint string\r\n * @param {Object} customConfig - allowed tags\r\n */\r\n sanitizer.clean = function (dirtyString, customConfig) {\r\n\r\n let janitorInstance = init_(customConfig);\r\n\r\n return janitorInstance.clean(dirtyString);\r\n\r\n };\r\n\r\n return sanitizer;\r\n\r\n})({});\n\n\n// WEBPACK FOOTER //\n// ./modules/sanitizer.js","(function (root, factory) {\n if (typeof define === 'function' && define.amd) {\n define('html-janitor', factory);\n } else if (typeof exports === 'object') {\n module.exports = factory();\n } else {\n root.HTMLJanitor = factory();\n }\n}(this, function () {\n\n /**\n * @param {Object} config.tags Dictionary of allowed tags.\n * @param {boolean} config.keepNestedBlockElements Default false.\n */\n function HTMLJanitor(config) {\n\n var tagDefinitions = config['tags'];\n var tags = Object.keys(tagDefinitions);\n\n var validConfigValues = tags\n .map(function(k) { return typeof tagDefinitions[k]; })\n .every(function(type) { return type === 'object' || type === 'boolean' || type === 'function'; });\n\n if(!validConfigValues) {\n throw new Error(\"The configuration was invalid\");\n }\n\n this.config = config;\n }\n\n // TODO: not exhaustive?\n var blockElementNames = ['P', 'LI', 'TD', 'TH', 'DIV', 'H1', 'H2', 'H3', 'H4', 'H5', 'H6', 'PRE'];\n function isBlockElement(node) {\n return blockElementNames.indexOf(node.nodeName) !== -1;\n }\n\n var inlineElementNames = ['A', 'B', 'STRONG', 'I', 'EM', 'SUB', 'SUP', 'U', 'STRIKE'];\n function isInlineElement(node) {\n return inlineElementNames.indexOf(node.nodeName) !== -1;\n }\n\n HTMLJanitor.prototype.clean = function (html) {\n var sandbox = document.createElement('div');\n sandbox.innerHTML = html;\n\n this._sanitize(sandbox);\n\n return sandbox.innerHTML;\n };\n\n HTMLJanitor.prototype._sanitize = function (parentNode) {\n var treeWalker = createTreeWalker(parentNode);\n var node = treeWalker.firstChild();\n if (!node) { return; }\n\n do {\n // Ignore nodes that have already been sanitized\n if (node._sanitized) {\n continue;\n }\n\n if (node.nodeType === Node.TEXT_NODE) {\n // If this text node is just whitespace and the previous or next element\n // sibling is a block element, remove it\n // N.B.: This heuristic could change. Very specific to a bug with\n // `contenteditable` in Firefox: http://jsbin.com/EyuKase/1/edit?js,output\n // FIXME: make this an option?\n if (node.data.trim() === ''\n && ((node.previousElementSibling && isBlockElement(node.previousElementSibling))\n || (node.nextElementSibling && isBlockElement(node.nextElementSibling)))) {\n parentNode.removeChild(node);\n this._sanitize(parentNode);\n break;\n } else {\n continue;\n }\n }\n\n // Remove all comments\n if (node.nodeType === Node.COMMENT_NODE) {\n parentNode.removeChild(node);\n this._sanitize(parentNode);\n break;\n }\n\n var isInline = isInlineElement(node);\n var containsBlockElement;\n if (isInline) {\n containsBlockElement = Array.prototype.some.call(node.childNodes, isBlockElement);\n }\n\n // Block elements should not be nested (e.g.

  • ...); if\n // they are, we want to unwrap the inner block element.\n var isNotTopContainer = !! parentNode.parentNode;\n var isNestedBlockElement =\n isBlockElement(parentNode) &&\n isBlockElement(node) &&\n isNotTopContainer;\n\n var nodeName = node.nodeName.toLowerCase();\n\n var allowedAttrs = getAllowedAttrs(this.config, nodeName, node);\n\n var isInvalid = isInline && containsBlockElement;\n\n // Drop tag entirely according to the whitelist *and* if the markup\n // is invalid.\n if (isInvalid || shouldRejectNode(node, allowedAttrs)\n || (!this.config.keepNestedBlockElements && isNestedBlockElement)) {\n // Do not keep the inner text of SCRIPT/STYLE elements.\n if (! (node.nodeName === 'SCRIPT' || node.nodeName === 'STYLE')) {\n while (node.childNodes.length > 0) {\n parentNode.insertBefore(node.childNodes[0], node);\n }\n }\n parentNode.removeChild(node);\n\n this._sanitize(parentNode);\n break;\n }\n\n // Sanitize attributes\n for (var a = 0; a < node.attributes.length; a += 1) {\n var attr = node.attributes[a];\n\n if (shouldRejectAttr(attr, allowedAttrs, node)) {\n node.removeAttribute(attr.name);\n // Shift the array to continue looping.\n a = a - 1;\n }\n }\n\n // Sanitize children\n this._sanitize(node);\n\n // Mark node as sanitized so it's ignored in future runs\n node._sanitized = true;\n } while ((node = treeWalker.nextSibling()));\n };\n\n function createTreeWalker(node) {\n return document.createTreeWalker(node,\n NodeFilter.SHOW_TEXT | NodeFilter.SHOW_ELEMENT | NodeFilter.SHOW_COMMENT,\n null, false);\n }\n\n function getAllowedAttrs(config, nodeName, node){\n if (typeof config.tags[nodeName] === 'function') {\n return config.tags[nodeName](node);\n } else {\n return config.tags[nodeName];\n }\n }\n\n function shouldRejectNode(node, allowedAttrs){\n if (typeof allowedAttrs === 'undefined') {\n return true;\n } else if (typeof allowedAttrs === 'boolean') {\n return !allowedAttrs;\n }\n\n return false;\n }\n\n function shouldRejectAttr(attr, allowedAttrs, node){\n var attrName = attr.name.toLowerCase();\n\n if (allowedAttrs === true){\n return false;\n } else if (typeof allowedAttrs[attrName] === 'function'){\n return !allowedAttrs[attrName](attr.value, node);\n } else if (typeof allowedAttrs[attrName] === 'undefined'){\n return true;\n } else if (allowedAttrs[attrName] === false) {\n return true;\n } else if (typeof allowedAttrs[attrName] === 'string') {\n return (allowedAttrs[attrName] !== attr.value);\n }\n\n return false;\n }\n\n return HTMLJanitor;\n\n}));\n\n\n\n//////////////////\n// WEBPACK FOOTER\n// ./~/html-janitor/src/html-janitor.js\n// module id = 18\n// module chunks = 0","/**\r\n * Codex Editor Listeners module\r\n *\r\n * @author Codex Team\r\n * @version 1.0\r\n */\r\n\r\n/**\r\n * Module-decorator for event listeners assignment\r\n */\r\nmodule.exports = function (listeners) {\r\n\r\n var allListeners = [];\r\n\r\n /**\r\n * Search methods\r\n *\r\n * byElement, byType and byHandler returns array of suitable listeners\r\n * one and all takes element, eventType, and handler and returns first (all) suitable listener\r\n *\r\n */\r\n listeners.search = function () {\r\n\r\n var byElement = function (element, context) {\r\n\r\n var listenersOnElement = [];\r\n\r\n context = context || allListeners;\r\n\r\n for (var i = 0; i < context.length; i++) {\r\n\r\n var listener = context[i];\r\n\r\n if (listener.element === element) {\r\n\r\n listenersOnElement.push(listener);\r\n\r\n }\r\n\r\n }\r\n\r\n return listenersOnElement;\r\n\r\n };\r\n\r\n var byType = function (eventType, context) {\r\n\r\n var listenersWithType = [];\r\n\r\n context = context || allListeners;\r\n\r\n for (var i = 0; i < context.length; i++) {\r\n\r\n var listener = context[i];\r\n\r\n if (listener.type === eventType) {\r\n\r\n listenersWithType.push(listener);\r\n\r\n }\r\n\r\n }\r\n\r\n return listenersWithType;\r\n\r\n };\r\n\r\n var byHandler = function (handler, context) {\r\n\r\n var listenersWithHandler = [];\r\n\r\n context = context || allListeners;\r\n\r\n for (var i = 0; i < context.length; i++) {\r\n\r\n var listener = context[i];\r\n\r\n if (listener.handler === handler) {\r\n\r\n listenersWithHandler.push(listener);\r\n\r\n }\r\n\r\n }\r\n\r\n return listenersWithHandler;\r\n\r\n };\r\n\r\n var one = function (element, eventType, handler) {\r\n\r\n var result = allListeners;\r\n\r\n if (element)\r\n result = byElement(element, result);\r\n\r\n if (eventType)\r\n result = byType(eventType, result);\r\n\r\n if (handler)\r\n result = byHandler(handler, result);\r\n\r\n return result[0];\r\n\r\n };\r\n\r\n var all = function (element, eventType, handler) {\r\n\r\n var result = allListeners;\r\n\r\n if (element)\r\n result = byElement(element, result);\r\n\r\n if (eventType)\r\n result = byType(eventType, result);\r\n\r\n if (handler)\r\n result = byHandler(handler, result);\r\n\r\n return result;\r\n\r\n };\r\n\r\n return {\r\n byElement : byElement,\r\n byType : byType,\r\n byHandler : byHandler,\r\n one : one,\r\n all : all\r\n };\r\n\r\n }();\r\n\r\n listeners.add = function (element, eventType, handler, isCapture) {\r\n\r\n element.addEventListener(eventType, handler, isCapture);\r\n\r\n var data = {\r\n element: element,\r\n type: eventType,\r\n handler: handler\r\n };\r\n\r\n var alreadyAddedListener = listeners.search.one(element, eventType, handler);\r\n\r\n if (!alreadyAddedListener) {\r\n\r\n allListeners.push(data);\r\n\r\n }\r\n\r\n };\r\n\r\n listeners.remove = function (element, eventType, handler) {\r\n\r\n element.removeEventListener(eventType, handler);\r\n\r\n var existingListeners = listeners.search.all(element, eventType, handler);\r\n\r\n for (var i = 0; i < existingListeners.length; i++) {\r\n\r\n var index = allListeners.indexOf(existingListeners[i]);\r\n\r\n if (index > 0) {\r\n\r\n allListeners.splice(index, 1);\r\n\r\n }\r\n\r\n }\r\n\r\n };\r\n\r\n listeners.removeAll = function () {\r\n\r\n allListeners.map(function (current) {\r\n\r\n listeners.remove(current.element, current.type, current.handler);\r\n\r\n });\r\n\r\n };\r\n\r\n listeners.get = function (element, eventType, handler) {\r\n\r\n return listeners.search.all(element, eventType, handler);\r\n\r\n };\r\n\r\n return listeners;\r\n\r\n}({});\n\n\n// WEBPACK FOOTER //\n// ./modules/listeners.js","/**\r\n * Codex Editor Destroyer module\r\n *\r\n * @auhor Codex Team\r\n * @version 1.0\r\n */\r\n\r\nmodule.exports = function (destroyer) {\r\n\r\n let editor = codex.editor;\r\n\r\n destroyer.removeNodes = function () {\r\n\r\n editor.nodes.wrapper.remove();\r\n editor.nodes.notifications.remove();\r\n\r\n };\r\n\r\n destroyer.destroyPlugins = function () {\r\n\r\n for (var tool in editor.tools) {\r\n\r\n if (typeof editor.tools[tool].destroy === 'function') {\r\n\r\n editor.tools[tool].destroy();\r\n\r\n }\r\n\r\n }\r\n\r\n };\r\n\r\n destroyer.destroyScripts = function () {\r\n\r\n var scripts = document.getElementsByTagName('SCRIPT');\r\n\r\n for (var i = 0; i < scripts.length; i++) {\r\n\r\n if (scripts[i].id.indexOf(editor.scriptPrefix) + 1) {\r\n\r\n scripts[i].remove();\r\n i--;\r\n\r\n }\r\n\r\n }\r\n\r\n };\r\n\r\n\r\n /**\r\n * Delete editor data from webpage.\r\n * You should send settings argument with boolean flags:\r\n * @param settings.ui- remove redactor event listeners and DOM nodes\r\n * @param settings.scripts - remove redactor scripts from DOM\r\n * @param settings.plugins - remove plugin's objects\r\n * @param settings.core - remove editor core. You can remove core only if UI and scripts flags is true\r\n * }\r\n *\r\n */\r\n destroyer.destroy = function (settings) {\r\n\r\n if (!settings || typeof settings !== 'object') {\r\n\r\n return;\r\n\r\n }\r\n\r\n if (settings.ui) {\r\n\r\n destroyer.removeNodes();\r\n editor.listeners.removeAll();\r\n\r\n }\r\n\r\n if (settings.scripts) {\r\n\r\n destroyer.destroyScripts();\r\n\r\n }\r\n\r\n if (settings.plugins) {\r\n\r\n destroyer.destroyPlugins();\r\n\r\n }\r\n\r\n if (settings.ui && settings.scripts && settings.core) {\r\n\r\n delete codex.editor;\r\n\r\n }\r\n\r\n };\r\n\r\n return destroyer;\r\n\r\n}({});\n\n\n// WEBPACK FOOTER //\n// ./modules/destroyer.js","/**\r\n * Codex Editor Paste module\r\n *\r\n * @author Codex Team\r\n * @version 1.1.1\r\n */\r\n\r\nmodule.exports = function (paste) {\r\n\r\n let editor = codex.editor;\r\n\r\n var patterns = [];\r\n\r\n paste.prepare = function () {\r\n\r\n var tools = editor.tools;\r\n\r\n for (var tool in tools) {\r\n\r\n if (!tools[tool].renderOnPastePatterns || !Array.isArray(tools[tool].renderOnPastePatterns)) {\r\n\r\n continue;\r\n\r\n }\r\n\r\n tools[tool].renderOnPastePatterns.map(function (pattern) {\r\n\r\n\r\n patterns.push(pattern);\r\n\r\n });\r\n\r\n }\r\n\r\n return Promise.resolve();\r\n\r\n };\r\n\r\n /**\r\n * Saves data\r\n * @param event\r\n */\r\n paste.pasted = function (event) {\r\n\r\n var clipBoardData = event.clipboardData || window.clipboardData,\r\n content = clipBoardData.getData('Text');\r\n\r\n var result = analize(content);\r\n\r\n if (result) {\r\n\r\n event.preventDefault();\r\n event.stopImmediatePropagation();\r\n\r\n }\r\n\r\n return result;\r\n\r\n };\r\n\r\n /**\r\n * Analizes pated string and calls necessary method\r\n */\r\n\r\n var analize = function (string) {\r\n\r\n var result = false,\r\n content = editor.content.currentNode,\r\n plugin = content.dataset.tool;\r\n\r\n patterns.map( function (pattern) {\r\n\r\n var execArray = pattern.regex.exec(string),\r\n match = execArray && execArray[0];\r\n\r\n if ( match && match === string.trim()) {\r\n\r\n /** current block is not empty */\r\n if ( content.textContent.trim() && plugin == editor.settings.initialBlockPlugin ) {\r\n\r\n pasteToNewBlock_();\r\n\r\n }\r\n\r\n pattern.callback(string, pattern);\r\n result = true;\r\n\r\n }\r\n\r\n });\r\n\r\n return result;\r\n\r\n };\r\n\r\n var pasteToNewBlock_ = function () {\r\n\r\n /** Create new initial block */\r\n editor.content.insertBlock({\r\n\r\n type : editor.settings.initialBlockPlugin,\r\n block : editor.tools[editor.settings.initialBlockPlugin].render({\r\n text : ''\r\n })\r\n\r\n }, false);\r\n\r\n };\r\n\r\n /**\r\n * This method prevents default behaviour.\r\n *\r\n * @param {Object} event\r\n * @protected\r\n *\r\n * @description We get from clipboard pasted data, sanitize, make a fragment that contains of this sanitized nodes.\r\n * Firstly, we need to memorize the caret position. We can do that by getting the range of selection.\r\n * After all, we insert clear fragment into caret placed position. Then, we should move the caret to the last node\r\n */\r\n paste.blockPasteCallback = function (event) {\r\n\r\n\r\n if (!needsToHandlePasteEvent(event.target)) {\r\n\r\n return;\r\n\r\n }\r\n\r\n /** get html pasted data - dirty data */\r\n var htmlData = event.clipboardData.getData('text/html'),\r\n plainData = event.clipboardData.getData('text/plain');\r\n\r\n /** Temporary DIV that is used to work with text's paragraphs as DOM-elements*/\r\n var paragraphs = editor.draw.node('DIV', '', {}),\r\n cleanData,\r\n wrappedData;\r\n\r\n /** Create fragment, that we paste to range after proccesing */\r\n cleanData = editor.sanitizer.clean(htmlData);\r\n\r\n /**\r\n * We wrap pasted text with

    tags to split it logically\r\n * @type {string}\r\n */\r\n wrappedData = editor.content.wrapTextWithParagraphs(cleanData, plainData);\r\n paragraphs.innerHTML = wrappedData;\r\n\r\n /**\r\n * If there only one paragraph, lets user agent paste it\r\n */\r\n if (paragraphs.childNodes.length == 1) {\r\n\r\n return;\r\n\r\n }\r\n\r\n /** Prevent default behaviour */\r\n event.preventDefault();\r\n\r\n insertPastedParagraphs(paragraphs.childNodes);\r\n\r\n };\r\n\r\n /**\r\n * Checks if we should handle paste event on block\r\n * @param block\r\n *\r\n * @return {boolean}\r\n */\r\n var needsToHandlePasteEvent = function (block) {\r\n\r\n /** If area is input or textarea then allow default behaviour */\r\n if ( editor.core.isNativeInput(block) ) {\r\n\r\n return false;\r\n\r\n }\r\n\r\n var editableParent = editor.content.getEditableParent(block);\r\n\r\n /** Allow paste when event target placed in Editable element */\r\n if (!editableParent) {\r\n\r\n return false;\r\n\r\n }\r\n\r\n return true;\r\n\r\n };\r\n\r\n var insertPastedParagraphs = function (paragraphs) {\r\n\r\n var NEW_BLOCK_TYPE = editor.settings.initialBlockPlugin,\r\n currentBlockContent = editor.content.currentNode.firstChild.firstChild;\r\n\r\n paragraphs.forEach(function (paragraph, index) {\r\n\r\n /**\r\n * If there was no data in working node, replace it with first paragraph of pasted text\r\n */\r\n if (index == 0 && currentBlockContent.innerHTML.trim() === '') {\r\n\r\n editor.content.switchBlock(editor.content.currentNode, editor.tools[NEW_BLOCK_TYPE].render({\r\n text : paragraph.innerHTML\r\n }), NEW_BLOCK_TYPE);\r\n\r\n return;\r\n\r\n }\r\n\r\n editor.content.insertBlock({\r\n type : NEW_BLOCK_TYPE,\r\n block : editor.tools[NEW_BLOCK_TYPE].render({\r\n text : paragraph.innerHTML\r\n })\r\n });\r\n\r\n editor.caret.inputIndex++;\r\n\r\n });\r\n\r\n editor.caret.setToPreviousBlock(editor.caret.getCurrentInputIndex() + 1);\r\n\r\n\r\n };\r\n\r\n\r\n return paste;\r\n\r\n}({});\n\n\n// WEBPACK FOOTER //\n// ./modules/paste.js"],"sourceRoot":""} \ No newline at end of file diff --git a/modules/callbacks.js b/modules/callbacks.js index c0b70530..e04024ec 100644 --- a/modules/callbacks.js +++ b/modules/callbacks.js @@ -801,7 +801,7 @@ module.exports = (function (callbacks) { selectionLength, firstLevelBlocksCount; - if (isNativeInput_(event.target)) { + if (editor.core.isNativeInput(event.target)) { /** If input value is empty - remove block */ if (event.target.value.trim() == '') { @@ -895,104 +895,6 @@ module.exports = (function (callbacks) { }; - /** - * This method prevents default behaviour. - * - * @param {Object} event - * @protected - * - * @description We get from clipboard pasted data, sanitize, make a fragment that contains of this sanitized nodes. - * Firstly, we need to memorize the caret position. We can do that by getting the range of selection. - * After all, we insert clear fragment into caret placed position. Then, we should move the caret to the last node - */ - callbacks.blockPasteCallback = function (event) { - - /** If area is input or textarea then allow default behaviour */ - if ( isNativeInput_(event.target) ) { - - return; - - } - - /** Prevent default behaviour */ - event.preventDefault(); - - var editableParent = editor.content.getEditableParent(event.target); - - /** Allow paste when event target placed in Editable element */ - if (!editableParent) { - - return; - - } - - /** get html pasted data - dirty data */ - var htmlData = event.clipboardData.getData('text/html'), - plainData = event.clipboardData.getData('text/plain'); - - - /** Temporary DIV that is used to work with childs as arrays item */ - var div = editor.draw.node('DIV', '', {}), - cleanData, - wrappedData; - - /** Create fragment, that we paste to range after proccesing */ - cleanData = editor.sanitizer.clean(htmlData); - - - /** - * We wrap pasted text with

    tags to split it logically - * - * @type {string} - */ - wrappedData = editor.content.wrapTextWithParagraphs(cleanData, plainData); - div.innerHTML = wrappedData; - - var NEW_BLOCK_TYPE = editor.settings.initialBlockPlugin, - currentBlockContent = editor.content.currentNode.firstChild.firstChild; - - /** - * If there only one paragraph, just insert it - */ - if (div.childNodes.length == 1) { - - editor.caret.insertNode(document.createTextNode(div.firstChild.innerHTML)); - return; - - } - - div.childNodes.forEach(function (paragraph, index) { - - - /** - * If there was no data in working node, replace it with first paragraph of pasted text - */ - if (index == 0 && currentBlockContent.innerHTML.trim() === '') { - - editor.content.switchBlock(editor.content.currentNode, editor.tools[NEW_BLOCK_TYPE].render({ - text : paragraph.innerHTML - }), NEW_BLOCK_TYPE); - - return; - - } - - editor.content.insertBlock({ - type : NEW_BLOCK_TYPE, - block : editor.tools[NEW_BLOCK_TYPE].render({ - text : paragraph.innerHTML - }) - }); - - editor.caret.inputIndex++; - - }); - - editor.caret.setToPreviousBlock(editor.caret.getCurrentInputIndex() + 1); - - - }; - /** * used by UI module * Clicks on block settings button @@ -1019,21 +921,6 @@ module.exports = (function (callbacks) { }; - /** - * Check block - * @param target - * @private - * - * @description Checks target is it native input - */ - var isNativeInput_ = function (target) { - - var nativeInputAreas = ['INPUT', 'TEXTAREA']; - - return (nativeInputAreas.indexOf(target.tagName) != -1); - - }; - return callbacks; })({}); \ No newline at end of file diff --git a/modules/core.js b/modules/core.js index eb593ef0..4a07b489 100644 --- a/modules/core.js +++ b/modules/core.js @@ -338,6 +338,20 @@ module.exports = (function (core) { }; + /** + * Check block + * @param target + * @description Checks target is it native input + */ + core.isNativeInput = function (target) { + + var nativeInputAreas = ['INPUT', 'TEXTAREA']; + + return (nativeInputAreas.indexOf(target.tagName) != -1); + + }; + + return core; })({}); diff --git a/modules/paste.js b/modules/paste.js index 9a64f24e..d9082ce5 100644 --- a/modules/paste.js +++ b/modules/paste.js @@ -107,6 +107,125 @@ module.exports = function (paste) { }; + /** + * This method prevents default behaviour. + * + * @param {Object} event + * @protected + * + * @description We get from clipboard pasted data, sanitize, make a fragment that contains of this sanitized nodes. + * Firstly, we need to memorize the caret position. We can do that by getting the range of selection. + * After all, we insert clear fragment into caret placed position. Then, we should move the caret to the last node + */ + paste.blockPasteCallback = function (event) { + + + if (!needsToHandlePasteEvent(event.target)) { + + return; + + } + + /** get html pasted data - dirty data */ + var htmlData = event.clipboardData.getData('text/html'), + plainData = event.clipboardData.getData('text/plain'); + + /** Temporary DIV that is used to work with text's paragraphs as DOM-elements*/ + var paragraphs = editor.draw.node('DIV', '', {}), + cleanData, + wrappedData; + + /** Create fragment, that we paste to range after proccesing */ + cleanData = editor.sanitizer.clean(htmlData); + + /** + * We wrap pasted text with

    tags to split it logically + * @type {string} + */ + wrappedData = editor.content.wrapTextWithParagraphs(cleanData, plainData); + paragraphs.innerHTML = wrappedData; + + /** + * If there only one paragraph, lets user agent paste it + */ + if (paragraphs.childNodes.length == 1) { + + return; + + } + + /** Prevent default behaviour */ + event.preventDefault(); + + insertPastedParagraphs(paragraphs.childNodes); + + }; + + /** + * Checks if we should handle paste event on block + * @param block + * + * @return {boolean} + */ + var needsToHandlePasteEvent = function (block) { + + /** If area is input or textarea then allow default behaviour */ + if ( editor.core.isNativeInput(block) ) { + + return false; + + } + + var editableParent = editor.content.getEditableParent(block); + + /** Allow paste when event target placed in Editable element */ + if (!editableParent) { + + return false; + + } + + return true; + + }; + + var insertPastedParagraphs = function (paragraphs) { + + var NEW_BLOCK_TYPE = editor.settings.initialBlockPlugin, + currentBlockContent = editor.content.currentNode.firstChild.firstChild; + + paragraphs.forEach(function (paragraph, index) { + + /** + * If there was no data in working node, replace it with first paragraph of pasted text + */ + if (index == 0 && currentBlockContent.innerHTML.trim() === '') { + + editor.content.switchBlock(editor.content.currentNode, editor.tools[NEW_BLOCK_TYPE].render({ + text : paragraph.innerHTML + }), NEW_BLOCK_TYPE); + + return; + + } + + editor.content.insertBlock({ + type : NEW_BLOCK_TYPE, + block : editor.tools[NEW_BLOCK_TYPE].render({ + text : paragraph.innerHTML + }) + }); + + editor.caret.inputIndex++; + + }); + + editor.caret.setToPreviousBlock(editor.caret.getCurrentInputIndex() + 1); + + + }; + + return paste; }({}); \ No newline at end of file diff --git a/modules/ui.js b/modules/ui.js index 22ab2158..3fade030 100644 --- a/modules/ui.js +++ b/modules/ui.js @@ -361,7 +361,7 @@ module.exports = (function (ui) { * @example editor.callback.blockPasteViaSanitize(event), the second method. * */ - editor.listeners.add(block, 'paste', editor.callback.blockPasteCallback, false); + editor.listeners.add(block, 'paste', editor.paste.blockPasteCallback, false); editor.listeners.add(block, 'mouseup', editor.toolbar.inline.show, false); From f4a6f1cd9cf2a0f820899095dea2f37915c396d7 Mon Sep 17 00:00:00 2001 From: George Berezhnoy Date: Mon, 24 Apr 2017 03:08:03 +0300 Subject: [PATCH 04/12] Don't allow empty paragraphs --- codex-editor.js | 4 ++-- codex-editor.js.map | 2 +- modules/content.js | 13 ++++++++++++- modules/paste.js | 27 ++++++++++++++++++++++++--- 4 files changed, 39 insertions(+), 7 deletions(-) diff --git a/codex-editor.js b/codex-editor.js index 8f816f10..a0cb6016 100644 --- a/codex-editor.js +++ b/codex-editor.js @@ -1,3 +1,3 @@ -var codex=codex||{};codex.editor=function(e){function t(o){if(n[o])return n[o].exports;var r=n[o]={exports:{},id:o,loaded:!1};return e[o].call(r.exports,r,r.exports,t),r.loaded=!0,r.exports}var n={};return t.m=e,t.c=n,t.p="",t(0)}([function(e,t,n){"use strict";e.exports=function(e){e.version="1.6.4",e.scriptPrefix="cdx-script-";var t=function(){e.core=n(1),e.tools=n(2),e.ui=n(3),e.transport=n(4),e.renderer=n(5),e.saver=n(6),e.content=n(7),e.toolbar=n(8),e.callback=n(12),e.draw=n(13),e.caret=n(14),e.notifications=n(15),e.parser=n(16),e.sanitizer=n(17),e.listeners=n(19),e.destroyer=n(20),e.paste=n(21)};return e.settings={tools:["paragraph","header","picture","list","quote","code","twitter","instagram","smile"],holderId:"codex-editor",initialBlockPlugin:"paragraph"},e.nodes={holder:null,wrapper:null,toolbar:null,inlineToolbar:{wrapper:null,buttons:null,actions:null},toolbox:null,notifications:null,plusButton:null,showSettingsButton:null,showTrashButton:null,blockSettings:null,pluginSettings:null,defaultSettings:null,toolbarButtons:{},redactor:null},e.state={jsonOutput:[],blocks:[],inputs:[]},e.tools={},e.start=function(n){t(),e.core.prepare(n).then(e.ui.prepare).then(e.tools.prepare).then(e.sanitizer.prepare).then(e.paste.prepare).then(e.transport.prepare).then(e.renderer.makeBlocksFromData).then(e.ui.saveInputs).catch(function(t){e.core.log("Initialization failed with error: %o","warn",t)})},e}({})},function(e,t){"use strict";var n="function"==typeof Symbol&&"symbol"==typeof Symbol.iterator?function(e){return typeof e}:function(e){return e&&"function"==typeof Symbol&&e.constructor===Symbol&&e!==Symbol.prototype?"symbol":typeof e};e.exports=function(e){var t=codex.editor;e.prepare=function(e){return new Promise(function(o,r){e&&(t.settings.tools=e.tools||t.settings.tools),e.data&&(t.state.blocks=e.data),e.initialBlockPlugin&&(t.settings.initialBlockPlugin=e.initialBlockPlugin),e.sanitizer&&(t.settings.sanitizer=e.sanitizer),t.hideToolbar=e.hideToolbar,t.nodes.holder=document.getElementById(e.holderId||t.settings.holderId),void 0===n(t.nodes.holder)||null===t.nodes.holder?r(Error("Holder wasn't found by ID: #"+e.holderId)):o()})},e.log=function(e,t,n){t=t||"log",n?e="[codex-editor]: "+e:(n=e||"undefined",e="[codex-editor]: %o");try{"console"in window&&window.console[t]&&(n?window.console[t](e,n):window.console[t](e))}catch(e){}},e.insertAfter=function(e,t){e.parentNode.insertBefore(t,e.nextSibling)},e.nodeTypes={TAG:1,TEXT:3,COMMENT:8},e.keys={BACKSPACE:8,TAB:9,ENTER:13,SHIFT:16,CTRL:17,ALT:18,ESC:27,SPACE:32,LEFT:37,UP:38,DOWN:40,RIGHT:39,DELETE:46,META:91},e.isDomNode=function(e){return e&&"object"===("undefined"==typeof e?"undefined":n(e))&&e.nodeType&&e.nodeType==this.nodeTypes.TAG},e.isEmpty=function(e){return 0===Object.keys(e).length},e.ajax=function(e){if(e&&e.url){var t,n,r,i=window.XMLHttpRequest?new XMLHttpRequest:new ActiveXObject("Microsoft.XMLHTTP");if(e.async=!0,e.type=e.type||"GET",e.data=e.data||"",e["content-type"]=e["content-type"]||"application/json; charset=utf-8","GET"==e.type&&e.data)e.url=/\?/.test(e.url)?e.url+"&"+e.data:e.url+"?"+e.data;else{t="";for(r in e.data)t+=r+"="+encodeURIComponent(e.data[r])+"&"}e.withCredentials&&(i.withCredentials=!0);var a=void 0,s=void 0;if("function"!=typeof e.beforeSend||(s=e.beforeSend.call(),s!==!1))return i.open(e.type,e.url,e.async),n=o(e.data),n||("POST"!==e.type?i.setRequestHeader("Content-type",e["content-type"]):i.setRequestHeader("Content-type","application/x-www-form-urlencoded")),i.setRequestHeader("X-Requested-With","XMLHttpRequest"),a=s||i,"function"==typeof e.progress&&(i.upload.onprogress=e.progress.bind(a)),i.onreadystatechange=function(){4===i.readyState&&(200===i.status?"function"==typeof e.success&&e.success.call(a,i.responseText):"function"==typeof e.error&&e.error.call(a,i.responseText,i.status))},n?i.send(e.data):i.send(t),i}},e.importScript=function(e,n){return new Promise(function(o,r){var i=void 0;n?document.getElementById(t.scriptPrefix+n)&&o(e):r("Instance name is missed"),i=document.createElement("SCRIPT"),i.async=!0,i.defer=!0,i.id=t.scriptPrefix+n,i.onload=function(){o(e)},i.onerror=function(){r(e)},i.src=e,document.head.appendChild(i)})};var o=function(e){return e instanceof FormData};return e.isNativeInput=function(e){var t=["INPUT","TEXTAREA"];return t.indexOf(e.tagName)!=-1},e}({})},function(e,t){"use strict";e.exports=function(){function e(){return new Promise(function(e,o){Promise.resolve().then(function(){var t=[],o=n.tools;for(var r in o){var i=o[r];i.prepare&&"function"!=typeof i.prepare||!i.prepare||t.push(i)}return t.length||e(),t}).then(t).then(function(){n.core.log("Plugins loaded","info"),e()}).catch(function(e){o(e)})})}function t(e){return new Promise(function(t){e.reduce(function(r,i,a){return r.then(function(){return new Promise(function(r){o(i).then(r).then(function(){i.available=!0}).catch(function(e){n.core.log("Plugin «"+i.type+"» was not loaded. Preparation failed because %o","warn",e),i.available=!1,i.loadingMessage=e,r()}).then(function(){a==e.length-1&&t()})})})},Promise.resolve())})}var n=codex.editor,o=function(e){return e.prepare(e.config||{})};return{prepare:e}}()},function(e,t){"use strict";e.exports=function(e){var t=codex.editor;e.className={BLOCK_CLASSNAME:"ce-block",BLOCK_CONTENT:"ce-block__content",BLOCK_STRETCHED:"ce-block--stretched",BLOCK_HIGHLIGHTED:"ce-block--focused",SETTINGS_ITEM:"ce-settings__item"},e.prepare=function(){return new Promise(function(e){var n=t.draw.wrapper(),r=t.draw.redactor(),i=o();n.appendChild(i),n.appendChild(r),t.nodes.wrapper=n,t.nodes.redactor=r,t.nodes.holder.appendChild(n),e()}).then(s).then(n).then(c).then(a).then(l).catch(function(){t.core.log("Can't draw editor interface")})};var n=function(){var e=t.draw.inlineToolbar();t.nodes.inlineToolbar.wrapper=e,t.nodes.inlineToolbar.buttons=t.draw.inlineToolbarButtons(),t.nodes.inlineToolbar.actions=t.draw.inlineToolbarActions(),t.nodes.inlineToolbar.wrapper.appendChild(t.nodes.inlineToolbar.buttons),t.nodes.inlineToolbar.wrapper.appendChild(t.nodes.inlineToolbar.actions),t.nodes.wrapper.appendChild(t.nodes.inlineToolbar.wrapper)},o=function(){var e=t.draw.toolbar(),n=i(),o=r();return e.appendChild(n),e.appendChild(o),t.nodes.toolbar=e,e},r=function(){var e=t.draw.toolbarContent(),n=t.draw.toolbox(),o=t.draw.plusButton();return e.appendChild(o),e.appendChild(n),t.nodes.toolbox=n,t.nodes.plusButton=o,e},i=function(){var e=t.draw.blockSettings(),n=t.draw.blockButtons(),o=t.draw.defaultSettings(),r=t.draw.settingsButton(),i=t.toolbar.settings.makeRemoveBlockButton(),a=t.draw.pluginsSettings();return e.appendChild(a),e.appendChild(o),n.appendChild(r),n.appendChild(i),n.appendChild(e),t.nodes.blockSettings=e,t.nodes.pluginSettings=a,t.nodes.defaultSettings=o,t.nodes.showSettingsButton=r,t.nodes.showTrashButton=i,n},a=function(){t.nodes.notifications=t.notifications.createHolder()},s=function(){var e,n,o;for(n in t.settings.tools)e=t.settings.tools[n],t.tools[n]=e,e.iconClassname||!e.displayInToolbox?"function"==typeof e.render?e.displayInToolbox&&(o=t.draw.toolbarButton(n,e.iconClassname),t.nodes.toolbox.appendChild(o),t.nodes.toolbarButtons[n]=o):t.core.log("render method missed. Tool %o skipped","warn",n):t.core.log("Toolbar icon classname missed. Tool %o skipped","warn",n)},c=function(){var e,n,o={bold:{icon:"ce-icon-bold",command:"bold"},italic:{icon:"ce-icon-italic",command:"italic"},underline:{icon:"ce-icon-underline",command:"underline"},link:{icon:"ce-icon-link",command:"createLink"}};for(var r in o)n=o[r],e=t.draw.toolbarButtonInline(r,n.icon),t.nodes.inlineToolbar.buttons.appendChild(e),t.ui.setInlineToolbarButtonBehaviour(e,n.command)},l=function(){t.core.log("ui.bindEvents fired","info"),t.listeners.add(document,"keydown",t.callback.globalKeydown,!1),t.listeners.add(t.nodes.redactor,"keydown",t.callback.redactorKeyDown,!1),t.listeners.add(document,"keyup",t.callback.globalKeyup,!1),t.listeners.add(t.nodes.redactor,"click",t.callback.redactorClicked,!1),t.listeners.add(t.nodes.plusButton,"click",t.callback.plusButtonClicked,!1),t.listeners.add(t.nodes.showSettingsButton,"click",t.callback.showSettingsButtonClicked,!1);for(var e in t.nodes.toolbarButtons)t.listeners.add(t.nodes.toolbarButtons[e],"click",t.callback.toolbarButtonClicked,!1)};return e.addBlockHandlers=function(e){e&&(t.listeners.add(e,"keydown",t.callback.blockKeydown,!1),t.listeners.add(e,"paste",t.paste.blockPasteCallback,!1),t.listeners.add(e,"mouseup",t.toolbar.inline.show,!1))},e.saveInputs=function(){var e=t.nodes.redactor;t.state.inputs=[];var n=e.querySelectorAll("[contenteditable], input, textarea");Array.prototype.map.call(n,function(e){e.type&&"text"!=e.type&&"textarea"!=e.type||t.state.inputs.push(e)})},e.addInitialBlock=function(){var e,n=t.settings.initialBlockPlugin;return t.tools[n]?(e=t.tools[n].render(),e.setAttribute("data-placeholder","Расскажите свою историю..."),t.content.insertBlock({type:n,block:e}),void t.content.workingNodeChanged(e)):void t.core.log("Plugin %o was not implemented and can't be used as initial block","warn",n)},e.setInlineToolbarButtonBehaviour=function(e,n){t.listeners.add(e,"mousedown",function(e){t.toolbar.inline.toolClicked(e,n)},!1)},e}({})},function(e,t){"use strict";e.exports=function(e){var t=codex.editor,n=null;return e.input=null,e.arguments=null,e.prepare=function(){var e=t.draw.node("INPUT","",{type:"file"});t.listeners.add(e,"change",t.transport.fileSelected),t.transport.input=e},e.clearInput=function(){e.input=null,e.prepare()},e.fileSelected=function(){var o,r=this,i=r.files,a=new FormData;if(t.transport.arguments.multiple===!0)for(o=0;o"+t.split("\n\n").join("

    ")+"

    ";var n,o,r,i,a=document.createElement("DIV"),s=document.createElement("DIV"),c=["DIV","P"];for(a.innerHTML=e,o=document.createElement("P"),n=0;n'}),o=t.draw.node("DIV","ce-toolbar__remove-confirmation",{}),r=t.draw.node("DIV","ce-toolbar__remove-confirm",{textContent:"Удалить блок"}),i=t.draw.node("DIV","ce-toolbar__remove-cancel",{textContent:"Отмена"});return t.listeners.add(n,"click",t.toolbar.settings.removeButtonClicked,!1),t.listeners.add(r,"click",t.toolbar.settings.confirmRemovingRequest,!1),t.listeners.add(i,"click",t.toolbar.settings.cancelRemovingRequest,!1),o.appendChild(r),o.appendChild(i),e.appendChild(n),e.appendChild(o),t.toolbar.settings.setting=n,t.toolbar.settings.actions=o,e},e.removeButtonClicked=function(){var e=t.toolbar.settings.actions;e.classList.contains("opened")?t.toolbar.settings.hideRemoveActions():t.toolbar.settings.showRemoveActions(),t.toolbar.toolbox.close(),t.toolbar.settings.close()},e.cancelRemovingRequest=function(){t.toolbar.settings.actions.classList.remove("opened")},e.confirmRemovingRequest=function(){var e,n=t.content.currentNode;n.remove(),e=t.nodes.redactor.childNodes.length,0===e&&(t.content.currentNode=null,t.ui.addInitialBlock()),t.ui.saveInputs(),t.toolbar.close()},e.showRemoveActions=function(){t.toolbar.settings.actions.classList.add("opened")},e.hideRemoveActions=function(){t.toolbar.settings.actions.classList.remove("opened")},e}({})},function(e,t){"use strict";e.exports=function(e){var t=codex.editor;e.buttonsOpened=null,e.actionsOpened=null,e.wrappersOffset=null,e.storedSelection=null,e.show=function(){var n,o=t.content.currentNode,r=o.dataset.tool;if(n=t.tools[r],n.showInlineToolbar){var i=e.getSelectionText(),a=t.nodes.inlineToolbar.wrapper;i.length>0&&(t.toolbar.inline.move(),a.classList.add("opened"),t.toolbar.inline.showButtons())}},e.close=function(){var e=t.nodes.inlineToolbar.wrapper;e.classList.remove("opened")},e.move=function(){this.wrappersOffset||(this.wrappersOffset=this.getWrappersOffset());var e,n,o=this.getSelectionCoords(),r=0,i=t.nodes.inlineToolbar.wrapper;0===i.offsetHeight&&(r=40),e=o.x-this.wrappersOffset.left,n=o.y+window.scrollY-this.wrappersOffset.top-r-i.offsetHeight,i.style.transform="translate3D("+Math.floor(e)+"px, "+Math.floor(n)+"px, 0)",t.toolbar.inline.closeButtons(),t.toolbar.inline.closeAction()},e.toolClicked=function(e,n){switch(n){case"createLink":t.toolbar.inline.createLinkAction(e,n);break;default:t.toolbar.inline.defaultToolAction(n)}t.nodes.inlineToolbar.buttons.childNodes.forEach(t.toolbar.inline.hightlight)},e.getWrappersOffset=function(){var e=t.nodes.wrapper,n=this.getOffset(e);return this.wrappersOffset=n,n},e.getOffset=function(e){for(var t=0,n=0;e&&!isNaN(e.offsetLeft)&&!isNaN(e.offsetTop);)t+=e.offsetLeft+e.clientLeft,n+=e.offsetTop+e.clientTop,e=e.offsetParent;return{top:n,left:t}},e.getSelectionCoords=function(){var e,t=document.selection,n=0,o=0;if(t)"Control"!=t.type&&(e=t.createRange(),e.collapse(!0),n=e.boundingLeft,o=e.boundingTop);else if(window.getSelection&&(t=window.getSelection(),t.rangeCount&&(e=t.getRangeAt(0).cloneRange(),e.getClientRects))){e.collapse(!0);var r=e.getClientRects()[0];if(!r)return;n=r.left,o=r.top}return{x:n,y:o}},e.getSelectionText=function(){var e="";return window.getSelection&&(e=window.getSelection().toString()),e},e.showButtons=function(){var e=t.nodes.inlineToolbar.buttons;e.classList.add("opened"),t.toolbar.inline.buttonsOpened=!0,t.nodes.inlineToolbar.buttons.childNodes.forEach(t.toolbar.inline.hightlight)},e.closeButtons=function(){var e=t.nodes.inlineToolbar.buttons;e.classList.remove("opened"),t.toolbar.inline.buttonsOpened=!1},e.showActions=function(){var e=t.nodes.inlineToolbar.actions;e.classList.add("opened"),t.toolbar.inline.actionsOpened=!0},e.closeAction=function(){var e=t.nodes.inlineToolbar.actions;e.innerHTML="",e.classList.remove("opened"),t.toolbar.inline.actionsOpened=!1};var n=function(e){if(e.keyCode==t.core.keys.ENTER){var n=t.content.currentNode,o=t.toolbar.inline.storedSelection;t.toolbar.inline.restoreSelection(n,o),t.toolbar.inline.setAnchor(this.value),e.preventDefault(),e.stopImmediatePropagation(),t.toolbar.inline.clearRange()}};return e.createLinkAction=function(e){var o=this.isLinkActive(),r=t.content.currentNode,i=t.toolbar.inline.saveSelection(r);if(t.toolbar.inline.storedSelection=i,o)t.toolbar.inline.restoreSelection(r,i),t.toolbar.inline.defaultToolAction("unlink");else{var a=t.draw.inputForLink();t.nodes.inlineToolbar.actions.appendChild(a),t.toolbar.inline.closeButtons(),t.toolbar.inline.showActions(),a.focus(),e.preventDefault(),t.listeners.add(a,"keydown",n,!1)}},e.isLinkActive=function(){var e=!1;return t.nodes.inlineToolbar.buttons.childNodes.forEach(function(t){var n=t.dataset.type;"link"==n&&t.classList.contains("hightlighted")&&(e=!0)}),e},e.defaultToolAction=function(e){document.execCommand(e,!1,null)},e.setAnchor=function(e){document.execCommand("createLink",!1,e),t.toolbar.inline.closeAction()},e.saveSelection=function(e){var t,n=window.getSelection().getRangeAt(0),o=n.cloneRange();return o.selectNodeContents(e),o.setEnd(n.startContainer,n.startOffset),t=o.toString().length,{start:t,end:t+n.toString().length}},e.restoreSelection=function(e,t){var n=document.createRange(),o=0;n.setStart(e,0),n.collapse(!0);for(var r,i,a=[e],s=!1,c=!1;!c&&(r=a.pop());)if(3==r.nodeType)i=o+r.length,!s&&t.start>=o&&t.start<=i&&(n.setStart(r,t.start-o),s=!0),s&&t.end>=o&&t.end<=i&&(n.setEnd(r,t.end-o),c=!0),o=i;else for(var l=r.childNodes.length;l--;)a.push(r.childNodes[l]);var d=window.getSelection();d.removeAllRanges(),d.addRange(n)},e.clearRange=function(){var e=window.getSelection();e.removeAllRanges()},e.hightlight=function(e){var n=e.dataset.type;document.queryCommandState(n)?t.toolbar.inline.setButtonHighlighted(e):t.toolbar.inline.removeButtonsHighLight(e);var o=window.getSelection(),r=o.anchorNode.parentNode;"A"==r.tagName&&"link"==n&&t.toolbar.inline.setButtonHighlighted(e)},e.setButtonHighlighted=function(e){if(e.classList.add("hightlighted"),"link"==e.dataset.type){var t=e.childNodes[0];t.classList.remove("ce-icon-link"),t.classList.add("ce-icon-unlink")}},e.removeButtonsHighLight=function(e){if(e.classList.remove("hightlighted"),"link"==e.dataset.type){var t=e.childNodes[0];t.classList.remove("ce-icon-unlink"),t.classList.add("ce-icon-link")}},e}({})},function(e,t){"use strict";e.exports=function(e){var t=codex.editor;return e.opened=!1,e.open=function(){t.toolbar.settings.opened&&t.toolbar.settings.close(),t.nodes.toolbox.classList.add("opened"),t.nodes.plusButton.classList.add("clicked"),t.toolbar.toolbox.opened=!0},e.close=function(){t.nodes.toolbox.classList.remove("opened"),t.nodes.plusButton.classList.remove("clicked"),t.toolbar.toolbox.opened=!1,t.toolbar.current=null},e.leaf=function(){var e=t.toolbar.current,n=Object.keys(t.tools),o=t.nodes.toolbarButtons,r=0,i=void 0,a=void 0,s=void 0;if(e)for(r=(n.indexOf(e)+1)%n.length,a=n[r];!t.tools[a].displayInToolbox;)r=(r+1)%n.length,a=n[r];else for(s in t.tools){if(t.tools[s].displayInToolbox)break;r++}i=n[r];for(var c in o)o[c].classList.remove("selected");o[i].classList.add("selected"),t.toolbar.current=i},e.toolClicked=function(e){var n,o,r,i=["image","link","list","instagram","twitter","embed"],a=t.tools[t.toolbar.current],s=t.content.currentNode,c=t.caret.inputIndex;n=a.render(),r={block:n,type:a.type,stretched:!1},s&&i.indexOf(s.dataset.tool)===-1&&""===s.textContent.trim()?t.content.switchBlock(s,n,a.type):(t.content.insertBlock(r),c++),o=a.appendCallback,o&&"function"==typeof o&&o.call(e),window.setTimeout(function(){t.caret.setToBlock(c)},10),t.content.workingNodeChanged(),t.toolbar.move()},e}({})},function(e,t){"use strict";e.exports=function(e){var t=codex.editor;e.globalKeydown=function(e){switch(e.keyCode){case t.core.keys.ENTER:o(e)}},e.redactorKeyDown=function(e){switch(e.keyCode){case t.core.keys.TAB:n(e);break;case t.core.keys.ENTER:i(e);break;case t.core.keys.ESC:a(e);break;default:c(e)}},e.globalKeyup=function(e){switch(e.keyCode){case t.core.keys.UP:case t.core.keys.LEFT:case t.core.keys.RIGHT:case t.core.keys.DOWN:s(e)}};var n=function(e){e.preventDefault();var n=t.content.currentNode.querySelectorAll("textarea, input"),o=!0,r=!t.content.currentNode.textContent.trim();Array.prototype.map.call(n,function(e){"textarea"!=e.type&&"text"!=e.type||(o=o&&!e.value.trim())});var i=r&&o;i&&(t.toolbar.opened||t.toolbar.open(),t.toolbar.opened&&!t.toolbar.toolbox.opened?t.toolbar.toolbox.open():t.toolbar.toolbox.leaf())},o=function(){t.content.editorAreaHightlighted&&(t.caret.inputIndex=-1,r())},r=function(){var e=t.settings.initialBlockPlugin;t.content.insertBlock({type:e,block:t.tools[e].render()},!0),t.toolbar.move(),t.toolbar.open()},i=function(e){"true"==e.target.contentEditable&&t.caret.saveCurrentInputIndex();var n=t.caret.getCurrentInputIndex()||0,o=t.content.currentNode,r=o.dataset.tool,i=t.toolbar.opened&&t.toolbar.current&&e.target==t.state.inputs[n],a=t.tools[r].enableLineBreaks,s=t.settings.initialBlockPlugin;if(i)return e.preventDefault(),t.toolbar.toolbox.toolClicked(e),t.toolbar.close(),e.stopPropagation(),void e.stopImmediatePropagation();if(e.shiftKey||a)return e.stopPropagation(),void e.stopImmediatePropagation();var c=window.getSelection(),l=c.anchorNode,d=t.caret.position.atTheEnd(),u=!1;if(e.shiftKey&&!a)return t.callback.enterPressedOnBlock(t.content.currentBlock,e),void e.preventDefault();if(u=l&&"true"!=l.parentNode.contentEditable,l.nodeType!=t.core.nodeTypes.TEXT||u||d){var p=t.content.isLastNode(l);p&&d&&(e.preventDefault(),e.stopPropagation(),e.stopImmediatePropagation(),t.core.log("ENTER clicked in last textNode. Create new BLOCK"),t.content.insertBlock({type:s,block:t.tools[s].render()},!0),t.toolbar.move(),t.toolbar.open(),t.toolbar.showPlusButton())}else e.preventDefault(),t.core.log("Splitting Text node..."),t.content.splitBlock(n),t.state.inputs[n+1].textContent.trim()||t.toolbar.showPlusButton();t.ui.saveInputs()},a=function(e){t.toolbar.close(),t.toolbar.toolbox.close(),e.preventDefault()},s=function(e){t.content.workingNodeChanged(),t.toolbar.close(),t.toolbar.move()},c=function(){t.toolbar.close(),t.toolbar.inline.actionsOpened||(t.toolbar.inline.close(),t.content.clearMark())};e.redactorClicked=function(e){l(),t.content.workingNodeChanged(e.target),t.ui.saveInputs();var n,o=t.toolbar.inline.getSelectionText();if(0===o.length&&t.toolbar.inline.close(),"true"==e.target.contentEditable&&t.caret.saveCurrentInputIndex(),null===t.content.currentNode){var r=t.state.inputs.length>0?t.state.inputs.length-1:0;if(t.state.inputs.length&&(n=t.content.getFirstLevelBlock(t.state.inputs[r])),t.state.inputs.length&&""===t.state.inputs[r].textContent&&n.dataset.tool==t.settings.initialBlockPlugin)t.caret.setToBlock(r);else{var i=t.settings.initialBlockPlugin;t.content.insertBlock({type:i,block:t.tools[i].render()}),1===t.state.inputs.length?t.caret.setToBlock(r):t.caret.setToNextBlock(r)}}else t.toolbar.settings.close(),t.toolbar.toolbox.close();t.toolbar.move(),t.toolbar.open();var a=!t.content.currentNode.textContent.trim(),s=t.content.currentNode.dataset.tool,c=s==t.settings.initialBlockPlugin;t.toolbar.hidePlusButton(),a||t.content.markBlock(),c&&a&&t.toolbar.showPlusButton()};var l=function(){var e=window.getSelection(),n=e.anchorNode,o=!1;if(0===e.rangeCount)t.content.editorAreaHightlighted=!0;else{for(t.core.isDomNode(n)||(n=n.parentNode),"true"==n.contentEditable&&(o=!0);"true"!=n.contentEditable&&(n=n.parentNode,"true"==n.contentEditable&&(o=!0),n!=document.body););t.content.editorAreaHightlighted=!o}};e.toolbarButtonClicked=function(e){var n=this;t.toolbar.current=n.dataset.type,t.toolbar.toolbox.toolClicked(e),t.toolbar.close()},e.plusButtonClicked=function(){t.nodes.toolbox.classList.contains("opened")?t.toolbar.toolbox.close():t.toolbar.toolbox.open()},e.blockKeydown=function(e){var n=e.target;switch(e.keyCode){case t.core.keys.DOWN:case t.core.keys.RIGHT:d(e);break;case t.core.keys.BACKSPACE:p(n,e);break;case t.core.keys.UP:case t.core.keys.LEFT:u(e)}};var d=function(e){var n,o=window.getSelection(),r=t.state.inputs,i=o.anchorNode;if(!i)return!1;for(;"true"!=i.contentEditable;)n=i.parentNode,i=n;for(var a=0;i!=r[a];)a++;if(!i.textContent)return void t.caret.setToNextBlock(a);var s,c,l=!1,d=!1;return s=i.childNodes[i.childNodes.length-1],c=t.core.isDomNode(s)?t.content.getDeepestTextNodeFromPosition(s,s.childNodes.length):s,l=o.anchorNode==c,d=c.length==o.anchorOffset,l&&d?void t.caret.setToNextBlock(a):(t.core.log("arrow [down|right] : caret does not reached the end"),!1)},u=function(e){var n,o=window.getSelection(),r=t.state.inputs,i=o.anchorNode;if(!i)return!1;if(0!==o.anchorOffset)return!1;for(;"true"!=i.contentEditable;)n=i.parentNode,i=n;for(var a=0;i!=r[a];)a++;var s,c,l=!1,d=!1;return i.textContent?(s=i.childNodes[0],c=t.core.isDomNode(s)?t.content.getDeepestTextNodeFromPosition(s,0):s,l=o.anchorNode==c,d=0===o.anchorOffset,void(l&&d&&t.caret.setToPreviousBlock(a))):void t.caret.setToPreviousBlock(a)},p=function(e,n){var o,r,i,a=t.caret.getCurrentInputIndex();if(t.core.isNativeInput(n.target)){if(""!=n.target.value.trim())return;e.remove()}if(e.textContent.trim()){if(o=t.content.getRange(),r=o.endOffset-o.startOffset,!t.caret.position.atStart()||r||!t.state.inputs[a-1])return;t.content.mergeBlocks(a)}r||e.remove(),i=t.nodes.redactor.childNodes.length,0===i?(t.content.currentNode=null,t.ui.addInitialBlock(),t.ui.saveInputs(),window.setTimeout(function(){ -t.caret.setToPreviousBlock(1)},10)):0!==t.caret.inputIndex?t.caret.setToPreviousBlock(t.caret.inputIndex):t.caret.setToNextBlock(t.caret.inputIndex),t.toolbar.move(),t.toolbar.opened||t.toolbar.open(),t.ui.saveInputs(),n.preventDefault()};return e.showSettingsButtonClicked=function(e){var n=t.content.currentNode.dataset.tool;t.toolbar.settings.toggle(n),t.toolbar.toolbox.close(),t.toolbar.settings.hideRemoveActions()},e}({})},function(e,t){"use strict";e.exports=function(e){return e.wrapper=function(){var e=document.createElement("div");return e.className+="codex-editor",e},e.redactor=function(){var e=document.createElement("div");return e.className+="ce-redactor",e},e.ceBlock=function(){var e=document.createElement("DIV");return e.className+="ce_block",e},e.toolbar=function(){var e=document.createElement("div");return e.className+="ce-toolbar",e},e.toolbarContent=function(){var e=document.createElement("DIV");return e.classList.add("ce-toolbar__content"),e},e.inlineToolbar=function(){var e=document.createElement("DIV");return e.className+="ce-toolbar-inline",e},e.inlineToolbarButtons=function(){var e=document.createElement("DIV");return e.className+="ce-toolbar-inline__buttons",e},e.inlineToolbarActions=function(){var e=document.createElement("DIV");return e.className+="ce-toolbar-inline__actions",e},e.inputForLink=function(){var e=document.createElement("INPUT");return e.type="input",e.className+="inputForLink",e.placeholder="Вставьте ссылку ...",e.setAttribute("form","defaultForm"),e.setAttribute("autofocus","autofocus"),e},e.blockButtons=function(){var e=document.createElement("div");return e.className+="ce-toolbar__actions",e},e.blockSettings=function(){var e=document.createElement("div");return e.className+="ce-settings",e},e.defaultSettings=function(){var e=document.createElement("div");return e.classList.add("ce-settings_default"),e},e.pluginsSettings=function(){var e=document.createElement("div");return e.classList.add("ce-settings_plugin"),e},e.plusButton=function(){var e=document.createElement("span");return e.className="ce-toolbar__plus",e},e.settingsButton=function(){var e=document.createElement("span");return e.className="ce-toolbar__settings-btn",e.innerHTML='',e},e.toolbox=function(){var e=document.createElement("div");return e.className="ce-toolbar__tools",e},e.toolbarButton=function(e,t){var n=document.createElement("li"),o=document.createElement("i"),r=document.createElement("span");return n.dataset.type=e,n.setAttribute("title",e),o.classList.add(t),r.classList.add("ce_toolbar_tools--title"),n.appendChild(o),n.appendChild(r),n},e.toolbarButtonInline=function(e,t){var n=document.createElement("BUTTON"),o=document.createElement("I");return n.type="button",n.dataset.type=e,o.classList.add(t),n.appendChild(o),n},e.block=function(e,t){var n=document.createElement(e);return n.innerHTML=t||"",n},e.node=function(e,t,n){var o=document.createElement(e);if(t&&(o.className=t),n)for(var r in n)o[r]=n[r];return o},e.unavailableBlock=function(){var e=document.createElement("DIV");return e.classList.add("cdx-unavailable-block"),e},e}({})},function(e,t){"use strict";e.exports=function(e){var t=codex.editor;return e.inputIndex=null,e.offset=null,e.focusedNodeIndex=null,e.set=function(n,o,r){r=r||e.offset||0,o=o||e.focusedNodeIndex||0;var i,a=n.childNodes;if(i=0===a.length?n:a[o],"true"!=n.contentEditable)return void n.focus();t.core.isDomNode(i)&&(i=t.content.getDeepestTextNodeFromPosition(i,i.childNodes.length));var s=document.createRange(),c=window.getSelection();window.setTimeout(function(){s.setStart(i,r),s.setEnd(i,r),c.removeAllRanges(),c.addRange(s),t.caret.saveCurrentInputIndex()},20)},e.saveCurrentInputIndex=function(){var n,o=window.getSelection(),r=t.state.inputs,i=o.anchorNode;if(i){for(;"true"!=i.contentEditable;)n=i.parentNode,i=n;for(var a=0;i!=r[a];)a++;e.inputIndex=a}},e.getCurrentInputIndex=function(){return e.inputIndex},e.setToNextBlock=function(e){var n=t.state.inputs,o=n[e+1];if(!o)return void t.core.log("We are reached the end");if(!o.childNodes.length){var r=document.createTextNode("");o.appendChild(r)}t.caret.inputIndex=e+1,t.caret.set(o,0,0),t.content.workingNodeChanged(o)},e.setToBlock=function(e){var n=t.state.inputs,o=n[e];if(o){if(!o.childNodes.length){var r=document.createTextNode("");o.appendChild(r)}t.caret.inputIndex=e,t.caret.set(o,0,0),t.content.workingNodeChanged(o)}},e.setToPreviousBlock=function(e){e=e||0;var n,o,r,i=t.state.inputs,a=i[e-1];return a?(n=t.content.getDeepestTextNodeFromPosition(a,a.childNodes.length),o=n.length,a.childNodes.length||(r=document.createTextNode(""),a.appendChild(r)),t.caret.inputIndex=e-1,t.caret.set(a,a.childNodes.length-1,o),void t.content.workingNodeChanged(i[e-1])):void t.core.log("We are reached first node")},e.position={atStart:function(){var e=window.getSelection(),n=e.anchorOffset,o=e.anchorNode,r=t.content.getFirstLevelBlock(o),i=r.childNodes[0];t.core.isDomNode(o)||(o=o.parentNode);var a=o===i.childNodes[0],s=0===n;return a&&s},atTheEnd:function(){var e=window.getSelection(),t=e.anchorOffset,n=e.anchorNode;return!n||!n.length||t===n.length}},e.insertNode=function(e){var t,n;t=window.getSelection(),n=t.getRangeAt(0),n.deleteContents(),n.insertNode(e),n.setStartAfter(e),n.collapse(!0),t.removeAllRanges(),t.addRange(n)},e}({})},function(e,t){"use strict";e.exports=function(e){var t=codex.editor,n=[],o=function(e){n.push(e);for(var t=0;t5;)"confirm"!=n[t].type&&"prompt"!=n[t].type?(n[t].close(),n.splice(t,1)):t++};return e.createHolder=function(){var e=t.draw.node("DIV","cdx-notifications-block");return t.nodes.notifications=document.body.appendChild(e),e},e.errorThrown=function(e,n){t.notifications.notification({message:"This action is not available currently",type:n.type})},e.notification=function(e){function n(e){if(!e||!e.message)return void t.core.log("Can't create notification. Message is missed");e.type=e.type||"alert",e.time=1e3*e.time||1e4;var n=t.draw.node("DIV","cdx-notification"),o=t.draw.node("DIV","cdx-notification__message"),r=t.draw.node("INPUT","cdx-notification__input"),f=t.draw.node("SPAN","cdx-notification__ok-btn"),g=t.draw.node("SPAN","cdx-notification__cancel-btn");o.textContent=e.message,f.textContent=e.okMsg||"ОК",g.textContent=e.cancelMsg||"Отмена",t.listeners.add(f,"click",u),t.listeners.add(g,"click",p),n.appendChild(o),"prompt"==e.type&&n.appendChild(r),n.appendChild(f),"prompt"!=e.type&&"confirm"!=e.type||n.appendChild(g),n.classList.add("cdx-notification-"+e.type),n.dataset.type=e.type,a=n,c=e.type,l=e.confirm,s=e.cancel,d=r,"prompt"!=e.type&&"confirm"!=e.type&&window.setTimeout(i,e.time)}function r(){t.nodes.notifications.appendChild(a),d.focus(),t.nodes.notifications.classList.add("cdx-notification__notification-appending"),window.setTimeout(function(){t.nodes.notifications.classList.remove("cdx-notification__notification-appending")},100),o({type:c,close:i})}function i(){a.remove()}var a=null,s=null,c=null,l=null,d=null,u=function(){if(i(),"function"==typeof l)return"prompt"==c?void l(d.value):void l()},p=function(){i(),"function"==typeof s&&s()};return e&&(n(e),r()),{create:n,send:r,close:i}},e.clear=function(){t.nodes.notifications.innerHTML="",n=[]},e}({})},function(e,t){"use strict";e.exports=function(e){var t=codex.editor;return e.insertPastedContent=function(e,n){t.content.insertBlock({type:e.type,block:e.render({text:n.innerHTML})})},e.isFirstLevelBlock=function(e){return e.nodeType==t.core.nodeTypes.TAG&&e.classList.contains(t.ui.className.BLOCK_CLASSNAME)},e}({})},function(e,t,n){"use strict";e.exports=function(e){var t=n(18),o=codex.editor;e.prepare=function(){o.settings.sanitizer&&!o.core.isEmpty(o.settings.sanitizer)&&(r.CUSTOM=o.settings.sanitizer)};var r={CUSTOM:null,BASIC:{tags:{p:{},a:{href:!0,target:"_blank",rel:"nofollow"},i:{},b:{},strong:{},em:{},span:{}}}};e.Config=r;var i=function(e){var n=e||r.CUSTOM||r.BASIC;return new t(n)};return e.clean=function(e,t){var n=i(t);return n.clean(e)},e}({})},function(e,t,n){var o,r;!function(i,a){o=a,r="function"==typeof o?o.call(t,n,t,e):o,!(void 0!==r&&(e.exports=r))}(this,function(){function e(e){var t=e.tags,n=Object.keys(t),o=n.map(function(e){return typeof t[e]}).every(function(e){return"object"===e||"boolean"===e||"function"===e});if(!o)throw new Error("The configuration was invalid");this.config=e}function t(e){return s.indexOf(e.nodeName)!==-1}function n(e){return c.indexOf(e.nodeName)!==-1}function o(e){return document.createTreeWalker(e,NodeFilter.SHOW_TEXT|NodeFilter.SHOW_ELEMENT|NodeFilter.SHOW_COMMENT,null,!1)}function r(e,t,n){return"function"==typeof e.tags[t]?e.tags[t](n):e.tags[t]}function i(e,t){return"undefined"==typeof t||"boolean"==typeof t&&!t}function a(e,t,n){var o=e.name.toLowerCase();return t!==!0&&("function"==typeof t[o]?!t[o](e.value,n):"undefined"==typeof t[o]||(t[o]===!1||"string"==typeof t[o]&&t[o]!==e.value))}var s=["P","LI","TD","TH","DIV","H1","H2","H3","H4","H5","H6","PRE"],c=["A","B","STRONG","I","EM","SUB","SUP","U","STRIKE"];return e.prototype.clean=function(e){var t=document.createElement("div");return t.innerHTML=e,this._sanitize(t),t.innerHTML},e.prototype._sanitize=function(e){var s=o(e),c=s.firstChild();if(c)do if(!c._sanitized)if(c.nodeType!==Node.TEXT_NODE){if(c.nodeType===Node.COMMENT_NODE){e.removeChild(c),this._sanitize(e);break}var l,d=n(c);d&&(l=Array.prototype.some.call(c.childNodes,t));var u=!!e.parentNode,p=t(e)&&t(c)&&u,f=c.nodeName.toLowerCase(),g=r(this.config,f,c),h=d&&l;if(h||i(c,g)||!this.config.keepNestedBlockElements&&p){if("SCRIPT"!==c.nodeName&&"STYLE"!==c.nodeName)for(;c.childNodes.length>0;)e.insertBefore(c.childNodes[0],c);e.removeChild(c),this._sanitize(e);break}for(var v=0;v0&&t.splice(s,1)}},e.removeAll=function(){t.map(function(t){e.remove(t.element,t.type,t.handler)})},e.get=function(t,n,o){return e.search.all(t,n,o)},e}({})},function(e,t){"use strict";var n="function"==typeof Symbol&&"symbol"==typeof Symbol.iterator?function(e){return typeof e}:function(e){return e&&"function"==typeof Symbol&&e.constructor===Symbol&&e!==Symbol.prototype?"symbol":typeof e};e.exports=function(e){var t=codex.editor;return e.removeNodes=function(){t.nodes.wrapper.remove(),t.nodes.notifications.remove()},e.destroyPlugins=function(){for(var e in t.tools)"function"==typeof t.tools[e].destroy&&t.tools[e].destroy()},e.destroyScripts=function(){for(var e=document.getElementsByTagName("SCRIPT"),n=0;n"+e.split("\n\n").join("

    ")+"

    "};return e.getEditableParent=function(e){for(;e&&"true"!=e.contentEditable;)e=e.parentNode;return e},e}({})},function(e,t,n){"use strict";e.exports=function(e){var t=codex.editor;return e.settings=n(9),e.inline=n(10),e.toolbox=n(11),e.defaultToolbarHeight=49,e.defaultOffset=34,e.opened=!1,e.current=null,e.open=function(){t.hideToolbar||(t.nodes.toolbar.classList.add("opened"),this.opened=!0)},e.close=function(){t.nodes.toolbar.classList.remove("opened"),e.opened=!1,e.current=null;for(var n in t.nodes.toolbarButtons)t.nodes.toolbarButtons[n].classList.remove("selected");t.toolbar.toolbox.close(),t.toolbar.settings.close()},e.toggle=function(){this.opened?this.close():this.open()},e.hidePlusButton=function(){t.nodes.plusButton.classList.add("hide")},e.showPlusButton=function(){t.nodes.plusButton.classList.remove("hide")},e.move=function(){if(t.toolbar.toolbox.close(),t.content.currentNode){var e=t.content.currentNode.offsetTop-t.toolbar.defaultToolbarHeight/2+t.toolbar.defaultOffset;t.nodes.toolbar.style.transform="translate3D(0, "+Math.floor(e)+"px, 0)",t.toolbar.settings.hideRemoveActions()}},e}({})},function(e,t){"use strict";e.exports=function(e){var t=codex.editor;return e.opened=!1,e.setting=null,e.actions=null,e.open=function(e){if(t.tools[e]&&t.tools[e].makeSettings){var n=t.tools[e].makeSettings();t.nodes.pluginSettings.appendChild(n)}else t.core.log("Plugin «"+e+"» has no settings","warn"),t.nodes.pluginSettings.innerHTML="Плагин «"+e+"» не имеет настроек";t.nodes.blockSettings.classList.add("opened"),this.opened=!0},e.close=function(){t.nodes.blockSettings.classList.remove("opened"),t.nodes.pluginSettings.innerHTML="",this.opened=!1},e.toggle=function(e){this.opened?this.close():this.open(e)},e.makeRemoveBlockButton=function(){var e=t.draw.node("SPAN","ce-toolbar__remove-btn",{}),n=t.draw.node("SPAN","ce-toolbar__remove-setting",{innerHTML:''}),o=t.draw.node("DIV","ce-toolbar__remove-confirmation",{}),r=t.draw.node("DIV","ce-toolbar__remove-confirm",{textContent:"Удалить блок"}),i=t.draw.node("DIV","ce-toolbar__remove-cancel",{textContent:"Отмена"});return t.listeners.add(n,"click",t.toolbar.settings.removeButtonClicked,!1),t.listeners.add(r,"click",t.toolbar.settings.confirmRemovingRequest,!1),t.listeners.add(i,"click",t.toolbar.settings.cancelRemovingRequest,!1),o.appendChild(r),o.appendChild(i),e.appendChild(n),e.appendChild(o),t.toolbar.settings.setting=n,t.toolbar.settings.actions=o,e},e.removeButtonClicked=function(){var e=t.toolbar.settings.actions;e.classList.contains("opened")?t.toolbar.settings.hideRemoveActions():t.toolbar.settings.showRemoveActions(),t.toolbar.toolbox.close(),t.toolbar.settings.close()},e.cancelRemovingRequest=function(){t.toolbar.settings.actions.classList.remove("opened")},e.confirmRemovingRequest=function(){var e,n=t.content.currentNode;n.remove(),e=t.nodes.redactor.childNodes.length,0===e&&(t.content.currentNode=null,t.ui.addInitialBlock()),t.ui.saveInputs(),t.toolbar.close()},e.showRemoveActions=function(){t.toolbar.settings.actions.classList.add("opened")},e.hideRemoveActions=function(){t.toolbar.settings.actions.classList.remove("opened")},e}({})},function(e,t){"use strict";e.exports=function(e){var t=codex.editor;e.buttonsOpened=null,e.actionsOpened=null,e.wrappersOffset=null,e.storedSelection=null,e.show=function(){var n,o=t.content.currentNode,r=o.dataset.tool;if(n=t.tools[r],n.showInlineToolbar){var i=e.getSelectionText(),a=t.nodes.inlineToolbar.wrapper;i.length>0&&(t.toolbar.inline.move(),a.classList.add("opened"),t.toolbar.inline.showButtons())}},e.close=function(){var e=t.nodes.inlineToolbar.wrapper;e.classList.remove("opened")},e.move=function(){this.wrappersOffset||(this.wrappersOffset=this.getWrappersOffset());var e,n,o=this.getSelectionCoords(),r=0,i=t.nodes.inlineToolbar.wrapper;0===i.offsetHeight&&(r=40),e=o.x-this.wrappersOffset.left,n=o.y+window.scrollY-this.wrappersOffset.top-r-i.offsetHeight,i.style.transform="translate3D("+Math.floor(e)+"px, "+Math.floor(n)+"px, 0)",t.toolbar.inline.closeButtons(),t.toolbar.inline.closeAction()},e.toolClicked=function(e,n){switch(n){case"createLink":t.toolbar.inline.createLinkAction(e,n);break;default:t.toolbar.inline.defaultToolAction(n)}t.nodes.inlineToolbar.buttons.childNodes.forEach(t.toolbar.inline.hightlight)},e.getWrappersOffset=function(){var e=t.nodes.wrapper,n=this.getOffset(e);return this.wrappersOffset=n,n},e.getOffset=function(e){for(var t=0,n=0;e&&!isNaN(e.offsetLeft)&&!isNaN(e.offsetTop);)t+=e.offsetLeft+e.clientLeft,n+=e.offsetTop+e.clientTop,e=e.offsetParent;return{top:n,left:t}},e.getSelectionCoords=function(){var e,t=document.selection,n=0,o=0;if(t)"Control"!=t.type&&(e=t.createRange(),e.collapse(!0),n=e.boundingLeft,o=e.boundingTop);else if(window.getSelection&&(t=window.getSelection(),t.rangeCount&&(e=t.getRangeAt(0).cloneRange(),e.getClientRects))){e.collapse(!0);var r=e.getClientRects()[0];if(!r)return;n=r.left,o=r.top}return{x:n,y:o}},e.getSelectionText=function(){var e="";return window.getSelection&&(e=window.getSelection().toString()),e},e.showButtons=function(){var e=t.nodes.inlineToolbar.buttons;e.classList.add("opened"),t.toolbar.inline.buttonsOpened=!0,t.nodes.inlineToolbar.buttons.childNodes.forEach(t.toolbar.inline.hightlight)},e.closeButtons=function(){var e=t.nodes.inlineToolbar.buttons;e.classList.remove("opened"),t.toolbar.inline.buttonsOpened=!1},e.showActions=function(){var e=t.nodes.inlineToolbar.actions;e.classList.add("opened"),t.toolbar.inline.actionsOpened=!0},e.closeAction=function(){var e=t.nodes.inlineToolbar.actions;e.innerHTML="",e.classList.remove("opened"),t.toolbar.inline.actionsOpened=!1};var n=function(e){if(e.keyCode==t.core.keys.ENTER){var n=t.content.currentNode,o=t.toolbar.inline.storedSelection;t.toolbar.inline.restoreSelection(n,o),t.toolbar.inline.setAnchor(this.value),e.preventDefault(),e.stopImmediatePropagation(),t.toolbar.inline.clearRange()}};return e.createLinkAction=function(e){var o=this.isLinkActive(),r=t.content.currentNode,i=t.toolbar.inline.saveSelection(r);if(t.toolbar.inline.storedSelection=i,o)t.toolbar.inline.restoreSelection(r,i),t.toolbar.inline.defaultToolAction("unlink");else{var a=t.draw.inputForLink();t.nodes.inlineToolbar.actions.appendChild(a),t.toolbar.inline.closeButtons(),t.toolbar.inline.showActions(),a.focus(),e.preventDefault(),t.listeners.add(a,"keydown",n,!1)}},e.isLinkActive=function(){var e=!1;return t.nodes.inlineToolbar.buttons.childNodes.forEach(function(t){var n=t.dataset.type;"link"==n&&t.classList.contains("hightlighted")&&(e=!0)}),e},e.defaultToolAction=function(e){document.execCommand(e,!1,null)},e.setAnchor=function(e){document.execCommand("createLink",!1,e),t.toolbar.inline.closeAction()},e.saveSelection=function(e){var t,n=window.getSelection().getRangeAt(0),o=n.cloneRange();return o.selectNodeContents(e),o.setEnd(n.startContainer,n.startOffset),t=o.toString().length,{start:t,end:t+n.toString().length}},e.restoreSelection=function(e,t){var n=document.createRange(),o=0;n.setStart(e,0),n.collapse(!0);for(var r,i,a=[e],s=!1,c=!1;!c&&(r=a.pop());)if(3==r.nodeType)i=o+r.length,!s&&t.start>=o&&t.start<=i&&(n.setStart(r,t.start-o),s=!0),s&&t.end>=o&&t.end<=i&&(n.setEnd(r,t.end-o),c=!0),o=i;else for(var l=r.childNodes.length;l--;)a.push(r.childNodes[l]);var d=window.getSelection();d.removeAllRanges(),d.addRange(n)},e.clearRange=function(){var e=window.getSelection();e.removeAllRanges()},e.hightlight=function(e){var n=e.dataset.type;document.queryCommandState(n)?t.toolbar.inline.setButtonHighlighted(e):t.toolbar.inline.removeButtonsHighLight(e);var o=window.getSelection(),r=o.anchorNode.parentNode;"A"==r.tagName&&"link"==n&&t.toolbar.inline.setButtonHighlighted(e)},e.setButtonHighlighted=function(e){if(e.classList.add("hightlighted"),"link"==e.dataset.type){var t=e.childNodes[0];t.classList.remove("ce-icon-link"),t.classList.add("ce-icon-unlink")}},e.removeButtonsHighLight=function(e){if(e.classList.remove("hightlighted"),"link"==e.dataset.type){var t=e.childNodes[0];t.classList.remove("ce-icon-unlink"),t.classList.add("ce-icon-link")}},e}({})},function(e,t){"use strict";e.exports=function(e){var t=codex.editor;return e.opened=!1,e.open=function(){t.toolbar.settings.opened&&t.toolbar.settings.close(),t.nodes.toolbox.classList.add("opened"),t.nodes.plusButton.classList.add("clicked"),t.toolbar.toolbox.opened=!0},e.close=function(){t.nodes.toolbox.classList.remove("opened"),t.nodes.plusButton.classList.remove("clicked"),t.toolbar.toolbox.opened=!1,t.toolbar.current=null},e.leaf=function(){var e=t.toolbar.current,n=Object.keys(t.tools),o=t.nodes.toolbarButtons,r=0,i=void 0,a=void 0,s=void 0;if(e)for(r=(n.indexOf(e)+1)%n.length,a=n[r];!t.tools[a].displayInToolbox;)r=(r+1)%n.length,a=n[r];else for(s in t.tools){if(t.tools[s].displayInToolbox)break;r++}i=n[r];for(var c in o)o[c].classList.remove("selected");o[i].classList.add("selected"),t.toolbar.current=i},e.toolClicked=function(e){var n,o,r,i=["image","link","list","instagram","twitter","embed"],a=t.tools[t.toolbar.current],s=t.content.currentNode,c=t.caret.inputIndex;n=a.render(),r={block:n,type:a.type,stretched:!1},s&&i.indexOf(s.dataset.tool)===-1&&""===s.textContent.trim()?t.content.switchBlock(s,n,a.type):(t.content.insertBlock(r),c++),o=a.appendCallback,o&&"function"==typeof o&&o.call(e),window.setTimeout(function(){t.caret.setToBlock(c)},10),t.content.workingNodeChanged(),t.toolbar.move()},e}({})},function(e,t){"use strict";e.exports=function(e){var t=codex.editor;e.globalKeydown=function(e){switch(e.keyCode){case t.core.keys.ENTER:o(e)}},e.redactorKeyDown=function(e){switch(e.keyCode){case t.core.keys.TAB:n(e);break;case t.core.keys.ENTER:i(e);break;case t.core.keys.ESC:a(e);break;default:c(e)}},e.globalKeyup=function(e){switch(e.keyCode){case t.core.keys.UP:case t.core.keys.LEFT:case t.core.keys.RIGHT:case t.core.keys.DOWN:s(e)}};var n=function(e){e.preventDefault();var n=t.content.currentNode.querySelectorAll("textarea, input"),o=!0,r=!t.content.currentNode.textContent.trim();Array.prototype.map.call(n,function(e){"textarea"!=e.type&&"text"!=e.type||(o=o&&!e.value.trim())});var i=r&&o;i&&(t.toolbar.opened||t.toolbar.open(),t.toolbar.opened&&!t.toolbar.toolbox.opened?t.toolbar.toolbox.open():t.toolbar.toolbox.leaf())},o=function(){t.content.editorAreaHightlighted&&(t.caret.inputIndex=-1,r())},r=function(){var e=t.settings.initialBlockPlugin;t.content.insertBlock({type:e,block:t.tools[e].render()},!0),t.toolbar.move(),t.toolbar.open()},i=function(e){"true"==e.target.contentEditable&&t.caret.saveCurrentInputIndex();var n=t.caret.getCurrentInputIndex()||0,o=t.content.currentNode,r=o.dataset.tool,i=t.toolbar.opened&&t.toolbar.current&&e.target==t.state.inputs[n],a=t.tools[r].enableLineBreaks,s=t.settings.initialBlockPlugin;if(i)return e.preventDefault(),t.toolbar.toolbox.toolClicked(e),t.toolbar.close(),e.stopPropagation(),void e.stopImmediatePropagation();if(e.shiftKey||a)return e.stopPropagation(),void e.stopImmediatePropagation();var c=window.getSelection(),l=c.anchorNode,d=t.caret.position.atTheEnd(),u=!1;if(e.shiftKey&&!a)return t.callback.enterPressedOnBlock(t.content.currentBlock,e),void e.preventDefault();if(u=l&&"true"!=l.parentNode.contentEditable,l.nodeType!=t.core.nodeTypes.TEXT||u||d){var p=t.content.isLastNode(l);p&&d&&(e.preventDefault(),e.stopPropagation(),e.stopImmediatePropagation(),t.core.log("ENTER clicked in last textNode. Create new BLOCK"),t.content.insertBlock({type:s,block:t.tools[s].render()},!0),t.toolbar.move(),t.toolbar.open(),t.toolbar.showPlusButton())}else e.preventDefault(),t.core.log("Splitting Text node..."),t.content.splitBlock(n),t.state.inputs[n+1].textContent.trim()||t.toolbar.showPlusButton();t.ui.saveInputs()},a=function(e){t.toolbar.close(),t.toolbar.toolbox.close(),e.preventDefault()},s=function(e){t.content.workingNodeChanged(),t.toolbar.close(),t.toolbar.move()},c=function(){t.toolbar.close(),t.toolbar.inline.actionsOpened||(t.toolbar.inline.close(),t.content.clearMark())};e.redactorClicked=function(e){l(),t.content.workingNodeChanged(e.target),t.ui.saveInputs();var n,o=t.toolbar.inline.getSelectionText();if(0===o.length&&t.toolbar.inline.close(),"true"==e.target.contentEditable&&t.caret.saveCurrentInputIndex(),null===t.content.currentNode){var r=t.state.inputs.length>0?t.state.inputs.length-1:0;if(t.state.inputs.length&&(n=t.content.getFirstLevelBlock(t.state.inputs[r])),t.state.inputs.length&&""===t.state.inputs[r].textContent&&n.dataset.tool==t.settings.initialBlockPlugin)t.caret.setToBlock(r);else{var i=t.settings.initialBlockPlugin;t.content.insertBlock({type:i,block:t.tools[i].render()}),1===t.state.inputs.length?t.caret.setToBlock(r):t.caret.setToNextBlock(r)}}else t.toolbar.settings.close(),t.toolbar.toolbox.close();t.toolbar.move(),t.toolbar.open();var a=!t.content.currentNode.textContent.trim(),s=t.content.currentNode.dataset.tool,c=s==t.settings.initialBlockPlugin;t.toolbar.hidePlusButton(),a||t.content.markBlock(),c&&a&&t.toolbar.showPlusButton()};var l=function(){var e=window.getSelection(),n=e.anchorNode,o=!1;if(0===e.rangeCount)t.content.editorAreaHightlighted=!0;else{for(t.core.isDomNode(n)||(n=n.parentNode),"true"==n.contentEditable&&(o=!0);"true"!=n.contentEditable&&(n=n.parentNode,"true"==n.contentEditable&&(o=!0),n!=document.body););t.content.editorAreaHightlighted=!o}};e.toolbarButtonClicked=function(e){var n=this;t.toolbar.current=n.dataset.type,t.toolbar.toolbox.toolClicked(e),t.toolbar.close()},e.plusButtonClicked=function(){t.nodes.toolbox.classList.contains("opened")?t.toolbar.toolbox.close():t.toolbar.toolbox.open()},e.blockKeydown=function(e){var n=e.target;switch(e.keyCode){case t.core.keys.DOWN:case t.core.keys.RIGHT:d(e);break;case t.core.keys.BACKSPACE:p(n,e);break;case t.core.keys.UP:case t.core.keys.LEFT:u(e)}};var d=function(e){var n,o=window.getSelection(),r=t.state.inputs,i=o.anchorNode;if(!i)return!1;for(;"true"!=i.contentEditable;)n=i.parentNode,i=n;for(var a=0;i!=r[a];)a++;if(!i.textContent)return void t.caret.setToNextBlock(a);var s,c,l=!1,d=!1;return s=i.childNodes[i.childNodes.length-1],c=t.core.isDomNode(s)?t.content.getDeepestTextNodeFromPosition(s,s.childNodes.length):s,l=o.anchorNode==c,d=c.length==o.anchorOffset,l&&d?void t.caret.setToNextBlock(a):(t.core.log("arrow [down|right] : caret does not reached the end"),!1)},u=function(e){var n,o=window.getSelection(),r=t.state.inputs,i=o.anchorNode;if(!i)return!1;if(0!==o.anchorOffset)return!1;for(;"true"!=i.contentEditable;)n=i.parentNode,i=n;for(var a=0;i!=r[a];)a++;var s,c,l=!1,d=!1;return i.textContent?(s=i.childNodes[0],c=t.core.isDomNode(s)?t.content.getDeepestTextNodeFromPosition(s,0):s,l=o.anchorNode==c,d=0===o.anchorOffset,void(l&&d&&t.caret.setToPreviousBlock(a))):void t.caret.setToPreviousBlock(a)},p=function(e,n){var o,r,i,a=t.caret.getCurrentInputIndex();if(t.core.isNativeInput(n.target)){if(""!=n.target.value.trim())return;e.remove()}if(e.textContent.trim()){if(o=t.content.getRange(),r=o.endOffset-o.startOffset,!t.caret.position.atStart()||r||!t.state.inputs[a-1])return;t.content.mergeBlocks(a)}r||e.remove(),i=t.nodes.redactor.childNodes.length,0===i?(t.content.currentNode=null,t.ui.addInitialBlock(),t.ui.saveInputs(), +window.setTimeout(function(){t.caret.setToPreviousBlock(1)},10)):0!==t.caret.inputIndex?t.caret.setToPreviousBlock(t.caret.inputIndex):t.caret.setToNextBlock(t.caret.inputIndex),t.toolbar.move(),t.toolbar.opened||t.toolbar.open(),t.ui.saveInputs(),n.preventDefault()};return e.showSettingsButtonClicked=function(e){var n=t.content.currentNode.dataset.tool;t.toolbar.settings.toggle(n),t.toolbar.toolbox.close(),t.toolbar.settings.hideRemoveActions()},e}({})},function(e,t){"use strict";e.exports=function(e){return e.wrapper=function(){var e=document.createElement("div");return e.className+="codex-editor",e},e.redactor=function(){var e=document.createElement("div");return e.className+="ce-redactor",e},e.ceBlock=function(){var e=document.createElement("DIV");return e.className+="ce_block",e},e.toolbar=function(){var e=document.createElement("div");return e.className+="ce-toolbar",e},e.toolbarContent=function(){var e=document.createElement("DIV");return e.classList.add("ce-toolbar__content"),e},e.inlineToolbar=function(){var e=document.createElement("DIV");return e.className+="ce-toolbar-inline",e},e.inlineToolbarButtons=function(){var e=document.createElement("DIV");return e.className+="ce-toolbar-inline__buttons",e},e.inlineToolbarActions=function(){var e=document.createElement("DIV");return e.className+="ce-toolbar-inline__actions",e},e.inputForLink=function(){var e=document.createElement("INPUT");return e.type="input",e.className+="inputForLink",e.placeholder="Вставьте ссылку ...",e.setAttribute("form","defaultForm"),e.setAttribute("autofocus","autofocus"),e},e.blockButtons=function(){var e=document.createElement("div");return e.className+="ce-toolbar__actions",e},e.blockSettings=function(){var e=document.createElement("div");return e.className+="ce-settings",e},e.defaultSettings=function(){var e=document.createElement("div");return e.classList.add("ce-settings_default"),e},e.pluginsSettings=function(){var e=document.createElement("div");return e.classList.add("ce-settings_plugin"),e},e.plusButton=function(){var e=document.createElement("span");return e.className="ce-toolbar__plus",e},e.settingsButton=function(){var e=document.createElement("span");return e.className="ce-toolbar__settings-btn",e.innerHTML='',e},e.toolbox=function(){var e=document.createElement("div");return e.className="ce-toolbar__tools",e},e.toolbarButton=function(e,t){var n=document.createElement("li"),o=document.createElement("i"),r=document.createElement("span");return n.dataset.type=e,n.setAttribute("title",e),o.classList.add(t),r.classList.add("ce_toolbar_tools--title"),n.appendChild(o),n.appendChild(r),n},e.toolbarButtonInline=function(e,t){var n=document.createElement("BUTTON"),o=document.createElement("I");return n.type="button",n.dataset.type=e,o.classList.add(t),n.appendChild(o),n},e.block=function(e,t){var n=document.createElement(e);return n.innerHTML=t||"",n},e.node=function(e,t,n){var o=document.createElement(e);if(t&&(o.className=t),n)for(var r in n)o[r]=n[r];return o},e.unavailableBlock=function(){var e=document.createElement("DIV");return e.classList.add("cdx-unavailable-block"),e},e}({})},function(e,t){"use strict";e.exports=function(e){var t=codex.editor;return e.inputIndex=null,e.offset=null,e.focusedNodeIndex=null,e.set=function(n,o,r){r=r||e.offset||0,o=o||e.focusedNodeIndex||0;var i,a=n.childNodes;if(i=0===a.length?n:a[o],"true"!=n.contentEditable)return void n.focus();t.core.isDomNode(i)&&(i=t.content.getDeepestTextNodeFromPosition(i,i.childNodes.length));var s=document.createRange(),c=window.getSelection();window.setTimeout(function(){s.setStart(i,r),s.setEnd(i,r),c.removeAllRanges(),c.addRange(s),t.caret.saveCurrentInputIndex()},20)},e.saveCurrentInputIndex=function(){var n,o=window.getSelection(),r=t.state.inputs,i=o.anchorNode;if(i){for(;"true"!=i.contentEditable;)n=i.parentNode,i=n;for(var a=0;i!=r[a];)a++;e.inputIndex=a}},e.getCurrentInputIndex=function(){return e.inputIndex},e.setToNextBlock=function(e){var n=t.state.inputs,o=n[e+1];if(!o)return void t.core.log("We are reached the end");if(!o.childNodes.length){var r=document.createTextNode("");o.appendChild(r)}t.caret.inputIndex=e+1,t.caret.set(o,0,0),t.content.workingNodeChanged(o)},e.setToBlock=function(e){var n=t.state.inputs,o=n[e];if(o){if(!o.childNodes.length){var r=document.createTextNode("");o.appendChild(r)}t.caret.inputIndex=e,t.caret.set(o,0,0),t.content.workingNodeChanged(o)}},e.setToPreviousBlock=function(e){e=e||0;var n,o,r,i=t.state.inputs,a=i[e-1];return a?(n=t.content.getDeepestTextNodeFromPosition(a,a.childNodes.length),o=n.length,a.childNodes.length||(r=document.createTextNode(""),a.appendChild(r)),t.caret.inputIndex=e-1,t.caret.set(a,a.childNodes.length-1,o),void t.content.workingNodeChanged(i[e-1])):void t.core.log("We are reached first node")},e.position={atStart:function(){var e=window.getSelection(),n=e.anchorOffset,o=e.anchorNode,r=t.content.getFirstLevelBlock(o),i=r.childNodes[0];t.core.isDomNode(o)||(o=o.parentNode);var a=o===i.childNodes[0],s=0===n;return a&&s},atTheEnd:function(){var e=window.getSelection(),t=e.anchorOffset,n=e.anchorNode;return!n||!n.length||t===n.length}},e.insertNode=function(e){var t,n;t=window.getSelection(),n=t.getRangeAt(0),n.deleteContents(),n.insertNode(e),n.setStartAfter(e),n.collapse(!0),t.removeAllRanges(),t.addRange(n)},e}({})},function(e,t){"use strict";e.exports=function(e){var t=codex.editor,n=[],o=function(e){n.push(e);for(var t=0;t5;)"confirm"!=n[t].type&&"prompt"!=n[t].type?(n[t].close(),n.splice(t,1)):t++};return e.createHolder=function(){var e=t.draw.node("DIV","cdx-notifications-block");return t.nodes.notifications=document.body.appendChild(e),e},e.errorThrown=function(e,n){t.notifications.notification({message:"This action is not available currently",type:n.type})},e.notification=function(e){function n(e){if(!e||!e.message)return void t.core.log("Can't create notification. Message is missed");e.type=e.type||"alert",e.time=1e3*e.time||1e4;var n=t.draw.node("DIV","cdx-notification"),o=t.draw.node("DIV","cdx-notification__message"),r=t.draw.node("INPUT","cdx-notification__input"),f=t.draw.node("SPAN","cdx-notification__ok-btn"),g=t.draw.node("SPAN","cdx-notification__cancel-btn");o.textContent=e.message,f.textContent=e.okMsg||"ОК",g.textContent=e.cancelMsg||"Отмена",t.listeners.add(f,"click",u),t.listeners.add(g,"click",p),n.appendChild(o),"prompt"==e.type&&n.appendChild(r),n.appendChild(f),"prompt"!=e.type&&"confirm"!=e.type||n.appendChild(g),n.classList.add("cdx-notification-"+e.type),n.dataset.type=e.type,a=n,c=e.type,l=e.confirm,s=e.cancel,d=r,"prompt"!=e.type&&"confirm"!=e.type&&window.setTimeout(i,e.time)}function r(){t.nodes.notifications.appendChild(a),d.focus(),t.nodes.notifications.classList.add("cdx-notification__notification-appending"),window.setTimeout(function(){t.nodes.notifications.classList.remove("cdx-notification__notification-appending")},100),o({type:c,close:i})}function i(){a.remove()}var a=null,s=null,c=null,l=null,d=null,u=function(){if(i(),"function"==typeof l)return"prompt"==c?void l(d.value):void l()},p=function(){i(),"function"==typeof s&&s()};return e&&(n(e),r()),{create:n,send:r,close:i}},e.clear=function(){t.nodes.notifications.innerHTML="",n=[]},e}({})},function(e,t){"use strict";e.exports=function(e){var t=codex.editor;return e.insertPastedContent=function(e,n){t.content.insertBlock({type:e.type,block:e.render({text:n.innerHTML})})},e.isFirstLevelBlock=function(e){return e.nodeType==t.core.nodeTypes.TAG&&e.classList.contains(t.ui.className.BLOCK_CLASSNAME)},e}({})},function(e,t,n){"use strict";e.exports=function(e){var t=n(18),o=codex.editor;e.prepare=function(){o.settings.sanitizer&&!o.core.isEmpty(o.settings.sanitizer)&&(r.CUSTOM=o.settings.sanitizer)};var r={CUSTOM:null,BASIC:{tags:{p:{},a:{href:!0,target:"_blank",rel:"nofollow"},i:{},b:{},strong:{},em:{},span:{}}}};e.Config=r;var i=function(e){var n=e||r.CUSTOM||r.BASIC;return new t(n)};return e.clean=function(e,t){var n=i(t);return n.clean(e)},e}({})},function(e,t,n){var o,r;!function(i,a){o=a,r="function"==typeof o?o.call(t,n,t,e):o,!(void 0!==r&&(e.exports=r))}(this,function(){function e(e){var t=e.tags,n=Object.keys(t),o=n.map(function(e){return typeof t[e]}).every(function(e){return"object"===e||"boolean"===e||"function"===e});if(!o)throw new Error("The configuration was invalid");this.config=e}function t(e){return s.indexOf(e.nodeName)!==-1}function n(e){return c.indexOf(e.nodeName)!==-1}function o(e){return document.createTreeWalker(e,NodeFilter.SHOW_TEXT|NodeFilter.SHOW_ELEMENT|NodeFilter.SHOW_COMMENT,null,!1)}function r(e,t,n){return"function"==typeof e.tags[t]?e.tags[t](n):e.tags[t]}function i(e,t){return"undefined"==typeof t||"boolean"==typeof t&&!t}function a(e,t,n){var o=e.name.toLowerCase();return t!==!0&&("function"==typeof t[o]?!t[o](e.value,n):"undefined"==typeof t[o]||(t[o]===!1||"string"==typeof t[o]&&t[o]!==e.value))}var s=["P","LI","TD","TH","DIV","H1","H2","H3","H4","H5","H6","PRE"],c=["A","B","STRONG","I","EM","SUB","SUP","U","STRIKE"];return e.prototype.clean=function(e){var t=document.createElement("div");return t.innerHTML=e,this._sanitize(t),t.innerHTML},e.prototype._sanitize=function(e){var s=o(e),c=s.firstChild();if(c)do if(!c._sanitized)if(c.nodeType!==Node.TEXT_NODE){if(c.nodeType===Node.COMMENT_NODE){e.removeChild(c),this._sanitize(e);break}var l,d=n(c);d&&(l=Array.prototype.some.call(c.childNodes,t));var u=!!e.parentNode,p=t(e)&&t(c)&&u,f=c.nodeName.toLowerCase(),g=r(this.config,f,c),h=d&&l;if(h||i(c,g)||!this.config.keepNestedBlockElements&&p){if("SCRIPT"!==c.nodeName&&"STYLE"!==c.nodeName)for(;c.childNodes.length>0;)e.insertBefore(c.childNodes[0],c);e.removeChild(c),this._sanitize(e);break}for(var v=0;v0&&t.splice(s,1)}},e.removeAll=function(){t.map(function(t){e.remove(t.element,t.type,t.handler)})},e.get=function(t,n,o){return e.search.all(t,n,o)},e}({})},function(e,t){"use strict";var n="function"==typeof Symbol&&"symbol"==typeof Symbol.iterator?function(e){return typeof e}:function(e){return e&&"function"==typeof Symbol&&e.constructor===Symbol&&e!==Symbol.prototype?"symbol":typeof e};e.exports=function(e){var t=codex.editor;return e.removeNodes=function(){t.nodes.wrapper.remove(),t.nodes.notifications.remove()},e.destroyPlugins=function(){for(var e in t.tools)"function"==typeof t.tools[e].destroy&&t.tools[e].destroy()},e.destroyScripts=function(){for(var e=document.getElementsByTagName("SCRIPT"),n=0;n nodeList\r\n\t */\n\t renderer.appendBlocks = function (data) {\n\t\n\t var blocks = data.items;\n\t\n\t /**\r\n\t * Sequence of one-by-one blocks appending\r\n\t * Uses to save blocks order after async-handler\r\n\t */\n\t var nodeSequence = Promise.resolve();\n\t\n\t for (var index = 0; index < blocks.length; index++) {\n\t\n\t /** Add node to sequence at specified index */\n\t editor.renderer.appendNodeAtIndex(nodeSequence, blocks, index);\n\t }\n\t };\n\t\n\t /**\r\n\t * Append node at specified index\r\n\t */\n\t renderer.appendNodeAtIndex = function (nodeSequence, blocks, index) {\n\t\n\t /** We need to append node to sequence */\n\t nodeSequence\n\t\n\t /** first, get node async-aware */\n\t .then(function () {\n\t\n\t return editor.renderer.getNodeAsync(blocks, index);\n\t })\n\t\n\t /**\r\n\t * second, compose editor-block from JSON object\r\n\t */\n\t .then(editor.renderer.createBlockFromData)\n\t\n\t /**\r\n\t * now insert block to redactor\r\n\t */\n\t .then(function (blockData) {\n\t\n\t /**\r\n\t * blockData has 'block', 'type' and 'stretched' information\r\n\t */\n\t editor.content.insertBlock(blockData);\n\t\n\t /** Pass created block to next step */\n\t return blockData.block;\n\t })\n\t\n\t /** Log if something wrong with node */\n\t .catch(function (error) {\n\t\n\t editor.core.log('Node skipped while parsing because %o', 'error', error);\n\t });\n\t };\n\t\n\t /**\r\n\t * Asynchronously returns block data from blocksList by index\r\n\t * @return Promise to node\r\n\t */\n\t renderer.getNodeAsync = function (blocksList, index) {\n\t\n\t return Promise.resolve().then(function () {\n\t\n\t return {\n\t tool: blocksList[index],\n\t position: index\n\t };\n\t });\n\t };\n\t\n\t /**\r\n\t * Creates editor block by JSON-data\r\n\t *\r\n\t * @uses render method of each plugin\r\n\t *\r\n\t * @param {Object} toolData.tool\r\n\t * { header : {\r\n\t * text: '',\r\n\t * type: 'H3', ...\r\n\t * }\r\n\t * }\r\n\t * @param {Number} toolData.position - index in input-blocks array\r\n\t * @return {Object} with type and Element\r\n\t */\n\t renderer.createBlockFromData = function (toolData) {\n\t\n\t /** New parser */\n\t var block,\n\t tool = toolData.tool,\n\t pluginName = tool.type;\n\t\n\t /** Get first key of object that stores plugin name */\n\t // for (var pluginName in blockData) break;\n\t\n\t /** Check for plugin existance */\n\t if (!editor.tools[pluginName]) {\n\t\n\t throw Error('Plugin \\xAB' + pluginName + '\\xBB not found');\n\t }\n\t\n\t /** Check for plugin having render method */\n\t if (typeof editor.tools[pluginName].render != 'function') {\n\t\n\t throw Error('Plugin \\xAB' + pluginName + '\\xBB must have \\xABrender\\xBB method');\n\t }\n\t\n\t if (editor.tools[pluginName].available === false) {\n\t\n\t block = editor.draw.unavailableBlock();\n\t\n\t block.innerHTML = editor.tools[pluginName].loadingMessage;\n\t\n\t /**\r\n\t * Saver will extract data from initial block data by position in array\r\n\t */\n\t block.dataset.inputPosition = toolData.position;\n\t } else {\n\t\n\t /** New Parser */\n\t block = editor.tools[pluginName].render(tool.data);\n\t }\n\t\n\t /** is first-level block stretched */\n\t var stretched = editor.tools[pluginName].isStretched || false;\n\t\n\t /** Retrun type and block */\n\t return {\n\t type: pluginName,\n\t block: block,\n\t stretched: stretched\n\t };\n\t };\n\t\n\t return renderer;\n\t}({});\n\n/***/ },\n/* 6 */\n/***/ function(module, exports) {\n\n\t'use strict';\n\t\n\t/**\r\n\t * Codex Editor Saver\r\n\t *\r\n\t * @author Codex Team\r\n\t * @version 1.0.2\r\n\t */\n\t\n\tmodule.exports = function (saver) {\n\t\n\t var editor = codex.editor;\n\t\n\t /**\r\n\t * Saves blocks\r\n\t * @private\r\n\t */\n\t saver.saveBlocks = function () {\n\t\n\t /** Save html content of redactor to memory */\n\t editor.state.html = editor.nodes.redactor.innerHTML;\n\t\n\t /** Empty jsonOutput state */\n\t editor.state.jsonOutput = [];\n\t\n\t Promise.resolve().then(function () {\n\t\n\t return editor.nodes.redactor.childNodes;\n\t })\n\t /** Making a sequence from separate blocks */\n\t .then(editor.saver.makeQueue).then(function () {\n\t // editor.nodes.textarea.innerHTML = editor.state.html;\n\t }).catch(function (error) {\n\t\n\t editor.core.log(error);\n\t });\n\t };\n\t\n\t saver.makeQueue = function (blocks) {\n\t\n\t var queue = Promise.resolve();\n\t\n\t for (var index = 0; index < blocks.length; index++) {\n\t\n\t /** Add node to sequence at specified index */\n\t editor.saver.getBlockData(queue, blocks, index);\n\t }\n\t };\n\t\n\t /** Gets every block and makes From Data */\n\t saver.getBlockData = function (queue, blocks, index) {\n\t\n\t queue.then(function () {\n\t\n\t return editor.saver.getNodeAsync(blocks, index);\n\t }).then(editor.saver.makeFormDataFromBlocks);\n\t };\n\t\n\t /**\r\n\t * Asynchronously returns block data from blocksList by index\r\n\t * @return Promise to node\r\n\t */\n\t saver.getNodeAsync = function (blocksList, index) {\n\t\n\t return Promise.resolve().then(function () {\n\t\n\t return blocksList[index];\n\t });\n\t };\n\t\n\t saver.makeFormDataFromBlocks = function (block) {\n\t\n\t var pluginName = block.dataset.tool;\n\t\n\t /** Check for plugin existance */\n\t if (!editor.tools[pluginName]) {\n\t\n\t throw Error('Plugin \\xAB' + pluginName + '\\xBB not found');\n\t }\n\t\n\t /** Check for plugin having render method */\n\t if (typeof editor.tools[pluginName].save != 'function') {\n\t\n\t throw Error('Plugin \\xAB' + pluginName + '\\xBB must have save method');\n\t }\n\t\n\t /** Result saver */\n\t var blockContent = block.childNodes[0],\n\t pluginsContent = blockContent.childNodes[0],\n\t savedData,\n\t position,\n\t output;\n\t\n\t /** If plugin wasn't available then return data from cache */\n\t if (editor.tools[pluginName].available === false) {\n\t\n\t position = pluginsContent.dataset.inputPosition;\n\t savedData = codex.editor.state.blocks.items[position].data;\n\t } else {\n\t\n\t savedData = editor.tools[pluginName].save(pluginsContent);\n\t\n\t if (editor.tools[pluginName].validate) {\n\t\n\t var result = editor.tools[pluginName].validate(savedData);\n\t\n\t /**\r\n\t * Do not allow invalid data\r\n\t */\n\t if (!result) return;\n\t }\n\t }\n\t\n\t output = {\n\t type: pluginName,\n\t data: savedData\n\t };\n\t\n\t editor.state.jsonOutput.push(output);\n\t };\n\t\n\t return saver;\n\t}({});\n\n/***/ },\n/* 7 */\n/***/ function(module, exports) {\n\n\t'use strict';\n\t\n\t/**\r\n\t * Codex Editor Content Module\r\n\t * Works with DOM\r\n\t *\r\n\t * @module Codex Editor content module\r\n\t *\r\n\t * @author Codex Team\r\n\t * @version 1.3.13\r\n\t *\r\n\t * @description Module works with Elements that have been appended to the main DOM\r\n\t */\n\t\n\tmodule.exports = function (content) {\n\t\n\t var editor = codex.editor;\n\t\n\t /**\r\n\t * Links to current active block\r\n\t * @type {null | Element}\r\n\t */\n\t content.currentNode = null;\n\t\n\t /**\r\n\t * clicked in redactor area\r\n\t * @type {null | Boolean}\r\n\t */\n\t content.editorAreaHightlighted = null;\n\t\n\t /**\r\n\t * @deprecated\r\n\t * Synchronizes redactor with original textarea\r\n\t */\n\t content.sync = function () {\n\t\n\t editor.core.log('syncing...');\n\t\n\t /**\r\n\t * Save redactor content to editor.state\r\n\t */\n\t editor.state.html = editor.nodes.redactor.innerHTML;\n\t };\n\t\n\t /**\r\n\t * Appends background to the block\r\n\t *\r\n\t * @description add CSS class to highlight visually first-level block area\r\n\t */\n\t content.markBlock = function () {\n\t\n\t editor.content.currentNode.classList.add(editor.ui.className.BLOCK_HIGHLIGHTED);\n\t };\n\t\n\t /**\r\n\t * Clear background\r\n\t *\r\n\t * @description clears styles that highlights block\r\n\t */\n\t content.clearMark = function () {\n\t\n\t if (editor.content.currentNode) {\n\t\n\t editor.content.currentNode.classList.remove(editor.ui.className.BLOCK_HIGHLIGHTED);\n\t }\n\t };\n\t\n\t /**\r\n\t * Finds first-level block\r\n\t *\r\n\t * @param {Element} node - selected or clicked in redactors area node\r\n\t * @protected\r\n\t *\r\n\t * @description looks for first-level block.\r\n\t * gets parent while node is not first-level\r\n\t */\n\t content.getFirstLevelBlock = function (node) {\n\t\n\t if (!editor.core.isDomNode(node)) {\n\t\n\t node = node.parentNode;\n\t }\n\t\n\t if (node === editor.nodes.redactor || node === document.body) {\n\t\n\t return null;\n\t } else {\n\t\n\t while (!node.classList.contains(editor.ui.className.BLOCK_CLASSNAME)) {\n\t\n\t node = node.parentNode;\n\t }\n\t\n\t return node;\n\t }\n\t };\n\t\n\t /**\r\n\t * Trigger this event when working node changed\r\n\t * @param {Element} targetNode - first-level of this node will be current\r\n\t * @protected\r\n\t *\r\n\t * @description If targetNode is first-level then we set it as current else we look for parents to find first-level\r\n\t */\n\t content.workingNodeChanged = function (targetNode) {\n\t\n\t /** Clear background from previous marked block before we change */\n\t editor.content.clearMark();\n\t\n\t if (!targetNode) {\n\t\n\t return;\n\t }\n\t\n\t content.currentNode = content.getFirstLevelBlock(targetNode);\n\t };\n\t\n\t /**\r\n\t * Replaces one redactor block with another\r\n\t * @protected\r\n\t * @param {Element} targetBlock - block to replace. Mostly currentNode.\r\n\t * @param {Element} newBlock\r\n\t * @param {string} newBlockType - type of new block; we need to store it to data-attribute\r\n\t *\r\n\t * [!] Function does not saves old block content.\r\n\t * You can get it manually and pass with newBlock.innerHTML\r\n\t */\n\t content.replaceBlock = function (targetBlock, newBlock) {\n\t\n\t if (!targetBlock || !newBlock) {\n\t\n\t editor.core.log('replaceBlock: missed params');\n\t return;\n\t }\n\t\n\t /** If target-block is not a frist-level block, then we iterate parents to find it */\n\t while (!targetBlock.classList.contains(editor.ui.className.BLOCK_CLASSNAME)) {\n\t\n\t targetBlock = targetBlock.parentNode;\n\t }\n\t\n\t /** Replacing */\n\t editor.nodes.redactor.replaceChild(newBlock, targetBlock);\n\t\n\t /**\r\n\t * Set new node as current\r\n\t */\n\t editor.content.workingNodeChanged(newBlock);\n\t\n\t /**\r\n\t * Add block handlers\r\n\t */\n\t editor.ui.addBlockHandlers(newBlock);\n\t\n\t /**\r\n\t * Save changes\r\n\t */\n\t editor.ui.saveInputs();\n\t };\n\t\n\t /**\r\n\t * @protected\r\n\t *\r\n\t * Inserts new block to redactor\r\n\t * Wrapps block into a DIV with BLOCK_CLASSNAME class\r\n\t *\r\n\t * @param blockData {object}\r\n\t * @param blockData.block {Element} element with block content\r\n\t * @param blockData.type {string} block plugin\r\n\t * @param needPlaceCaret {bool} pass true to set caret in new block\r\n\t *\r\n\t */\n\t content.insertBlock = function (blockData, needPlaceCaret) {\n\t\n\t var workingBlock = editor.content.currentNode,\n\t newBlockContent = blockData.block,\n\t blockType = blockData.type,\n\t isStretched = blockData.stretched;\n\t\n\t var newBlock = composeNewBlock_(newBlockContent, blockType, isStretched);\n\t\n\t if (workingBlock) {\n\t\n\t editor.core.insertAfter(workingBlock, newBlock);\n\t } else {\n\t\n\t /**\r\n\t * If redactor is empty, append as first child\r\n\t */\n\t editor.nodes.redactor.appendChild(newBlock);\n\t }\n\t\n\t /**\r\n\t * Block handler\r\n\t */\n\t editor.ui.addBlockHandlers(newBlock);\n\t\n\t /**\r\n\t * Set new node as current\r\n\t */\n\t editor.content.workingNodeChanged(newBlock);\n\t\n\t /**\r\n\t * Save changes\r\n\t */\n\t editor.ui.saveInputs();\n\t\n\t if (needPlaceCaret) {\n\t\n\t /**\r\n\t * If we don't know input index then we set default value -1\r\n\t */\n\t var currentInputIndex = editor.caret.getCurrentInputIndex() || -1;\n\t\n\t if (currentInputIndex == -1) {\n\t\n\t var editableElement = newBlock.querySelector('[contenteditable]'),\n\t emptyText = document.createTextNode('');\n\t\n\t editableElement.appendChild(emptyText);\n\t editor.caret.set(editableElement, 0, 0);\n\t\n\t editor.toolbar.move();\n\t editor.toolbar.showPlusButton();\n\t } else {\n\t\n\t if (currentInputIndex === editor.state.inputs.length - 1) return;\n\t\n\t /** Timeout for browsers execution */\n\t window.setTimeout(function () {\n\t\n\t /** Setting to the new input */\n\t editor.caret.setToNextBlock(currentInputIndex);\n\t editor.toolbar.move();\n\t editor.toolbar.open();\n\t }, 10);\n\t }\n\t }\n\t\n\t /**\r\n\t * Block is inserted, wait for new click that defined focusing on editors area\r\n\t * @type {boolean}\r\n\t */\n\t content.editorAreaHightlighted = false;\n\t };\n\t\n\t /**\r\n\t * Replaces blocks with saving content\r\n\t * @protected\r\n\t * @param {Element} noteToReplace\r\n\t * @param {Element} newNode\r\n\t * @param {Element} blockType\r\n\t */\n\t content.switchBlock = function (blockToReplace, newBlock, tool) {\n\t\n\t tool = tool || editor.content.currentNode.dataset.tool;\n\t var newBlockComposed = composeNewBlock_(newBlock, tool);\n\t\n\t /** Replacing */\n\t editor.content.replaceBlock(blockToReplace, newBlockComposed);\n\t\n\t /** Save new Inputs when block is changed */\n\t editor.ui.saveInputs();\n\t };\n\t\n\t /**\r\n\t * Iterates between child noted and looking for #text node on deepest level\r\n\t * @protected\r\n\t *\r\n\t * @param {Element} block - node where find\r\n\t * @param {int} postiton - starting postion\r\n\t * Example: childNodex.length to find from the end\r\n\t * or 0 to find from the start\r\n\t * @return {Text} block\r\n\t * @uses DFS\r\n\t */\n\t content.getDeepestTextNodeFromPosition = function (block, position) {\n\t\n\t /**\r\n\t * Clear Block from empty and useless spaces with trim.\r\n\t * Such nodes we should remove\r\n\t */\n\t var blockChilds = block.childNodes,\n\t index,\n\t node,\n\t text;\n\t\n\t for (index = 0; index < blockChilds.length; index++) {\n\t\n\t node = blockChilds[index];\n\t\n\t if (node.nodeType == editor.core.nodeTypes.TEXT) {\n\t\n\t text = node.textContent.trim();\n\t\n\t /** Text is empty. We should remove this child from node before we start DFS\r\n\t * decrease the quantity of childs.\r\n\t */\n\t if (text === '') {\n\t\n\t block.removeChild(node);\n\t position--;\n\t }\n\t }\n\t }\n\t\n\t if (block.childNodes.length === 0) {\n\t\n\t return document.createTextNode('');\n\t }\n\t\n\t /** Setting default position when we deleted all empty nodes */\n\t if (position < 0) position = 1;\n\t\n\t var lookingFromStart = false;\n\t\n\t /** For looking from START */\n\t if (position === 0) {\n\t\n\t lookingFromStart = true;\n\t position = 1;\n\t }\n\t\n\t while (position) {\n\t\n\t /** initial verticle of node. */\n\t if (lookingFromStart) {\n\t\n\t block = block.childNodes[0];\n\t } else {\n\t\n\t block = block.childNodes[position - 1];\n\t }\n\t\n\t if (block.nodeType == editor.core.nodeTypes.TAG) {\n\t\n\t position = block.childNodes.length;\n\t } else if (block.nodeType == editor.core.nodeTypes.TEXT) {\n\t\n\t position = 0;\n\t }\n\t }\n\t\n\t return block;\n\t };\n\t\n\t /**\r\n\t * @private\r\n\t * @param {Element} block - current plugins render\r\n\t * @param {String} tool - plugins name\r\n\t * @param {Boolean} isStretched - make stretched block or not\r\n\t *\r\n\t * @description adds necessary information to wrap new created block by first-level holder\r\n\t */\n\t var composeNewBlock_ = function composeNewBlock_(block, tool, isStretched) {\n\t\n\t var newBlock = editor.draw.node('DIV', editor.ui.className.BLOCK_CLASSNAME, {}),\n\t blockContent = editor.draw.node('DIV', editor.ui.className.BLOCK_CONTENT, {});\n\t\n\t blockContent.appendChild(block);\n\t newBlock.appendChild(blockContent);\n\t\n\t if (isStretched) {\n\t\n\t blockContent.classList.add(editor.ui.className.BLOCK_STRETCHED);\n\t }\n\t\n\t newBlock.dataset.tool = tool;\n\t return newBlock;\n\t };\n\t\n\t /**\r\n\t * Returns Range object of current selection\r\n\t * @protected\r\n\t */\n\t content.getRange = function () {\n\t\n\t var selection = window.getSelection().getRangeAt(0);\n\t\n\t return selection;\n\t };\n\t\n\t /**\r\n\t * Divides block in two blocks (after and before caret)\r\n\t *\r\n\t * @protected\r\n\t * @param {int} inputIndex - target input index\r\n\t *\r\n\t * @description splits current input content to the separate blocks\r\n\t * When enter is pressed among the words, that text will be splited.\r\n\t */\n\t content.splitBlock = function (inputIndex) {\n\t\n\t var selection = window.getSelection(),\n\t anchorNode = selection.anchorNode,\n\t anchorNodeText = anchorNode.textContent,\n\t caretOffset = selection.anchorOffset,\n\t textBeforeCaret,\n\t textNodeBeforeCaret,\n\t textAfterCaret,\n\t textNodeAfterCaret;\n\t\n\t var currentBlock = editor.content.currentNode.querySelector('[contentEditable]');\n\t\n\t textBeforeCaret = anchorNodeText.substring(0, caretOffset);\n\t textAfterCaret = anchorNodeText.substring(caretOffset);\n\t\n\t textNodeBeforeCaret = document.createTextNode(textBeforeCaret);\n\t\n\t if (textAfterCaret) {\n\t\n\t textNodeAfterCaret = document.createTextNode(textAfterCaret);\n\t }\n\t\n\t var previousChilds = [],\n\t nextChilds = [],\n\t reachedCurrent = false;\n\t\n\t if (textNodeAfterCaret) {\n\t\n\t nextChilds.push(textNodeAfterCaret);\n\t }\n\t\n\t for (var i = 0, child; !!(child = currentBlock.childNodes[i]); i++) {\n\t\n\t if (child != anchorNode) {\n\t\n\t if (!reachedCurrent) {\n\t\n\t previousChilds.push(child);\n\t } else {\n\t\n\t nextChilds.push(child);\n\t }\n\t } else {\n\t\n\t reachedCurrent = true;\n\t }\n\t }\n\t\n\t /** Clear current input */\n\t editor.state.inputs[inputIndex].innerHTML = '';\n\t\n\t /**\r\n\t * Append all childs founded before anchorNode\r\n\t */\n\t var previousChildsLength = previousChilds.length;\n\t\n\t for (i = 0; i < previousChildsLength; i++) {\n\t\n\t editor.state.inputs[inputIndex].appendChild(previousChilds[i]);\n\t }\n\t\n\t editor.state.inputs[inputIndex].appendChild(textNodeBeforeCaret);\n\t\n\t /**\r\n\t * Append text node which is after caret\r\n\t */\n\t var nextChildsLength = nextChilds.length,\n\t newNode = document.createElement('div');\n\t\n\t for (i = 0; i < nextChildsLength; i++) {\n\t\n\t newNode.appendChild(nextChilds[i]);\n\t }\n\t\n\t newNode = newNode.innerHTML;\n\t\n\t /** This type of block creates when enter is pressed */\n\t var NEW_BLOCK_TYPE = editor.settings.initialBlockPlugin;\n\t\n\t /**\r\n\t * Make new paragraph with text after caret\r\n\t */\n\t editor.content.insertBlock({\n\t type: NEW_BLOCK_TYPE,\n\t block: editor.tools[NEW_BLOCK_TYPE].render({\n\t text: newNode\n\t })\n\t }, true);\n\t };\n\t\n\t /**\r\n\t * Merges two blocks — current and target\r\n\t * If target index is not exist, then previous will be as target\r\n\t *\r\n\t * @protected\r\n\t * @param {int} currentInputIndex\r\n\t * @param {int} targetInputIndex\r\n\t *\r\n\t * @description gets two inputs indexes and merges into one\r\n\t */\n\t content.mergeBlocks = function (currentInputIndex, targetInputIndex) {\n\t\n\t /** If current input index is zero, then prevent method execution */\n\t if (currentInputIndex === 0) {\n\t\n\t return;\n\t }\n\t\n\t var targetInput,\n\t currentInputContent = editor.state.inputs[currentInputIndex].innerHTML;\n\t\n\t if (!targetInputIndex) {\n\t\n\t targetInput = editor.state.inputs[currentInputIndex - 1];\n\t } else {\n\t\n\t targetInput = editor.state.inputs[targetInputIndex];\n\t }\n\t\n\t targetInput.innerHTML += currentInputContent;\n\t };\n\t\n\t /**\r\n\t * Iterates all right siblings and parents, which has right siblings\r\n\t * while it does not reached the first-level block\r\n\t *\r\n\t * @param {Element} node\r\n\t * @return {boolean}\r\n\t */\n\t content.isLastNode = function (node) {\n\t\n\t // console.log('погнали перебор родителей');\n\t\n\t var allChecked = false;\n\t\n\t while (!allChecked) {\n\t\n\t // console.log('Смотрим на %o', node);\n\t // console.log('Проверим, пустые ли соседи справа');\n\t\n\t if (!allSiblingsEmpty_(node)) {\n\t\n\t // console.log('Есть непустые соседи. Узел не последний. Выходим.');\n\t return false;\n\t }\n\t\n\t node = node.parentNode;\n\t\n\t /**\r\n\t * Проверяем родителей до тех пор, пока не найдем блок первого уровня\r\n\t */\n\t if (node.classList.contains(editor.ui.className.BLOCK_CONTENT)) {\n\t\n\t allChecked = true;\n\t }\n\t }\n\t\n\t return true;\n\t };\n\t\n\t /**\r\n\t * Checks if all element right siblings is empty\r\n\t * @param node\r\n\t */\n\t var allSiblingsEmpty_ = function allSiblingsEmpty_(node) {\n\t\n\t /**\r\n\t * Нужно убедиться, что после пустого соседа ничего нет\r\n\t */\n\t var sibling = node.nextSibling;\n\t\n\t while (sibling) {\n\t\n\t if (sibling.textContent.length) {\n\t\n\t return false;\n\t }\n\t\n\t sibling = sibling.nextSibling;\n\t }\n\t\n\t return true;\n\t };\n\t\n\t /**\r\n\t * @public\r\n\t *\r\n\t * @param {string} htmlData - html content as string\r\n\t * @param {string} plainData - plain text\r\n\t * @return {string} - html content as string\r\n\t */\n\t content.wrapTextWithParagraphs = function (htmlData, plainData) {\n\t\n\t if (!htmlData) {\n\t\n\t return '

    ' + plainData.split('\\n\\n').join('

    ') + '

    ';\n\t }\n\t\n\t var wrapper = document.createElement('DIV'),\n\t newWrapper = document.createElement('DIV'),\n\t i,\n\t paragraph,\n\t firstLevelBlocks = ['DIV', 'P'],\n\t blockTyped,\n\t node;\n\t\n\t /**\r\n\t * Make HTML Element to Wrap Text\r\n\t * It allows us to work with input data as HTML content\r\n\t */\n\t wrapper.innerHTML = htmlData;\n\t paragraph = document.createElement('P');\n\t\n\t for (i = 0; i < wrapper.childNodes.length; i++) {\n\t\n\t node = wrapper.childNodes[i];\n\t\n\t blockTyped = firstLevelBlocks.indexOf(node.tagName) != -1;\n\t\n\t /**\r\n\t * If node is first-levet\r\n\t * we add this node to our new wrapper\r\n\t */\n\t if (blockTyped) {\n\t\n\t /**\r\n\t * If we had splitted inline nodes to paragraph before\r\n\t */\n\t if (paragraph.childNodes.length) {\n\t\n\t newWrapper.appendChild(paragraph.cloneNode(true));\n\t\n\t /** empty paragraph */\n\t paragraph = null;\n\t paragraph = document.createElement('P');\n\t }\n\t\n\t newWrapper.appendChild(node.cloneNode(true));\n\t } else {\n\t\n\t /** Collect all inline nodes to one as paragraph */\n\t paragraph.appendChild(node.cloneNode(true));\n\t\n\t /** if node is last we should append this node to paragraph and paragraph to new wrapper */\n\t if (i == wrapper.childNodes.length - 1) {\n\t\n\t newWrapper.appendChild(paragraph.cloneNode(true));\n\t }\n\t }\n\t }\n\t\n\t return newWrapper.innerHTML;\n\t };\n\t\n\t /**\r\n\t * Finds closest Contenteditable parent from Element\r\n\t * @param {Element} node element looking from\r\n\t * @return {Element} node contenteditable\r\n\t */\n\t content.getEditableParent = function (node) {\n\t\n\t while (node && node.contentEditable != 'true') {\n\t\n\t node = node.parentNode;\n\t }\n\t\n\t return node;\n\t };\n\t\n\t return content;\n\t}({});\n\n/***/ },\n/* 8 */\n/***/ function(module, exports, __webpack_require__) {\n\n\t'use strict';\n\t\n\t/**\r\n\t * Codex Editor toolbar module\r\n\t *\r\n\t * Contains:\r\n\t * - Inline toolbox\r\n\t * - Toolbox within plus button\r\n\t * - Settings section\r\n\t *\r\n\t * @author Codex Team\r\n\t * @version 1.0\r\n\t */\n\t\n\tmodule.exports = function (toolbar) {\n\t\n\t var editor = codex.editor;\n\t\n\t toolbar.settings = __webpack_require__(9);\n\t toolbar.inline = __webpack_require__(10);\n\t toolbar.toolbox = __webpack_require__(11);\n\t\n\t /**\r\n\t * Margin between focused node and toolbar\r\n\t */\n\t toolbar.defaultToolbarHeight = 49;\n\t\n\t toolbar.defaultOffset = 34;\n\t\n\t toolbar.opened = false;\n\t\n\t toolbar.current = null;\n\t\n\t /**\r\n\t * @protected\r\n\t */\n\t toolbar.open = function () {\n\t\n\t if (editor.hideToolbar) {\n\t\n\t return;\n\t }\n\t\n\t editor.nodes.toolbar.classList.add('opened');\n\t this.opened = true;\n\t };\n\t\n\t /**\r\n\t * @protected\r\n\t */\n\t toolbar.close = function () {\n\t\n\t editor.nodes.toolbar.classList.remove('opened');\n\t\n\t toolbar.opened = false;\n\t toolbar.current = null;\n\t\n\t for (var button in editor.nodes.toolbarButtons) {\n\t\n\t editor.nodes.toolbarButtons[button].classList.remove('selected');\n\t }\n\t\n\t /** Close toolbox when toolbar is not displayed */\n\t editor.toolbar.toolbox.close();\n\t editor.toolbar.settings.close();\n\t };\n\t\n\t toolbar.toggle = function () {\n\t\n\t if (!this.opened) {\n\t\n\t this.open();\n\t } else {\n\t\n\t this.close();\n\t }\n\t };\n\t\n\t toolbar.hidePlusButton = function () {\n\t\n\t editor.nodes.plusButton.classList.add('hide');\n\t };\n\t\n\t toolbar.showPlusButton = function () {\n\t\n\t editor.nodes.plusButton.classList.remove('hide');\n\t };\n\t\n\t /**\r\n\t * Moving toolbar to the specified node\r\n\t */\n\t toolbar.move = function () {\n\t\n\t /** Close Toolbox when we move toolbar */\n\t editor.toolbar.toolbox.close();\n\t\n\t if (!editor.content.currentNode) {\n\t\n\t return;\n\t }\n\t\n\t var newYCoordinate = editor.content.currentNode.offsetTop - editor.toolbar.defaultToolbarHeight / 2 + editor.toolbar.defaultOffset;\n\t\n\t editor.nodes.toolbar.style.transform = 'translate3D(0, ' + Math.floor(newYCoordinate) + 'px, 0)';\n\t\n\t /** Close trash actions */\n\t editor.toolbar.settings.hideRemoveActions();\n\t };\n\t\n\t return toolbar;\n\t}({});\n\n/***/ },\n/* 9 */\n/***/ function(module, exports) {\n\n\t'use strict';\n\t\n\t/**\r\n\t * Toolbar settings\r\n\t *\r\n\t * @version 1.0.4\r\n\t */\n\t\n\tmodule.exports = function (settings) {\n\t\n\t var editor = codex.editor;\n\t\n\t settings.opened = false;\n\t\n\t settings.setting = null;\n\t settings.actions = null;\n\t\n\t /**\r\n\t * Append and open settings\r\n\t */\n\t settings.open = function (toolType) {\n\t\n\t /**\r\n\t * Append settings content\r\n\t * It's stored in tool.settings\r\n\t */\n\t if (!editor.tools[toolType] || !editor.tools[toolType].makeSettings) {\n\t\n\t editor.core.log('Plugin \\xAB' + toolType + '\\xBB has no settings', 'warn');\n\t editor.nodes.pluginSettings.innerHTML = '\\u041F\\u043B\\u0430\\u0433\\u0438\\u043D \\xAB' + toolType + '\\xBB \\u043D\\u0435 \\u0438\\u043C\\u0435\\u0435\\u0442 \\u043D\\u0430\\u0441\\u0442\\u0440\\u043E\\u0435\\u043A';\n\t } else {\n\t\n\t /**\r\n\t * Draw settings block\r\n\t */\n\t var settingsBlock = editor.tools[toolType].makeSettings();\n\t\n\t editor.nodes.pluginSettings.appendChild(settingsBlock);\n\t }\n\t\n\t /** Open settings block */\n\t editor.nodes.blockSettings.classList.add('opened');\n\t this.opened = true;\n\t };\n\t\n\t /**\r\n\t * Close and clear settings\r\n\t */\n\t settings.close = function () {\n\t\n\t editor.nodes.blockSettings.classList.remove('opened');\n\t editor.nodes.pluginSettings.innerHTML = '';\n\t\n\t this.opened = false;\n\t };\n\t\n\t /**\r\n\t * @param {string} toolType - plugin type\r\n\t */\n\t settings.toggle = function (toolType) {\n\t\n\t if (!this.opened) {\n\t\n\t this.open(toolType);\n\t } else {\n\t\n\t this.close();\n\t }\n\t };\n\t\n\t /**\r\n\t * Here we will draw buttons and add listeners to components\r\n\t */\n\t settings.makeRemoveBlockButton = function () {\n\t\n\t var removeBlockWrapper = editor.draw.node('SPAN', 'ce-toolbar__remove-btn', {}),\n\t settingButton = editor.draw.node('SPAN', 'ce-toolbar__remove-setting', { innerHTML: '' }),\n\t actionWrapper = editor.draw.node('DIV', 'ce-toolbar__remove-confirmation', {}),\n\t confirmAction = editor.draw.node('DIV', 'ce-toolbar__remove-confirm', { textContent: 'Удалить блок' }),\n\t cancelAction = editor.draw.node('DIV', 'ce-toolbar__remove-cancel', { textContent: 'Отмена' });\n\t\n\t editor.listeners.add(settingButton, 'click', editor.toolbar.settings.removeButtonClicked, false);\n\t\n\t editor.listeners.add(confirmAction, 'click', editor.toolbar.settings.confirmRemovingRequest, false);\n\t\n\t editor.listeners.add(cancelAction, 'click', editor.toolbar.settings.cancelRemovingRequest, false);\n\t\n\t actionWrapper.appendChild(confirmAction);\n\t actionWrapper.appendChild(cancelAction);\n\t\n\t removeBlockWrapper.appendChild(settingButton);\n\t removeBlockWrapper.appendChild(actionWrapper);\n\t\n\t /** Save setting */\n\t editor.toolbar.settings.setting = settingButton;\n\t editor.toolbar.settings.actions = actionWrapper;\n\t\n\t return removeBlockWrapper;\n\t };\n\t\n\t settings.removeButtonClicked = function () {\n\t\n\t var action = editor.toolbar.settings.actions;\n\t\n\t if (action.classList.contains('opened')) {\n\t\n\t editor.toolbar.settings.hideRemoveActions();\n\t } else {\n\t\n\t editor.toolbar.settings.showRemoveActions();\n\t }\n\t\n\t editor.toolbar.toolbox.close();\n\t editor.toolbar.settings.close();\n\t };\n\t\n\t settings.cancelRemovingRequest = function () {\n\t\n\t editor.toolbar.settings.actions.classList.remove('opened');\n\t };\n\t\n\t settings.confirmRemovingRequest = function () {\n\t\n\t var currentBlock = editor.content.currentNode,\n\t firstLevelBlocksCount;\n\t\n\t currentBlock.remove();\n\t\n\t firstLevelBlocksCount = editor.nodes.redactor.childNodes.length;\n\t\n\t /**\r\n\t * If all blocks are removed\r\n\t */\n\t if (firstLevelBlocksCount === 0) {\n\t\n\t /** update currentNode variable */\n\t editor.content.currentNode = null;\n\t\n\t /** Inserting new empty initial block */\n\t editor.ui.addInitialBlock();\n\t }\n\t\n\t editor.ui.saveInputs();\n\t\n\t editor.toolbar.close();\n\t };\n\t\n\t settings.showRemoveActions = function () {\n\t\n\t editor.toolbar.settings.actions.classList.add('opened');\n\t };\n\t\n\t settings.hideRemoveActions = function () {\n\t\n\t editor.toolbar.settings.actions.classList.remove('opened');\n\t };\n\t\n\t return settings;\n\t}({});\n\n/***/ },\n/* 10 */\n/***/ function(module, exports) {\n\n\t'use strict';\n\t\n\t/**\r\n\t * Inline toolbar\r\n\t *\r\n\t * Contains from tools:\r\n\t * Bold, Italic, Underline and Anchor\r\n\t *\r\n\t * @author Codex Team\r\n\t * @version 1.0\r\n\t */\n\t\n\tmodule.exports = function (inline) {\n\t\n\t var editor = codex.editor;\n\t\n\t inline.buttonsOpened = null;\n\t inline.actionsOpened = null;\n\t inline.wrappersOffset = null;\n\t\n\t /**\r\n\t * saving selection that need for execCommand for styling\r\n\t *\r\n\t */\n\t inline.storedSelection = null;\n\t\n\t /**\r\n\t * @protected\r\n\t *\r\n\t * Open inline toobar\r\n\t */\n\t inline.show = function () {\n\t\n\t var currentNode = editor.content.currentNode,\n\t tool = currentNode.dataset.tool,\n\t plugin;\n\t\n\t /**\r\n\t * tool allowed to open inline toolbar\r\n\t */\n\t plugin = editor.tools[tool];\n\t\n\t if (!plugin.showInlineToolbar) return;\n\t\n\t var selectedText = inline.getSelectionText(),\n\t toolbar = editor.nodes.inlineToolbar.wrapper;\n\t\n\t if (selectedText.length > 0) {\n\t\n\t /** Move toolbar and open */\n\t editor.toolbar.inline.move();\n\t\n\t /** Open inline toolbar */\n\t toolbar.classList.add('opened');\n\t\n\t /** show buttons of inline toolbar */\n\t editor.toolbar.inline.showButtons();\n\t }\n\t };\n\t\n\t /**\r\n\t * @protected\r\n\t *\r\n\t * Closes inline toolbar\r\n\t */\n\t inline.close = function () {\n\t\n\t var toolbar = editor.nodes.inlineToolbar.wrapper;\n\t\n\t toolbar.classList.remove('opened');\n\t };\n\t\n\t /**\r\n\t * @private\r\n\t *\r\n\t * Moving toolbar\r\n\t */\n\t inline.move = function () {\n\t\n\t if (!this.wrappersOffset) {\n\t\n\t this.wrappersOffset = this.getWrappersOffset();\n\t }\n\t\n\t var coords = this.getSelectionCoords(),\n\t defaultOffset = 0,\n\t toolbar = editor.nodes.inlineToolbar.wrapper,\n\t newCoordinateX,\n\t newCoordinateY;\n\t\n\t if (toolbar.offsetHeight === 0) {\n\t\n\t defaultOffset = 40;\n\t }\n\t\n\t newCoordinateX = coords.x - this.wrappersOffset.left;\n\t newCoordinateY = coords.y + window.scrollY - this.wrappersOffset.top - defaultOffset - toolbar.offsetHeight;\n\t\n\t toolbar.style.transform = 'translate3D(' + Math.floor(newCoordinateX) + 'px, ' + Math.floor(newCoordinateY) + 'px, 0)';\n\t\n\t /** Close everything */\n\t editor.toolbar.inline.closeButtons();\n\t editor.toolbar.inline.closeAction();\n\t };\n\t\n\t /**\r\n\t * @private\r\n\t *\r\n\t * Tool Clicked\r\n\t */\n\t\n\t inline.toolClicked = function (event, type) {\n\t\n\t /**\r\n\t * For simple tools we use default browser function\r\n\t * For more complicated tools, we should write our own behavior\r\n\t */\n\t switch (type) {\n\t case 'createLink':\n\t editor.toolbar.inline.createLinkAction(event, type);break;\n\t default:\n\t editor.toolbar.inline.defaultToolAction(type);break;\n\t }\n\t\n\t /**\r\n\t * highlight buttons\r\n\t * after making some action\r\n\t */\n\t editor.nodes.inlineToolbar.buttons.childNodes.forEach(editor.toolbar.inline.hightlight);\n\t };\n\t\n\t /**\r\n\t * @private\r\n\t *\r\n\t * Saving wrappers offset in DOM\r\n\t */\n\t inline.getWrappersOffset = function () {\n\t\n\t var wrapper = editor.nodes.wrapper,\n\t offset = this.getOffset(wrapper);\n\t\n\t this.wrappersOffset = offset;\n\t return offset;\n\t };\n\t\n\t /**\r\n\t * @private\r\n\t *\r\n\t * Calculates offset of DOM element\r\n\t *\r\n\t * @param el\r\n\t * @returns {{top: number, left: number}}\r\n\t */\n\t inline.getOffset = function (el) {\n\t\n\t var _x = 0;\n\t var _y = 0;\n\t\n\t while (el && !isNaN(el.offsetLeft) && !isNaN(el.offsetTop)) {\n\t\n\t _x += el.offsetLeft + el.clientLeft;\n\t _y += el.offsetTop + el.clientTop;\n\t el = el.offsetParent;\n\t }\n\t return { top: _y, left: _x };\n\t };\n\t\n\t /**\r\n\t * @private\r\n\t *\r\n\t * Calculates position of selected text\r\n\t * @returns {{x: number, y: number}}\r\n\t */\n\t inline.getSelectionCoords = function () {\n\t\n\t var sel = document.selection,\n\t range;\n\t var x = 0,\n\t y = 0;\n\t\n\t if (sel) {\n\t\n\t if (sel.type != 'Control') {\n\t\n\t range = sel.createRange();\n\t range.collapse(true);\n\t x = range.boundingLeft;\n\t y = range.boundingTop;\n\t }\n\t } else if (window.getSelection) {\n\t\n\t sel = window.getSelection();\n\t\n\t if (sel.rangeCount) {\n\t\n\t range = sel.getRangeAt(0).cloneRange();\n\t if (range.getClientRects) {\n\t\n\t range.collapse(true);\n\t var rect = range.getClientRects()[0];\n\t\n\t if (!rect) {\n\t\n\t return;\n\t }\n\t\n\t x = rect.left;\n\t y = rect.top;\n\t }\n\t }\n\t }\n\t return { x: x, y: y };\n\t };\n\t\n\t /**\r\n\t * @private\r\n\t *\r\n\t * Returns selected text as String\r\n\t * @returns {string}\r\n\t */\n\t inline.getSelectionText = function () {\n\t\n\t var selectedText = '';\n\t\n\t // all modern browsers and IE9+\n\t if (window.getSelection) {\n\t\n\t selectedText = window.getSelection().toString();\n\t }\n\t\n\t return selectedText;\n\t };\n\t\n\t /** Opens buttons block */\n\t inline.showButtons = function () {\n\t\n\t var buttons = editor.nodes.inlineToolbar.buttons;\n\t\n\t buttons.classList.add('opened');\n\t\n\t editor.toolbar.inline.buttonsOpened = true;\n\t\n\t /** highlight buttons */\n\t editor.nodes.inlineToolbar.buttons.childNodes.forEach(editor.toolbar.inline.hightlight);\n\t };\n\t\n\t /** Makes buttons disappear */\n\t inline.closeButtons = function () {\n\t\n\t var buttons = editor.nodes.inlineToolbar.buttons;\n\t\n\t buttons.classList.remove('opened');\n\t\n\t editor.toolbar.inline.buttonsOpened = false;\n\t };\n\t\n\t /** Open buttons defined action if exist */\n\t inline.showActions = function () {\n\t\n\t var action = editor.nodes.inlineToolbar.actions;\n\t\n\t action.classList.add('opened');\n\t\n\t editor.toolbar.inline.actionsOpened = true;\n\t };\n\t\n\t /** Close actions block */\n\t inline.closeAction = function () {\n\t\n\t var action = editor.nodes.inlineToolbar.actions;\n\t\n\t action.innerHTML = '';\n\t action.classList.remove('opened');\n\t editor.toolbar.inline.actionsOpened = false;\n\t };\n\t\n\t /**\r\n\t * Callback for keydowns in inline toolbar \"Insert link...\" input\r\n\t */\n\t var inlineToolbarAnchorInputKeydown_ = function inlineToolbarAnchorInputKeydown_(event) {\n\t\n\t if (event.keyCode != editor.core.keys.ENTER) {\n\t\n\t return;\n\t }\n\t\n\t var editable = editor.content.currentNode,\n\t storedSelection = editor.toolbar.inline.storedSelection;\n\t\n\t editor.toolbar.inline.restoreSelection(editable, storedSelection);\n\t editor.toolbar.inline.setAnchor(this.value);\n\t\n\t /**\r\n\t * Preventing events that will be able to happen\r\n\t */\n\t event.preventDefault();\n\t event.stopImmediatePropagation();\n\t\n\t editor.toolbar.inline.clearRange();\n\t };\n\t\n\t /** Action for link creation or for setting anchor */\n\t inline.createLinkAction = function (event) {\n\t\n\t var isActive = this.isLinkActive();\n\t\n\t var editable = editor.content.currentNode,\n\t storedSelection = editor.toolbar.inline.saveSelection(editable);\n\t\n\t /** Save globally selection */\n\t editor.toolbar.inline.storedSelection = storedSelection;\n\t\n\t if (isActive) {\n\t\n\t /**\r\n\t * Changing stored selection. if we want to remove anchor from word\r\n\t * we should remove anchor from whole word, not only selected part.\r\n\t * The solution is than we get the length of current link\r\n\t * Change start position to - end of selection minus length of anchor\r\n\t */\n\t editor.toolbar.inline.restoreSelection(editable, storedSelection);\n\t\n\t editor.toolbar.inline.defaultToolAction('unlink');\n\t } else {\n\t\n\t /** Create input and close buttons */\n\t var action = editor.draw.inputForLink();\n\t\n\t editor.nodes.inlineToolbar.actions.appendChild(action);\n\t\n\t editor.toolbar.inline.closeButtons();\n\t editor.toolbar.inline.showActions();\n\t\n\t /**\r\n\t * focus to input\r\n\t * Solution: https://developer.mozilla.org/ru/docs/Web/API/HTMLElement/focus\r\n\t * Prevents event after showing input and when we need to focus an input which is in unexisted form\r\n\t */\n\t action.focus();\n\t event.preventDefault();\n\t\n\t /** Callback to link action */\n\t editor.listeners.add(action, 'keydown', inlineToolbarAnchorInputKeydown_, false);\n\t }\n\t };\n\t\n\t inline.isLinkActive = function () {\n\t\n\t var isActive = false;\n\t\n\t editor.nodes.inlineToolbar.buttons.childNodes.forEach(function (tool) {\n\t\n\t var dataType = tool.dataset.type;\n\t\n\t if (dataType == 'link' && tool.classList.contains('hightlighted')) {\n\t\n\t isActive = true;\n\t }\n\t });\n\t\n\t return isActive;\n\t };\n\t\n\t /** default action behavior of tool */\n\t inline.defaultToolAction = function (type) {\n\t\n\t document.execCommand(type, false, null);\n\t };\n\t\n\t /**\r\n\t * @private\r\n\t *\r\n\t * Sets URL\r\n\t *\r\n\t * @param {String} url - URL\r\n\t */\n\t inline.setAnchor = function (url) {\n\t\n\t document.execCommand('createLink', false, url);\n\t\n\t /** Close after URL inserting */\n\t editor.toolbar.inline.closeAction();\n\t };\n\t\n\t /**\r\n\t * @private\r\n\t *\r\n\t * Saves selection\r\n\t */\n\t inline.saveSelection = function (containerEl) {\n\t\n\t var range = window.getSelection().getRangeAt(0),\n\t preSelectionRange = range.cloneRange(),\n\t start;\n\t\n\t preSelectionRange.selectNodeContents(containerEl);\n\t preSelectionRange.setEnd(range.startContainer, range.startOffset);\n\t\n\t start = preSelectionRange.toString().length;\n\t\n\t return {\n\t start: start,\n\t end: start + range.toString().length\n\t };\n\t };\n\t\n\t /**\r\n\t * @private\r\n\t *\r\n\t * Sets to previous selection (Range)\r\n\t *\r\n\t * @param {Element} containerEl - editable element where we restore range\r\n\t * @param {Object} savedSel - range basic information to restore\r\n\t */\n\t inline.restoreSelection = function (containerEl, savedSel) {\n\t\n\t var range = document.createRange(),\n\t charIndex = 0;\n\t\n\t range.setStart(containerEl, 0);\n\t range.collapse(true);\n\t\n\t var nodeStack = [containerEl],\n\t node,\n\t foundStart = false,\n\t stop = false,\n\t nextCharIndex;\n\t\n\t while (!stop && (node = nodeStack.pop())) {\n\t\n\t if (node.nodeType == 3) {\n\t\n\t nextCharIndex = charIndex + node.length;\n\t\n\t if (!foundStart && savedSel.start >= charIndex && savedSel.start <= nextCharIndex) {\n\t\n\t range.setStart(node, savedSel.start - charIndex);\n\t foundStart = true;\n\t }\n\t if (foundStart && savedSel.end >= charIndex && savedSel.end <= nextCharIndex) {\n\t\n\t range.setEnd(node, savedSel.end - charIndex);\n\t stop = true;\n\t }\n\t charIndex = nextCharIndex;\n\t } else {\n\t\n\t var i = node.childNodes.length;\n\t\n\t while (i--) {\n\t\n\t nodeStack.push(node.childNodes[i]);\n\t }\n\t }\n\t }\n\t\n\t var sel = window.getSelection();\n\t\n\t sel.removeAllRanges();\n\t sel.addRange(range);\n\t };\n\t\n\t /**\r\n\t * @private\r\n\t *\r\n\t * Removes all ranges from window selection\r\n\t */\n\t inline.clearRange = function () {\n\t\n\t var selection = window.getSelection();\n\t\n\t selection.removeAllRanges();\n\t };\n\t\n\t /**\r\n\t * @private\r\n\t *\r\n\t * sets or removes hightlight\r\n\t */\n\t inline.hightlight = function (tool) {\n\t\n\t var dataType = tool.dataset.type;\n\t\n\t if (document.queryCommandState(dataType)) {\n\t\n\t editor.toolbar.inline.setButtonHighlighted(tool);\n\t } else {\n\t\n\t editor.toolbar.inline.removeButtonsHighLight(tool);\n\t }\n\t\n\t /**\r\n\t *\r\n\t * hightlight for anchors\r\n\t */\n\t var selection = window.getSelection(),\n\t tag = selection.anchorNode.parentNode;\n\t\n\t if (tag.tagName == 'A' && dataType == 'link') {\n\t\n\t editor.toolbar.inline.setButtonHighlighted(tool);\n\t }\n\t };\n\t\n\t /**\r\n\t * @private\r\n\t *\r\n\t * Mark button if text is already executed\r\n\t */\n\t inline.setButtonHighlighted = function (button) {\n\t\n\t button.classList.add('hightlighted');\n\t\n\t /** At link tool we also change icon */\n\t if (button.dataset.type == 'link') {\n\t\n\t var icon = button.childNodes[0];\n\t\n\t icon.classList.remove('ce-icon-link');\n\t icon.classList.add('ce-icon-unlink');\n\t }\n\t };\n\t\n\t /**\r\n\t * @private\r\n\t *\r\n\t * Removes hightlight\r\n\t */\n\t inline.removeButtonsHighLight = function (button) {\n\t\n\t button.classList.remove('hightlighted');\n\t\n\t /** At link tool we also change icon */\n\t if (button.dataset.type == 'link') {\n\t\n\t var icon = button.childNodes[0];\n\t\n\t icon.classList.remove('ce-icon-unlink');\n\t icon.classList.add('ce-icon-link');\n\t }\n\t };\n\t\n\t return inline;\n\t}({});\n\n/***/ },\n/* 11 */\n/***/ function(module, exports) {\n\n\t'use strict';\n\t\n\t/**\r\n\t * Codex Editor toolbox\r\n\t *\r\n\t * All tools be able to appended here\r\n\t *\r\n\t * @author Codex Team\r\n\t * @version 1.0\r\n\t */\n\t\n\tmodule.exports = function (toolbox) {\n\t\n\t var editor = codex.editor;\n\t\n\t toolbox.opened = false;\n\t\n\t /** Shows toolbox */\n\t toolbox.open = function () {\n\t\n\t /** Close setting if toolbox is opened */\n\t if (editor.toolbar.settings.opened) {\n\t\n\t editor.toolbar.settings.close();\n\t }\n\t\n\t /** display toolbox */\n\t editor.nodes.toolbox.classList.add('opened');\n\t\n\t /** Animate plus button */\n\t editor.nodes.plusButton.classList.add('clicked');\n\t\n\t /** toolbox state */\n\t editor.toolbar.toolbox.opened = true;\n\t };\n\t\n\t /** Closes toolbox */\n\t toolbox.close = function () {\n\t\n\t /** Makes toolbox disapear */\n\t editor.nodes.toolbox.classList.remove('opened');\n\t\n\t /** Rotate plus button */\n\t editor.nodes.plusButton.classList.remove('clicked');\n\t\n\t /** toolbox state */\n\t editor.toolbar.toolbox.opened = false;\n\t\n\t editor.toolbar.current = null;\n\t };\n\t\n\t toolbox.leaf = function () {\n\t\n\t var currentTool = editor.toolbar.current,\n\t tools = Object.keys(editor.tools),\n\t barButtons = editor.nodes.toolbarButtons,\n\t nextToolIndex = 0,\n\t toolToSelect = void 0,\n\t visibleTool = void 0,\n\t tool = void 0;\n\t\n\t if (!currentTool) {\n\t\n\t /** Get first tool from object*/\n\t for (tool in editor.tools) {\n\t\n\t if (editor.tools[tool].displayInToolbox) {\n\t\n\t break;\n\t }\n\t\n\t nextToolIndex++;\n\t }\n\t } else {\n\t\n\t nextToolIndex = (tools.indexOf(currentTool) + 1) % tools.length;\n\t visibleTool = tools[nextToolIndex];\n\t\n\t while (!editor.tools[visibleTool].displayInToolbox) {\n\t\n\t nextToolIndex = (nextToolIndex + 1) % tools.length;\n\t visibleTool = tools[nextToolIndex];\n\t }\n\t }\n\t\n\t toolToSelect = tools[nextToolIndex];\n\t\n\t for (var button in barButtons) {\n\t\n\t barButtons[button].classList.remove('selected');\n\t }\n\t\n\t barButtons[toolToSelect].classList.add('selected');\n\t editor.toolbar.current = toolToSelect;\n\t };\n\t\n\t /**\r\n\t * Transforming selected node type into selected toolbar element type\r\n\t * @param {event} event\r\n\t */\n\t toolbox.toolClicked = function (event) {\n\t\n\t /**\r\n\t * UNREPLACEBLE_TOOLS this types of tools are forbidden to replace even they are empty\r\n\t */\n\t var UNREPLACEBLE_TOOLS = ['image', 'link', 'list', 'instagram', 'twitter', 'embed'],\n\t tool = editor.tools[editor.toolbar.current],\n\t workingNode = editor.content.currentNode,\n\t currentInputIndex = editor.caret.inputIndex,\n\t newBlockContent,\n\t appendCallback,\n\t blockData;\n\t\n\t /** Make block from plugin */\n\t newBlockContent = tool.render();\n\t\n\t /** information about block */\n\t blockData = {\n\t block: newBlockContent,\n\t type: tool.type,\n\t stretched: false\n\t };\n\t\n\t if (workingNode && UNREPLACEBLE_TOOLS.indexOf(workingNode.dataset.tool) === -1 && workingNode.textContent.trim() === '') {\n\t\n\t /** Replace current block */\n\t editor.content.switchBlock(workingNode, newBlockContent, tool.type);\n\t } else {\n\t\n\t /** Insert new Block from plugin */\n\t editor.content.insertBlock(blockData);\n\t\n\t /** increase input index */\n\t currentInputIndex++;\n\t }\n\t\n\t /** Fire tool append callback */\n\t appendCallback = tool.appendCallback;\n\t\n\t if (appendCallback && typeof appendCallback == 'function') {\n\t\n\t appendCallback.call(event);\n\t }\n\t\n\t window.setTimeout(function () {\n\t\n\t /** Set caret to current block */\n\t editor.caret.setToBlock(currentInputIndex);\n\t }, 10);\n\t\n\t /**\r\n\t * Changing current Node\r\n\t */\n\t editor.content.workingNodeChanged();\n\t\n\t /**\r\n\t * Move toolbar when node is changed\r\n\t */\n\t editor.toolbar.move();\n\t };\n\t\n\t return toolbox;\n\t}({});\n\n/***/ },\n/* 12 */\n/***/ function(module, exports) {\n\n\t'use strict';\n\t\n\t/**\r\n\t * @module Codex Editor Callbacks module\r\n\t * @description Module works with editor added Elements\r\n\t *\r\n\t * @author Codex Team\r\n\t * @version 1.4.0\r\n\t */\n\t\n\tmodule.exports = function (callbacks) {\n\t\n\t var editor = codex.editor;\n\t\n\t /**\r\n\t * used by UI module\r\n\t * @description Routes all keydowns on document\r\n\t * @param {Object} event\r\n\t */\n\t callbacks.globalKeydown = function (event) {\n\t\n\t switch (event.keyCode) {\n\t case editor.core.keys.ENTER:\n\t enterKeyPressed_(event);break;\n\t }\n\t };\n\t\n\t /**\r\n\t * used by UI module\r\n\t * @description Routes all keydowns on redactors area\r\n\t * @param {Object} event\r\n\t */\n\t callbacks.redactorKeyDown = function (event) {\n\t\n\t switch (event.keyCode) {\n\t case editor.core.keys.TAB:\n\t tabKeyPressedOnRedactorsZone_(event);break;\n\t case editor.core.keys.ENTER:\n\t enterKeyPressedOnRedactorsZone_(event);break;\n\t case editor.core.keys.ESC:\n\t escapeKeyPressedOnRedactorsZone_(event);break;\n\t default:\n\t defaultKeyPressedOnRedactorsZone_(event);break;\n\t }\n\t };\n\t\n\t /**\r\n\t * used by UI module\r\n\t * @description Routes all keyup events\r\n\t * @param {Object} event\r\n\t */\n\t callbacks.globalKeyup = function (event) {\n\t\n\t switch (event.keyCode) {\n\t case editor.core.keys.UP:\n\t case editor.core.keys.LEFT:\n\t case editor.core.keys.RIGHT:\n\t case editor.core.keys.DOWN:\n\t arrowKeyPressed_(event);break;\n\t }\n\t };\n\t\n\t /**\r\n\t * @param {Object} event\r\n\t * @private\r\n\t *\r\n\t * Handles behaviour when tab pressed\r\n\t * @description if Content is empty show toolbox (if it is closed) or leaf tools\r\n\t * uses Toolbars toolbox module to handle the situation\r\n\t */\n\t var tabKeyPressedOnRedactorsZone_ = function tabKeyPressedOnRedactorsZone_(event) {\n\t\n\t /**\r\n\t * Wait for solution. Would like to know the behaviour\r\n\t * @todo Add spaces\r\n\t */\n\t event.preventDefault();\n\t\n\t var nativeInputs = editor.content.currentNode.querySelectorAll('textarea, input'),\n\t nativeInputsAreEmpty = true,\n\t textContentIsEmpty = !editor.content.currentNode.textContent.trim();\n\t\n\t Array.prototype.map.call(nativeInputs, function (input) {\n\t\n\t if (input.type == 'textarea' || input.type == 'text') {\n\t\n\t nativeInputsAreEmpty = nativeInputsAreEmpty && !input.value.trim();\n\t }\n\t });\n\t\n\t var blockIsEmpty = textContentIsEmpty && nativeInputsAreEmpty;\n\t\n\t if (!blockIsEmpty) {\n\t\n\t return;\n\t }\n\t\n\t if (!editor.toolbar.opened) {\n\t\n\t editor.toolbar.open();\n\t }\n\t\n\t if (editor.toolbar.opened && !editor.toolbar.toolbox.opened) {\n\t\n\t editor.toolbar.toolbox.open();\n\t } else {\n\t\n\t editor.toolbar.toolbox.leaf();\n\t }\n\t };\n\t\n\t /**\r\n\t * Handles global EnterKey Press\r\n\t * @see enterPressedOnBlock_\r\n\t * @param {Object} event\r\n\t */\n\t var enterKeyPressed_ = function enterKeyPressed_() {\n\t\n\t if (editor.content.editorAreaHightlighted) {\n\t\n\t /**\r\n\t * it means that we lose input index, saved index before is not correct\r\n\t * therefore we need to set caret when we insert new block\r\n\t */\n\t editor.caret.inputIndex = -1;\n\t\n\t enterPressedOnBlock_();\n\t }\n\t };\n\t\n\t /**\r\n\t * Callback for enter key pressing in first-level block area\r\n\t *\r\n\t * @param {Event} event\r\n\t * @private\r\n\t *\r\n\t * @description Inserts new block with initial type from settings\r\n\t */\n\t var enterPressedOnBlock_ = function enterPressedOnBlock_() {\n\t\n\t var NEW_BLOCK_TYPE = editor.settings.initialBlockPlugin;\n\t\n\t editor.content.insertBlock({\n\t type: NEW_BLOCK_TYPE,\n\t block: editor.tools[NEW_BLOCK_TYPE].render()\n\t }, true);\n\t\n\t editor.toolbar.move();\n\t editor.toolbar.open();\n\t };\n\t\n\t /**\r\n\t * ENTER key handler\r\n\t *\r\n\t * @param {Object} event\r\n\t * @private\r\n\t *\r\n\t * @description Makes new block with initial type from settings\r\n\t */\n\t var enterKeyPressedOnRedactorsZone_ = function enterKeyPressedOnRedactorsZone_(event) {\n\t\n\t if (event.target.contentEditable == 'true') {\n\t\n\t /** Update input index */\n\t editor.caret.saveCurrentInputIndex();\n\t }\n\t\n\t var currentInputIndex = editor.caret.getCurrentInputIndex() || 0,\n\t workingNode = editor.content.currentNode,\n\t tool = workingNode.dataset.tool,\n\t isEnterPressedOnToolbar = editor.toolbar.opened && editor.toolbar.current && event.target == editor.state.inputs[currentInputIndex];\n\t\n\t /** The list of tools which needs the default browser behaviour */\n\t var enableLineBreaks = editor.tools[tool].enableLineBreaks;\n\t\n\t /** This type of block creates when enter is pressed */\n\t var NEW_BLOCK_TYPE = editor.settings.initialBlockPlugin;\n\t\n\t /**\r\n\t * When toolbar is opened, select tool instead of making new paragraph\r\n\t */\n\t if (isEnterPressedOnToolbar) {\n\t\n\t event.preventDefault();\n\t\n\t editor.toolbar.toolbox.toolClicked(event);\n\t\n\t editor.toolbar.close();\n\t\n\t /**\r\n\t * Stop other listeners callback executions\r\n\t */\n\t event.stopPropagation();\n\t event.stopImmediatePropagation();\n\t\n\t return;\n\t }\n\t\n\t /**\r\n\t * Allow paragraph lineBreaks with shift enter\r\n\t * Or if shiftkey pressed and enter and enabledLineBreaks, the let new block creation\r\n\t */\n\t if (event.shiftKey || enableLineBreaks) {\n\t\n\t event.stopPropagation();\n\t event.stopImmediatePropagation();\n\t return;\n\t }\n\t\n\t var currentSelection = window.getSelection(),\n\t currentSelectedNode = currentSelection.anchorNode,\n\t caretAtTheEndOfText = editor.caret.position.atTheEnd(),\n\t isTextNodeHasParentBetweenContenteditable = false;\n\t\n\t /**\r\n\t * Allow making new

    in same block by SHIFT+ENTER and forbids to prevent default browser behaviour\r\n\t */\n\t if (event.shiftKey && !enableLineBreaks) {\n\t\n\t editor.callback.enterPressedOnBlock(editor.content.currentBlock, event);\n\t event.preventDefault();\n\t return;\n\t }\n\t\n\t /**\r\n\t * Workaround situation when caret at the Text node that has some wrapper Elements\r\n\t * Split block cant handle this.\r\n\t * We need to save default behavior\r\n\t */\n\t isTextNodeHasParentBetweenContenteditable = currentSelectedNode && currentSelectedNode.parentNode.contentEditable != 'true';\n\t\n\t /**\r\n\t * Split blocks when input has several nodes and caret placed in textNode\r\n\t */\n\t if (currentSelectedNode.nodeType == editor.core.nodeTypes.TEXT && !isTextNodeHasParentBetweenContenteditable && !caretAtTheEndOfText) {\n\t\n\t event.preventDefault();\n\t\n\t editor.core.log('Splitting Text node...');\n\t\n\t editor.content.splitBlock(currentInputIndex);\n\t\n\t /** Show plus button when next input after split is empty*/\n\t if (!editor.state.inputs[currentInputIndex + 1].textContent.trim()) {\n\t\n\t editor.toolbar.showPlusButton();\n\t }\n\t } else {\n\t\n\t var islastNode = editor.content.isLastNode(currentSelectedNode);\n\t\n\t if (islastNode && caretAtTheEndOfText) {\n\t\n\t event.preventDefault();\n\t event.stopPropagation();\n\t event.stopImmediatePropagation();\n\t\n\t editor.core.log('ENTER clicked in last textNode. Create new BLOCK');\n\t\n\t editor.content.insertBlock({\n\t type: NEW_BLOCK_TYPE,\n\t block: editor.tools[NEW_BLOCK_TYPE].render()\n\t }, true);\n\t\n\t editor.toolbar.move();\n\t editor.toolbar.open();\n\t\n\t /** Show plus button with empty block */\n\t editor.toolbar.showPlusButton();\n\t }\n\t }\n\t\n\t /** get all inputs after new appending block */\n\t editor.ui.saveInputs();\n\t };\n\t\n\t /**\r\n\t * Escape behaviour\r\n\t * @param event\r\n\t * @private\r\n\t *\r\n\t * @description Closes toolbox and toolbar. Prevents default behaviour\r\n\t */\n\t var escapeKeyPressedOnRedactorsZone_ = function escapeKeyPressedOnRedactorsZone_(event) {\n\t\n\t /** Close all toolbar */\n\t editor.toolbar.close();\n\t\n\t /** Close toolbox */\n\t editor.toolbar.toolbox.close();\n\t\n\t event.preventDefault();\n\t };\n\t\n\t /**\r\n\t * @param {Event} event\r\n\t * @private\r\n\t *\r\n\t * closes and moves toolbar\r\n\t */\n\t var arrowKeyPressed_ = function arrowKeyPressed_(event) {\n\t\n\t editor.content.workingNodeChanged();\n\t\n\t /* Closing toolbar */\n\t editor.toolbar.close();\n\t editor.toolbar.move();\n\t };\n\t\n\t /**\r\n\t * @private\r\n\t * @param {Event} event\r\n\t *\r\n\t * @description Closes all opened bars from toolbar.\r\n\t * If block is mark, clears highlightning\r\n\t */\n\t var defaultKeyPressedOnRedactorsZone_ = function defaultKeyPressedOnRedactorsZone_() {\n\t\n\t editor.toolbar.close();\n\t\n\t if (!editor.toolbar.inline.actionsOpened) {\n\t\n\t editor.toolbar.inline.close();\n\t editor.content.clearMark();\n\t }\n\t };\n\t\n\t /**\r\n\t * Handler when clicked on redactors area\r\n\t *\r\n\t * @protected\r\n\t * @param event\r\n\t *\r\n\t * @description Detects clicked area. If it is first-level block area, marks as detected and\r\n\t * on next enter press will be inserted new block\r\n\t * Otherwise, save carets position (input index) and put caret to the editable zone.\r\n\t *\r\n\t * @see detectWhenClickedOnFirstLevelBlockArea_\r\n\t *\r\n\t */\n\t callbacks.redactorClicked = function (event) {\n\t\n\t detectWhenClickedOnFirstLevelBlockArea_();\n\t\n\t editor.content.workingNodeChanged(event.target);\n\t editor.ui.saveInputs();\n\t\n\t var selectedText = editor.toolbar.inline.getSelectionText(),\n\t firstLevelBlock;\n\t\n\t /** If selection range took off, then we hide inline toolbar */\n\t if (selectedText.length === 0) {\n\t\n\t editor.toolbar.inline.close();\n\t }\n\t\n\t /** Update current input index in memory when caret focused into existed input */\n\t if (event.target.contentEditable == 'true') {\n\t\n\t editor.caret.saveCurrentInputIndex();\n\t }\n\t\n\t if (editor.content.currentNode === null) {\n\t\n\t /**\r\n\t * If inputs in redactor does not exits, then we put input index 0 not -1\r\n\t */\n\t var indexOfLastInput = editor.state.inputs.length > 0 ? editor.state.inputs.length - 1 : 0;\n\t\n\t /** If we have any inputs */\n\t if (editor.state.inputs.length) {\n\t\n\t /** getting firstlevel parent of input */\n\t firstLevelBlock = editor.content.getFirstLevelBlock(editor.state.inputs[indexOfLastInput]);\n\t }\n\t\n\t /** If input is empty, then we set caret to the last input */\n\t if (editor.state.inputs.length && editor.state.inputs[indexOfLastInput].textContent === '' && firstLevelBlock.dataset.tool == editor.settings.initialBlockPlugin) {\n\t\n\t editor.caret.setToBlock(indexOfLastInput);\n\t } else {\n\t\n\t /** Create new input when caret clicked in redactors area */\n\t var NEW_BLOCK_TYPE = editor.settings.initialBlockPlugin;\n\t\n\t editor.content.insertBlock({\n\t type: NEW_BLOCK_TYPE,\n\t block: editor.tools[NEW_BLOCK_TYPE].render()\n\t });\n\t\n\t /** If there is no inputs except inserted */\n\t if (editor.state.inputs.length === 1) {\n\t\n\t editor.caret.setToBlock(indexOfLastInput);\n\t } else {\n\t\n\t /** Set caret to this appended input */\n\t editor.caret.setToNextBlock(indexOfLastInput);\n\t }\n\t }\n\t } else {\n\t\n\t /** Close all panels */\n\t editor.toolbar.settings.close();\n\t editor.toolbar.toolbox.close();\n\t }\n\t\n\t /**\r\n\t * Move toolbar and open\r\n\t */\n\t editor.toolbar.move();\n\t editor.toolbar.open();\n\t\n\t var inputIsEmpty = !editor.content.currentNode.textContent.trim(),\n\t currentNodeType = editor.content.currentNode.dataset.tool,\n\t isInitialType = currentNodeType == editor.settings.initialBlockPlugin;\n\t\n\t /** Hide plus buttons */\n\t editor.toolbar.hidePlusButton();\n\t\n\t if (!inputIsEmpty) {\n\t\n\t /** Mark current block */\n\t editor.content.markBlock();\n\t }\n\t\n\t if (isInitialType && inputIsEmpty) {\n\t\n\t /** Show plus button */\n\t editor.toolbar.showPlusButton();\n\t }\n\t };\n\t\n\t /**\r\n\t * This method allows to define, is caret in contenteditable element or not.\r\n\t *\r\n\t * @private\r\n\t *\r\n\t * @description Otherwise, if we get TEXT node from range container, that will means we have input index.\r\n\t * In this case we use default browsers behaviour (if plugin allows that) or overwritten action.\r\n\t * Therefore, to be sure that we've clicked first-level block area, we should have currentNode, which always\r\n\t * specifies to the first-level block. Other cases we just ignore.\r\n\t */\n\t var detectWhenClickedOnFirstLevelBlockArea_ = function detectWhenClickedOnFirstLevelBlockArea_() {\n\t\n\t var selection = window.getSelection(),\n\t anchorNode = selection.anchorNode,\n\t flag = false;\n\t\n\t if (selection.rangeCount === 0) {\n\t\n\t editor.content.editorAreaHightlighted = true;\n\t } else {\n\t\n\t if (!editor.core.isDomNode(anchorNode)) {\n\t\n\t anchorNode = anchorNode.parentNode;\n\t }\n\t\n\t /** Already founded, without loop */\n\t if (anchorNode.contentEditable == 'true') {\n\t\n\t flag = true;\n\t }\n\t\n\t while (anchorNode.contentEditable != 'true') {\n\t\n\t anchorNode = anchorNode.parentNode;\n\t\n\t if (anchorNode.contentEditable == 'true') {\n\t\n\t flag = true;\n\t }\n\t\n\t if (anchorNode == document.body) {\n\t\n\t break;\n\t }\n\t }\n\t\n\t /** If editable element founded, flag is \"TRUE\", Therefore we return \"FALSE\" */\n\t editor.content.editorAreaHightlighted = !flag;\n\t }\n\t };\n\t\n\t /**\r\n\t * Toolbar button click handler\r\n\t *\r\n\t * @param {Object} event - cursor to the button\r\n\t * @protected\r\n\t *\r\n\t * @description gets current tool and calls render method\r\n\t */\n\t callbacks.toolbarButtonClicked = function (event) {\n\t\n\t var button = this;\n\t\n\t editor.toolbar.current = button.dataset.type;\n\t\n\t editor.toolbar.toolbox.toolClicked(event);\n\t editor.toolbar.close();\n\t };\n\t\n\t /**\r\n\t * Show or Hide toolbox when plus button is clicked\r\n\t */\n\t callbacks.plusButtonClicked = function () {\n\t\n\t if (!editor.nodes.toolbox.classList.contains('opened')) {\n\t\n\t editor.toolbar.toolbox.open();\n\t } else {\n\t\n\t editor.toolbar.toolbox.close();\n\t }\n\t };\n\t\n\t /**\r\n\t * Block handlers for KeyDown events\r\n\t *\r\n\t * @protected\r\n\t * @param {Object} event\r\n\t *\r\n\t * Handles keydowns on block\r\n\t * @see blockRightOrDownArrowPressed_\r\n\t * @see backspacePressed_\r\n\t * @see blockLeftOrUpArrowPressed_\r\n\t */\n\t callbacks.blockKeydown = function (event) {\n\t\n\t var block = event.target; // event.target is input\n\t\n\t switch (event.keyCode) {\n\t\n\t case editor.core.keys.DOWN:\n\t case editor.core.keys.RIGHT:\n\t blockRightOrDownArrowPressed_(event);\n\t break;\n\t\n\t case editor.core.keys.BACKSPACE:\n\t backspacePressed_(block, event);\n\t break;\n\t\n\t case editor.core.keys.UP:\n\t case editor.core.keys.LEFT:\n\t blockLeftOrUpArrowPressed_(event);\n\t break;\n\t\n\t }\n\t };\n\t\n\t /**\r\n\t * RIGHT or DOWN keydowns on block\r\n\t *\r\n\t * @param {Object} event\r\n\t * @private\r\n\t *\r\n\t * @description watches the selection and gets closest editable element.\r\n\t * Uses method getDeepestTextNodeFromPosition to get the last node of next block\r\n\t * Sets caret if it is contenteditable\r\n\t */\n\t var blockRightOrDownArrowPressed_ = function blockRightOrDownArrowPressed_(event) {\n\t\n\t var selection = window.getSelection(),\n\t inputs = editor.state.inputs,\n\t focusedNode = selection.anchorNode,\n\t focusedNodeHolder;\n\t\n\t /** Check for caret existance */\n\t if (!focusedNode) {\n\t\n\t return false;\n\t }\n\t\n\t /** Looking for closest (parent) contentEditable element of focused node */\n\t while (focusedNode.contentEditable != 'true') {\n\t\n\t focusedNodeHolder = focusedNode.parentNode;\n\t focusedNode = focusedNodeHolder;\n\t }\n\t\n\t /** Input index in DOM level */\n\t var editableElementIndex = 0;\n\t\n\t while (focusedNode != inputs[editableElementIndex]) {\n\t\n\t editableElementIndex++;\n\t }\n\t\n\t /**\r\n\t * Founded contentEditable element doesn't have childs\r\n\t * Or maybe New created block\r\n\t */\n\t if (!focusedNode.textContent) {\n\t\n\t editor.caret.setToNextBlock(editableElementIndex);\n\t return;\n\t }\n\t\n\t /**\r\n\t * Do nothing when caret doesn not reaches the end of last child\r\n\t */\n\t var caretInLastChild = false,\n\t caretAtTheEndOfText = false;\n\t\n\t var lastChild, deepestTextnode;\n\t\n\t lastChild = focusedNode.childNodes[focusedNode.childNodes.length - 1];\n\t\n\t if (editor.core.isDomNode(lastChild)) {\n\t\n\t deepestTextnode = editor.content.getDeepestTextNodeFromPosition(lastChild, lastChild.childNodes.length);\n\t } else {\n\t\n\t deepestTextnode = lastChild;\n\t }\n\t\n\t caretInLastChild = selection.anchorNode == deepestTextnode;\n\t caretAtTheEndOfText = deepestTextnode.length == selection.anchorOffset;\n\t\n\t if (!caretInLastChild || !caretAtTheEndOfText) {\n\t\n\t editor.core.log('arrow [down|right] : caret does not reached the end');\n\t return false;\n\t }\n\t\n\t editor.caret.setToNextBlock(editableElementIndex);\n\t };\n\t\n\t /**\r\n\t * LEFT or UP keydowns on block\r\n\t *\r\n\t * @param {Object} event\r\n\t * @private\r\n\t *\r\n\t * watches the selection and gets closest editable element.\r\n\t * Uses method getDeepestTextNodeFromPosition to get the last node of previous block\r\n\t * Sets caret if it is contenteditable\r\n\t *\r\n\t */\n\t var blockLeftOrUpArrowPressed_ = function blockLeftOrUpArrowPressed_(event) {\n\t\n\t var selection = window.getSelection(),\n\t inputs = editor.state.inputs,\n\t focusedNode = selection.anchorNode,\n\t focusedNodeHolder;\n\t\n\t /** Check for caret existance */\n\t if (!focusedNode) {\n\t\n\t return false;\n\t }\n\t\n\t /**\r\n\t * LEFT or UP not at the beginning\r\n\t */\n\t if (selection.anchorOffset !== 0) {\n\t\n\t return false;\n\t }\n\t\n\t /** Looking for parent contentEditable block */\n\t while (focusedNode.contentEditable != 'true') {\n\t\n\t focusedNodeHolder = focusedNode.parentNode;\n\t focusedNode = focusedNodeHolder;\n\t }\n\t\n\t /** Input index in DOM level */\n\t var editableElementIndex = 0;\n\t\n\t while (focusedNode != inputs[editableElementIndex]) {\n\t\n\t editableElementIndex++;\n\t }\n\t\n\t /**\r\n\t * Do nothing if caret is not at the beginning of first child\r\n\t */\n\t var caretInFirstChild = false,\n\t caretAtTheBeginning = false;\n\t\n\t var firstChild, deepestTextnode;\n\t\n\t /**\r\n\t * Founded contentEditable element doesn't have childs\r\n\t * Or maybe New created block\r\n\t */\n\t if (!focusedNode.textContent) {\n\t\n\t editor.caret.setToPreviousBlock(editableElementIndex);\n\t return;\n\t }\n\t\n\t firstChild = focusedNode.childNodes[0];\n\t\n\t if (editor.core.isDomNode(firstChild)) {\n\t\n\t deepestTextnode = editor.content.getDeepestTextNodeFromPosition(firstChild, 0);\n\t } else {\n\t\n\t deepestTextnode = firstChild;\n\t }\n\t\n\t caretInFirstChild = selection.anchorNode == deepestTextnode;\n\t caretAtTheBeginning = selection.anchorOffset === 0;\n\t\n\t if (caretInFirstChild && caretAtTheBeginning) {\n\t\n\t editor.caret.setToPreviousBlock(editableElementIndex);\n\t }\n\t };\n\t\n\t /**\r\n\t * Handles backspace keydown\r\n\t *\r\n\t * @param {Element} block\r\n\t * @param {Object} event\r\n\t * @private\r\n\t *\r\n\t * @description if block is empty, delete the block and set caret to the previous block\r\n\t * If block is not empty, try to merge two blocks - current and previous\r\n\t * But it we try'n to remove first block, then we should set caret to the next block, not previous.\r\n\t * If we removed the last block, create new one\r\n\t */\n\t var backspacePressed_ = function backspacePressed_(block, event) {\n\t\n\t var currentInputIndex = editor.caret.getCurrentInputIndex(),\n\t range,\n\t selectionLength,\n\t firstLevelBlocksCount;\n\t\n\t if (editor.core.isNativeInput(event.target)) {\n\t\n\t /** If input value is empty - remove block */\n\t if (event.target.value.trim() == '') {\n\t\n\t block.remove();\n\t } else {\n\t\n\t return;\n\t }\n\t }\n\t\n\t if (block.textContent.trim()) {\n\t\n\t range = editor.content.getRange();\n\t selectionLength = range.endOffset - range.startOffset;\n\t\n\t if (editor.caret.position.atStart() && !selectionLength && editor.state.inputs[currentInputIndex - 1]) {\n\t\n\t editor.content.mergeBlocks(currentInputIndex);\n\t } else {\n\t\n\t return;\n\t }\n\t }\n\t\n\t if (!selectionLength) {\n\t\n\t block.remove();\n\t }\n\t\n\t firstLevelBlocksCount = editor.nodes.redactor.childNodes.length;\n\t\n\t /**\r\n\t * If all blocks are removed\r\n\t */\n\t if (firstLevelBlocksCount === 0) {\n\t\n\t /** update currentNode variable */\n\t editor.content.currentNode = null;\n\t\n\t /** Inserting new empty initial block */\n\t editor.ui.addInitialBlock();\n\t\n\t /** Updating inputs state after deleting last block */\n\t editor.ui.saveInputs();\n\t\n\t /** Set to current appended block */\n\t window.setTimeout(function () {\n\t\n\t editor.caret.setToPreviousBlock(1);\n\t }, 10);\n\t } else {\n\t\n\t if (editor.caret.inputIndex !== 0) {\n\t\n\t /** Target block is not first */\n\t editor.caret.setToPreviousBlock(editor.caret.inputIndex);\n\t } else {\n\t\n\t /** If we try to delete first block */\n\t editor.caret.setToNextBlock(editor.caret.inputIndex);\n\t }\n\t }\n\t\n\t editor.toolbar.move();\n\t\n\t if (!editor.toolbar.opened) {\n\t\n\t editor.toolbar.open();\n\t }\n\t\n\t /** Updating inputs state */\n\t editor.ui.saveInputs();\n\t\n\t /** Prevent default browser behaviour */\n\t event.preventDefault();\n\t };\n\t\n\t /**\r\n\t * used by UI module\r\n\t * Clicks on block settings button\r\n\t *\r\n\t * @param {Object} event\r\n\t * @protected\r\n\t * @description Opens toolbar settings\r\n\t */\n\t callbacks.showSettingsButtonClicked = function (event) {\n\t\n\t /**\r\n\t * Get type of current block\r\n\t * It uses to append settings from tool.settings property.\r\n\t * ...\r\n\t * Type is stored in data-type attribute on block\r\n\t */\n\t var currentToolType = editor.content.currentNode.dataset.tool;\n\t\n\t editor.toolbar.settings.toggle(currentToolType);\n\t\n\t /** Close toolbox when settings button is active */\n\t editor.toolbar.toolbox.close();\n\t editor.toolbar.settings.hideRemoveActions();\n\t };\n\t\n\t return callbacks;\n\t}({});\n\n/***/ },\n/* 13 */\n/***/ function(module, exports) {\n\n\t'use strict';\n\t\n\t/**\r\n\t * Codex Editor Draw module\r\n\t *\r\n\t * @author Codex Team\r\n\t * @version 1.0.\r\n\t */\n\t\n\tmodule.exports = function (draw) {\n\t\n\t /**\r\n\t * Base editor wrapper\r\n\t */\n\t draw.wrapper = function () {\n\t\n\t var wrapper = document.createElement('div');\n\t\n\t wrapper.className += 'codex-editor';\n\t\n\t return wrapper;\n\t };\n\t\n\t /**\r\n\t * Content-editable holder\r\n\t */\n\t draw.redactor = function () {\n\t\n\t var redactor = document.createElement('div');\n\t\n\t redactor.className += 'ce-redactor';\n\t\n\t return redactor;\n\t };\n\t\n\t draw.ceBlock = function () {\n\t\n\t var block = document.createElement('DIV');\n\t\n\t block.className += 'ce_block';\n\t\n\t return block;\n\t };\n\t\n\t /**\r\n\t * Empty toolbar with toggler\r\n\t */\n\t draw.toolbar = function () {\n\t\n\t var bar = document.createElement('div');\n\t\n\t bar.className += 'ce-toolbar';\n\t\n\t return bar;\n\t };\n\t\n\t draw.toolbarContent = function () {\n\t\n\t var wrapper = document.createElement('DIV');\n\t\n\t wrapper.classList.add('ce-toolbar__content');\n\t\n\t return wrapper;\n\t };\n\t\n\t /**\r\n\t * Inline toolbar\r\n\t */\n\t draw.inlineToolbar = function () {\n\t\n\t var bar = document.createElement('DIV');\n\t\n\t bar.className += 'ce-toolbar-inline';\n\t\n\t return bar;\n\t };\n\t\n\t /**\r\n\t * Wrapper for inline toobar buttons\r\n\t */\n\t draw.inlineToolbarButtons = function () {\n\t\n\t var wrapper = document.createElement('DIV');\n\t\n\t wrapper.className += 'ce-toolbar-inline__buttons';\n\t\n\t return wrapper;\n\t };\n\t\n\t /**\r\n\t * For some actions\r\n\t */\n\t draw.inlineToolbarActions = function () {\n\t\n\t var wrapper = document.createElement('DIV');\n\t\n\t wrapper.className += 'ce-toolbar-inline__actions';\n\t\n\t return wrapper;\n\t };\n\t\n\t draw.inputForLink = function () {\n\t\n\t var input = document.createElement('INPUT');\n\t\n\t input.type = 'input';\n\t input.className += 'inputForLink';\n\t input.placeholder = 'Вставьте ссылку ...';\n\t input.setAttribute('form', 'defaultForm');\n\t\n\t input.setAttribute('autofocus', 'autofocus');\n\t\n\t return input;\n\t };\n\t\n\t /**\r\n\t * @todo Desc\r\n\t */\n\t draw.blockButtons = function () {\n\t\n\t var block = document.createElement('div');\n\t\n\t block.className += 'ce-toolbar__actions';\n\t\n\t return block;\n\t };\n\t\n\t /**\r\n\t * Block settings panel\r\n\t */\n\t draw.blockSettings = function () {\n\t\n\t var settings = document.createElement('div');\n\t\n\t settings.className += 'ce-settings';\n\t\n\t return settings;\n\t };\n\t\n\t draw.defaultSettings = function () {\n\t\n\t var div = document.createElement('div');\n\t\n\t div.classList.add('ce-settings_default');\n\t\n\t return div;\n\t };\n\t\n\t draw.pluginsSettings = function () {\n\t\n\t var div = document.createElement('div');\n\t\n\t div.classList.add('ce-settings_plugin');\n\t\n\t return div;\n\t };\n\t\n\t draw.plusButton = function () {\n\t\n\t var button = document.createElement('span');\n\t\n\t button.className = 'ce-toolbar__plus';\n\t // button.innerHTML = '';\n\t\n\t return button;\n\t };\n\t\n\t /**\r\n\t * Settings button in toolbar\r\n\t */\n\t draw.settingsButton = function () {\n\t\n\t var toggler = document.createElement('span');\n\t\n\t toggler.className = 'ce-toolbar__settings-btn';\n\t\n\t /** Toggler button*/\n\t toggler.innerHTML = '';\n\t\n\t return toggler;\n\t };\n\t\n\t /**\r\n\t * Redactor tools wrapper\r\n\t */\n\t\n\t draw.toolbox = function () {\n\t\n\t var wrapper = document.createElement('div');\n\t\n\t wrapper.className = 'ce-toolbar__tools';\n\t\n\t return wrapper;\n\t };\n\t\n\t /**\r\n\t * @protected\r\n\t *\r\n\t * Draws tool buttons for toolbox\r\n\t *\r\n\t * @param {String} type\r\n\t * @param {String} classname\r\n\t * @returns {Element}\r\n\t */\n\t draw.toolbarButton = function (type, classname) {\n\t\n\t var button = document.createElement('li'),\n\t toolIcon = document.createElement('i'),\n\t toolTitle = document.createElement('span');\n\t\n\t button.dataset.type = type;\n\t button.setAttribute('title', type);\n\t\n\t toolIcon.classList.add(classname);\n\t toolTitle.classList.add('ce_toolbar_tools--title');\n\t\n\t button.appendChild(toolIcon);\n\t button.appendChild(toolTitle);\n\t\n\t return button;\n\t };\n\t\n\t /**\r\n\t * @protected\r\n\t *\r\n\t * Draws tools for inline toolbar\r\n\t *\r\n\t * @param {String} type\r\n\t * @param {String} classname\r\n\t */\n\t draw.toolbarButtonInline = function (type, classname) {\n\t\n\t var button = document.createElement('BUTTON'),\n\t toolIcon = document.createElement('I');\n\t\n\t button.type = 'button';\n\t button.dataset.type = type;\n\t toolIcon.classList.add(classname);\n\t\n\t button.appendChild(toolIcon);\n\t\n\t return button;\n\t };\n\t\n\t /**\r\n\t * Redactor block\r\n\t */\n\t draw.block = function (tagName, content) {\n\t\n\t var node = document.createElement(tagName);\n\t\n\t node.innerHTML = content || '';\n\t\n\t return node;\n\t };\n\t\n\t /**\r\n\t * Creates Node with passed tagName and className\r\n\t * @param {string} tagName\r\n\t * @param {string} className\r\n\t * @param {object} properties - allow to assign properties\r\n\t */\n\t draw.node = function (tagName, className, properties) {\n\t\n\t var el = document.createElement(tagName);\n\t\n\t if (className) el.className = className;\n\t\n\t if (properties) {\n\t\n\t for (var name in properties) {\n\t\n\t el[name] = properties[name];\n\t }\n\t }\n\t\n\t return el;\n\t };\n\t\n\t /**\r\n\t * Unavailable plugin block\r\n\t */\n\t draw.unavailableBlock = function () {\n\t\n\t var wrapper = document.createElement('DIV');\n\t\n\t wrapper.classList.add('cdx-unavailable-block');\n\t\n\t return wrapper;\n\t };\n\t\n\t return draw;\n\t}({});\n\n/***/ },\n/* 14 */\n/***/ function(module, exports) {\n\n\t'use strict';\n\t\n\t/**\r\n\t * Codex Editor Caret Module\r\n\t *\r\n\t * @author Codex Team\r\n\t * @version 1.0\r\n\t */\n\t\n\tmodule.exports = function (caret) {\n\t\n\t var editor = codex.editor;\n\t\n\t /**\r\n\t * @var {int} InputIndex - editable element in DOM\r\n\t */\n\t caret.inputIndex = null;\n\t\n\t /**\r\n\t * @var {int} offset - caret position in a text node.\r\n\t */\n\t caret.offset = null;\n\t\n\t /**\r\n\t * @var {int} focusedNodeIndex - we get index of child node from first-level block\r\n\t */\n\t caret.focusedNodeIndex = null;\n\t\n\t /**\r\n\t * Creates Document Range and sets caret to the element.\r\n\t * @protected\r\n\t * @uses caret.save — if you need to save caret position\r\n\t * @param {Element} el - Changed Node.\r\n\t */\n\t caret.set = function (el, index, offset) {\n\t\n\t offset = offset || caret.offset || 0;\n\t index = index || caret.focusedNodeIndex || 0;\n\t\n\t var childs = el.childNodes,\n\t nodeToSet;\n\t\n\t if (childs.length === 0) {\n\t\n\t nodeToSet = el;\n\t } else {\n\t\n\t nodeToSet = childs[index];\n\t }\n\t\n\t /** If Element is INPUT */\n\t if (el.contentEditable != 'true') {\n\t\n\t el.focus();\n\t return;\n\t }\n\t\n\t if (editor.core.isDomNode(nodeToSet)) {\n\t\n\t nodeToSet = editor.content.getDeepestTextNodeFromPosition(nodeToSet, nodeToSet.childNodes.length);\n\t }\n\t\n\t var range = document.createRange(),\n\t selection = window.getSelection();\n\t\n\t window.setTimeout(function () {\n\t\n\t range.setStart(nodeToSet, offset);\n\t range.setEnd(nodeToSet, offset);\n\t\n\t selection.removeAllRanges();\n\t selection.addRange(range);\n\t\n\t editor.caret.saveCurrentInputIndex();\n\t }, 20);\n\t };\n\t\n\t /**\r\n\t * @protected\r\n\t * Updates index of input and saves it in caret object\r\n\t */\n\t caret.saveCurrentInputIndex = function () {\n\t\n\t /** Index of Input that we paste sanitized content */\n\t var selection = window.getSelection(),\n\t inputs = editor.state.inputs,\n\t focusedNode = selection.anchorNode,\n\t focusedNodeHolder;\n\t\n\t if (!focusedNode) {\n\t\n\t return;\n\t }\n\t\n\t /** Looking for parent contentEditable block */\n\t while (focusedNode.contentEditable != 'true') {\n\t\n\t focusedNodeHolder = focusedNode.parentNode;\n\t focusedNode = focusedNodeHolder;\n\t }\n\t\n\t /** Input index in DOM level */\n\t var editableElementIndex = 0;\n\t\n\t while (focusedNode != inputs[editableElementIndex]) {\n\t\n\t editableElementIndex++;\n\t }\n\t\n\t caret.inputIndex = editableElementIndex;\n\t };\n\t\n\t /**\r\n\t * Returns current input index (caret object)\r\n\t */\n\t caret.getCurrentInputIndex = function () {\n\t\n\t return caret.inputIndex;\n\t };\n\t\n\t /**\r\n\t * @param {int} index - index of first-level block after that we set caret into next input\r\n\t */\n\t caret.setToNextBlock = function (index) {\n\t\n\t var inputs = editor.state.inputs,\n\t nextInput = inputs[index + 1];\n\t\n\t if (!nextInput) {\n\t\n\t editor.core.log('We are reached the end');\n\t return;\n\t }\n\t\n\t /**\r\n\t * When new Block created or deleted content of input\r\n\t * We should add some text node to set caret\r\n\t */\n\t if (!nextInput.childNodes.length) {\n\t\n\t var emptyTextElement = document.createTextNode('');\n\t\n\t nextInput.appendChild(emptyTextElement);\n\t }\n\t\n\t editor.caret.inputIndex = index + 1;\n\t editor.caret.set(nextInput, 0, 0);\n\t editor.content.workingNodeChanged(nextInput);\n\t };\n\t\n\t /**\r\n\t * @param {int} index - index of target input.\r\n\t * Sets caret to input with this index\r\n\t */\n\t caret.setToBlock = function (index) {\n\t\n\t var inputs = editor.state.inputs,\n\t targetInput = inputs[index];\n\t\n\t if (!targetInput) {\n\t\n\t return;\n\t }\n\t\n\t /**\r\n\t * When new Block created or deleted content of input\r\n\t * We should add some text node to set caret\r\n\t */\n\t if (!targetInput.childNodes.length) {\n\t\n\t var emptyTextElement = document.createTextNode('');\n\t\n\t targetInput.appendChild(emptyTextElement);\n\t }\n\t\n\t editor.caret.inputIndex = index;\n\t editor.caret.set(targetInput, 0, 0);\n\t editor.content.workingNodeChanged(targetInput);\n\t };\n\t\n\t /**\r\n\t * @param {int} index - index of input\r\n\t */\n\t caret.setToPreviousBlock = function (index) {\n\t\n\t index = index || 0;\n\t\n\t var inputs = editor.state.inputs,\n\t previousInput = inputs[index - 1],\n\t lastChildNode,\n\t lengthOfLastChildNode,\n\t emptyTextElement;\n\t\n\t if (!previousInput) {\n\t\n\t editor.core.log('We are reached first node');\n\t return;\n\t }\n\t\n\t lastChildNode = editor.content.getDeepestTextNodeFromPosition(previousInput, previousInput.childNodes.length);\n\t lengthOfLastChildNode = lastChildNode.length;\n\t\n\t /**\r\n\t * When new Block created or deleted content of input\r\n\t * We should add some text node to set caret\r\n\t */\n\t if (!previousInput.childNodes.length) {\n\t\n\t emptyTextElement = document.createTextNode('');\n\t previousInput.appendChild(emptyTextElement);\n\t }\n\t editor.caret.inputIndex = index - 1;\n\t editor.caret.set(previousInput, previousInput.childNodes.length - 1, lengthOfLastChildNode);\n\t editor.content.workingNodeChanged(inputs[index - 1]);\n\t };\n\t\n\t caret.position = {\n\t\n\t atStart: function atStart() {\n\t\n\t var selection = window.getSelection(),\n\t anchorOffset = selection.anchorOffset,\n\t anchorNode = selection.anchorNode,\n\t firstLevelBlock = editor.content.getFirstLevelBlock(anchorNode),\n\t pluginsRender = firstLevelBlock.childNodes[0];\n\t\n\t if (!editor.core.isDomNode(anchorNode)) {\n\t\n\t anchorNode = anchorNode.parentNode;\n\t }\n\t\n\t var isFirstNode = anchorNode === pluginsRender.childNodes[0],\n\t isOffsetZero = anchorOffset === 0;\n\t\n\t return isFirstNode && isOffsetZero;\n\t },\n\t\n\t atTheEnd: function atTheEnd() {\n\t\n\t var selection = window.getSelection(),\n\t anchorOffset = selection.anchorOffset,\n\t anchorNode = selection.anchorNode;\n\t\n\t /** Caret is at the end of input */\n\t return !anchorNode || !anchorNode.length || anchorOffset === anchorNode.length;\n\t }\n\t };\n\t\n\t /**\r\n\t * Inserts node at the caret location\r\n\t * @param node\r\n\t */\n\t caret.insertNode = function (node) {\n\t\n\t var selection, range;\n\t\n\t selection = window.getSelection();\n\t\n\t range = selection.getRangeAt(0);\n\t range.deleteContents();\n\t\n\t range.insertNode(node);\n\t\n\t range.setStartAfter(node);\n\t range.collapse(true);\n\t\n\t selection.removeAllRanges();\n\t selection.addRange(range);\n\t };\n\t\n\t return caret;\n\t}({});\n\n/***/ },\n/* 15 */\n/***/ function(module, exports) {\n\n\t'use strict';\n\t\n\t/**\r\n\t * Codex Editor Notification Module\r\n\t *\r\n\t * @author Codex Team\r\n\t * @version 1.0\r\n\t */\n\t\n\tmodule.exports = function (notifications) {\n\t\n\t var editor = codex.editor;\n\t\n\t var queue = [];\n\t\n\t var addToQueue = function addToQueue(settings) {\n\t\n\t queue.push(settings);\n\t\n\t var index = 0;\n\t\n\t while (index < queue.length && queue.length > 5) {\n\t\n\t if (queue[index].type == 'confirm' || queue[index].type == 'prompt') {\n\t\n\t index++;\n\t continue;\n\t }\n\t\n\t queue[index].close();\n\t queue.splice(index, 1);\n\t }\n\t };\n\t\n\t notifications.createHolder = function () {\n\t\n\t var holder = editor.draw.node('DIV', 'cdx-notifications-block');\n\t\n\t editor.nodes.notifications = document.body.appendChild(holder);\n\t\n\t return holder;\n\t };\n\t\n\t /**\r\n\t * Error notificator. Shows block with message\r\n\t * @protected\r\n\t */\n\t notifications.errorThrown = function (errorMsg, event) {\n\t\n\t editor.notifications.notification({ message: 'This action is not available currently', type: event.type });\n\t };\n\t\n\t /**\r\n\t *\r\n\t * Appends notification\r\n\t *\r\n\t * settings = {\r\n\t * type - notification type (reserved types: alert, confirm, prompt). Just add class 'cdx-notification-'+type\r\n\t * message - notification message\r\n\t * okMsg - confirm button text (default - 'Ok')\r\n\t * cancelBtn - cancel button text (default - 'Cancel'). Only for confirm and prompt types\r\n\t * confirm - function-handler for ok button click\r\n\t * cancel - function-handler for cancel button click. Only for confirm and prompt types\r\n\t * time - time (in seconds) after which notification will close (default - 10s)\r\n\t * }\r\n\t *\r\n\t * @param settings\r\n\t */\n\t notifications.notification = function (constructorSettings) {\n\t\n\t /** Private vars and methods */\n\t var notification = null,\n\t cancel = null,\n\t type = null,\n\t confirm = null,\n\t inputField = null;\n\t\n\t var confirmHandler = function confirmHandler() {\n\t\n\t close();\n\t\n\t if (typeof confirm !== 'function') {\n\t\n\t return;\n\t }\n\t\n\t if (type == 'prompt') {\n\t\n\t confirm(inputField.value);\n\t return;\n\t }\n\t\n\t confirm();\n\t };\n\t\n\t var cancelHandler = function cancelHandler() {\n\t\n\t close();\n\t\n\t if (typeof cancel !== 'function') {\n\t\n\t return;\n\t }\n\t\n\t cancel();\n\t };\n\t\n\t /** Public methods */\n\t function create(settings) {\n\t\n\t if (!(settings && settings.message)) {\n\t\n\t editor.core.log('Can\\'t create notification. Message is missed');\n\t return;\n\t }\n\t\n\t settings.type = settings.type || 'alert';\n\t settings.time = settings.time * 1000 || 10000;\n\t\n\t var wrapper = editor.draw.node('DIV', 'cdx-notification'),\n\t message = editor.draw.node('DIV', 'cdx-notification__message'),\n\t input = editor.draw.node('INPUT', 'cdx-notification__input'),\n\t okBtn = editor.draw.node('SPAN', 'cdx-notification__ok-btn'),\n\t cancelBtn = editor.draw.node('SPAN', 'cdx-notification__cancel-btn');\n\t\n\t message.textContent = settings.message;\n\t okBtn.textContent = settings.okMsg || 'ОК';\n\t cancelBtn.textContent = settings.cancelMsg || 'Отмена';\n\t\n\t editor.listeners.add(okBtn, 'click', confirmHandler);\n\t editor.listeners.add(cancelBtn, 'click', cancelHandler);\n\t\n\t wrapper.appendChild(message);\n\t\n\t if (settings.type == 'prompt') {\n\t\n\t wrapper.appendChild(input);\n\t }\n\t\n\t wrapper.appendChild(okBtn);\n\t\n\t if (settings.type == 'prompt' || settings.type == 'confirm') {\n\t\n\t wrapper.appendChild(cancelBtn);\n\t }\n\t\n\t wrapper.classList.add('cdx-notification-' + settings.type);\n\t wrapper.dataset.type = settings.type;\n\t\n\t notification = wrapper;\n\t type = settings.type;\n\t confirm = settings.confirm;\n\t cancel = settings.cancel;\n\t inputField = input;\n\t\n\t if (settings.type != 'prompt' && settings.type != 'confirm') {\n\t\n\t window.setTimeout(close, settings.time);\n\t }\n\t };\n\t\n\t function send() {\n\t\n\t editor.nodes.notifications.appendChild(notification);\n\t inputField.focus();\n\t\n\t editor.nodes.notifications.classList.add('cdx-notification__notification-appending');\n\t\n\t window.setTimeout(function () {\n\t\n\t editor.nodes.notifications.classList.remove('cdx-notification__notification-appending');\n\t }, 100);\n\t\n\t addToQueue({ type: type, close: close });\n\t };\n\t\n\t function close() {\n\t\n\t notification.remove();\n\t };\n\t\n\t if (constructorSettings) {\n\t\n\t create(constructorSettings);\n\t send();\n\t }\n\t\n\t return {\n\t create: create,\n\t send: send,\n\t close: close\n\t };\n\t };\n\t\n\t notifications.clear = function () {\n\t\n\t editor.nodes.notifications.innerHTML = '';\n\t queue = [];\n\t };\n\t\n\t return notifications;\n\t}({});\n\n/***/ },\n/* 16 */\n/***/ function(module, exports) {\n\n\t\"use strict\";\n\t\n\t/**\r\n\t * Codex Editor Parser Module\r\n\t *\r\n\t * @author Codex Team\r\n\t * @version 1.1\r\n\t */\n\t\n\tmodule.exports = function (parser) {\n\t\n\t var editor = codex.editor;\n\t\n\t /** inserting text */\n\t parser.insertPastedContent = function (blockType, tag) {\n\t\n\t editor.content.insertBlock({\n\t type: blockType.type,\n\t block: blockType.render({\n\t text: tag.innerHTML\n\t })\n\t });\n\t };\n\t\n\t /**\r\n\t * Check DOM node for display style: separated block or child-view\r\n\t */\n\t parser.isFirstLevelBlock = function (node) {\n\t\n\t return node.nodeType == editor.core.nodeTypes.TAG && node.classList.contains(editor.ui.className.BLOCK_CLASSNAME);\n\t };\n\t\n\t return parser;\n\t}({});\n\n/***/ },\n/* 17 */\n/***/ function(module, exports, __webpack_require__) {\n\n\t'use strict';\n\t\n\t/**\r\n\t * Codex Sanitizer\r\n\t */\n\t\n\tmodule.exports = function (sanitizer) {\n\t\n\t /** HTML Janitor library */\n\t var janitor = __webpack_require__(18);\n\t\n\t /** Codex Editor */\n\t var editor = codex.editor;\n\t\n\t sanitizer.prepare = function () {\n\t\n\t if (editor.settings.sanitizer && !editor.core.isEmpty(editor.settings.sanitizer)) {\n\t\n\t Config.CUSTOM = editor.settings.sanitizer;\n\t }\n\t };\n\t\n\t /**\r\n\t * Basic config\r\n\t */\n\t var Config = {\n\t\n\t /** User configuration */\n\t CUSTOM: null,\n\t\n\t BASIC: {\n\t\n\t tags: {\n\t p: {},\n\t a: {\n\t href: true,\n\t target: '_blank',\n\t rel: 'nofollow'\n\t },\n\t i: {},\n\t b: {},\n\t strong: {},\n\t em: {},\n\t span: {}\n\t }\n\t }\n\t };\n\t\n\t sanitizer.Config = Config;\n\t\n\t /**\r\n\t *\r\n\t * @param userCustomConfig\r\n\t * @returns {*}\r\n\t * @private\r\n\t *\r\n\t * @description If developer uses editor's API, then he can customize sane restrictions.\r\n\t * Or, sane config can be defined globally in editors initialization. That config will be used everywhere\r\n\t * At least, if there is no config overrides, that API uses BASIC Default configation\r\n\t */\n\t var init_ = function init_(userCustomConfig) {\n\t\n\t var configuration = userCustomConfig || Config.CUSTOM || Config.BASIC;\n\t\n\t return new janitor(configuration);\n\t };\n\t\n\t /**\r\n\t * Cleans string from unwanted tags\r\n\t * @protected\r\n\t * @param {String} dirtyString - taint string\r\n\t * @param {Object} customConfig - allowed tags\r\n\t */\n\t sanitizer.clean = function (dirtyString, customConfig) {\n\t\n\t var janitorInstance = init_(customConfig);\n\t\n\t return janitorInstance.clean(dirtyString);\n\t };\n\t\n\t return sanitizer;\n\t}({});\n\n/***/ },\n/* 18 */\n/***/ function(module, exports, __webpack_require__) {\n\n\tvar __WEBPACK_AMD_DEFINE_FACTORY__, __WEBPACK_AMD_DEFINE_RESULT__;(function (root, factory) {\n\t if (true) {\n\t !(__WEBPACK_AMD_DEFINE_FACTORY__ = (factory), __WEBPACK_AMD_DEFINE_RESULT__ = (typeof __WEBPACK_AMD_DEFINE_FACTORY__ === 'function' ? (__WEBPACK_AMD_DEFINE_FACTORY__.call(exports, __webpack_require__, exports, module)) : __WEBPACK_AMD_DEFINE_FACTORY__), __WEBPACK_AMD_DEFINE_RESULT__ !== undefined && (module.exports = __WEBPACK_AMD_DEFINE_RESULT__));\n\t } else if (typeof exports === 'object') {\n\t module.exports = factory();\n\t } else {\n\t root.HTMLJanitor = factory();\n\t }\n\t}(this, function () {\n\t\n\t /**\n\t * @param {Object} config.tags Dictionary of allowed tags.\n\t * @param {boolean} config.keepNestedBlockElements Default false.\n\t */\n\t function HTMLJanitor(config) {\n\t\n\t var tagDefinitions = config['tags'];\n\t var tags = Object.keys(tagDefinitions);\n\t\n\t var validConfigValues = tags\n\t .map(function(k) { return typeof tagDefinitions[k]; })\n\t .every(function(type) { return type === 'object' || type === 'boolean' || type === 'function'; });\n\t\n\t if(!validConfigValues) {\n\t throw new Error(\"The configuration was invalid\");\n\t }\n\t\n\t this.config = config;\n\t }\n\t\n\t // TODO: not exhaustive?\n\t var blockElementNames = ['P', 'LI', 'TD', 'TH', 'DIV', 'H1', 'H2', 'H3', 'H4', 'H5', 'H6', 'PRE'];\n\t function isBlockElement(node) {\n\t return blockElementNames.indexOf(node.nodeName) !== -1;\n\t }\n\t\n\t var inlineElementNames = ['A', 'B', 'STRONG', 'I', 'EM', 'SUB', 'SUP', 'U', 'STRIKE'];\n\t function isInlineElement(node) {\n\t return inlineElementNames.indexOf(node.nodeName) !== -1;\n\t }\n\t\n\t HTMLJanitor.prototype.clean = function (html) {\n\t var sandbox = document.createElement('div');\n\t sandbox.innerHTML = html;\n\t\n\t this._sanitize(sandbox);\n\t\n\t return sandbox.innerHTML;\n\t };\n\t\n\t HTMLJanitor.prototype._sanitize = function (parentNode) {\n\t var treeWalker = createTreeWalker(parentNode);\n\t var node = treeWalker.firstChild();\n\t if (!node) { return; }\n\t\n\t do {\n\t // Ignore nodes that have already been sanitized\n\t if (node._sanitized) {\n\t continue;\n\t }\n\t\n\t if (node.nodeType === Node.TEXT_NODE) {\n\t // If this text node is just whitespace and the previous or next element\n\t // sibling is a block element, remove it\n\t // N.B.: This heuristic could change. Very specific to a bug with\n\t // `contenteditable` in Firefox: http://jsbin.com/EyuKase/1/edit?js,output\n\t // FIXME: make this an option?\n\t if (node.data.trim() === ''\n\t && ((node.previousElementSibling && isBlockElement(node.previousElementSibling))\n\t || (node.nextElementSibling && isBlockElement(node.nextElementSibling)))) {\n\t parentNode.removeChild(node);\n\t this._sanitize(parentNode);\n\t break;\n\t } else {\n\t continue;\n\t }\n\t }\n\t\n\t // Remove all comments\n\t if (node.nodeType === Node.COMMENT_NODE) {\n\t parentNode.removeChild(node);\n\t this._sanitize(parentNode);\n\t break;\n\t }\n\t\n\t var isInline = isInlineElement(node);\n\t var containsBlockElement;\n\t if (isInline) {\n\t containsBlockElement = Array.prototype.some.call(node.childNodes, isBlockElement);\n\t }\n\t\n\t // Block elements should not be nested (e.g.

  • ...); if\n\t // they are, we want to unwrap the inner block element.\n\t var isNotTopContainer = !! parentNode.parentNode;\n\t var isNestedBlockElement =\n\t isBlockElement(parentNode) &&\n\t isBlockElement(node) &&\n\t isNotTopContainer;\n\t\n\t var nodeName = node.nodeName.toLowerCase();\n\t\n\t var allowedAttrs = getAllowedAttrs(this.config, nodeName, node);\n\t\n\t var isInvalid = isInline && containsBlockElement;\n\t\n\t // Drop tag entirely according to the whitelist *and* if the markup\n\t // is invalid.\n\t if (isInvalid || shouldRejectNode(node, allowedAttrs)\n\t || (!this.config.keepNestedBlockElements && isNestedBlockElement)) {\n\t // Do not keep the inner text of SCRIPT/STYLE elements.\n\t if (! (node.nodeName === 'SCRIPT' || node.nodeName === 'STYLE')) {\n\t while (node.childNodes.length > 0) {\n\t parentNode.insertBefore(node.childNodes[0], node);\n\t }\n\t }\n\t parentNode.removeChild(node);\n\t\n\t this._sanitize(parentNode);\n\t break;\n\t }\n\t\n\t // Sanitize attributes\n\t for (var a = 0; a < node.attributes.length; a += 1) {\n\t var attr = node.attributes[a];\n\t\n\t if (shouldRejectAttr(attr, allowedAttrs, node)) {\n\t node.removeAttribute(attr.name);\n\t // Shift the array to continue looping.\n\t a = a - 1;\n\t }\n\t }\n\t\n\t // Sanitize children\n\t this._sanitize(node);\n\t\n\t // Mark node as sanitized so it's ignored in future runs\n\t node._sanitized = true;\n\t } while ((node = treeWalker.nextSibling()));\n\t };\n\t\n\t function createTreeWalker(node) {\n\t return document.createTreeWalker(node,\n\t NodeFilter.SHOW_TEXT | NodeFilter.SHOW_ELEMENT | NodeFilter.SHOW_COMMENT,\n\t null, false);\n\t }\n\t\n\t function getAllowedAttrs(config, nodeName, node){\n\t if (typeof config.tags[nodeName] === 'function') {\n\t return config.tags[nodeName](node);\n\t } else {\n\t return config.tags[nodeName];\n\t }\n\t }\n\t\n\t function shouldRejectNode(node, allowedAttrs){\n\t if (typeof allowedAttrs === 'undefined') {\n\t return true;\n\t } else if (typeof allowedAttrs === 'boolean') {\n\t return !allowedAttrs;\n\t }\n\t\n\t return false;\n\t }\n\t\n\t function shouldRejectAttr(attr, allowedAttrs, node){\n\t var attrName = attr.name.toLowerCase();\n\t\n\t if (allowedAttrs === true){\n\t return false;\n\t } else if (typeof allowedAttrs[attrName] === 'function'){\n\t return !allowedAttrs[attrName](attr.value, node);\n\t } else if (typeof allowedAttrs[attrName] === 'undefined'){\n\t return true;\n\t } else if (allowedAttrs[attrName] === false) {\n\t return true;\n\t } else if (typeof allowedAttrs[attrName] === 'string') {\n\t return (allowedAttrs[attrName] !== attr.value);\n\t }\n\t\n\t return false;\n\t }\n\t\n\t return HTMLJanitor;\n\t\n\t}));\n\n\n/***/ },\n/* 19 */\n/***/ function(module, exports) {\n\n\t\"use strict\";\n\t\n\t/**\r\n\t * Codex Editor Listeners module\r\n\t *\r\n\t * @author Codex Team\r\n\t * @version 1.0\r\n\t */\n\t\n\t/**\r\n\t * Module-decorator for event listeners assignment\r\n\t */\n\tmodule.exports = function (listeners) {\n\t\n\t var allListeners = [];\n\t\n\t /**\r\n\t * Search methods\r\n\t *\r\n\t * byElement, byType and byHandler returns array of suitable listeners\r\n\t * one and all takes element, eventType, and handler and returns first (all) suitable listener\r\n\t *\r\n\t */\n\t listeners.search = function () {\n\t\n\t var byElement = function byElement(element, context) {\n\t\n\t var listenersOnElement = [];\n\t\n\t context = context || allListeners;\n\t\n\t for (var i = 0; i < context.length; i++) {\n\t\n\t var listener = context[i];\n\t\n\t if (listener.element === element) {\n\t\n\t listenersOnElement.push(listener);\n\t }\n\t }\n\t\n\t return listenersOnElement;\n\t };\n\t\n\t var byType = function byType(eventType, context) {\n\t\n\t var listenersWithType = [];\n\t\n\t context = context || allListeners;\n\t\n\t for (var i = 0; i < context.length; i++) {\n\t\n\t var listener = context[i];\n\t\n\t if (listener.type === eventType) {\n\t\n\t listenersWithType.push(listener);\n\t }\n\t }\n\t\n\t return listenersWithType;\n\t };\n\t\n\t var byHandler = function byHandler(handler, context) {\n\t\n\t var listenersWithHandler = [];\n\t\n\t context = context || allListeners;\n\t\n\t for (var i = 0; i < context.length; i++) {\n\t\n\t var listener = context[i];\n\t\n\t if (listener.handler === handler) {\n\t\n\t listenersWithHandler.push(listener);\n\t }\n\t }\n\t\n\t return listenersWithHandler;\n\t };\n\t\n\t var one = function one(element, eventType, handler) {\n\t\n\t var result = allListeners;\n\t\n\t if (element) result = byElement(element, result);\n\t\n\t if (eventType) result = byType(eventType, result);\n\t\n\t if (handler) result = byHandler(handler, result);\n\t\n\t return result[0];\n\t };\n\t\n\t var all = function all(element, eventType, handler) {\n\t\n\t var result = allListeners;\n\t\n\t if (element) result = byElement(element, result);\n\t\n\t if (eventType) result = byType(eventType, result);\n\t\n\t if (handler) result = byHandler(handler, result);\n\t\n\t return result;\n\t };\n\t\n\t return {\n\t byElement: byElement,\n\t byType: byType,\n\t byHandler: byHandler,\n\t one: one,\n\t all: all\n\t };\n\t }();\n\t\n\t listeners.add = function (element, eventType, handler, isCapture) {\n\t\n\t element.addEventListener(eventType, handler, isCapture);\n\t\n\t var data = {\n\t element: element,\n\t type: eventType,\n\t handler: handler\n\t };\n\t\n\t var alreadyAddedListener = listeners.search.one(element, eventType, handler);\n\t\n\t if (!alreadyAddedListener) {\n\t\n\t allListeners.push(data);\n\t }\n\t };\n\t\n\t listeners.remove = function (element, eventType, handler) {\n\t\n\t element.removeEventListener(eventType, handler);\n\t\n\t var existingListeners = listeners.search.all(element, eventType, handler);\n\t\n\t for (var i = 0; i < existingListeners.length; i++) {\n\t\n\t var index = allListeners.indexOf(existingListeners[i]);\n\t\n\t if (index > 0) {\n\t\n\t allListeners.splice(index, 1);\n\t }\n\t }\n\t };\n\t\n\t listeners.removeAll = function () {\n\t\n\t allListeners.map(function (current) {\n\t\n\t listeners.remove(current.element, current.type, current.handler);\n\t });\n\t };\n\t\n\t listeners.get = function (element, eventType, handler) {\n\t\n\t return listeners.search.all(element, eventType, handler);\n\t };\n\t\n\t return listeners;\n\t}({});\n\n/***/ },\n/* 20 */\n/***/ function(module, exports) {\n\n\t'use strict';\n\t\n\tvar _typeof = typeof Symbol === \"function\" && typeof Symbol.iterator === \"symbol\" ? function (obj) { return typeof obj; } : function (obj) { return obj && typeof Symbol === \"function\" && obj.constructor === Symbol && obj !== Symbol.prototype ? \"symbol\" : typeof obj; };\n\t\n\t/**\r\n\t * Codex Editor Destroyer module\r\n\t *\r\n\t * @auhor Codex Team\r\n\t * @version 1.0\r\n\t */\n\t\n\tmodule.exports = function (destroyer) {\n\t\n\t var editor = codex.editor;\n\t\n\t destroyer.removeNodes = function () {\n\t\n\t editor.nodes.wrapper.remove();\n\t editor.nodes.notifications.remove();\n\t };\n\t\n\t destroyer.destroyPlugins = function () {\n\t\n\t for (var tool in editor.tools) {\n\t\n\t if (typeof editor.tools[tool].destroy === 'function') {\n\t\n\t editor.tools[tool].destroy();\n\t }\n\t }\n\t };\n\t\n\t destroyer.destroyScripts = function () {\n\t\n\t var scripts = document.getElementsByTagName('SCRIPT');\n\t\n\t for (var i = 0; i < scripts.length; i++) {\n\t\n\t if (scripts[i].id.indexOf(editor.scriptPrefix) + 1) {\n\t\n\t scripts[i].remove();\n\t i--;\n\t }\n\t }\n\t };\n\t\n\t /**\r\n\t * Delete editor data from webpage.\r\n\t * You should send settings argument with boolean flags:\r\n\t * @param settings.ui- remove redactor event listeners and DOM nodes\r\n\t * @param settings.scripts - remove redactor scripts from DOM\r\n\t * @param settings.plugins - remove plugin's objects\r\n\t * @param settings.core - remove editor core. You can remove core only if UI and scripts flags is true\r\n\t * }\r\n\t *\r\n\t */\n\t destroyer.destroy = function (settings) {\n\t\n\t if (!settings || (typeof settings === 'undefined' ? 'undefined' : _typeof(settings)) !== 'object') {\n\t\n\t return;\n\t }\n\t\n\t if (settings.ui) {\n\t\n\t destroyer.removeNodes();\n\t editor.listeners.removeAll();\n\t }\n\t\n\t if (settings.scripts) {\n\t\n\t destroyer.destroyScripts();\n\t }\n\t\n\t if (settings.plugins) {\n\t\n\t destroyer.destroyPlugins();\n\t }\n\t\n\t if (settings.ui && settings.scripts && settings.core) {\n\t\n\t delete codex.editor;\n\t }\n\t };\n\t\n\t return destroyer;\n\t}({});\n\n/***/ },\n/* 21 */\n/***/ function(module, exports) {\n\n\t'use strict';\n\t\n\t/**\r\n\t * Codex Editor Paste module\r\n\t *\r\n\t * @author Codex Team\r\n\t * @version 1.1.1\r\n\t */\n\t\n\tmodule.exports = function (paste) {\n\t\n\t var editor = codex.editor;\n\t\n\t var patterns = [];\n\t\n\t paste.prepare = function () {\n\t\n\t var tools = editor.tools;\n\t\n\t for (var tool in tools) {\n\t\n\t if (!tools[tool].renderOnPastePatterns || !Array.isArray(tools[tool].renderOnPastePatterns)) {\n\t\n\t continue;\n\t }\n\t\n\t tools[tool].renderOnPastePatterns.map(function (pattern) {\n\t\n\t patterns.push(pattern);\n\t });\n\t }\n\t\n\t return Promise.resolve();\n\t };\n\t\n\t /**\r\n\t * Saves data\r\n\t * @param event\r\n\t */\n\t paste.pasted = function (event) {\n\t\n\t var clipBoardData = event.clipboardData || window.clipboardData,\n\t content = clipBoardData.getData('Text');\n\t\n\t var result = analize(content);\n\t\n\t if (result) {\n\t\n\t event.preventDefault();\n\t event.stopImmediatePropagation();\n\t }\n\t\n\t return result;\n\t };\n\t\n\t /**\r\n\t * Analizes pated string and calls necessary method\r\n\t */\n\t\n\t var analize = function analize(string) {\n\t\n\t var result = false,\n\t content = editor.content.currentNode,\n\t plugin = content.dataset.tool;\n\t\n\t patterns.map(function (pattern) {\n\t\n\t var execArray = pattern.regex.exec(string),\n\t match = execArray && execArray[0];\n\t\n\t if (match && match === string.trim()) {\n\t\n\t /** current block is not empty */\n\t if (content.textContent.trim() && plugin == editor.settings.initialBlockPlugin) {\n\t\n\t pasteToNewBlock_();\n\t }\n\t\n\t pattern.callback(string, pattern);\n\t result = true;\n\t }\n\t });\n\t\n\t return result;\n\t };\n\t\n\t var pasteToNewBlock_ = function pasteToNewBlock_() {\n\t\n\t /** Create new initial block */\n\t editor.content.insertBlock({\n\t\n\t type: editor.settings.initialBlockPlugin,\n\t block: editor.tools[editor.settings.initialBlockPlugin].render({\n\t text: ''\n\t })\n\t\n\t }, false);\n\t };\n\t\n\t /**\r\n\t * This method prevents default behaviour.\r\n\t *\r\n\t * @param {Object} event\r\n\t * @protected\r\n\t *\r\n\t * @description We get from clipboard pasted data, sanitize, make a fragment that contains of this sanitized nodes.\r\n\t * Firstly, we need to memorize the caret position. We can do that by getting the range of selection.\r\n\t * After all, we insert clear fragment into caret placed position. Then, we should move the caret to the last node\r\n\t */\n\t paste.blockPasteCallback = function (event) {\n\t\n\t if (!needsToHandlePasteEvent(event.target)) {\n\t\n\t return;\n\t }\n\t\n\t /** get html pasted data - dirty data */\n\t var htmlData = event.clipboardData.getData('text/html'),\n\t plainData = event.clipboardData.getData('text/plain');\n\t\n\t /** Temporary DIV that is used to work with text's paragraphs as DOM-elements*/\n\t var paragraphs = editor.draw.node('DIV', '', {}),\n\t cleanData,\n\t wrappedData;\n\t\n\t /** Create fragment, that we paste to range after proccesing */\n\t cleanData = editor.sanitizer.clean(htmlData);\n\t\n\t /**\r\n\t * We wrap pasted text with

    tags to split it logically\r\n\t * @type {string}\r\n\t */\n\t wrappedData = editor.content.wrapTextWithParagraphs(cleanData, plainData);\n\t paragraphs.innerHTML = wrappedData;\n\t\n\t /**\r\n\t * If there only one paragraph, lets user agent paste it\r\n\t */\n\t if (paragraphs.childNodes.length == 1) {\n\t\n\t return;\n\t }\n\t\n\t /** Prevent default behaviour */\n\t event.preventDefault();\n\t\n\t insertPastedParagraphs(paragraphs.childNodes);\n\t };\n\t\n\t /**\r\n\t * Checks if we should handle paste event on block\r\n\t * @param block\r\n\t *\r\n\t * @return {boolean}\r\n\t */\n\t var needsToHandlePasteEvent = function needsToHandlePasteEvent(block) {\n\t\n\t /** If area is input or textarea then allow default behaviour */\n\t if (editor.core.isNativeInput(block)) {\n\t\n\t return false;\n\t }\n\t\n\t var editableParent = editor.content.getEditableParent(block);\n\t\n\t /** Allow paste when event target placed in Editable element */\n\t if (!editableParent) {\n\t\n\t return false;\n\t }\n\t\n\t return true;\n\t };\n\t\n\t var insertPastedParagraphs = function insertPastedParagraphs(paragraphs) {\n\t\n\t var NEW_BLOCK_TYPE = editor.settings.initialBlockPlugin,\n\t currentBlockContent = editor.content.currentNode.firstChild.firstChild;\n\t\n\t paragraphs.forEach(function (paragraph, index) {\n\t\n\t /**\r\n\t * If there was no data in working node, replace it with first paragraph of pasted text\r\n\t */\n\t if (index == 0 && currentBlockContent.innerHTML.trim() === '') {\n\t\n\t editor.content.switchBlock(editor.content.currentNode, editor.tools[NEW_BLOCK_TYPE].render({\n\t text: paragraph.innerHTML\n\t }), NEW_BLOCK_TYPE);\n\t\n\t return;\n\t }\n\t\n\t editor.content.insertBlock({\n\t type: NEW_BLOCK_TYPE,\n\t block: editor.tools[NEW_BLOCK_TYPE].render({\n\t text: paragraph.innerHTML\n\t })\n\t });\n\t\n\t editor.caret.inputIndex++;\n\t });\n\t\n\t editor.caret.setToPreviousBlock(editor.caret.getCurrentInputIndex() + 1);\n\t };\n\t\n\t return paste;\n\t}({});\n\n/***/ }\n/******/ ]);\n\n\n// WEBPACK FOOTER //\n// codex-editor.js"," \t// The module cache\n \tvar installedModules = {};\n\n \t// The require function\n \tfunction __webpack_require__(moduleId) {\n\n \t\t// Check if module is in cache\n \t\tif(installedModules[moduleId])\n \t\t\treturn installedModules[moduleId].exports;\n\n \t\t// Create a new module (and put it into the cache)\n \t\tvar module = installedModules[moduleId] = {\n \t\t\texports: {},\n \t\t\tid: moduleId,\n \t\t\tloaded: false\n \t\t};\n\n \t\t// Execute the module function\n \t\tmodules[moduleId].call(module.exports, module, module.exports, __webpack_require__);\n\n \t\t// Flag the module as loaded\n \t\tmodule.loaded = true;\n\n \t\t// Return the exports of the module\n \t\treturn module.exports;\n \t}\n\n\n \t// expose the modules object (__webpack_modules__)\n \t__webpack_require__.m = modules;\n\n \t// expose the module cache\n \t__webpack_require__.c = installedModules;\n\n \t// __webpack_public_path__\n \t__webpack_require__.p = \"\";\n\n \t// Load entry module and return exports\n \treturn __webpack_require__(0);\n\n\n\n// WEBPACK FOOTER //\n// webpack/bootstrap ad1af9a0031475a14714","/**\r\n *\r\n * Codex Editor\r\n *\r\n * @author Codex Team\r\n */\r\n\r\nmodule.exports = (function (editor) {\r\n\r\n 'use strict';\r\n\r\n editor.version = VERSION;\r\n editor.scriptPrefix = 'cdx-script-';\r\n\r\n var init = function () {\r\n\r\n editor.core = require('./modules/core');\r\n editor.tools = require('./modules/tools');\r\n editor.ui = require('./modules/ui');\r\n editor.transport = require('./modules/transport');\r\n editor.renderer = require('./modules/renderer');\r\n editor.saver = require('./modules/saver');\r\n editor.content = require('./modules/content');\r\n editor.toolbar = require('./modules/toolbar/toolbar');\r\n editor.callback = require('./modules/callbacks');\r\n editor.draw = require('./modules/draw');\r\n editor.caret = require('./modules/caret');\r\n editor.notifications = require('./modules/notifications');\r\n editor.parser = require('./modules/parser');\r\n editor.sanitizer = require('./modules/sanitizer');\r\n editor.listeners = require('./modules/listeners');\r\n editor.destroyer = require('./modules/destroyer');\r\n editor.paste = require('./modules/paste');\r\n\r\n };\r\n\r\n /**\r\n * @public\r\n * holds initial settings\r\n */\r\n editor.settings = {\r\n tools : ['paragraph', 'header', 'picture', 'list', 'quote', 'code', 'twitter', 'instagram', 'smile'],\r\n holderId : 'codex-editor',\r\n\r\n // Type of block showing on empty editor\r\n initialBlockPlugin: 'paragraph'\r\n };\r\n\r\n /**\r\n * public\r\n *\r\n * Static nodes\r\n */\r\n editor.nodes = {\r\n holder : null,\r\n wrapper : null,\r\n toolbar : null,\r\n inlineToolbar : {\r\n wrapper : null,\r\n buttons : null,\r\n actions : null\r\n },\r\n toolbox : null,\r\n notifications : null,\r\n plusButton : null,\r\n showSettingsButton: null,\r\n showTrashButton : null,\r\n blockSettings : null,\r\n pluginSettings : null,\r\n defaultSettings : null,\r\n toolbarButtons : {}, // { type : DomEl, ... }\r\n redactor : null\r\n };\r\n\r\n /**\r\n * @public\r\n *\r\n * Output state\r\n */\r\n editor.state = {\r\n jsonOutput : [],\r\n blocks : [],\r\n inputs : []\r\n };\r\n\r\n /**\r\n * @public\r\n * Editor plugins\r\n */\r\n editor.tools = {};\r\n\r\n /**\r\n * Initialization\r\n * @uses Promise cEditor.core.prepare\r\n * @param {Object} userSettings\r\n * @param {Array} userSettings.tools list of plugins\r\n * @param {String} userSettings.holderId Element's id to append editor\r\n *\r\n * Load user defined tools\r\n * Tools must contain this important objects :\r\n * @param {String} type - this is a type of plugin. It can be used as plugin name\r\n * @param {String} iconClassname - this a icon in toolbar\r\n * @param {Object} make - what should plugin do, when it is clicked\r\n * @param {Object} appendCallback - callback after clicking\r\n * @param {Element} settings - what settings does it have\r\n * @param {Object} render - plugin get JSON, and should return HTML\r\n * @param {Object} save - plugin gets HTML content, returns JSON\r\n * @param {Boolean} displayInToolbox - will be displayed in toolbox. Default value is TRUE\r\n * @param {Boolean} enableLineBreaks - inserts new block or break lines. Default value is FALSE\r\n *\r\n * @example\r\n * - type : 'header',\r\n * - iconClassname : 'ce-icon-header',\r\n * - make : headerTool.make,\r\n * - appendCallback : headerTool.appendCallback,\r\n * - settings : headerTool.makeSettings(),\r\n * - render : headerTool.render,\r\n * - save : headerTool.save,\r\n * - displayInToolbox : true,\r\n * - enableLineBreaks : false\r\n */\r\n editor.start = function (userSettings) {\r\n\r\n init();\r\n\r\n editor.core.prepare(userSettings)\r\n\r\n // If all ok, make UI, bind events and parse initial-content\r\n .then(editor.ui.prepare)\r\n .then(editor.tools.prepare)\r\n .then(editor.sanitizer.prepare)\r\n .then(editor.paste.prepare)\r\n .then(editor.transport.prepare)\r\n .then(editor.renderer.makeBlocksFromData)\r\n .then(editor.ui.saveInputs)\r\n .catch(function (error) {\r\n\r\n editor.core.log('Initialization failed with error: %o', 'warn', error);\r\n\r\n });\r\n\r\n };\r\n\r\n return editor;\r\n\r\n})({});\r\n\n\n\n// WEBPACK FOOTER //\n// ./codex.js","/**\r\n * Codex Editor Core\r\n *\r\n * @author Codex Team\r\n * @version 1.1.3\r\n */\r\n\r\nmodule.exports = (function (core) {\r\n\r\n let editor = codex.editor;\r\n\r\n /**\r\n * @public\r\n *\r\n * Editor preparing method\r\n * @return Promise\r\n */\r\n core.prepare = function (userSettings) {\r\n\r\n return new Promise(function (resolve, reject) {\r\n\r\n if ( userSettings ) {\r\n\r\n editor.settings.tools = userSettings.tools || editor.settings.tools;\r\n\r\n }\r\n\r\n if (userSettings.data) {\r\n\r\n editor.state.blocks = userSettings.data;\r\n\r\n }\r\n\r\n if (userSettings.initialBlockPlugin) {\r\n\r\n editor.settings.initialBlockPlugin = userSettings.initialBlockPlugin;\r\n\r\n }\r\n\r\n if (userSettings.sanitizer) {\r\n\r\n editor.settings.sanitizer = userSettings.sanitizer;\r\n\r\n }\r\n\r\n editor.hideToolbar = userSettings.hideToolbar;\r\n\r\n editor.nodes.holder = document.getElementById(userSettings.holderId || editor.settings.holderId);\r\n\r\n if (typeof editor.nodes.holder === undefined || editor.nodes.holder === null) {\r\n\r\n reject(Error(\"Holder wasn't found by ID: #\" + userSettings.holderId));\r\n\r\n } else {\r\n\r\n resolve();\r\n\r\n }\r\n\r\n });\r\n\r\n };\r\n\r\n /**\r\n * Logging method\r\n * @param type = ['log', 'info', 'warn']\r\n */\r\n core.log = function (msg, type, arg) {\r\n\r\n type = type || 'log';\r\n\r\n if (!arg) {\r\n\r\n arg = msg || 'undefined';\r\n msg = '[codex-editor]: %o';\r\n\r\n } else {\r\n\r\n msg = '[codex-editor]: ' + msg;\r\n\r\n }\r\n\r\n try{\r\n\r\n if ( 'console' in window && window.console[ type ] ) {\r\n\r\n if ( arg ) window.console[ type ]( msg, arg );\r\n else window.console[ type ]( msg );\r\n\r\n }\r\n\r\n }catch(e) {}\r\n\r\n };\r\n\r\n /**\r\n * @protected\r\n *\r\n * Helper for insert one element after another\r\n */\r\n core.insertAfter = function (target, element) {\r\n\r\n target.parentNode.insertBefore(element, target.nextSibling);\r\n\r\n };\r\n\r\n /**\r\n * @const\r\n *\r\n * Readable DOM-node types map\r\n */\r\n core.nodeTypes = {\r\n TAG : 1,\r\n TEXT : 3,\r\n COMMENT : 8\r\n };\r\n\r\n /**\r\n * @const\r\n * Readable keys map\r\n */\r\n core.keys = { BACKSPACE: 8, TAB: 9, ENTER: 13, SHIFT: 16, CTRL: 17, ALT: 18, ESC: 27, SPACE: 32, LEFT: 37, UP: 38, DOWN: 40, RIGHT: 39, DELETE: 46, META: 91 };\r\n\r\n /**\r\n * @protected\r\n *\r\n * Check object for DOM node\r\n */\r\n core.isDomNode = function (el) {\r\n\r\n return el && typeof el === 'object' && el.nodeType && el.nodeType == this.nodeTypes.TAG;\r\n\r\n };\r\n\r\n /**\r\n * Checks passed object for emptiness\r\n * @require ES5 - Object.keys\r\n * @param {object}\r\n */\r\n core.isEmpty = function ( obj ) {\r\n\r\n return Object.keys(obj).length === 0;\r\n\r\n };\r\n\r\n /**\r\n * Native Ajax\r\n * @param {String} settings.url - request URL\r\n * @param {function} settings.beforeSend - returned value will be passed as context to the Success, Error and Progress callbacks\r\n * @param {function} settings.success\r\n * @param {function} settings.progress\r\n */\r\n core.ajax = function (settings) {\r\n\r\n if (!settings || !settings.url) {\r\n\r\n return;\r\n\r\n }\r\n\r\n var XMLHTTP = window.XMLHttpRequest ? new XMLHttpRequest() : new ActiveXObject('Microsoft.XMLHTTP'),\r\n encodedString,\r\n isFormData,\r\n prop;\r\n\r\n\r\n settings.async = true;\r\n settings.type = settings.type || 'GET';\r\n settings.data = settings.data || '';\r\n settings['content-type'] = settings['content-type'] || 'application/json; charset=utf-8';\r\n\r\n if (settings.type == 'GET' && settings.data) {\r\n\r\n settings.url = /\\?/.test(settings.url) ? settings.url + '&' + settings.data : settings.url + '?' + settings.data;\r\n\r\n } else {\r\n\r\n encodedString = '';\r\n for(prop in settings.data) {\r\n\r\n encodedString += (prop + '=' + encodeURIComponent(settings.data[prop]) + '&');\r\n\r\n }\r\n\r\n }\r\n\r\n if (settings.withCredentials) {\r\n\r\n XMLHTTP.withCredentials = true;\r\n\r\n }\r\n\r\n /**\r\n * Value returned in beforeSend funtion will be passed as context to the other response callbacks\r\n * If beforeSend returns false, AJAX will be blocked\r\n */\r\n let responseContext,\r\n beforeSendResult;\r\n\r\n if (typeof settings.beforeSend === 'function') {\r\n\r\n beforeSendResult = settings.beforeSend.call();\r\n\r\n if (beforeSendResult === false) {\r\n\r\n return;\r\n\r\n }\r\n\r\n }\r\n\r\n XMLHTTP.open( settings.type, settings.url, settings.async );\r\n\r\n /**\r\n * If we send FormData, we need no content-type header\r\n */\r\n isFormData = isFormData_(settings.data);\r\n\r\n if (!isFormData) {\r\n\r\n if (settings.type !== 'POST') {\r\n\r\n XMLHTTP.setRequestHeader('Content-type', settings['content-type']);\r\n\r\n } else {\r\n\r\n XMLHTTP.setRequestHeader('Content-type', 'application/x-www-form-urlencoded');\r\n\r\n }\r\n\r\n }\r\n\r\n XMLHTTP.setRequestHeader('X-Requested-With', 'XMLHttpRequest');\r\n\r\n responseContext = beforeSendResult || XMLHTTP;\r\n\r\n if (typeof settings.progress === 'function') {\r\n\r\n XMLHTTP.upload.onprogress = settings.progress.bind(responseContext);\r\n\r\n }\r\n\r\n XMLHTTP.onreadystatechange = function () {\r\n\r\n if (XMLHTTP.readyState === 4) {\r\n\r\n if (XMLHTTP.status === 200) {\r\n\r\n if (typeof settings.success === 'function') {\r\n\r\n settings.success.call(responseContext, XMLHTTP.responseText);\r\n\r\n }\r\n\r\n } else {\r\n\r\n if (typeof settings.error === 'function') {\r\n\r\n settings.error.call(responseContext, XMLHTTP.responseText, XMLHTTP.status);\r\n\r\n }\r\n\r\n }\r\n\r\n }\r\n\r\n };\r\n\r\n if (isFormData) {\r\n\r\n // Sending FormData\r\n XMLHTTP.send(settings.data);\r\n\r\n } else {\r\n\r\n // POST requests\r\n XMLHTTP.send(encodedString);\r\n\r\n }\r\n\r\n return XMLHTTP;\r\n\r\n };\r\n\r\n /**\r\n * Appends script to head of document\r\n * @return Promise\r\n */\r\n core.importScript = function (scriptPath, instanceName) {\r\n\r\n return new Promise(function (resolve, reject) {\r\n\r\n let script;\r\n\r\n /** Script is already loaded */\r\n if ( !instanceName ) {\r\n\r\n reject('Instance name is missed');\r\n\r\n } else if ( document.getElementById(editor.scriptPrefix + instanceName) ) {\r\n\r\n resolve(scriptPath);\r\n\r\n }\r\n\r\n script = document.createElement('SCRIPT');\r\n script.async = true;\r\n script.defer = true;\r\n script.id = editor.scriptPrefix + instanceName;\r\n\r\n script.onload = function () {\r\n\r\n resolve(scriptPath);\r\n\r\n };\r\n\r\n script.onerror = function () {\r\n\r\n reject(scriptPath);\r\n\r\n };\r\n\r\n script.src = scriptPath;\r\n document.head.appendChild(script);\r\n\r\n });\r\n\r\n };\r\n\r\n /**\r\n * Function for checking is it FormData object to send.\r\n * @param {Object} object to check\r\n * @return boolean\r\n */\r\n var isFormData_ = function (object) {\r\n\r\n return object instanceof FormData;\r\n\r\n };\r\n\r\n /**\r\n * Check block\r\n * @param target\r\n * @description Checks target is it native input\r\n */\r\n core.isNativeInput = function (target) {\r\n\r\n var nativeInputAreas = ['INPUT', 'TEXTAREA'];\r\n\r\n return (nativeInputAreas.indexOf(target.tagName) != -1);\r\n\r\n };\r\n\r\n\r\n return core;\r\n\r\n})({});\r\n\n\n\n// WEBPACK FOOTER //\n// ./modules/core.js","/**\r\n* Module working with plugins\r\n*/\r\nmodule.exports = (function () {\r\n\r\n let editor = codex.editor;\r\n\r\n /**\r\n * Initialize plugins before using\r\n * Ex. Load scripts or call some internal methods\r\n * @return Promise\r\n */\r\n function prepare() {\r\n\r\n return new Promise(function (resolve_, reject_) {\r\n\r\n Promise.resolve()\r\n\r\n /**\r\n * Compose a sequence of plugins that requires preparation\r\n */\r\n .then(function () {\r\n\r\n let pluginsRequiresPreparation = [],\r\n allPlugins = editor.tools;\r\n\r\n for ( let pluginName in allPlugins ) {\r\n\r\n let plugin = allPlugins[pluginName];\r\n\r\n if (plugin.prepare && typeof plugin.prepare != 'function' || !plugin.prepare) {\r\n\r\n continue;\r\n\r\n }\r\n\r\n pluginsRequiresPreparation.push(plugin);\r\n\r\n }\r\n\r\n /**\r\n * If no one passed plugins requires preparation, finish prepare() and go ahead\r\n */\r\n if (!pluginsRequiresPreparation.length) {\r\n\r\n resolve_();\r\n\r\n }\r\n\r\n return pluginsRequiresPreparation;\r\n\r\n })\r\n\r\n /** Wait plugins while they prepares */\r\n .then(waitAllPluginsPreparation_)\r\n\r\n .then(function () {\r\n\r\n editor.core.log('Plugins loaded', 'info');\r\n resolve_();\r\n\r\n }).catch(function (error) {\r\n\r\n reject_(error);\r\n\r\n });\r\n\r\n });\r\n\r\n }\r\n\r\n /**\r\n * @param {array} plugins - list of tools that requires preparation\r\n * @return {Promise} resolved while all plugins will be ready or failed\r\n */\r\n function waitAllPluginsPreparation_(plugins) {\r\n\r\n /**\r\n * @calls allPluginsProcessed__ when all plugins prepared or failed\r\n */\r\n return new Promise (function (allPluginsProcessed__) {\r\n\r\n /**\r\n * pluck each element from queue\r\n * First, send resolved Promise as previous value\r\n * Each plugins \"prepare\" method returns a Promise, that's why\r\n * reduce current element will not be able to continue while can't get\r\n * a resolved Promise\r\n *\r\n * If last plugin is \"prepared\" then go to the next stage of initialization\r\n */\r\n plugins.reduce(function (previousValue, plugin, iteration) {\r\n\r\n return previousValue.then(function () {\r\n\r\n /**\r\n * Wait till plugins prepared\r\n * @calls pluginIsReady__ when plugin is ready or failed\r\n */\r\n return new Promise ( function (pluginIsReady__) {\r\n\r\n callPluginsPrepareMethod_( plugin )\r\n\r\n .then( pluginIsReady__ )\r\n .then( function () {\r\n\r\n plugin.available = true;\r\n\r\n })\r\n\r\n .catch(function (error) {\r\n\r\n editor.core.log(`Plugin «${plugin.type}» was not loaded. Preparation failed because %o`, 'warn', error);\r\n plugin.available = false;\r\n plugin.loadingMessage = error;\r\n\r\n /** Go ahead even some plugin has problems */\r\n pluginIsReady__();\r\n\r\n })\r\n\r\n .then(function () {\r\n\r\n /** If last plugin has problems then just ignore and continue */\r\n if (iteration == plugins.length - 1) {\r\n\r\n allPluginsProcessed__();\r\n\r\n }\r\n\r\n });\r\n\r\n });\r\n\r\n });\r\n\r\n }, Promise.resolve() );\r\n\r\n });\r\n\r\n }\r\n\r\n var callPluginsPrepareMethod_ = function (plugin) {\r\n\r\n return plugin.prepare( plugin.config || {} );\r\n\r\n };\r\n\r\n return {\r\n prepare: prepare\r\n };\r\n\r\n}());\n\n\n// WEBPACK FOOTER //\n// ./modules/tools.js","/**\r\n * Codex Editor UI module\r\n *\r\n * @author Codex Team\r\n * @version 1.2.0\r\n */\r\n\r\nmodule.exports = (function (ui) {\r\n\r\n let editor = codex.editor;\r\n\r\n /**\r\n * Basic editor classnames\r\n */\r\n ui.className = {\r\n\r\n /**\r\n * @const {string} BLOCK_CLASSNAME - redactor blocks name\r\n */\r\n BLOCK_CLASSNAME : 'ce-block',\r\n\r\n /**\r\n * @const {String} wrapper for plugins content\r\n */\r\n BLOCK_CONTENT : 'ce-block__content',\r\n\r\n /**\r\n * @const {String} BLOCK_STRETCHED - makes block stretched\r\n */\r\n BLOCK_STRETCHED : 'ce-block--stretched',\r\n\r\n /**\r\n * @const {String} BLOCK_HIGHLIGHTED - adds background\r\n */\r\n BLOCK_HIGHLIGHTED : 'ce-block--focused',\r\n\r\n /**\r\n * @const {String} - for all default settings\r\n */\r\n SETTINGS_ITEM : 'ce-settings__item'\r\n\r\n };\r\n\r\n /**\r\n * @protected\r\n *\r\n * Making main interface\r\n */\r\n ui.prepare = function () {\r\n\r\n return new Promise(function (resolve) {\r\n\r\n let wrapper = editor.draw.wrapper(),\r\n redactor = editor.draw.redactor(),\r\n toolbar = makeToolBar_();\r\n\r\n wrapper.appendChild(toolbar);\r\n wrapper.appendChild(redactor);\r\n\r\n /** Save created ui-elements to static nodes state */\r\n editor.nodes.wrapper = wrapper;\r\n editor.nodes.redactor = redactor;\r\n\r\n /** Append editor wrapper with redactor zone into holder */\r\n editor.nodes.holder.appendChild(wrapper);\r\n\r\n resolve();\r\n\r\n })\r\n\r\n /** Add toolbox tools */\r\n .then(addTools_)\r\n\r\n /** Make container for inline toolbar */\r\n .then(makeInlineToolbar_)\r\n\r\n /** Add inline toolbar tools */\r\n .then(addInlineToolbarTools_)\r\n\r\n /** Draw wrapper for notifications */\r\n .then(makeNotificationHolder_)\r\n\r\n /** Add eventlisteners to redactor elements */\r\n .then(bindEvents_)\r\n\r\n .catch( function () {\r\n\r\n editor.core.log(\"Can't draw editor interface\");\r\n\r\n });\r\n\r\n };\r\n\r\n /**\r\n * @private\r\n * Draws inline toolbar zone\r\n */\r\n var makeInlineToolbar_ = function () {\r\n\r\n var container = editor.draw.inlineToolbar();\r\n\r\n /** Append to redactor new inline block */\r\n editor.nodes.inlineToolbar.wrapper = container;\r\n\r\n /** Draw toolbar buttons */\r\n editor.nodes.inlineToolbar.buttons = editor.draw.inlineToolbarButtons();\r\n\r\n /** Buttons action or settings */\r\n editor.nodes.inlineToolbar.actions = editor.draw.inlineToolbarActions();\r\n\r\n /** Append to inline toolbar buttons as part of it */\r\n editor.nodes.inlineToolbar.wrapper.appendChild(editor.nodes.inlineToolbar.buttons);\r\n editor.nodes.inlineToolbar.wrapper.appendChild(editor.nodes.inlineToolbar.actions);\r\n\r\n editor.nodes.wrapper.appendChild(editor.nodes.inlineToolbar.wrapper);\r\n\r\n };\r\n\r\n var makeToolBar_ = function () {\r\n\r\n let toolbar = editor.draw.toolbar(),\r\n blockButtons = makeToolbarSettings_(),\r\n toolbarContent = makeToolbarContent_();\r\n\r\n /** Appending first-level block buttons */\r\n toolbar.appendChild(blockButtons);\r\n\r\n /** Append toolbarContent to toolbar */\r\n toolbar.appendChild(toolbarContent);\r\n\r\n /** Make toolbar global */\r\n editor.nodes.toolbar = toolbar;\r\n\r\n return toolbar;\r\n\r\n };\r\n\r\n var makeToolbarContent_ = function () {\r\n\r\n let toolbarContent = editor.draw.toolbarContent(),\r\n toolbox = editor.draw.toolbox(),\r\n plusButton = editor.draw.plusButton();\r\n\r\n /** Append plus button */\r\n toolbarContent.appendChild(plusButton);\r\n\r\n /** Appending toolbar tools */\r\n toolbarContent.appendChild(toolbox);\r\n\r\n /** Make Toolbox and plusButton global */\r\n editor.nodes.toolbox = toolbox;\r\n editor.nodes.plusButton = plusButton;\r\n\r\n return toolbarContent;\r\n\r\n };\r\n\r\n var makeToolbarSettings_ = function () {\r\n\r\n let blockSettings = editor.draw.blockSettings(),\r\n blockButtons = editor.draw.blockButtons(),\r\n defaultSettings = editor.draw.defaultSettings(),\r\n showSettingsButton = editor.draw.settingsButton(),\r\n showTrashButton = editor.toolbar.settings.makeRemoveBlockButton(),\r\n pluginSettings = editor.draw.pluginsSettings();\r\n\r\n /** Add default and plugins settings */\r\n blockSettings.appendChild(pluginSettings);\r\n blockSettings.appendChild(defaultSettings);\r\n\r\n /**\r\n * Make blocks buttons\r\n * This block contains settings button and remove block button\r\n */\r\n blockButtons.appendChild(showSettingsButton);\r\n blockButtons.appendChild(showTrashButton);\r\n blockButtons.appendChild(blockSettings);\r\n\r\n /** Make BlockSettings, PluginSettings, DefaultSettings global */\r\n editor.nodes.blockSettings = blockSettings;\r\n editor.nodes.pluginSettings = pluginSettings;\r\n editor.nodes.defaultSettings = defaultSettings;\r\n editor.nodes.showSettingsButton = showSettingsButton;\r\n editor.nodes.showTrashButton = showTrashButton;\r\n\r\n return blockButtons;\r\n\r\n };\r\n\r\n /** Draw notifications holder */\r\n var makeNotificationHolder_ = function () {\r\n\r\n /** Append block with notifications to the document */\r\n editor.nodes.notifications = editor.notifications.createHolder();\r\n\r\n };\r\n\r\n /**\r\n * @private\r\n * Append tools passed in editor.tools\r\n */\r\n var addTools_ = function () {\r\n\r\n var tool,\r\n toolName,\r\n toolButton;\r\n\r\n for ( toolName in editor.settings.tools ) {\r\n\r\n tool = editor.settings.tools[toolName];\r\n\r\n editor.tools[toolName] = tool;\r\n\r\n if (!tool.iconClassname && tool.displayInToolbox) {\r\n\r\n editor.core.log('Toolbar icon classname missed. Tool %o skipped', 'warn', toolName);\r\n continue;\r\n\r\n }\r\n\r\n if (typeof tool.render != 'function') {\r\n\r\n editor.core.log('render method missed. Tool %o skipped', 'warn', toolName);\r\n continue;\r\n\r\n }\r\n\r\n if (!tool.displayInToolbox) {\r\n\r\n continue;\r\n\r\n } else {\r\n\r\n /** if tools is for toolbox */\r\n toolButton = editor.draw.toolbarButton(toolName, tool.iconClassname);\r\n\r\n editor.nodes.toolbox.appendChild(toolButton);\r\n\r\n editor.nodes.toolbarButtons[toolName] = toolButton;\r\n\r\n }\r\n\r\n }\r\n\r\n };\r\n\r\n var addInlineToolbarTools_ = function () {\r\n\r\n var tools = {\r\n\r\n bold: {\r\n icon : 'ce-icon-bold',\r\n command : 'bold'\r\n },\r\n\r\n italic: {\r\n icon : 'ce-icon-italic',\r\n command : 'italic'\r\n },\r\n\r\n underline: {\r\n icon : 'ce-icon-underline',\r\n command : 'underline'\r\n },\r\n\r\n link: {\r\n icon : 'ce-icon-link',\r\n command : 'createLink'\r\n }\r\n };\r\n\r\n var toolButton,\r\n tool;\r\n\r\n for(var name in tools) {\r\n\r\n tool = tools[name];\r\n\r\n toolButton = editor.draw.toolbarButtonInline(name, tool.icon);\r\n\r\n editor.nodes.inlineToolbar.buttons.appendChild(toolButton);\r\n /**\r\n * Add callbacks to this buttons\r\n */\r\n editor.ui.setInlineToolbarButtonBehaviour(toolButton, tool.command);\r\n\r\n }\r\n\r\n };\r\n\r\n /**\r\n * @private\r\n * Bind editor UI events\r\n */\r\n var bindEvents_ = function () {\r\n\r\n editor.core.log('ui.bindEvents fired', 'info');\r\n\r\n // window.addEventListener('error', function (errorMsg, url, lineNumber) {\r\n // editor.notifications.errorThrown(errorMsg, event);\r\n // }, false );\r\n\r\n /** All keydowns on Document */\r\n editor.listeners.add(document, 'keydown', editor.callback.globalKeydown, false);\r\n\r\n /** All keydowns on Redactor zone */\r\n editor.listeners.add(editor.nodes.redactor, 'keydown', editor.callback.redactorKeyDown, false);\r\n\r\n /** All keydowns on Document */\r\n editor.listeners.add(document, 'keyup', editor.callback.globalKeyup, false );\r\n\r\n /**\r\n * Mouse click to radactor\r\n */\r\n editor.listeners.add(editor.nodes.redactor, 'click', editor.callback.redactorClicked, false );\r\n\r\n /**\r\n * Clicks to the Plus button\r\n */\r\n editor.listeners.add(editor.nodes.plusButton, 'click', editor.callback.plusButtonClicked, false);\r\n\r\n /**\r\n * Clicks to SETTINGS button in toolbar\r\n */\r\n editor.listeners.add(editor.nodes.showSettingsButton, 'click', editor.callback.showSettingsButtonClicked, false );\r\n\r\n /** Bind click listeners on toolbar buttons */\r\n for (var button in editor.nodes.toolbarButtons) {\r\n\r\n editor.listeners.add(editor.nodes.toolbarButtons[button], 'click', editor.callback.toolbarButtonClicked, false);\r\n\r\n }\r\n\r\n };\r\n\r\n ui.addBlockHandlers = function (block) {\r\n\r\n if (!block) return;\r\n\r\n /**\r\n * Block keydowns\r\n */\r\n editor.listeners.add(block, 'keydown', editor.callback.blockKeydown, false);\r\n\r\n /**\r\n * Pasting content from another source\r\n * We have two type of sanitization\r\n * First - uses deep-first search algorithm to get sub nodes,\r\n * sanitizes whole Block_content and replaces cleared nodes\r\n * This method is deprecated\r\n * Method is used in editor.callback.blockPaste(event)\r\n *\r\n * Secont - uses Mutation observer.\r\n * Observer \"observe\" DOM changes and send changings to callback.\r\n * Callback gets changed node, not whole Block_content.\r\n * Inserted or changed node, which we've gotten have been cleared and replaced with diry node\r\n *\r\n * Method is used in editor.callback.blockPasteViaSanitize(event)\r\n *\r\n * @uses html-janitor\r\n * @example editor.callback.blockPasteViaSanitize(event), the second method.\r\n *\r\n */\r\n editor.listeners.add(block, 'paste', editor.paste.blockPasteCallback, false);\r\n\r\n editor.listeners.add(block, 'mouseup', editor.toolbar.inline.show, false);\r\n\r\n };\r\n\r\n /** getting all contenteditable elements */\r\n ui.saveInputs = function () {\r\n\r\n var redactor = editor.nodes.redactor;\r\n\r\n editor.state.inputs = [];\r\n\r\n /** Save all inputs in global variable state */\r\n var inputs = redactor.querySelectorAll('[contenteditable], input, textarea');\r\n\r\n Array.prototype.map.call(inputs, function (current) {\r\n\r\n if (!current.type || current.type == 'text' || current.type == 'textarea') {\r\n\r\n editor.state.inputs.push(current);\r\n\r\n }\r\n\r\n });\r\n\r\n };\r\n\r\n /**\r\n * Adds first initial block on empty redactor\r\n */\r\n ui.addInitialBlock = function () {\r\n\r\n var initialBlockType = editor.settings.initialBlockPlugin,\r\n initialBlock;\r\n\r\n if ( !editor.tools[initialBlockType] ) {\r\n\r\n editor.core.log('Plugin %o was not implemented and can\\'t be used as initial block', 'warn', initialBlockType);\r\n return;\r\n\r\n }\r\n\r\n initialBlock = editor.tools[initialBlockType].render();\r\n\r\n initialBlock.setAttribute('data-placeholder', 'Расскажите свою историю...');\r\n\r\n editor.content.insertBlock({\r\n type : initialBlockType,\r\n block : initialBlock\r\n });\r\n\r\n editor.content.workingNodeChanged(initialBlock);\r\n\r\n };\r\n\r\n ui.setInlineToolbarButtonBehaviour = function (button, type) {\r\n\r\n editor.listeners.add(button, 'mousedown', function (event) {\r\n\r\n editor.toolbar.inline.toolClicked(event, type);\r\n\r\n }, false);\r\n\r\n };\r\n\r\n return ui;\r\n\r\n})({});\r\n\n\n\n// WEBPACK FOOTER //\n// ./modules/ui.js","/**\r\n *\r\n * Codex.Editor Transport Module\r\n *\r\n * @copyright 2017 Codex-Team\r\n * @version 1.2.0\r\n */\r\n\r\nmodule.exports = (function (transport) {\r\n\r\n let editor = codex.editor;\r\n\r\n\r\n /**\r\n * @private {Object} current XmlHttpRequest instance\r\n */\r\n var currentRequest = null;\r\n\r\n\r\n /**\r\n * @type {null} | {DOMElement} input - keeps input element in memory\r\n */\r\n transport.input = null;\r\n\r\n /**\r\n * @property {Object} arguments - keep plugin settings and defined callbacks\r\n */\r\n transport.arguments = null;\r\n\r\n /**\r\n * Prepares input element where will be files\r\n */\r\n transport.prepare = function () {\r\n\r\n let input = editor.draw.node( 'INPUT', '', { type : 'file' } );\r\n\r\n editor.listeners.add(input, 'change', editor.transport.fileSelected);\r\n editor.transport.input = input;\r\n\r\n };\r\n\r\n /** Clear input when files is uploaded */\r\n transport.clearInput = function () {\r\n\r\n /** Remove old input */\r\n transport.input = null;\r\n\r\n /** Prepare new one */\r\n transport.prepare();\r\n\r\n };\r\n\r\n /**\r\n * Callback for file selection\r\n * @param {Event} event\r\n */\r\n transport.fileSelected = function () {\r\n\r\n var input = this,\r\n i,\r\n files = input.files,\r\n formData = new FormData();\r\n\r\n if (editor.transport.arguments.multiple === true) {\r\n\r\n for ( i = 0; i < files.length; i++) {\r\n\r\n formData.append('files[]', files[i], files[i].name);\r\n\r\n }\r\n\r\n } else {\r\n\r\n formData.append('files', files[0], files[0].name);\r\n\r\n }\r\n\r\n currentRequest = editor.core.ajax({\r\n type : 'POST',\r\n data : formData,\r\n url : editor.transport.arguments.url,\r\n beforeSend : editor.transport.arguments.beforeSend,\r\n success : editor.transport.arguments.success,\r\n error : editor.transport.arguments.error,\r\n progress : editor.transport.arguments.progress\r\n });\r\n\r\n /** Clear input */\r\n transport.clearInput();\r\n\r\n };\r\n\r\n /**\r\n * Use plugin callbacks\r\n * @protected\r\n *\r\n * @param {Object} args - can have :\r\n * @param {String} args.url - fetch URL\r\n * @param {Function} args.beforeSend - function calls before sending ajax\r\n * @param {Function} args.success - success callback\r\n * @param {Function} args.error - on error handler\r\n * @param {Function} args.progress - xhr onprogress handler\r\n * @param {Boolean} args.multiple - allow select several files\r\n * @param {String} args.accept - adds accept attribute\r\n */\r\n transport.selectAndUpload = function (args) {\r\n\r\n transport.arguments = args;\r\n\r\n if ( args.multiple === true) {\r\n\r\n transport.input.setAttribute('multiple', 'multiple');\r\n\r\n }\r\n\r\n if ( args.accept ) {\r\n\r\n transport.input.setAttribute('accept', args.accept);\r\n\r\n }\r\n\r\n transport.input.click();\r\n\r\n };\r\n\r\n transport.abort = function () {\r\n\r\n currentRequest.abort();\r\n\r\n currentRequest = null;\r\n\r\n };\r\n\r\n return transport;\r\n\r\n})({});\n\n\n// WEBPACK FOOTER //\n// ./modules/transport.js","/**\r\n * Codex Editor Renderer Module\r\n *\r\n * @author Codex Team\r\n * @version 1.0\r\n */\r\n\r\nmodule.exports = (function (renderer) {\r\n\r\n let editor = codex.editor;\r\n\r\n /**\r\n * Asyncronously parses input JSON to redactor blocks\r\n */\r\n renderer.makeBlocksFromData = function () {\r\n\r\n /**\r\n * If redactor is empty, add first paragraph to start writing\r\n */\r\n if (editor.core.isEmpty(editor.state.blocks) || !editor.state.blocks.items.length) {\r\n\r\n editor.ui.addInitialBlock();\r\n return;\r\n\r\n }\r\n\r\n Promise.resolve()\r\n\r\n /** First, get JSON from state */\r\n .then(function () {\r\n\r\n return editor.state.blocks;\r\n\r\n })\r\n\r\n /** Then, start to iterate they */\r\n .then(editor.renderer.appendBlocks)\r\n\r\n /** Write log if something goes wrong */\r\n .catch(function (error) {\r\n\r\n editor.core.log('Error while parsing JSON: %o', 'error', error);\r\n\r\n });\r\n\r\n };\r\n\r\n /**\r\n * Parses JSON to blocks\r\n * @param {object} data\r\n * @return Primise -> nodeList\r\n */\r\n renderer.appendBlocks = function (data) {\r\n\r\n var blocks = data.items;\r\n\r\n /**\r\n * Sequence of one-by-one blocks appending\r\n * Uses to save blocks order after async-handler\r\n */\r\n var nodeSequence = Promise.resolve();\r\n\r\n for (var index = 0; index < blocks.length ; index++ ) {\r\n\r\n /** Add node to sequence at specified index */\r\n editor.renderer.appendNodeAtIndex(nodeSequence, blocks, index);\r\n\r\n }\r\n\r\n };\r\n\r\n /**\r\n * Append node at specified index\r\n */\r\n renderer.appendNodeAtIndex = function (nodeSequence, blocks, index) {\r\n\r\n /** We need to append node to sequence */\r\n nodeSequence\r\n\r\n /** first, get node async-aware */\r\n .then(function () {\r\n\r\n return editor.renderer.getNodeAsync(blocks, index);\r\n\r\n })\r\n\r\n /**\r\n * second, compose editor-block from JSON object\r\n */\r\n .then(editor.renderer.createBlockFromData)\r\n\r\n /**\r\n * now insert block to redactor\r\n */\r\n .then(function (blockData) {\r\n\r\n /**\r\n * blockData has 'block', 'type' and 'stretched' information\r\n */\r\n editor.content.insertBlock(blockData);\r\n\r\n /** Pass created block to next step */\r\n return blockData.block;\r\n\r\n })\r\n\r\n /** Log if something wrong with node */\r\n .catch(function (error) {\r\n\r\n editor.core.log('Node skipped while parsing because %o', 'error', error);\r\n\r\n });\r\n\r\n };\r\n\r\n /**\r\n * Asynchronously returns block data from blocksList by index\r\n * @return Promise to node\r\n */\r\n renderer.getNodeAsync = function (blocksList, index) {\r\n\r\n return Promise.resolve().then(function () {\r\n\r\n return {\r\n tool : blocksList[index],\r\n position : index\r\n };\r\n\r\n });\r\n\r\n };\r\n\r\n /**\r\n * Creates editor block by JSON-data\r\n *\r\n * @uses render method of each plugin\r\n *\r\n * @param {Object} toolData.tool\r\n * { header : {\r\n * text: '',\r\n * type: 'H3', ...\r\n * }\r\n * }\r\n * @param {Number} toolData.position - index in input-blocks array\r\n * @return {Object} with type and Element\r\n */\r\n renderer.createBlockFromData = function ( toolData ) {\r\n\r\n /** New parser */\r\n var block,\r\n tool = toolData.tool,\r\n pluginName = tool.type;\r\n\r\n /** Get first key of object that stores plugin name */\r\n // for (var pluginName in blockData) break;\r\n\r\n /** Check for plugin existance */\r\n if (!editor.tools[pluginName]) {\r\n\r\n throw Error(`Plugin «${pluginName}» not found`);\r\n\r\n }\r\n\r\n /** Check for plugin having render method */\r\n if (typeof editor.tools[pluginName].render != 'function') {\r\n\r\n throw Error(`Plugin «${pluginName}» must have «render» method`);\r\n\r\n }\r\n\r\n if ( editor.tools[pluginName].available === false ) {\r\n\r\n block = editor.draw.unavailableBlock();\r\n\r\n block.innerHTML = editor.tools[pluginName].loadingMessage;\r\n\r\n /**\r\n * Saver will extract data from initial block data by position in array\r\n */\r\n block.dataset.inputPosition = toolData.position;\r\n\r\n } else {\r\n\r\n /** New Parser */\r\n block = editor.tools[pluginName].render(tool.data);\r\n\r\n }\r\n\r\n /** is first-level block stretched */\r\n var stretched = editor.tools[pluginName].isStretched || false;\r\n\r\n /** Retrun type and block */\r\n return {\r\n type : pluginName,\r\n block : block,\r\n stretched : stretched\r\n };\r\n\r\n };\r\n\r\n return renderer;\r\n\r\n})({});\n\n\n// WEBPACK FOOTER //\n// ./modules/renderer.js","/**\r\n * Codex Editor Saver\r\n *\r\n * @author Codex Team\r\n * @version 1.0.2\r\n */\r\n\r\nmodule.exports = (function (saver) {\r\n\r\n let editor = codex.editor;\r\n\r\n /**\r\n * Saves blocks\r\n * @private\r\n */\r\n saver.saveBlocks = function () {\r\n\r\n /** Save html content of redactor to memory */\r\n editor.state.html = editor.nodes.redactor.innerHTML;\r\n\r\n /** Empty jsonOutput state */\r\n editor.state.jsonOutput = [];\r\n\r\n Promise.resolve()\r\n\r\n .then(function () {\r\n\r\n return editor.nodes.redactor.childNodes;\r\n\r\n })\r\n /** Making a sequence from separate blocks */\r\n .then(editor.saver.makeQueue)\r\n\r\n .then(function () {\r\n // editor.nodes.textarea.innerHTML = editor.state.html;\r\n })\r\n\r\n .catch( function (error) {\r\n\r\n editor.core.log(error);\r\n\r\n });\r\n\r\n };\r\n\r\n saver.makeQueue = function (blocks) {\r\n\r\n var queue = Promise.resolve();\r\n\r\n for(var index = 0; index < blocks.length; index++) {\r\n\r\n /** Add node to sequence at specified index */\r\n editor.saver.getBlockData(queue, blocks, index);\r\n\r\n }\r\n\r\n };\r\n\r\n /** Gets every block and makes From Data */\r\n saver.getBlockData = function (queue, blocks, index) {\r\n\r\n queue.then(function () {\r\n\r\n return editor.saver.getNodeAsync(blocks, index);\r\n\r\n })\r\n\r\n .then(editor.saver.makeFormDataFromBlocks);\r\n\r\n };\r\n\r\n\r\n /**\r\n * Asynchronously returns block data from blocksList by index\r\n * @return Promise to node\r\n */\r\n saver.getNodeAsync = function (blocksList, index) {\r\n\r\n return Promise.resolve().then(function () {\r\n\r\n return blocksList[index];\r\n\r\n });\r\n\r\n };\r\n\r\n saver.makeFormDataFromBlocks = function (block) {\r\n\r\n var pluginName = block.dataset.tool;\r\n\r\n /** Check for plugin existance */\r\n if (!editor.tools[pluginName]) {\r\n\r\n throw Error(`Plugin «${pluginName}» not found`);\r\n\r\n }\r\n\r\n /** Check for plugin having render method */\r\n if (typeof editor.tools[pluginName].save != 'function') {\r\n\r\n throw Error(`Plugin «${pluginName}» must have save method`);\r\n\r\n }\r\n\r\n /** Result saver */\r\n var blockContent = block.childNodes[0],\r\n pluginsContent = blockContent.childNodes[0],\r\n savedData,\r\n position,\r\n output;\r\n\r\n /** If plugin wasn't available then return data from cache */\r\n if ( editor.tools[pluginName].available === false ) {\r\n\r\n position = pluginsContent.dataset.inputPosition;\r\n savedData = codex.editor.state.blocks.items[position].data;\r\n\r\n } else {\r\n\r\n savedData = editor.tools[pluginName].save(pluginsContent);\r\n\r\n if (editor.tools[pluginName].validate) {\r\n\r\n var result = editor.tools[pluginName].validate(savedData);\r\n\r\n /**\r\n * Do not allow invalid data\r\n */\r\n if (!result)\r\n return;\r\n\r\n }\r\n\r\n }\r\n\r\n output = {\r\n type : pluginName,\r\n data : savedData\r\n };\r\n\r\n editor.state.jsonOutput.push(output);\r\n\r\n };\r\n\r\n return saver;\r\n\r\n})({});\n\n\n// WEBPACK FOOTER //\n// ./modules/saver.js","/**\r\n * Codex Editor Content Module\r\n * Works with DOM\r\n *\r\n * @module Codex Editor content module\r\n *\r\n * @author Codex Team\r\n * @version 1.3.13\r\n *\r\n * @description Module works with Elements that have been appended to the main DOM\r\n */\r\n\r\nmodule.exports = (function (content) {\r\n\r\n let editor = codex.editor;\r\n\r\n /**\r\n * Links to current active block\r\n * @type {null | Element}\r\n */\r\n content.currentNode = null;\r\n\r\n /**\r\n * clicked in redactor area\r\n * @type {null | Boolean}\r\n */\r\n content.editorAreaHightlighted = null;\r\n\r\n /**\r\n * @deprecated\r\n * Synchronizes redactor with original textarea\r\n */\r\n content.sync = function () {\r\n\r\n editor.core.log('syncing...');\r\n\r\n /**\r\n * Save redactor content to editor.state\r\n */\r\n editor.state.html = editor.nodes.redactor.innerHTML;\r\n\r\n };\r\n\r\n /**\r\n * Appends background to the block\r\n *\r\n * @description add CSS class to highlight visually first-level block area\r\n */\r\n content.markBlock = function () {\r\n\r\n editor.content.currentNode.classList.add(editor.ui.className.BLOCK_HIGHLIGHTED);\r\n\r\n };\r\n\r\n /**\r\n * Clear background\r\n *\r\n * @description clears styles that highlights block\r\n */\r\n content.clearMark = function () {\r\n\r\n if (editor.content.currentNode) {\r\n\r\n editor.content.currentNode.classList.remove(editor.ui.className.BLOCK_HIGHLIGHTED);\r\n\r\n }\r\n\r\n };\r\n\r\n /**\r\n * Finds first-level block\r\n *\r\n * @param {Element} node - selected or clicked in redactors area node\r\n * @protected\r\n *\r\n * @description looks for first-level block.\r\n * gets parent while node is not first-level\r\n */\r\n content.getFirstLevelBlock = function (node) {\r\n\r\n if (!editor.core.isDomNode(node)) {\r\n\r\n node = node.parentNode;\r\n\r\n }\r\n\r\n if (node === editor.nodes.redactor || node === document.body) {\r\n\r\n return null;\r\n\r\n } else {\r\n\r\n while(!node.classList.contains(editor.ui.className.BLOCK_CLASSNAME)) {\r\n\r\n node = node.parentNode;\r\n\r\n }\r\n\r\n return node;\r\n\r\n }\r\n\r\n };\r\n\r\n /**\r\n * Trigger this event when working node changed\r\n * @param {Element} targetNode - first-level of this node will be current\r\n * @protected\r\n *\r\n * @description If targetNode is first-level then we set it as current else we look for parents to find first-level\r\n */\r\n content.workingNodeChanged = function (targetNode) {\r\n\r\n /** Clear background from previous marked block before we change */\r\n editor.content.clearMark();\r\n\r\n if (!targetNode) {\r\n\r\n return;\r\n\r\n }\r\n\r\n content.currentNode = content.getFirstLevelBlock(targetNode);\r\n\r\n };\r\n\r\n /**\r\n * Replaces one redactor block with another\r\n * @protected\r\n * @param {Element} targetBlock - block to replace. Mostly currentNode.\r\n * @param {Element} newBlock\r\n * @param {string} newBlockType - type of new block; we need to store it to data-attribute\r\n *\r\n * [!] Function does not saves old block content.\r\n * You can get it manually and pass with newBlock.innerHTML\r\n */\r\n content.replaceBlock = function (targetBlock, newBlock) {\r\n\r\n if (!targetBlock || !newBlock) {\r\n\r\n editor.core.log('replaceBlock: missed params');\r\n return;\r\n\r\n }\r\n\r\n /** If target-block is not a frist-level block, then we iterate parents to find it */\r\n while(!targetBlock.classList.contains(editor.ui.className.BLOCK_CLASSNAME)) {\r\n\r\n targetBlock = targetBlock.parentNode;\r\n\r\n }\r\n\r\n /** Replacing */\r\n editor.nodes.redactor.replaceChild(newBlock, targetBlock);\r\n\r\n /**\r\n * Set new node as current\r\n */\r\n editor.content.workingNodeChanged(newBlock);\r\n\r\n /**\r\n * Add block handlers\r\n */\r\n editor.ui.addBlockHandlers(newBlock);\r\n\r\n /**\r\n * Save changes\r\n */\r\n editor.ui.saveInputs();\r\n\r\n };\r\n\r\n /**\r\n * @protected\r\n *\r\n * Inserts new block to redactor\r\n * Wrapps block into a DIV with BLOCK_CLASSNAME class\r\n *\r\n * @param blockData {object}\r\n * @param blockData.block {Element} element with block content\r\n * @param blockData.type {string} block plugin\r\n * @param needPlaceCaret {bool} pass true to set caret in new block\r\n *\r\n */\r\n content.insertBlock = function ( blockData, needPlaceCaret ) {\r\n\r\n var workingBlock = editor.content.currentNode,\r\n newBlockContent = blockData.block,\r\n blockType = blockData.type,\r\n isStretched = blockData.stretched;\r\n\r\n var newBlock = composeNewBlock_(newBlockContent, blockType, isStretched);\r\n\r\n if (workingBlock) {\r\n\r\n editor.core.insertAfter(workingBlock, newBlock);\r\n\r\n } else {\r\n\r\n /**\r\n * If redactor is empty, append as first child\r\n */\r\n editor.nodes.redactor.appendChild(newBlock);\r\n\r\n }\r\n\r\n /**\r\n * Block handler\r\n */\r\n editor.ui.addBlockHandlers(newBlock);\r\n\r\n /**\r\n * Set new node as current\r\n */\r\n editor.content.workingNodeChanged(newBlock);\r\n\r\n /**\r\n * Save changes\r\n */\r\n editor.ui.saveInputs();\r\n\r\n\r\n if ( needPlaceCaret ) {\r\n\r\n /**\r\n * If we don't know input index then we set default value -1\r\n */\r\n var currentInputIndex = editor.caret.getCurrentInputIndex() || -1;\r\n\r\n\r\n if (currentInputIndex == -1) {\r\n\r\n\r\n var editableElement = newBlock.querySelector('[contenteditable]'),\r\n emptyText = document.createTextNode('');\r\n\r\n editableElement.appendChild(emptyText);\r\n editor.caret.set(editableElement, 0, 0);\r\n\r\n editor.toolbar.move();\r\n editor.toolbar.showPlusButton();\r\n\r\n\r\n } else {\r\n\r\n if (currentInputIndex === editor.state.inputs.length - 1)\r\n return;\r\n\r\n /** Timeout for browsers execution */\r\n window.setTimeout(function () {\r\n\r\n /** Setting to the new input */\r\n editor.caret.setToNextBlock(currentInputIndex);\r\n editor.toolbar.move();\r\n editor.toolbar.open();\r\n\r\n }, 10);\r\n\r\n }\r\n\r\n }\r\n\r\n /**\r\n * Block is inserted, wait for new click that defined focusing on editors area\r\n * @type {boolean}\r\n */\r\n content.editorAreaHightlighted = false;\r\n\r\n };\r\n\r\n /**\r\n * Replaces blocks with saving content\r\n * @protected\r\n * @param {Element} noteToReplace\r\n * @param {Element} newNode\r\n * @param {Element} blockType\r\n */\r\n content.switchBlock = function (blockToReplace, newBlock, tool) {\r\n\r\n tool = tool || editor.content.currentNode.dataset.tool;\r\n var newBlockComposed = composeNewBlock_(newBlock, tool);\r\n\r\n /** Replacing */\r\n editor.content.replaceBlock(blockToReplace, newBlockComposed);\r\n\r\n /** Save new Inputs when block is changed */\r\n editor.ui.saveInputs();\r\n\r\n };\r\n\r\n /**\r\n * Iterates between child noted and looking for #text node on deepest level\r\n * @protected\r\n *\r\n * @param {Element} block - node where find\r\n * @param {int} postiton - starting postion\r\n * Example: childNodex.length to find from the end\r\n * or 0 to find from the start\r\n * @return {Text} block\r\n * @uses DFS\r\n */\r\n content.getDeepestTextNodeFromPosition = function (block, position) {\r\n\r\n /**\r\n * Clear Block from empty and useless spaces with trim.\r\n * Such nodes we should remove\r\n */\r\n var blockChilds = block.childNodes,\r\n index,\r\n node,\r\n text;\r\n\r\n for(index = 0; index < blockChilds.length; index++) {\r\n\r\n node = blockChilds[index];\r\n\r\n if (node.nodeType == editor.core.nodeTypes.TEXT) {\r\n\r\n text = node.textContent.trim();\r\n\r\n /** Text is empty. We should remove this child from node before we start DFS\r\n * decrease the quantity of childs.\r\n */\r\n if (text === '') {\r\n\r\n block.removeChild(node);\r\n position--;\r\n\r\n }\r\n\r\n }\r\n\r\n }\r\n\r\n if (block.childNodes.length === 0) {\r\n\r\n return document.createTextNode('');\r\n\r\n }\r\n\r\n /** Setting default position when we deleted all empty nodes */\r\n if ( position < 0 )\r\n position = 1;\r\n\r\n var lookingFromStart = false;\r\n\r\n /** For looking from START */\r\n if (position === 0) {\r\n\r\n lookingFromStart = true;\r\n position = 1;\r\n\r\n }\r\n\r\n while ( position ) {\r\n\r\n /** initial verticle of node. */\r\n if ( lookingFromStart ) {\r\n\r\n block = block.childNodes[0];\r\n\r\n } else {\r\n\r\n block = block.childNodes[position - 1];\r\n\r\n }\r\n\r\n if ( block.nodeType == editor.core.nodeTypes.TAG ) {\r\n\r\n position = block.childNodes.length;\r\n\r\n } else if (block.nodeType == editor.core.nodeTypes.TEXT ) {\r\n\r\n position = 0;\r\n\r\n }\r\n\r\n }\r\n\r\n return block;\r\n\r\n };\r\n\r\n /**\r\n * @private\r\n * @param {Element} block - current plugins render\r\n * @param {String} tool - plugins name\r\n * @param {Boolean} isStretched - make stretched block or not\r\n *\r\n * @description adds necessary information to wrap new created block by first-level holder\r\n */\r\n var composeNewBlock_ = function (block, tool, isStretched) {\r\n\r\n var newBlock = editor.draw.node('DIV', editor.ui.className.BLOCK_CLASSNAME, {}),\r\n blockContent = editor.draw.node('DIV', editor.ui.className.BLOCK_CONTENT, {});\r\n\r\n blockContent.appendChild(block);\r\n newBlock.appendChild(blockContent);\r\n\r\n if (isStretched) {\r\n\r\n blockContent.classList.add(editor.ui.className.BLOCK_STRETCHED);\r\n\r\n }\r\n\r\n newBlock.dataset.tool = tool;\r\n return newBlock;\r\n\r\n };\r\n\r\n /**\r\n * Returns Range object of current selection\r\n * @protected\r\n */\r\n content.getRange = function () {\r\n\r\n var selection = window.getSelection().getRangeAt(0);\r\n\r\n return selection;\r\n\r\n };\r\n\r\n /**\r\n * Divides block in two blocks (after and before caret)\r\n *\r\n * @protected\r\n * @param {int} inputIndex - target input index\r\n *\r\n * @description splits current input content to the separate blocks\r\n * When enter is pressed among the words, that text will be splited.\r\n */\r\n content.splitBlock = function (inputIndex) {\r\n\r\n var selection = window.getSelection(),\r\n anchorNode = selection.anchorNode,\r\n anchorNodeText = anchorNode.textContent,\r\n caretOffset = selection.anchorOffset,\r\n textBeforeCaret,\r\n textNodeBeforeCaret,\r\n textAfterCaret,\r\n textNodeAfterCaret;\r\n\r\n var currentBlock = editor.content.currentNode.querySelector('[contentEditable]');\r\n\r\n\r\n textBeforeCaret = anchorNodeText.substring(0, caretOffset);\r\n textAfterCaret = anchorNodeText.substring(caretOffset);\r\n\r\n textNodeBeforeCaret = document.createTextNode(textBeforeCaret);\r\n\r\n if (textAfterCaret) {\r\n\r\n textNodeAfterCaret = document.createTextNode(textAfterCaret);\r\n\r\n }\r\n\r\n var previousChilds = [],\r\n nextChilds = [],\r\n reachedCurrent = false;\r\n\r\n if (textNodeAfterCaret) {\r\n\r\n nextChilds.push(textNodeAfterCaret);\r\n\r\n }\r\n\r\n for ( var i = 0, child; !!(child = currentBlock.childNodes[i]); i++) {\r\n\r\n if ( child != anchorNode ) {\r\n\r\n if ( !reachedCurrent ) {\r\n\r\n previousChilds.push(child);\r\n\r\n } else {\r\n\r\n nextChilds.push(child);\r\n\r\n }\r\n\r\n } else {\r\n\r\n reachedCurrent = true;\r\n\r\n }\r\n\r\n }\r\n\r\n /** Clear current input */\r\n editor.state.inputs[inputIndex].innerHTML = '';\r\n\r\n /**\r\n * Append all childs founded before anchorNode\r\n */\r\n var previousChildsLength = previousChilds.length;\r\n\r\n for(i = 0; i < previousChildsLength; i++) {\r\n\r\n editor.state.inputs[inputIndex].appendChild(previousChilds[i]);\r\n\r\n }\r\n\r\n editor.state.inputs[inputIndex].appendChild(textNodeBeforeCaret);\r\n\r\n /**\r\n * Append text node which is after caret\r\n */\r\n var nextChildsLength = nextChilds.length,\r\n newNode = document.createElement('div');\r\n\r\n for(i = 0; i < nextChildsLength; i++) {\r\n\r\n newNode.appendChild(nextChilds[i]);\r\n\r\n }\r\n\r\n newNode = newNode.innerHTML;\r\n\r\n /** This type of block creates when enter is pressed */\r\n var NEW_BLOCK_TYPE = editor.settings.initialBlockPlugin;\r\n\r\n /**\r\n * Make new paragraph with text after caret\r\n */\r\n editor.content.insertBlock({\r\n type : NEW_BLOCK_TYPE,\r\n block : editor.tools[NEW_BLOCK_TYPE].render({\r\n text : newNode\r\n })\r\n }, true );\r\n\r\n };\r\n\r\n /**\r\n * Merges two blocks — current and target\r\n * If target index is not exist, then previous will be as target\r\n *\r\n * @protected\r\n * @param {int} currentInputIndex\r\n * @param {int} targetInputIndex\r\n *\r\n * @description gets two inputs indexes and merges into one\r\n */\r\n content.mergeBlocks = function (currentInputIndex, targetInputIndex) {\r\n\r\n /** If current input index is zero, then prevent method execution */\r\n if (currentInputIndex === 0) {\r\n\r\n return;\r\n\r\n }\r\n\r\n var targetInput,\r\n currentInputContent = editor.state.inputs[currentInputIndex].innerHTML;\r\n\r\n if (!targetInputIndex) {\r\n\r\n targetInput = editor.state.inputs[currentInputIndex - 1];\r\n\r\n } else {\r\n\r\n targetInput = editor.state.inputs[targetInputIndex];\r\n\r\n }\r\n\r\n targetInput.innerHTML += currentInputContent;\r\n\r\n };\r\n\r\n /**\r\n * Iterates all right siblings and parents, which has right siblings\r\n * while it does not reached the first-level block\r\n *\r\n * @param {Element} node\r\n * @return {boolean}\r\n */\r\n content.isLastNode = function (node) {\r\n\r\n // console.log('погнали перебор родителей');\r\n\r\n var allChecked = false;\r\n\r\n while ( !allChecked ) {\r\n\r\n // console.log('Смотрим на %o', node);\r\n // console.log('Проверим, пустые ли соседи справа');\r\n\r\n if ( !allSiblingsEmpty_(node) ) {\r\n\r\n // console.log('Есть непустые соседи. Узел не последний. Выходим.');\r\n return false;\r\n\r\n }\r\n\r\n node = node.parentNode;\r\n\r\n /**\r\n * Проверяем родителей до тех пор, пока не найдем блок первого уровня\r\n */\r\n if ( node.classList.contains(editor.ui.className.BLOCK_CONTENT) ) {\r\n\r\n allChecked = true;\r\n\r\n }\r\n\r\n }\r\n\r\n return true;\r\n\r\n };\r\n\r\n /**\r\n * Checks if all element right siblings is empty\r\n * @param node\r\n */\r\n var allSiblingsEmpty_ = function (node) {\r\n\r\n /**\r\n * Нужно убедиться, что после пустого соседа ничего нет\r\n */\r\n var sibling = node.nextSibling;\r\n\r\n while ( sibling ) {\r\n\r\n if (sibling.textContent.length) {\r\n\r\n return false;\r\n\r\n }\r\n\r\n sibling = sibling.nextSibling;\r\n\r\n }\r\n\r\n return true;\r\n\r\n };\r\n\r\n /**\r\n * @public\r\n *\r\n * @param {string} htmlData - html content as string\r\n * @param {string} plainData - plain text\r\n * @return {string} - html content as string\r\n */\r\n content.wrapTextWithParagraphs = function (htmlData, plainData) {\r\n\r\n if (!htmlData) {\r\n\r\n return '

    ' + plainData.split('\\n\\n').join('

    ') + '

    ';\r\n\r\n }\r\n\r\n var wrapper = document.createElement('DIV'),\r\n newWrapper = document.createElement('DIV'),\r\n i,\r\n paragraph,\r\n firstLevelBlocks = ['DIV', 'P'],\r\n blockTyped,\r\n node;\r\n\r\n /**\r\n * Make HTML Element to Wrap Text\r\n * It allows us to work with input data as HTML content\r\n */\r\n wrapper.innerHTML = htmlData;\r\n paragraph = document.createElement('P');\r\n\r\n for (i = 0; i < wrapper.childNodes.length; i++) {\r\n\r\n node = wrapper.childNodes[i];\r\n\r\n blockTyped = firstLevelBlocks.indexOf(node.tagName) != -1;\r\n\r\n /**\r\n * If node is first-levet\r\n * we add this node to our new wrapper\r\n */\r\n if ( blockTyped ) {\r\n\r\n /**\r\n * If we had splitted inline nodes to paragraph before\r\n */\r\n if ( paragraph.childNodes.length ) {\r\n\r\n newWrapper.appendChild(paragraph.cloneNode(true));\r\n\r\n /** empty paragraph */\r\n paragraph = null;\r\n paragraph = document.createElement('P');\r\n\r\n }\r\n\r\n newWrapper.appendChild(node.cloneNode(true));\r\n\r\n } else {\r\n\r\n /** Collect all inline nodes to one as paragraph */\r\n paragraph.appendChild(node.cloneNode(true));\r\n\r\n /** if node is last we should append this node to paragraph and paragraph to new wrapper */\r\n if ( i == wrapper.childNodes.length - 1 ) {\r\n\r\n newWrapper.appendChild(paragraph.cloneNode(true));\r\n\r\n }\r\n\r\n }\r\n\r\n }\r\n\r\n return newWrapper.innerHTML;\r\n\r\n };\r\n\r\n /**\r\n * Finds closest Contenteditable parent from Element\r\n * @param {Element} node element looking from\r\n * @return {Element} node contenteditable\r\n */\r\n content.getEditableParent = function (node) {\r\n\r\n while (node && node.contentEditable != 'true') {\r\n\r\n node = node.parentNode;\r\n\r\n }\r\n\r\n return node;\r\n\r\n };\r\n\r\n return content;\r\n\r\n})({});\n\n\n// WEBPACK FOOTER //\n// ./modules/content.js","/**\r\n * Codex Editor toolbar module\r\n *\r\n * Contains:\r\n * - Inline toolbox\r\n * - Toolbox within plus button\r\n * - Settings section\r\n *\r\n * @author Codex Team\r\n * @version 1.0\r\n */\r\n\r\nmodule.exports = (function (toolbar) {\r\n\r\n let editor = codex.editor;\r\n\r\n toolbar.settings = require('./settings');\r\n toolbar.inline = require('./inline');\r\n toolbar.toolbox = require('./toolbox');\r\n\r\n /**\r\n * Margin between focused node and toolbar\r\n */\r\n toolbar.defaultToolbarHeight = 49;\r\n\r\n toolbar.defaultOffset = 34;\r\n\r\n toolbar.opened = false;\r\n\r\n toolbar.current = null;\r\n\r\n /**\r\n * @protected\r\n */\r\n toolbar.open = function () {\r\n\r\n if (editor.hideToolbar) {\r\n\r\n return;\r\n\r\n }\r\n\r\n editor.nodes.toolbar.classList.add('opened');\r\n this.opened = true;\r\n\r\n };\r\n\r\n /**\r\n * @protected\r\n */\r\n toolbar.close = function () {\r\n\r\n editor.nodes.toolbar.classList.remove('opened');\r\n\r\n toolbar.opened = false;\r\n toolbar.current = null;\r\n\r\n for (var button in editor.nodes.toolbarButtons) {\r\n\r\n editor.nodes.toolbarButtons[button].classList.remove('selected');\r\n\r\n }\r\n\r\n /** Close toolbox when toolbar is not displayed */\r\n editor.toolbar.toolbox.close();\r\n editor.toolbar.settings.close();\r\n\r\n };\r\n\r\n toolbar.toggle = function () {\r\n\r\n if ( !this.opened ) {\r\n\r\n this.open();\r\n\r\n } else {\r\n\r\n this.close();\r\n\r\n }\r\n\r\n };\r\n\r\n toolbar.hidePlusButton = function () {\r\n\r\n editor.nodes.plusButton.classList.add('hide');\r\n\r\n };\r\n\r\n toolbar.showPlusButton = function () {\r\n\r\n editor.nodes.plusButton.classList.remove('hide');\r\n\r\n };\r\n\r\n /**\r\n * Moving toolbar to the specified node\r\n */\r\n toolbar.move = function () {\r\n\r\n /** Close Toolbox when we move toolbar */\r\n editor.toolbar.toolbox.close();\r\n\r\n if (!editor.content.currentNode) {\r\n\r\n return;\r\n\r\n }\r\n\r\n var newYCoordinate = editor.content.currentNode.offsetTop - (editor.toolbar.defaultToolbarHeight / 2) + editor.toolbar.defaultOffset;\r\n\r\n editor.nodes.toolbar.style.transform = `translate3D(0, ${Math.floor(newYCoordinate)}px, 0)`;\r\n\r\n /** Close trash actions */\r\n editor.toolbar.settings.hideRemoveActions();\r\n\r\n };\r\n\r\n return toolbar;\r\n\r\n})({});\n\n\n// WEBPACK FOOTER //\n// ./modules/toolbar/toolbar.js","/**\r\n * Toolbar settings\r\n *\r\n * @version 1.0.4\r\n */\r\n\r\nmodule.exports = (function (settings) {\r\n\r\n let editor = codex.editor;\r\n\r\n settings.opened = false;\r\n\r\n settings.setting = null;\r\n settings.actions = null;\r\n\r\n /**\r\n * Append and open settings\r\n */\r\n settings.open = function (toolType) {\r\n\r\n /**\r\n * Append settings content\r\n * It's stored in tool.settings\r\n */\r\n if (!editor.tools[toolType] || !editor.tools[toolType].makeSettings ) {\r\n\r\n editor.core.log(`Plugin «${toolType}» has no settings`, 'warn');\r\n editor.nodes.pluginSettings.innerHTML = `Плагин «${toolType}» не имеет настроек`;\r\n\r\n } else {\r\n\r\n /**\r\n * Draw settings block\r\n */\r\n var settingsBlock = editor.tools[toolType].makeSettings();\r\n\r\n editor.nodes.pluginSettings.appendChild(settingsBlock);\r\n\r\n }\r\n\r\n /** Open settings block */\r\n editor.nodes.blockSettings.classList.add('opened');\r\n this.opened = true;\r\n\r\n };\r\n\r\n /**\r\n * Close and clear settings\r\n */\r\n settings.close = function () {\r\n\r\n editor.nodes.blockSettings.classList.remove('opened');\r\n editor.nodes.pluginSettings.innerHTML = '';\r\n\r\n this.opened = false;\r\n\r\n };\r\n\r\n /**\r\n * @param {string} toolType - plugin type\r\n */\r\n settings.toggle = function ( toolType ) {\r\n\r\n if ( !this.opened ) {\r\n\r\n this.open(toolType);\r\n\r\n } else {\r\n\r\n this.close();\r\n\r\n }\r\n\r\n };\r\n\r\n /**\r\n * Here we will draw buttons and add listeners to components\r\n */\r\n settings.makeRemoveBlockButton = function () {\r\n\r\n var removeBlockWrapper = editor.draw.node('SPAN', 'ce-toolbar__remove-btn', {}),\r\n settingButton = editor.draw.node('SPAN', 'ce-toolbar__remove-setting', { innerHTML : '' }),\r\n actionWrapper = editor.draw.node('DIV', 'ce-toolbar__remove-confirmation', {}),\r\n confirmAction = editor.draw.node('DIV', 'ce-toolbar__remove-confirm', { textContent : 'Удалить блок' }),\r\n cancelAction = editor.draw.node('DIV', 'ce-toolbar__remove-cancel', { textContent : 'Отмена' });\r\n\r\n editor.listeners.add(settingButton, 'click', editor.toolbar.settings.removeButtonClicked, false);\r\n\r\n editor.listeners.add(confirmAction, 'click', editor.toolbar.settings.confirmRemovingRequest, false);\r\n\r\n editor.listeners.add(cancelAction, 'click', editor.toolbar.settings.cancelRemovingRequest, false);\r\n\r\n actionWrapper.appendChild(confirmAction);\r\n actionWrapper.appendChild(cancelAction);\r\n\r\n removeBlockWrapper.appendChild(settingButton);\r\n removeBlockWrapper.appendChild(actionWrapper);\r\n\r\n /** Save setting */\r\n editor.toolbar.settings.setting = settingButton;\r\n editor.toolbar.settings.actions = actionWrapper;\r\n\r\n return removeBlockWrapper;\r\n\r\n };\r\n\r\n settings.removeButtonClicked = function () {\r\n\r\n var action = editor.toolbar.settings.actions;\r\n\r\n if (action.classList.contains('opened')) {\r\n\r\n editor.toolbar.settings.hideRemoveActions();\r\n\r\n } else {\r\n\r\n editor.toolbar.settings.showRemoveActions();\r\n\r\n }\r\n\r\n editor.toolbar.toolbox.close();\r\n editor.toolbar.settings.close();\r\n\r\n };\r\n\r\n settings.cancelRemovingRequest = function () {\r\n\r\n editor.toolbar.settings.actions.classList.remove('opened');\r\n\r\n };\r\n\r\n settings.confirmRemovingRequest = function () {\r\n\r\n var currentBlock = editor.content.currentNode,\r\n firstLevelBlocksCount;\r\n\r\n currentBlock.remove();\r\n\r\n firstLevelBlocksCount = editor.nodes.redactor.childNodes.length;\r\n\r\n /**\r\n * If all blocks are removed\r\n */\r\n if (firstLevelBlocksCount === 0) {\r\n\r\n /** update currentNode variable */\r\n editor.content.currentNode = null;\r\n\r\n /** Inserting new empty initial block */\r\n editor.ui.addInitialBlock();\r\n\r\n }\r\n\r\n editor.ui.saveInputs();\r\n\r\n editor.toolbar.close();\r\n\r\n };\r\n\r\n settings.showRemoveActions = function () {\r\n\r\n editor.toolbar.settings.actions.classList.add('opened');\r\n\r\n };\r\n\r\n settings.hideRemoveActions = function () {\r\n\r\n editor.toolbar.settings.actions.classList.remove('opened');\r\n\r\n };\r\n\r\n return settings;\r\n\r\n})({});\n\n\n// WEBPACK FOOTER //\n// ./modules/toolbar/settings.js","/**\r\n * Inline toolbar\r\n *\r\n * Contains from tools:\r\n * Bold, Italic, Underline and Anchor\r\n *\r\n * @author Codex Team\r\n * @version 1.0\r\n */\r\n\r\nmodule.exports = (function (inline) {\r\n\r\n let editor = codex.editor;\r\n\r\n inline.buttonsOpened = null;\r\n inline.actionsOpened = null;\r\n inline.wrappersOffset = null;\r\n\r\n /**\r\n * saving selection that need for execCommand for styling\r\n *\r\n */\r\n inline.storedSelection = null;\r\n\r\n /**\r\n * @protected\r\n *\r\n * Open inline toobar\r\n */\r\n inline.show = function () {\r\n\r\n var currentNode = editor.content.currentNode,\r\n tool = currentNode.dataset.tool,\r\n plugin;\r\n\r\n /**\r\n * tool allowed to open inline toolbar\r\n */\r\n plugin = editor.tools[tool];\r\n\r\n if (!plugin.showInlineToolbar)\r\n return;\r\n\r\n var selectedText = inline.getSelectionText(),\r\n toolbar = editor.nodes.inlineToolbar.wrapper;\r\n\r\n if (selectedText.length > 0) {\r\n\r\n /** Move toolbar and open */\r\n editor.toolbar.inline.move();\r\n\r\n /** Open inline toolbar */\r\n toolbar.classList.add('opened');\r\n\r\n /** show buttons of inline toolbar */\r\n editor.toolbar.inline.showButtons();\r\n\r\n }\r\n\r\n };\r\n\r\n /**\r\n * @protected\r\n *\r\n * Closes inline toolbar\r\n */\r\n inline.close = function () {\r\n\r\n var toolbar = editor.nodes.inlineToolbar.wrapper;\r\n\r\n toolbar.classList.remove('opened');\r\n\r\n };\r\n\r\n /**\r\n * @private\r\n *\r\n * Moving toolbar\r\n */\r\n inline.move = function () {\r\n\r\n if (!this.wrappersOffset) {\r\n\r\n this.wrappersOffset = this.getWrappersOffset();\r\n\r\n }\r\n\r\n var coords = this.getSelectionCoords(),\r\n defaultOffset = 0,\r\n toolbar = editor.nodes.inlineToolbar.wrapper,\r\n newCoordinateX,\r\n newCoordinateY;\r\n\r\n if (toolbar.offsetHeight === 0) {\r\n\r\n defaultOffset = 40;\r\n\r\n }\r\n\r\n newCoordinateX = coords.x - this.wrappersOffset.left;\r\n newCoordinateY = coords.y + window.scrollY - this.wrappersOffset.top - defaultOffset - toolbar.offsetHeight;\r\n\r\n toolbar.style.transform = `translate3D(${Math.floor(newCoordinateX)}px, ${Math.floor(newCoordinateY)}px, 0)`;\r\n\r\n /** Close everything */\r\n editor.toolbar.inline.closeButtons();\r\n editor.toolbar.inline.closeAction();\r\n\r\n };\r\n\r\n /**\r\n * @private\r\n *\r\n * Tool Clicked\r\n */\r\n\r\n inline.toolClicked = function (event, type) {\r\n\r\n /**\r\n * For simple tools we use default browser function\r\n * For more complicated tools, we should write our own behavior\r\n */\r\n switch (type) {\r\n case 'createLink' : editor.toolbar.inline.createLinkAction(event, type); break;\r\n default : editor.toolbar.inline.defaultToolAction(type); break;\r\n }\r\n\r\n /**\r\n * highlight buttons\r\n * after making some action\r\n */\r\n editor.nodes.inlineToolbar.buttons.childNodes.forEach(editor.toolbar.inline.hightlight);\r\n\r\n };\r\n\r\n /**\r\n * @private\r\n *\r\n * Saving wrappers offset in DOM\r\n */\r\n inline.getWrappersOffset = function () {\r\n\r\n var wrapper = editor.nodes.wrapper,\r\n offset = this.getOffset(wrapper);\r\n\r\n this.wrappersOffset = offset;\r\n return offset;\r\n\r\n };\r\n\r\n /**\r\n * @private\r\n *\r\n * Calculates offset of DOM element\r\n *\r\n * @param el\r\n * @returns {{top: number, left: number}}\r\n */\r\n inline.getOffset = function ( el ) {\r\n\r\n var _x = 0;\r\n var _y = 0;\r\n\r\n while( el && !isNaN( el.offsetLeft ) && !isNaN( el.offsetTop ) ) {\r\n\r\n _x += (el.offsetLeft + el.clientLeft);\r\n _y += (el.offsetTop + el.clientTop);\r\n el = el.offsetParent;\r\n\r\n }\r\n return { top: _y, left: _x };\r\n\r\n };\r\n\r\n /**\r\n * @private\r\n *\r\n * Calculates position of selected text\r\n * @returns {{x: number, y: number}}\r\n */\r\n inline.getSelectionCoords = function () {\r\n\r\n var sel = document.selection, range;\r\n var x = 0, y = 0;\r\n\r\n if (sel) {\r\n\r\n if (sel.type != 'Control') {\r\n\r\n range = sel.createRange();\r\n range.collapse(true);\r\n x = range.boundingLeft;\r\n y = range.boundingTop;\r\n\r\n }\r\n\r\n } else if (window.getSelection) {\r\n\r\n sel = window.getSelection();\r\n\r\n if (sel.rangeCount) {\r\n\r\n range = sel.getRangeAt(0).cloneRange();\r\n if (range.getClientRects) {\r\n\r\n range.collapse(true);\r\n var rect = range.getClientRects()[0];\r\n\r\n if (!rect) {\r\n\r\n return;\r\n\r\n }\r\n\r\n x = rect.left;\r\n y = rect.top;\r\n\r\n }\r\n\r\n }\r\n\r\n }\r\n return { x: x, y: y };\r\n\r\n };\r\n\r\n /**\r\n * @private\r\n *\r\n * Returns selected text as String\r\n * @returns {string}\r\n */\r\n inline.getSelectionText = function () {\r\n\r\n var selectedText = '';\r\n\r\n // all modern browsers and IE9+\r\n if (window.getSelection) {\r\n\r\n selectedText = window.getSelection().toString();\r\n\r\n }\r\n\r\n return selectedText;\r\n\r\n };\r\n\r\n /** Opens buttons block */\r\n inline.showButtons = function () {\r\n\r\n var buttons = editor.nodes.inlineToolbar.buttons;\r\n\r\n buttons.classList.add('opened');\r\n\r\n editor.toolbar.inline.buttonsOpened = true;\r\n\r\n /** highlight buttons */\r\n editor.nodes.inlineToolbar.buttons.childNodes.forEach(editor.toolbar.inline.hightlight);\r\n\r\n };\r\n\r\n /** Makes buttons disappear */\r\n inline.closeButtons = function () {\r\n\r\n var buttons = editor.nodes.inlineToolbar.buttons;\r\n\r\n buttons.classList.remove('opened');\r\n\r\n editor.toolbar.inline.buttonsOpened = false;\r\n\r\n };\r\n\r\n /** Open buttons defined action if exist */\r\n inline.showActions = function () {\r\n\r\n var action = editor.nodes.inlineToolbar.actions;\r\n\r\n action.classList.add('opened');\r\n\r\n editor.toolbar.inline.actionsOpened = true;\r\n\r\n };\r\n\r\n /** Close actions block */\r\n inline.closeAction = function () {\r\n\r\n var action = editor.nodes.inlineToolbar.actions;\r\n\r\n action.innerHTML = '';\r\n action.classList.remove('opened');\r\n editor.toolbar.inline.actionsOpened = false;\r\n\r\n };\r\n\r\n\r\n /**\r\n * Callback for keydowns in inline toolbar \"Insert link...\" input\r\n */\r\n let inlineToolbarAnchorInputKeydown_ = function (event) {\r\n\r\n if (event.keyCode != editor.core.keys.ENTER) {\r\n\r\n return;\r\n\r\n }\r\n\r\n let editable = editor.content.currentNode,\r\n storedSelection = editor.toolbar.inline.storedSelection;\r\n\r\n editor.toolbar.inline.restoreSelection(editable, storedSelection);\r\n editor.toolbar.inline.setAnchor(this.value);\r\n\r\n /**\r\n * Preventing events that will be able to happen\r\n */\r\n event.preventDefault();\r\n event.stopImmediatePropagation();\r\n\r\n editor.toolbar.inline.clearRange();\r\n\r\n };\r\n\r\n /** Action for link creation or for setting anchor */\r\n inline.createLinkAction = function (event) {\r\n\r\n var isActive = this.isLinkActive();\r\n\r\n var editable = editor.content.currentNode,\r\n storedSelection = editor.toolbar.inline.saveSelection(editable);\r\n\r\n /** Save globally selection */\r\n editor.toolbar.inline.storedSelection = storedSelection;\r\n\r\n if (isActive) {\r\n\r\n\r\n /**\r\n * Changing stored selection. if we want to remove anchor from word\r\n * we should remove anchor from whole word, not only selected part.\r\n * The solution is than we get the length of current link\r\n * Change start position to - end of selection minus length of anchor\r\n */\r\n editor.toolbar.inline.restoreSelection(editable, storedSelection);\r\n\r\n editor.toolbar.inline.defaultToolAction('unlink');\r\n\r\n } else {\r\n\r\n /** Create input and close buttons */\r\n var action = editor.draw.inputForLink();\r\n\r\n editor.nodes.inlineToolbar.actions.appendChild(action);\r\n\r\n editor.toolbar.inline.closeButtons();\r\n editor.toolbar.inline.showActions();\r\n\r\n /**\r\n * focus to input\r\n * Solution: https://developer.mozilla.org/ru/docs/Web/API/HTMLElement/focus\r\n * Prevents event after showing input and when we need to focus an input which is in unexisted form\r\n */\r\n action.focus();\r\n event.preventDefault();\r\n\r\n /** Callback to link action */\r\n editor.listeners.add(action, 'keydown', inlineToolbarAnchorInputKeydown_, false);\r\n\r\n }\r\n\r\n };\r\n\r\n inline.isLinkActive = function () {\r\n\r\n var isActive = false;\r\n\r\n editor.nodes.inlineToolbar.buttons.childNodes.forEach(function (tool) {\r\n\r\n var dataType = tool.dataset.type;\r\n\r\n if (dataType == 'link' && tool.classList.contains('hightlighted')) {\r\n\r\n isActive = true;\r\n\r\n }\r\n\r\n });\r\n\r\n return isActive;\r\n\r\n };\r\n\r\n /** default action behavior of tool */\r\n inline.defaultToolAction = function (type) {\r\n\r\n document.execCommand(type, false, null);\r\n\r\n };\r\n\r\n /**\r\n * @private\r\n *\r\n * Sets URL\r\n *\r\n * @param {String} url - URL\r\n */\r\n inline.setAnchor = function (url) {\r\n\r\n document.execCommand('createLink', false, url);\r\n\r\n /** Close after URL inserting */\r\n editor.toolbar.inline.closeAction();\r\n\r\n };\r\n\r\n /**\r\n * @private\r\n *\r\n * Saves selection\r\n */\r\n inline.saveSelection = function (containerEl) {\r\n\r\n var range = window.getSelection().getRangeAt(0),\r\n preSelectionRange = range.cloneRange(),\r\n start;\r\n\r\n preSelectionRange.selectNodeContents(containerEl);\r\n preSelectionRange.setEnd(range.startContainer, range.startOffset);\r\n\r\n start = preSelectionRange.toString().length;\r\n\r\n return {\r\n start: start,\r\n end: start + range.toString().length\r\n };\r\n\r\n };\r\n\r\n /**\r\n * @private\r\n *\r\n * Sets to previous selection (Range)\r\n *\r\n * @param {Element} containerEl - editable element where we restore range\r\n * @param {Object} savedSel - range basic information to restore\r\n */\r\n inline.restoreSelection = function (containerEl, savedSel) {\r\n\r\n var range = document.createRange(),\r\n charIndex = 0;\r\n\r\n range.setStart(containerEl, 0);\r\n range.collapse(true);\r\n\r\n var nodeStack = [ containerEl ],\r\n node,\r\n foundStart = false,\r\n stop = false,\r\n nextCharIndex;\r\n\r\n while (!stop && (node = nodeStack.pop())) {\r\n\r\n if (node.nodeType == 3) {\r\n\r\n nextCharIndex = charIndex + node.length;\r\n\r\n if (!foundStart && savedSel.start >= charIndex && savedSel.start <= nextCharIndex) {\r\n\r\n range.setStart(node, savedSel.start - charIndex);\r\n foundStart = true;\r\n\r\n }\r\n if (foundStart && savedSel.end >= charIndex && savedSel.end <= nextCharIndex) {\r\n\r\n range.setEnd(node, savedSel.end - charIndex);\r\n stop = true;\r\n\r\n }\r\n charIndex = nextCharIndex;\r\n\r\n } else {\r\n\r\n var i = node.childNodes.length;\r\n\r\n while (i--) {\r\n\r\n nodeStack.push(node.childNodes[i]);\r\n\r\n }\r\n\r\n }\r\n\r\n }\r\n\r\n var sel = window.getSelection();\r\n\r\n sel.removeAllRanges();\r\n sel.addRange(range);\r\n\r\n };\r\n\r\n /**\r\n * @private\r\n *\r\n * Removes all ranges from window selection\r\n */\r\n inline.clearRange = function () {\r\n\r\n var selection = window.getSelection();\r\n\r\n selection.removeAllRanges();\r\n\r\n };\r\n\r\n /**\r\n * @private\r\n *\r\n * sets or removes hightlight\r\n */\r\n inline.hightlight = function (tool) {\r\n\r\n var dataType = tool.dataset.type;\r\n\r\n if (document.queryCommandState(dataType)) {\r\n\r\n editor.toolbar.inline.setButtonHighlighted(tool);\r\n\r\n } else {\r\n\r\n editor.toolbar.inline.removeButtonsHighLight(tool);\r\n\r\n }\r\n\r\n /**\r\n *\r\n * hightlight for anchors\r\n */\r\n var selection = window.getSelection(),\r\n tag = selection.anchorNode.parentNode;\r\n\r\n if (tag.tagName == 'A' && dataType == 'link') {\r\n\r\n editor.toolbar.inline.setButtonHighlighted(tool);\r\n\r\n }\r\n\r\n };\r\n\r\n /**\r\n * @private\r\n *\r\n * Mark button if text is already executed\r\n */\r\n inline.setButtonHighlighted = function (button) {\r\n\r\n button.classList.add('hightlighted');\r\n\r\n /** At link tool we also change icon */\r\n if (button.dataset.type == 'link') {\r\n\r\n var icon = button.childNodes[0];\r\n\r\n icon.classList.remove('ce-icon-link');\r\n icon.classList.add('ce-icon-unlink');\r\n\r\n }\r\n\r\n };\r\n\r\n /**\r\n * @private\r\n *\r\n * Removes hightlight\r\n */\r\n inline.removeButtonsHighLight = function (button) {\r\n\r\n button.classList.remove('hightlighted');\r\n\r\n /** At link tool we also change icon */\r\n if (button.dataset.type == 'link') {\r\n\r\n var icon = button.childNodes[0];\r\n\r\n icon.classList.remove('ce-icon-unlink');\r\n icon.classList.add('ce-icon-link');\r\n\r\n }\r\n\r\n };\r\n\r\n\r\n return inline;\r\n\r\n})({});\n\n\n// WEBPACK FOOTER //\n// ./modules/toolbar/inline.js","/**\r\n * Codex Editor toolbox\r\n *\r\n * All tools be able to appended here\r\n *\r\n * @author Codex Team\r\n * @version 1.0\r\n */\r\n\r\nmodule.exports = (function (toolbox) {\r\n\r\n let editor = codex.editor;\r\n\r\n toolbox.opened = false;\r\n\r\n /** Shows toolbox */\r\n toolbox.open = function () {\r\n\r\n /** Close setting if toolbox is opened */\r\n if (editor.toolbar.settings.opened) {\r\n\r\n editor.toolbar.settings.close();\r\n\r\n }\r\n\r\n /** display toolbox */\r\n editor.nodes.toolbox.classList.add('opened');\r\n\r\n /** Animate plus button */\r\n editor.nodes.plusButton.classList.add('clicked');\r\n\r\n /** toolbox state */\r\n editor.toolbar.toolbox.opened = true;\r\n\r\n };\r\n\r\n /** Closes toolbox */\r\n toolbox.close = function () {\r\n\r\n /** Makes toolbox disapear */\r\n editor.nodes.toolbox.classList.remove('opened');\r\n\r\n /** Rotate plus button */\r\n editor.nodes.plusButton.classList.remove('clicked');\r\n\r\n /** toolbox state */\r\n editor.toolbar.toolbox.opened = false;\r\n\r\n editor.toolbar.current = null;\r\n\r\n };\r\n\r\n toolbox.leaf = function () {\r\n\r\n let currentTool = editor.toolbar.current,\r\n tools = Object.keys(editor.tools),\r\n barButtons = editor.nodes.toolbarButtons,\r\n nextToolIndex = 0,\r\n toolToSelect,\r\n visibleTool,\r\n tool;\r\n\r\n if ( !currentTool ) {\r\n\r\n /** Get first tool from object*/\r\n for(tool in editor.tools) {\r\n\r\n if (editor.tools[tool].displayInToolbox) {\r\n\r\n break;\r\n\r\n }\r\n\r\n nextToolIndex ++;\r\n\r\n }\r\n\r\n } else {\r\n\r\n nextToolIndex = (tools.indexOf(currentTool) + 1) % tools.length;\r\n visibleTool = tools[nextToolIndex];\r\n\r\n while (!editor.tools[visibleTool].displayInToolbox) {\r\n\r\n nextToolIndex = (nextToolIndex + 1) % tools.length;\r\n visibleTool = tools[nextToolIndex];\r\n\r\n }\r\n\r\n }\r\n\r\n toolToSelect = tools[nextToolIndex];\r\n\r\n for ( var button in barButtons ) {\r\n\r\n barButtons[button].classList.remove('selected');\r\n\r\n }\r\n\r\n barButtons[toolToSelect].classList.add('selected');\r\n editor.toolbar.current = toolToSelect;\r\n\r\n };\r\n\r\n /**\r\n * Transforming selected node type into selected toolbar element type\r\n * @param {event} event\r\n */\r\n toolbox.toolClicked = function (event) {\r\n\r\n /**\r\n * UNREPLACEBLE_TOOLS this types of tools are forbidden to replace even they are empty\r\n */\r\n var UNREPLACEBLE_TOOLS = ['image', 'link', 'list', 'instagram', 'twitter', 'embed'],\r\n tool = editor.tools[editor.toolbar.current],\r\n workingNode = editor.content.currentNode,\r\n currentInputIndex = editor.caret.inputIndex,\r\n newBlockContent,\r\n appendCallback,\r\n blockData;\r\n\r\n /** Make block from plugin */\r\n newBlockContent = tool.render();\r\n\r\n /** information about block */\r\n blockData = {\r\n block : newBlockContent,\r\n type : tool.type,\r\n stretched : false\r\n };\r\n\r\n if (\r\n workingNode &&\r\n UNREPLACEBLE_TOOLS.indexOf(workingNode.dataset.tool) === -1 &&\r\n workingNode.textContent.trim() === ''\r\n ) {\r\n\r\n /** Replace current block */\r\n editor.content.switchBlock(workingNode, newBlockContent, tool.type);\r\n\r\n } else {\r\n\r\n /** Insert new Block from plugin */\r\n editor.content.insertBlock(blockData);\r\n\r\n /** increase input index */\r\n currentInputIndex++;\r\n\r\n }\r\n\r\n /** Fire tool append callback */\r\n appendCallback = tool.appendCallback;\r\n\r\n if (appendCallback && typeof appendCallback == 'function') {\r\n\r\n appendCallback.call(event);\r\n\r\n }\r\n\r\n window.setTimeout(function () {\r\n\r\n /** Set caret to current block */\r\n editor.caret.setToBlock(currentInputIndex);\r\n\r\n }, 10);\r\n\r\n\r\n /**\r\n * Changing current Node\r\n */\r\n editor.content.workingNodeChanged();\r\n\r\n /**\r\n * Move toolbar when node is changed\r\n */\r\n editor.toolbar.move();\r\n\r\n };\r\n\r\n return toolbox;\r\n\r\n})({});\n\n\n// WEBPACK FOOTER //\n// ./modules/toolbar/toolbox.js","/**\r\n * @module Codex Editor Callbacks module\r\n * @description Module works with editor added Elements\r\n *\r\n * @author Codex Team\r\n * @version 1.4.0\r\n */\r\n\r\nmodule.exports = (function (callbacks) {\r\n\r\n let editor = codex.editor;\r\n\r\n /**\r\n * used by UI module\r\n * @description Routes all keydowns on document\r\n * @param {Object} event\r\n */\r\n callbacks.globalKeydown = function (event) {\r\n\r\n switch (event.keyCode) {\r\n case editor.core.keys.ENTER : enterKeyPressed_(event); break;\r\n }\r\n\r\n };\r\n\r\n /**\r\n * used by UI module\r\n * @description Routes all keydowns on redactors area\r\n * @param {Object} event\r\n */\r\n callbacks.redactorKeyDown = function (event) {\r\n\r\n switch (event.keyCode) {\r\n case editor.core.keys.TAB : tabKeyPressedOnRedactorsZone_(event); break;\r\n case editor.core.keys.ENTER : enterKeyPressedOnRedactorsZone_(event); break;\r\n case editor.core.keys.ESC : escapeKeyPressedOnRedactorsZone_(event); break;\r\n default : defaultKeyPressedOnRedactorsZone_(event); break;\r\n }\r\n\r\n };\r\n\r\n /**\r\n * used by UI module\r\n * @description Routes all keyup events\r\n * @param {Object} event\r\n */\r\n callbacks.globalKeyup = function (event) {\r\n\r\n switch (event.keyCode) {\r\n case editor.core.keys.UP :\r\n case editor.core.keys.LEFT :\r\n case editor.core.keys.RIGHT :\r\n case editor.core.keys.DOWN : arrowKeyPressed_(event); break;\r\n }\r\n\r\n };\r\n\r\n /**\r\n * @param {Object} event\r\n * @private\r\n *\r\n * Handles behaviour when tab pressed\r\n * @description if Content is empty show toolbox (if it is closed) or leaf tools\r\n * uses Toolbars toolbox module to handle the situation\r\n */\r\n var tabKeyPressedOnRedactorsZone_ = function (event) {\r\n\r\n /**\r\n * Wait for solution. Would like to know the behaviour\r\n * @todo Add spaces\r\n */\r\n event.preventDefault();\r\n\r\n var nativeInputs = editor.content.currentNode.querySelectorAll('textarea, input'),\r\n nativeInputsAreEmpty = true,\r\n textContentIsEmpty = !editor.content.currentNode.textContent.trim();\r\n\r\n Array.prototype.map.call(nativeInputs, function (input) {\r\n\r\n if (input.type == 'textarea' || input.type == 'text') {\r\n\r\n nativeInputsAreEmpty = nativeInputsAreEmpty && !input.value.trim();\r\n\r\n }\r\n\r\n });\r\n\r\n var blockIsEmpty = textContentIsEmpty && nativeInputsAreEmpty;\r\n\r\n if (!blockIsEmpty) {\r\n\r\n return;\r\n\r\n }\r\n\r\n if ( !editor.toolbar.opened ) {\r\n\r\n editor.toolbar.open();\r\n\r\n }\r\n\r\n if (editor.toolbar.opened && !editor.toolbar.toolbox.opened) {\r\n\r\n editor.toolbar.toolbox.open();\r\n\r\n } else {\r\n\r\n editor.toolbar.toolbox.leaf();\r\n\r\n }\r\n\r\n };\r\n\r\n /**\r\n * Handles global EnterKey Press\r\n * @see enterPressedOnBlock_\r\n * @param {Object} event\r\n */\r\n var enterKeyPressed_ = function () {\r\n\r\n if (editor.content.editorAreaHightlighted) {\r\n\r\n /**\r\n * it means that we lose input index, saved index before is not correct\r\n * therefore we need to set caret when we insert new block\r\n */\r\n editor.caret.inputIndex = -1;\r\n\r\n enterPressedOnBlock_();\r\n\r\n }\r\n\r\n };\r\n\r\n /**\r\n * Callback for enter key pressing in first-level block area\r\n *\r\n * @param {Event} event\r\n * @private\r\n *\r\n * @description Inserts new block with initial type from settings\r\n */\r\n var enterPressedOnBlock_ = function () {\r\n\r\n var NEW_BLOCK_TYPE = editor.settings.initialBlockPlugin;\r\n\r\n editor.content.insertBlock({\r\n type : NEW_BLOCK_TYPE,\r\n block : editor.tools[NEW_BLOCK_TYPE].render()\r\n }, true );\r\n\r\n editor.toolbar.move();\r\n editor.toolbar.open();\r\n\r\n };\r\n\r\n\r\n /**\r\n * ENTER key handler\r\n *\r\n * @param {Object} event\r\n * @private\r\n *\r\n * @description Makes new block with initial type from settings\r\n */\r\n var enterKeyPressedOnRedactorsZone_ = function (event) {\r\n\r\n if (event.target.contentEditable == 'true') {\r\n\r\n /** Update input index */\r\n editor.caret.saveCurrentInputIndex();\r\n\r\n }\r\n\r\n var currentInputIndex = editor.caret.getCurrentInputIndex() || 0,\r\n workingNode = editor.content.currentNode,\r\n tool = workingNode.dataset.tool,\r\n isEnterPressedOnToolbar = editor.toolbar.opened &&\r\n editor.toolbar.current &&\r\n event.target == editor.state.inputs[currentInputIndex];\r\n\r\n /** The list of tools which needs the default browser behaviour */\r\n var enableLineBreaks = editor.tools[tool].enableLineBreaks;\r\n\r\n /** This type of block creates when enter is pressed */\r\n var NEW_BLOCK_TYPE = editor.settings.initialBlockPlugin;\r\n\r\n /**\r\n * When toolbar is opened, select tool instead of making new paragraph\r\n */\r\n if ( isEnterPressedOnToolbar ) {\r\n\r\n event.preventDefault();\r\n\r\n editor.toolbar.toolbox.toolClicked(event);\r\n\r\n editor.toolbar.close();\r\n\r\n /**\r\n * Stop other listeners callback executions\r\n */\r\n event.stopPropagation();\r\n event.stopImmediatePropagation();\r\n\r\n return;\r\n\r\n }\r\n\r\n /**\r\n * Allow paragraph lineBreaks with shift enter\r\n * Or if shiftkey pressed and enter and enabledLineBreaks, the let new block creation\r\n */\r\n if ( event.shiftKey || enableLineBreaks ) {\r\n\r\n event.stopPropagation();\r\n event.stopImmediatePropagation();\r\n return;\r\n\r\n }\r\n\r\n var currentSelection = window.getSelection(),\r\n currentSelectedNode = currentSelection.anchorNode,\r\n caretAtTheEndOfText = editor.caret.position.atTheEnd(),\r\n isTextNodeHasParentBetweenContenteditable = false;\r\n\r\n /**\r\n * Allow making new

    in same block by SHIFT+ENTER and forbids to prevent default browser behaviour\r\n */\r\n if ( event.shiftKey && !enableLineBreaks ) {\r\n\r\n editor.callback.enterPressedOnBlock(editor.content.currentBlock, event);\r\n event.preventDefault();\r\n return;\r\n\r\n }\r\n\r\n /**\r\n * Workaround situation when caret at the Text node that has some wrapper Elements\r\n * Split block cant handle this.\r\n * We need to save default behavior\r\n */\r\n isTextNodeHasParentBetweenContenteditable = currentSelectedNode && currentSelectedNode.parentNode.contentEditable != 'true';\r\n\r\n /**\r\n * Split blocks when input has several nodes and caret placed in textNode\r\n */\r\n if (\r\n currentSelectedNode.nodeType == editor.core.nodeTypes.TEXT &&\r\n !isTextNodeHasParentBetweenContenteditable &&\r\n !caretAtTheEndOfText\r\n ) {\r\n\r\n event.preventDefault();\r\n\r\n editor.core.log('Splitting Text node...');\r\n\r\n editor.content.splitBlock(currentInputIndex);\r\n\r\n /** Show plus button when next input after split is empty*/\r\n if (!editor.state.inputs[currentInputIndex + 1].textContent.trim()) {\r\n\r\n editor.toolbar.showPlusButton();\r\n\r\n }\r\n\r\n } else {\r\n\r\n var islastNode = editor.content.isLastNode(currentSelectedNode);\r\n\r\n if ( islastNode && caretAtTheEndOfText ) {\r\n\r\n event.preventDefault();\r\n event.stopPropagation();\r\n event.stopImmediatePropagation();\r\n\r\n editor.core.log('ENTER clicked in last textNode. Create new BLOCK');\r\n\r\n editor.content.insertBlock({\r\n type: NEW_BLOCK_TYPE,\r\n block: editor.tools[NEW_BLOCK_TYPE].render()\r\n }, true);\r\n\r\n editor.toolbar.move();\r\n editor.toolbar.open();\r\n\r\n /** Show plus button with empty block */\r\n editor.toolbar.showPlusButton();\r\n\r\n }\r\n\r\n }\r\n\r\n /** get all inputs after new appending block */\r\n editor.ui.saveInputs();\r\n\r\n };\r\n\r\n /**\r\n * Escape behaviour\r\n * @param event\r\n * @private\r\n *\r\n * @description Closes toolbox and toolbar. Prevents default behaviour\r\n */\r\n var escapeKeyPressedOnRedactorsZone_ = function (event) {\r\n\r\n /** Close all toolbar */\r\n editor.toolbar.close();\r\n\r\n /** Close toolbox */\r\n editor.toolbar.toolbox.close();\r\n\r\n event.preventDefault();\r\n\r\n };\r\n\r\n /**\r\n * @param {Event} event\r\n * @private\r\n *\r\n * closes and moves toolbar\r\n */\r\n var arrowKeyPressed_ = function (event) {\r\n\r\n editor.content.workingNodeChanged();\r\n\r\n /* Closing toolbar */\r\n editor.toolbar.close();\r\n editor.toolbar.move();\r\n\r\n };\r\n\r\n /**\r\n * @private\r\n * @param {Event} event\r\n *\r\n * @description Closes all opened bars from toolbar.\r\n * If block is mark, clears highlightning\r\n */\r\n var defaultKeyPressedOnRedactorsZone_ = function () {\r\n\r\n editor.toolbar.close();\r\n\r\n if (!editor.toolbar.inline.actionsOpened) {\r\n\r\n editor.toolbar.inline.close();\r\n editor.content.clearMark();\r\n\r\n }\r\n\r\n };\r\n\r\n /**\r\n * Handler when clicked on redactors area\r\n *\r\n * @protected\r\n * @param event\r\n *\r\n * @description Detects clicked area. If it is first-level block area, marks as detected and\r\n * on next enter press will be inserted new block\r\n * Otherwise, save carets position (input index) and put caret to the editable zone.\r\n *\r\n * @see detectWhenClickedOnFirstLevelBlockArea_\r\n *\r\n */\r\n callbacks.redactorClicked = function (event) {\r\n\r\n detectWhenClickedOnFirstLevelBlockArea_();\r\n\r\n editor.content.workingNodeChanged(event.target);\r\n editor.ui.saveInputs();\r\n\r\n var selectedText = editor.toolbar.inline.getSelectionText(),\r\n firstLevelBlock;\r\n\r\n /** If selection range took off, then we hide inline toolbar */\r\n if (selectedText.length === 0) {\r\n\r\n editor.toolbar.inline.close();\r\n\r\n }\r\n\r\n /** Update current input index in memory when caret focused into existed input */\r\n if (event.target.contentEditable == 'true') {\r\n\r\n editor.caret.saveCurrentInputIndex();\r\n\r\n }\r\n\r\n if (editor.content.currentNode === null) {\r\n\r\n /**\r\n * If inputs in redactor does not exits, then we put input index 0 not -1\r\n */\r\n var indexOfLastInput = editor.state.inputs.length > 0 ? editor.state.inputs.length - 1 : 0;\r\n\r\n /** If we have any inputs */\r\n if (editor.state.inputs.length) {\r\n\r\n /** getting firstlevel parent of input */\r\n firstLevelBlock = editor.content.getFirstLevelBlock(editor.state.inputs[indexOfLastInput]);\r\n\r\n }\r\n\r\n /** If input is empty, then we set caret to the last input */\r\n if (editor.state.inputs.length && editor.state.inputs[indexOfLastInput].textContent === '' && firstLevelBlock.dataset.tool == editor.settings.initialBlockPlugin) {\r\n\r\n editor.caret.setToBlock(indexOfLastInput);\r\n\r\n } else {\r\n\r\n /** Create new input when caret clicked in redactors area */\r\n var NEW_BLOCK_TYPE = editor.settings.initialBlockPlugin;\r\n\r\n editor.content.insertBlock({\r\n type : NEW_BLOCK_TYPE,\r\n block : editor.tools[NEW_BLOCK_TYPE].render()\r\n });\r\n\r\n /** If there is no inputs except inserted */\r\n if (editor.state.inputs.length === 1) {\r\n\r\n editor.caret.setToBlock(indexOfLastInput);\r\n\r\n } else {\r\n\r\n /** Set caret to this appended input */\r\n editor.caret.setToNextBlock(indexOfLastInput);\r\n\r\n }\r\n\r\n }\r\n\r\n } else {\r\n\r\n /** Close all panels */\r\n editor.toolbar.settings.close();\r\n editor.toolbar.toolbox.close();\r\n\r\n }\r\n\r\n /**\r\n * Move toolbar and open\r\n */\r\n editor.toolbar.move();\r\n editor.toolbar.open();\r\n\r\n var inputIsEmpty = !editor.content.currentNode.textContent.trim(),\r\n currentNodeType = editor.content.currentNode.dataset.tool,\r\n isInitialType = currentNodeType == editor.settings.initialBlockPlugin;\r\n\r\n\r\n /** Hide plus buttons */\r\n editor.toolbar.hidePlusButton();\r\n\r\n if (!inputIsEmpty) {\r\n\r\n /** Mark current block */\r\n editor.content.markBlock();\r\n\r\n }\r\n\r\n if ( isInitialType && inputIsEmpty ) {\r\n\r\n /** Show plus button */\r\n editor.toolbar.showPlusButton();\r\n\r\n }\r\n\r\n\r\n };\r\n\r\n /**\r\n * This method allows to define, is caret in contenteditable element or not.\r\n *\r\n * @private\r\n *\r\n * @description Otherwise, if we get TEXT node from range container, that will means we have input index.\r\n * In this case we use default browsers behaviour (if plugin allows that) or overwritten action.\r\n * Therefore, to be sure that we've clicked first-level block area, we should have currentNode, which always\r\n * specifies to the first-level block. Other cases we just ignore.\r\n */\r\n var detectWhenClickedOnFirstLevelBlockArea_ = function () {\r\n\r\n var selection = window.getSelection(),\r\n anchorNode = selection.anchorNode,\r\n flag = false;\r\n\r\n if (selection.rangeCount === 0) {\r\n\r\n editor.content.editorAreaHightlighted = true;\r\n\r\n } else {\r\n\r\n if (!editor.core.isDomNode(anchorNode)) {\r\n\r\n anchorNode = anchorNode.parentNode;\r\n\r\n }\r\n\r\n /** Already founded, without loop */\r\n if (anchorNode.contentEditable == 'true') {\r\n\r\n flag = true;\r\n\r\n }\r\n\r\n while (anchorNode.contentEditable != 'true') {\r\n\r\n anchorNode = anchorNode.parentNode;\r\n\r\n if (anchorNode.contentEditable == 'true') {\r\n\r\n flag = true;\r\n\r\n }\r\n\r\n if (anchorNode == document.body) {\r\n\r\n break;\r\n\r\n }\r\n\r\n }\r\n\r\n /** If editable element founded, flag is \"TRUE\", Therefore we return \"FALSE\" */\r\n editor.content.editorAreaHightlighted = !flag;\r\n\r\n }\r\n\r\n };\r\n\r\n /**\r\n * Toolbar button click handler\r\n *\r\n * @param {Object} event - cursor to the button\r\n * @protected\r\n *\r\n * @description gets current tool and calls render method\r\n */\r\n callbacks.toolbarButtonClicked = function (event) {\r\n\r\n var button = this;\r\n\r\n editor.toolbar.current = button.dataset.type;\r\n\r\n editor.toolbar.toolbox.toolClicked(event);\r\n editor.toolbar.close();\r\n\r\n };\r\n\r\n /**\r\n * Show or Hide toolbox when plus button is clicked\r\n */\r\n callbacks.plusButtonClicked = function () {\r\n\r\n if (!editor.nodes.toolbox.classList.contains('opened')) {\r\n\r\n editor.toolbar.toolbox.open();\r\n\r\n } else {\r\n\r\n editor.toolbar.toolbox.close();\r\n\r\n }\r\n\r\n };\r\n\r\n /**\r\n * Block handlers for KeyDown events\r\n *\r\n * @protected\r\n * @param {Object} event\r\n *\r\n * Handles keydowns on block\r\n * @see blockRightOrDownArrowPressed_\r\n * @see backspacePressed_\r\n * @see blockLeftOrUpArrowPressed_\r\n */\r\n callbacks.blockKeydown = function (event) {\r\n\r\n let block = event.target; // event.target is input\r\n\r\n switch (event.keyCode) {\r\n\r\n case editor.core.keys.DOWN:\r\n case editor.core.keys.RIGHT:\r\n blockRightOrDownArrowPressed_(event);\r\n break;\r\n\r\n case editor.core.keys.BACKSPACE:\r\n backspacePressed_(block, event);\r\n break;\r\n\r\n case editor.core.keys.UP:\r\n case editor.core.keys.LEFT:\r\n blockLeftOrUpArrowPressed_(event);\r\n break;\r\n\r\n }\r\n\r\n };\r\n\r\n /**\r\n * RIGHT or DOWN keydowns on block\r\n *\r\n * @param {Object} event\r\n * @private\r\n *\r\n * @description watches the selection and gets closest editable element.\r\n * Uses method getDeepestTextNodeFromPosition to get the last node of next block\r\n * Sets caret if it is contenteditable\r\n */\r\n var blockRightOrDownArrowPressed_ = function (event) {\r\n\r\n var selection = window.getSelection(),\r\n inputs = editor.state.inputs,\r\n focusedNode = selection.anchorNode,\r\n focusedNodeHolder;\r\n\r\n /** Check for caret existance */\r\n if (!focusedNode) {\r\n\r\n return false;\r\n\r\n }\r\n\r\n /** Looking for closest (parent) contentEditable element of focused node */\r\n while (focusedNode.contentEditable != 'true') {\r\n\r\n focusedNodeHolder = focusedNode.parentNode;\r\n focusedNode = focusedNodeHolder;\r\n\r\n }\r\n\r\n /** Input index in DOM level */\r\n var editableElementIndex = 0;\r\n\r\n while (focusedNode != inputs[editableElementIndex]) {\r\n\r\n editableElementIndex ++;\r\n\r\n }\r\n\r\n /**\r\n * Founded contentEditable element doesn't have childs\r\n * Or maybe New created block\r\n */\r\n if (!focusedNode.textContent) {\r\n\r\n editor.caret.setToNextBlock(editableElementIndex);\r\n return;\r\n\r\n }\r\n\r\n /**\r\n * Do nothing when caret doesn not reaches the end of last child\r\n */\r\n var caretInLastChild = false,\r\n caretAtTheEndOfText = false;\r\n\r\n var lastChild,\r\n deepestTextnode;\r\n\r\n lastChild = focusedNode.childNodes[focusedNode.childNodes.length - 1 ];\r\n\r\n if (editor.core.isDomNode(lastChild)) {\r\n\r\n deepestTextnode = editor.content.getDeepestTextNodeFromPosition(lastChild, lastChild.childNodes.length);\r\n\r\n } else {\r\n\r\n deepestTextnode = lastChild;\r\n\r\n }\r\n\r\n caretInLastChild = selection.anchorNode == deepestTextnode;\r\n caretAtTheEndOfText = deepestTextnode.length == selection.anchorOffset;\r\n\r\n if ( !caretInLastChild || !caretAtTheEndOfText ) {\r\n\r\n editor.core.log('arrow [down|right] : caret does not reached the end');\r\n return false;\r\n\r\n }\r\n\r\n editor.caret.setToNextBlock(editableElementIndex);\r\n\r\n };\r\n\r\n /**\r\n * LEFT or UP keydowns on block\r\n *\r\n * @param {Object} event\r\n * @private\r\n *\r\n * watches the selection and gets closest editable element.\r\n * Uses method getDeepestTextNodeFromPosition to get the last node of previous block\r\n * Sets caret if it is contenteditable\r\n *\r\n */\r\n var blockLeftOrUpArrowPressed_ = function (event) {\r\n\r\n var selection = window.getSelection(),\r\n inputs = editor.state.inputs,\r\n focusedNode = selection.anchorNode,\r\n focusedNodeHolder;\r\n\r\n /** Check for caret existance */\r\n if (!focusedNode) {\r\n\r\n return false;\r\n\r\n }\r\n\r\n /**\r\n * LEFT or UP not at the beginning\r\n */\r\n if ( selection.anchorOffset !== 0) {\r\n\r\n return false;\r\n\r\n }\r\n\r\n /** Looking for parent contentEditable block */\r\n while (focusedNode.contentEditable != 'true') {\r\n\r\n focusedNodeHolder = focusedNode.parentNode;\r\n focusedNode = focusedNodeHolder;\r\n\r\n }\r\n\r\n /** Input index in DOM level */\r\n var editableElementIndex = 0;\r\n\r\n while (focusedNode != inputs[editableElementIndex]) {\r\n\r\n editableElementIndex ++;\r\n\r\n }\r\n\r\n /**\r\n * Do nothing if caret is not at the beginning of first child\r\n */\r\n var caretInFirstChild = false,\r\n caretAtTheBeginning = false;\r\n\r\n var firstChild,\r\n deepestTextnode;\r\n\r\n /**\r\n * Founded contentEditable element doesn't have childs\r\n * Or maybe New created block\r\n */\r\n if (!focusedNode.textContent) {\r\n\r\n editor.caret.setToPreviousBlock(editableElementIndex);\r\n return;\r\n\r\n }\r\n\r\n firstChild = focusedNode.childNodes[0];\r\n\r\n if (editor.core.isDomNode(firstChild)) {\r\n\r\n deepestTextnode = editor.content.getDeepestTextNodeFromPosition(firstChild, 0);\r\n\r\n } else {\r\n\r\n deepestTextnode = firstChild;\r\n\r\n }\r\n\r\n caretInFirstChild = selection.anchorNode == deepestTextnode;\r\n caretAtTheBeginning = selection.anchorOffset === 0;\r\n\r\n if ( caretInFirstChild && caretAtTheBeginning ) {\r\n\r\n editor.caret.setToPreviousBlock(editableElementIndex);\r\n\r\n }\r\n\r\n };\r\n\r\n /**\r\n * Handles backspace keydown\r\n *\r\n * @param {Element} block\r\n * @param {Object} event\r\n * @private\r\n *\r\n * @description if block is empty, delete the block and set caret to the previous block\r\n * If block is not empty, try to merge two blocks - current and previous\r\n * But it we try'n to remove first block, then we should set caret to the next block, not previous.\r\n * If we removed the last block, create new one\r\n */\r\n var backspacePressed_ = function (block, event) {\r\n\r\n var currentInputIndex = editor.caret.getCurrentInputIndex(),\r\n range,\r\n selectionLength,\r\n firstLevelBlocksCount;\r\n\r\n if (editor.core.isNativeInput(event.target)) {\r\n\r\n /** If input value is empty - remove block */\r\n if (event.target.value.trim() == '') {\r\n\r\n block.remove();\r\n\r\n } else {\r\n\r\n return;\r\n\r\n }\r\n\r\n }\r\n\r\n if (block.textContent.trim()) {\r\n\r\n range = editor.content.getRange();\r\n selectionLength = range.endOffset - range.startOffset;\r\n\r\n if (editor.caret.position.atStart() && !selectionLength && editor.state.inputs[currentInputIndex - 1]) {\r\n\r\n editor.content.mergeBlocks(currentInputIndex);\r\n\r\n } else {\r\n\r\n return;\r\n\r\n }\r\n\r\n }\r\n\r\n if (!selectionLength) {\r\n\r\n block.remove();\r\n\r\n }\r\n\r\n\r\n firstLevelBlocksCount = editor.nodes.redactor.childNodes.length;\r\n\r\n /**\r\n * If all blocks are removed\r\n */\r\n if (firstLevelBlocksCount === 0) {\r\n\r\n /** update currentNode variable */\r\n editor.content.currentNode = null;\r\n\r\n /** Inserting new empty initial block */\r\n editor.ui.addInitialBlock();\r\n\r\n /** Updating inputs state after deleting last block */\r\n editor.ui.saveInputs();\r\n\r\n /** Set to current appended block */\r\n window.setTimeout(function () {\r\n\r\n editor.caret.setToPreviousBlock(1);\r\n\r\n }, 10);\r\n\r\n } else {\r\n\r\n if (editor.caret.inputIndex !== 0) {\r\n\r\n /** Target block is not first */\r\n editor.caret.setToPreviousBlock(editor.caret.inputIndex);\r\n\r\n } else {\r\n\r\n /** If we try to delete first block */\r\n editor.caret.setToNextBlock(editor.caret.inputIndex);\r\n\r\n }\r\n\r\n }\r\n\r\n editor.toolbar.move();\r\n\r\n if (!editor.toolbar.opened) {\r\n\r\n editor.toolbar.open();\r\n\r\n }\r\n\r\n /** Updating inputs state */\r\n editor.ui.saveInputs();\r\n\r\n /** Prevent default browser behaviour */\r\n event.preventDefault();\r\n\r\n };\r\n\r\n /**\r\n * used by UI module\r\n * Clicks on block settings button\r\n *\r\n * @param {Object} event\r\n * @protected\r\n * @description Opens toolbar settings\r\n */\r\n callbacks.showSettingsButtonClicked = function (event) {\r\n\r\n /**\r\n * Get type of current block\r\n * It uses to append settings from tool.settings property.\r\n * ...\r\n * Type is stored in data-type attribute on block\r\n */\r\n var currentToolType = editor.content.currentNode.dataset.tool;\r\n\r\n editor.toolbar.settings.toggle(currentToolType);\r\n\r\n /** Close toolbox when settings button is active */\r\n editor.toolbar.toolbox.close();\r\n editor.toolbar.settings.hideRemoveActions();\r\n\r\n };\r\n\r\n return callbacks;\r\n\r\n})({});\n\n\n// WEBPACK FOOTER //\n// ./modules/callbacks.js","/**\r\n * Codex Editor Draw module\r\n *\r\n * @author Codex Team\r\n * @version 1.0.\r\n */\r\n\r\nmodule.exports = (function (draw) {\r\n\r\n /**\r\n * Base editor wrapper\r\n */\r\n draw.wrapper = function () {\r\n\r\n var wrapper = document.createElement('div');\r\n\r\n wrapper.className += 'codex-editor';\r\n\r\n return wrapper;\r\n\r\n };\r\n\r\n /**\r\n * Content-editable holder\r\n */\r\n draw.redactor = function () {\r\n\r\n var redactor = document.createElement('div');\r\n\r\n redactor.className += 'ce-redactor';\r\n\r\n return redactor;\r\n\r\n };\r\n\r\n draw.ceBlock = function () {\r\n\r\n var block = document.createElement('DIV');\r\n\r\n block.className += 'ce_block';\r\n\r\n return block;\r\n\r\n };\r\n\r\n /**\r\n * Empty toolbar with toggler\r\n */\r\n draw.toolbar = function () {\r\n\r\n var bar = document.createElement('div');\r\n\r\n bar.className += 'ce-toolbar';\r\n\r\n return bar;\r\n\r\n };\r\n\r\n draw.toolbarContent = function () {\r\n\r\n var wrapper = document.createElement('DIV');\r\n\r\n wrapper.classList.add('ce-toolbar__content');\r\n\r\n return wrapper;\r\n\r\n };\r\n\r\n /**\r\n * Inline toolbar\r\n */\r\n draw.inlineToolbar = function () {\r\n\r\n var bar = document.createElement('DIV');\r\n\r\n bar.className += 'ce-toolbar-inline';\r\n\r\n return bar;\r\n\r\n };\r\n\r\n /**\r\n * Wrapper for inline toobar buttons\r\n */\r\n draw.inlineToolbarButtons = function () {\r\n\r\n var wrapper = document.createElement('DIV');\r\n\r\n wrapper.className += 'ce-toolbar-inline__buttons';\r\n\r\n return wrapper;\r\n\r\n };\r\n\r\n /**\r\n * For some actions\r\n */\r\n draw.inlineToolbarActions = function () {\r\n\r\n var wrapper = document.createElement('DIV');\r\n\r\n wrapper.className += 'ce-toolbar-inline__actions';\r\n\r\n return wrapper;\r\n\r\n };\r\n\r\n draw.inputForLink = function () {\r\n\r\n var input = document.createElement('INPUT');\r\n\r\n input.type = 'input';\r\n input.className += 'inputForLink';\r\n input.placeholder = 'Вставьте ссылку ...';\r\n input.setAttribute('form', 'defaultForm');\r\n\r\n input.setAttribute('autofocus', 'autofocus');\r\n\r\n return input;\r\n\r\n };\r\n\r\n /**\r\n * @todo Desc\r\n */\r\n draw.blockButtons = function () {\r\n\r\n var block = document.createElement('div');\r\n\r\n block.className += 'ce-toolbar__actions';\r\n\r\n return block;\r\n\r\n };\r\n\r\n /**\r\n * Block settings panel\r\n */\r\n draw.blockSettings = function () {\r\n\r\n var settings = document.createElement('div');\r\n\r\n settings.className += 'ce-settings';\r\n\r\n return settings;\r\n\r\n };\r\n\r\n draw.defaultSettings = function () {\r\n\r\n var div = document.createElement('div');\r\n\r\n div.classList.add('ce-settings_default');\r\n\r\n return div;\r\n\r\n };\r\n\r\n draw.pluginsSettings = function () {\r\n\r\n var div = document.createElement('div');\r\n\r\n div.classList.add('ce-settings_plugin');\r\n\r\n return div;\r\n\r\n };\r\n\r\n draw.plusButton = function () {\r\n\r\n var button = document.createElement('span');\r\n\r\n button.className = 'ce-toolbar__plus';\r\n // button.innerHTML = '';\r\n\r\n return button;\r\n\r\n };\r\n\r\n /**\r\n * Settings button in toolbar\r\n */\r\n draw.settingsButton = function () {\r\n\r\n var toggler = document.createElement('span');\r\n\r\n toggler.className = 'ce-toolbar__settings-btn';\r\n\r\n /** Toggler button*/\r\n toggler.innerHTML = '';\r\n\r\n return toggler;\r\n\r\n };\r\n\r\n /**\r\n * Redactor tools wrapper\r\n */\r\n\r\n draw.toolbox = function () {\r\n\r\n var wrapper = document.createElement('div');\r\n\r\n wrapper.className = 'ce-toolbar__tools';\r\n\r\n return wrapper;\r\n\r\n };\r\n\r\n /**\r\n * @protected\r\n *\r\n * Draws tool buttons for toolbox\r\n *\r\n * @param {String} type\r\n * @param {String} classname\r\n * @returns {Element}\r\n */\r\n draw.toolbarButton = function (type, classname) {\r\n\r\n var button = document.createElement('li'),\r\n toolIcon = document.createElement('i'),\r\n toolTitle = document.createElement('span');\r\n\r\n button.dataset.type = type;\r\n button.setAttribute('title', type);\r\n\r\n toolIcon.classList.add(classname);\r\n toolTitle.classList.add('ce_toolbar_tools--title');\r\n\r\n\r\n button.appendChild(toolIcon);\r\n button.appendChild(toolTitle);\r\n\r\n return button;\r\n\r\n };\r\n\r\n /**\r\n * @protected\r\n *\r\n * Draws tools for inline toolbar\r\n *\r\n * @param {String} type\r\n * @param {String} classname\r\n */\r\n draw.toolbarButtonInline = function (type, classname) {\r\n\r\n var button = document.createElement('BUTTON'),\r\n toolIcon = document.createElement('I');\r\n\r\n button.type = 'button';\r\n button.dataset.type = type;\r\n toolIcon.classList.add(classname);\r\n\r\n button.appendChild(toolIcon);\r\n\r\n return button;\r\n\r\n };\r\n\r\n /**\r\n * Redactor block\r\n */\r\n draw.block = function (tagName, content) {\r\n\r\n var node = document.createElement(tagName);\r\n\r\n node.innerHTML = content || '';\r\n\r\n return node;\r\n\r\n };\r\n\r\n /**\r\n * Creates Node with passed tagName and className\r\n * @param {string} tagName\r\n * @param {string} className\r\n * @param {object} properties - allow to assign properties\r\n */\r\n draw.node = function ( tagName, className, properties ) {\r\n\r\n var el = document.createElement( tagName );\r\n\r\n if ( className ) el.className = className;\r\n\r\n if ( properties ) {\r\n\r\n for (var name in properties) {\r\n\r\n el[name] = properties[name];\r\n\r\n }\r\n\r\n }\r\n\r\n return el;\r\n\r\n };\r\n\r\n /**\r\n * Unavailable plugin block\r\n */\r\n draw.unavailableBlock = function () {\r\n\r\n var wrapper = document.createElement('DIV');\r\n\r\n wrapper.classList.add('cdx-unavailable-block');\r\n\r\n return wrapper;\r\n\r\n };\r\n\r\n return draw;\r\n\r\n})({});\n\n\n// WEBPACK FOOTER //\n// ./modules/draw.js","/**\r\n * Codex Editor Caret Module\r\n *\r\n * @author Codex Team\r\n * @version 1.0\r\n */\r\n\r\nmodule.exports = (function (caret) {\r\n\r\n let editor = codex.editor;\r\n\r\n /**\r\n * @var {int} InputIndex - editable element in DOM\r\n */\r\n caret.inputIndex = null;\r\n\r\n /**\r\n * @var {int} offset - caret position in a text node.\r\n */\r\n caret.offset = null;\r\n\r\n /**\r\n * @var {int} focusedNodeIndex - we get index of child node from first-level block\r\n */\r\n caret.focusedNodeIndex = null;\r\n\r\n /**\r\n * Creates Document Range and sets caret to the element.\r\n * @protected\r\n * @uses caret.save — if you need to save caret position\r\n * @param {Element} el - Changed Node.\r\n */\r\n caret.set = function ( el, index, offset) {\r\n\r\n offset = offset || caret.offset || 0;\r\n index = index || caret.focusedNodeIndex || 0;\r\n\r\n var childs = el.childNodes,\r\n nodeToSet;\r\n\r\n if ( childs.length === 0 ) {\r\n\r\n nodeToSet = el;\r\n\r\n } else {\r\n\r\n nodeToSet = childs[index];\r\n\r\n }\r\n\r\n /** If Element is INPUT */\r\n if (el.contentEditable != 'true') {\r\n\r\n el.focus();\r\n return;\r\n\r\n }\r\n\r\n if (editor.core.isDomNode(nodeToSet)) {\r\n\r\n nodeToSet = editor.content.getDeepestTextNodeFromPosition(nodeToSet, nodeToSet.childNodes.length);\r\n\r\n }\r\n\r\n var range = document.createRange(),\r\n selection = window.getSelection();\r\n\r\n window.setTimeout(function () {\r\n\r\n range.setStart(nodeToSet, offset);\r\n range.setEnd(nodeToSet, offset);\r\n\r\n selection.removeAllRanges();\r\n selection.addRange(range);\r\n\r\n editor.caret.saveCurrentInputIndex();\r\n\r\n }, 20);\r\n\r\n };\r\n\r\n /**\r\n * @protected\r\n * Updates index of input and saves it in caret object\r\n */\r\n caret.saveCurrentInputIndex = function () {\r\n\r\n /** Index of Input that we paste sanitized content */\r\n var selection = window.getSelection(),\r\n inputs = editor.state.inputs,\r\n focusedNode = selection.anchorNode,\r\n focusedNodeHolder;\r\n\r\n if (!focusedNode) {\r\n\r\n return;\r\n\r\n }\r\n\r\n /** Looking for parent contentEditable block */\r\n while (focusedNode.contentEditable != 'true') {\r\n\r\n focusedNodeHolder = focusedNode.parentNode;\r\n focusedNode = focusedNodeHolder;\r\n\r\n }\r\n\r\n /** Input index in DOM level */\r\n var editableElementIndex = 0;\r\n\r\n while (focusedNode != inputs[editableElementIndex]) {\r\n\r\n editableElementIndex ++;\r\n\r\n }\r\n\r\n caret.inputIndex = editableElementIndex;\r\n\r\n };\r\n\r\n /**\r\n * Returns current input index (caret object)\r\n */\r\n caret.getCurrentInputIndex = function () {\r\n\r\n return caret.inputIndex;\r\n\r\n };\r\n\r\n /**\r\n * @param {int} index - index of first-level block after that we set caret into next input\r\n */\r\n caret.setToNextBlock = function (index) {\r\n\r\n var inputs = editor.state.inputs,\r\n nextInput = inputs[index + 1];\r\n\r\n if (!nextInput) {\r\n\r\n editor.core.log('We are reached the end');\r\n return;\r\n\r\n }\r\n\r\n /**\r\n * When new Block created or deleted content of input\r\n * We should add some text node to set caret\r\n */\r\n if (!nextInput.childNodes.length) {\r\n\r\n var emptyTextElement = document.createTextNode('');\r\n\r\n nextInput.appendChild(emptyTextElement);\r\n\r\n }\r\n\r\n editor.caret.inputIndex = index + 1;\r\n editor.caret.set(nextInput, 0, 0);\r\n editor.content.workingNodeChanged(nextInput);\r\n\r\n };\r\n\r\n /**\r\n * @param {int} index - index of target input.\r\n * Sets caret to input with this index\r\n */\r\n caret.setToBlock = function (index) {\r\n\r\n var inputs = editor.state.inputs,\r\n targetInput = inputs[index];\r\n\r\n if ( !targetInput ) {\r\n\r\n return;\r\n\r\n }\r\n\r\n /**\r\n * When new Block created or deleted content of input\r\n * We should add some text node to set caret\r\n */\r\n if (!targetInput.childNodes.length) {\r\n\r\n var emptyTextElement = document.createTextNode('');\r\n\r\n targetInput.appendChild(emptyTextElement);\r\n\r\n }\r\n\r\n editor.caret.inputIndex = index;\r\n editor.caret.set(targetInput, 0, 0);\r\n editor.content.workingNodeChanged(targetInput);\r\n\r\n };\r\n\r\n /**\r\n * @param {int} index - index of input\r\n */\r\n caret.setToPreviousBlock = function (index) {\r\n\r\n index = index || 0;\r\n\r\n var inputs = editor.state.inputs,\r\n previousInput = inputs[index - 1],\r\n lastChildNode,\r\n lengthOfLastChildNode,\r\n emptyTextElement;\r\n\r\n\r\n if (!previousInput) {\r\n\r\n editor.core.log('We are reached first node');\r\n return;\r\n\r\n }\r\n\r\n lastChildNode = editor.content.getDeepestTextNodeFromPosition(previousInput, previousInput.childNodes.length);\r\n lengthOfLastChildNode = lastChildNode.length;\r\n\r\n /**\r\n * When new Block created or deleted content of input\r\n * We should add some text node to set caret\r\n */\r\n if (!previousInput.childNodes.length) {\r\n\r\n emptyTextElement = document.createTextNode('');\r\n previousInput.appendChild(emptyTextElement);\r\n\r\n }\r\n editor.caret.inputIndex = index - 1;\r\n editor.caret.set(previousInput, previousInput.childNodes.length - 1, lengthOfLastChildNode);\r\n editor.content.workingNodeChanged(inputs[index - 1]);\r\n\r\n };\r\n\r\n caret.position = {\r\n\r\n atStart : function () {\r\n\r\n var selection = window.getSelection(),\r\n anchorOffset = selection.anchorOffset,\r\n anchorNode = selection.anchorNode,\r\n firstLevelBlock = editor.content.getFirstLevelBlock(anchorNode),\r\n pluginsRender = firstLevelBlock.childNodes[0];\r\n\r\n if (!editor.core.isDomNode(anchorNode)) {\r\n\r\n anchorNode = anchorNode.parentNode;\r\n\r\n }\r\n\r\n var isFirstNode = anchorNode === pluginsRender.childNodes[0],\r\n isOffsetZero = anchorOffset === 0;\r\n\r\n return isFirstNode && isOffsetZero;\r\n\r\n },\r\n\r\n atTheEnd : function () {\r\n\r\n var selection = window.getSelection(),\r\n anchorOffset = selection.anchorOffset,\r\n anchorNode = selection.anchorNode;\r\n\r\n /** Caret is at the end of input */\r\n return !anchorNode || !anchorNode.length || anchorOffset === anchorNode.length;\r\n\r\n }\r\n };\r\n\r\n\r\n /**\r\n * Inserts node at the caret location\r\n * @param node\r\n */\r\n caret.insertNode = function (node) {\r\n\r\n var selection, range;\r\n\r\n selection = window.getSelection();\r\n\r\n range = selection.getRangeAt(0);\r\n range.deleteContents();\r\n\r\n range.insertNode(node);\r\n\r\n range.setStartAfter(node);\r\n range.collapse(true);\r\n\r\n selection.removeAllRanges();\r\n selection.addRange(range);\r\n\r\n\r\n };\r\n\r\n return caret;\r\n\r\n})({});\n\n\n// WEBPACK FOOTER //\n// ./modules/caret.js","/**\r\n * Codex Editor Notification Module\r\n *\r\n * @author Codex Team\r\n * @version 1.0\r\n */\r\n\r\nmodule.exports = (function (notifications) {\r\n\r\n let editor = codex.editor;\r\n\r\n var queue = [];\r\n\r\n var addToQueue = function (settings) {\r\n\r\n queue.push(settings);\r\n\r\n var index = 0;\r\n\r\n while ( index < queue.length && queue.length > 5) {\r\n\r\n if (queue[index].type == 'confirm' || queue[index].type == 'prompt') {\r\n\r\n index++;\r\n continue;\r\n\r\n }\r\n\r\n queue[index].close();\r\n queue.splice(index, 1);\r\n\r\n }\r\n\r\n };\r\n\r\n notifications.createHolder = function () {\r\n\r\n var holder = editor.draw.node('DIV', 'cdx-notifications-block');\r\n\r\n editor.nodes.notifications = document.body.appendChild(holder);\r\n\r\n return holder;\r\n\r\n };\r\n\r\n\r\n /**\r\n * Error notificator. Shows block with message\r\n * @protected\r\n */\r\n notifications.errorThrown = function (errorMsg, event) {\r\n\r\n editor.notifications.notification({message: 'This action is not available currently', type: event.type});\r\n\r\n };\r\n\r\n /**\r\n *\r\n * Appends notification\r\n *\r\n * settings = {\r\n * type - notification type (reserved types: alert, confirm, prompt). Just add class 'cdx-notification-'+type\r\n * message - notification message\r\n * okMsg - confirm button text (default - 'Ok')\r\n * cancelBtn - cancel button text (default - 'Cancel'). Only for confirm and prompt types\r\n * confirm - function-handler for ok button click\r\n * cancel - function-handler for cancel button click. Only for confirm and prompt types\r\n * time - time (in seconds) after which notification will close (default - 10s)\r\n * }\r\n *\r\n * @param settings\r\n */\r\n notifications.notification = function (constructorSettings) {\r\n\r\n /** Private vars and methods */\r\n var notification = null,\r\n cancel = null,\r\n type = null,\r\n confirm = null,\r\n inputField = null;\r\n\r\n var confirmHandler = function () {\r\n\r\n close();\r\n\r\n if (typeof confirm !== 'function' ) {\r\n\r\n return;\r\n\r\n }\r\n\r\n if (type == 'prompt') {\r\n\r\n confirm(inputField.value);\r\n return;\r\n\r\n }\r\n\r\n confirm();\r\n\r\n };\r\n\r\n var cancelHandler = function () {\r\n\r\n close();\r\n\r\n if (typeof cancel !== 'function' ) {\r\n\r\n return;\r\n\r\n }\r\n\r\n cancel();\r\n\r\n };\r\n\r\n\r\n /** Public methods */\r\n function create(settings) {\r\n\r\n if (!(settings && settings.message)) {\r\n\r\n editor.core.log('Can\\'t create notification. Message is missed');\r\n return;\r\n\r\n }\r\n\r\n settings.type = settings.type || 'alert';\r\n settings.time = settings.time*1000 || 10000;\r\n\r\n var wrapper = editor.draw.node('DIV', 'cdx-notification'),\r\n message = editor.draw.node('DIV', 'cdx-notification__message'),\r\n input = editor.draw.node('INPUT', 'cdx-notification__input'),\r\n okBtn = editor.draw.node('SPAN', 'cdx-notification__ok-btn'),\r\n cancelBtn = editor.draw.node('SPAN', 'cdx-notification__cancel-btn');\r\n\r\n message.textContent = settings.message;\r\n okBtn.textContent = settings.okMsg || 'ОК';\r\n cancelBtn.textContent = settings.cancelMsg || 'Отмена';\r\n\r\n editor.listeners.add(okBtn, 'click', confirmHandler);\r\n editor.listeners.add(cancelBtn, 'click', cancelHandler);\r\n\r\n wrapper.appendChild(message);\r\n\r\n if (settings.type == 'prompt') {\r\n\r\n wrapper.appendChild(input);\r\n\r\n }\r\n\r\n wrapper.appendChild(okBtn);\r\n\r\n if (settings.type == 'prompt' || settings.type == 'confirm') {\r\n\r\n wrapper.appendChild(cancelBtn);\r\n\r\n }\r\n\r\n wrapper.classList.add('cdx-notification-' + settings.type);\r\n wrapper.dataset.type = settings.type;\r\n\r\n notification = wrapper;\r\n type = settings.type;\r\n confirm = settings.confirm;\r\n cancel = settings.cancel;\r\n inputField = input;\r\n\r\n if (settings.type != 'prompt' && settings.type != 'confirm') {\r\n\r\n window.setTimeout(close, settings.time);\r\n\r\n }\r\n\r\n };\r\n\r\n function send() {\r\n\r\n editor.nodes.notifications.appendChild(notification);\r\n inputField.focus();\r\n\r\n editor.nodes.notifications.classList.add('cdx-notification__notification-appending');\r\n\r\n window.setTimeout(function () {\r\n\r\n editor.nodes.notifications.classList.remove('cdx-notification__notification-appending');\r\n\r\n }, 100);\r\n\r\n addToQueue({type: type, close: close});\r\n\r\n };\r\n\r\n function close() {\r\n\r\n notification.remove();\r\n\r\n };\r\n\r\n\r\n if (constructorSettings) {\r\n\r\n create(constructorSettings);\r\n send();\r\n\r\n }\r\n\r\n return {\r\n create: create,\r\n send: send,\r\n close: close\r\n };\r\n\r\n };\r\n\r\n notifications.clear = function () {\r\n\r\n editor.nodes.notifications.innerHTML = '';\r\n queue = [];\r\n\r\n };\r\n\r\n return notifications;\r\n\r\n})({});\n\n\n// WEBPACK FOOTER //\n// ./modules/notifications.js","/**\r\n * Codex Editor Parser Module\r\n *\r\n * @author Codex Team\r\n * @version 1.1\r\n */\r\n\r\nmodule.exports = (function (parser) {\r\n\r\n let editor = codex.editor;\r\n\r\n /** inserting text */\r\n parser.insertPastedContent = function (blockType, tag) {\r\n\r\n editor.content.insertBlock({\r\n type : blockType.type,\r\n block : blockType.render({\r\n text : tag.innerHTML\r\n })\r\n });\r\n\r\n };\r\n\r\n /**\r\n * Check DOM node for display style: separated block or child-view\r\n */\r\n parser.isFirstLevelBlock = function (node) {\r\n\r\n return node.nodeType == editor.core.nodeTypes.TAG &&\r\n node.classList.contains(editor.ui.className.BLOCK_CLASSNAME);\r\n\r\n };\r\n\r\n return parser;\r\n\r\n})({});\r\n\n\n\n// WEBPACK FOOTER //\n// ./modules/parser.js","/**\r\n * Codex Sanitizer\r\n */\r\n\r\nmodule.exports = (function (sanitizer) {\r\n\r\n /** HTML Janitor library */\r\n let janitor = require('html-janitor');\r\n\r\n /** Codex Editor */\r\n let editor = codex.editor;\r\n\r\n sanitizer.prepare = function () {\r\n\r\n if (editor.settings.sanitizer && !editor.core.isEmpty(editor.settings.sanitizer)) {\r\n\r\n Config.CUSTOM = editor.settings.sanitizer;\r\n\r\n }\r\n\r\n };\r\n\r\n /**\r\n * Basic config\r\n */\r\n var Config = {\r\n\r\n /** User configuration */\r\n CUSTOM : null,\r\n\r\n BASIC : {\r\n\r\n tags: {\r\n p: {},\r\n a: {\r\n href: true,\r\n target: '_blank',\r\n rel: 'nofollow'\r\n },\r\n i: {},\r\n b: {},\r\n strong: {},\r\n em: {},\r\n span: {}\r\n }\r\n }\r\n };\r\n\r\n sanitizer.Config = Config;\r\n\r\n /**\r\n *\r\n * @param userCustomConfig\r\n * @returns {*}\r\n * @private\r\n *\r\n * @description If developer uses editor's API, then he can customize sane restrictions.\r\n * Or, sane config can be defined globally in editors initialization. That config will be used everywhere\r\n * At least, if there is no config overrides, that API uses BASIC Default configation\r\n */\r\n let init_ = function (userCustomConfig) {\r\n\r\n let configuration = userCustomConfig || Config.CUSTOM || Config.BASIC;\r\n\r\n return new janitor(configuration);\r\n\r\n };\r\n\r\n /**\r\n * Cleans string from unwanted tags\r\n * @protected\r\n * @param {String} dirtyString - taint string\r\n * @param {Object} customConfig - allowed tags\r\n */\r\n sanitizer.clean = function (dirtyString, customConfig) {\r\n\r\n let janitorInstance = init_(customConfig);\r\n\r\n return janitorInstance.clean(dirtyString);\r\n\r\n };\r\n\r\n return sanitizer;\r\n\r\n})({});\n\n\n// WEBPACK FOOTER //\n// ./modules/sanitizer.js","(function (root, factory) {\n if (typeof define === 'function' && define.amd) {\n define('html-janitor', factory);\n } else if (typeof exports === 'object') {\n module.exports = factory();\n } else {\n root.HTMLJanitor = factory();\n }\n}(this, function () {\n\n /**\n * @param {Object} config.tags Dictionary of allowed tags.\n * @param {boolean} config.keepNestedBlockElements Default false.\n */\n function HTMLJanitor(config) {\n\n var tagDefinitions = config['tags'];\n var tags = Object.keys(tagDefinitions);\n\n var validConfigValues = tags\n .map(function(k) { return typeof tagDefinitions[k]; })\n .every(function(type) { return type === 'object' || type === 'boolean' || type === 'function'; });\n\n if(!validConfigValues) {\n throw new Error(\"The configuration was invalid\");\n }\n\n this.config = config;\n }\n\n // TODO: not exhaustive?\n var blockElementNames = ['P', 'LI', 'TD', 'TH', 'DIV', 'H1', 'H2', 'H3', 'H4', 'H5', 'H6', 'PRE'];\n function isBlockElement(node) {\n return blockElementNames.indexOf(node.nodeName) !== -1;\n }\n\n var inlineElementNames = ['A', 'B', 'STRONG', 'I', 'EM', 'SUB', 'SUP', 'U', 'STRIKE'];\n function isInlineElement(node) {\n return inlineElementNames.indexOf(node.nodeName) !== -1;\n }\n\n HTMLJanitor.prototype.clean = function (html) {\n var sandbox = document.createElement('div');\n sandbox.innerHTML = html;\n\n this._sanitize(sandbox);\n\n return sandbox.innerHTML;\n };\n\n HTMLJanitor.prototype._sanitize = function (parentNode) {\n var treeWalker = createTreeWalker(parentNode);\n var node = treeWalker.firstChild();\n if (!node) { return; }\n\n do {\n // Ignore nodes that have already been sanitized\n if (node._sanitized) {\n continue;\n }\n\n if (node.nodeType === Node.TEXT_NODE) {\n // If this text node is just whitespace and the previous or next element\n // sibling is a block element, remove it\n // N.B.: This heuristic could change. Very specific to a bug with\n // `contenteditable` in Firefox: http://jsbin.com/EyuKase/1/edit?js,output\n // FIXME: make this an option?\n if (node.data.trim() === ''\n && ((node.previousElementSibling && isBlockElement(node.previousElementSibling))\n || (node.nextElementSibling && isBlockElement(node.nextElementSibling)))) {\n parentNode.removeChild(node);\n this._sanitize(parentNode);\n break;\n } else {\n continue;\n }\n }\n\n // Remove all comments\n if (node.nodeType === Node.COMMENT_NODE) {\n parentNode.removeChild(node);\n this._sanitize(parentNode);\n break;\n }\n\n var isInline = isInlineElement(node);\n var containsBlockElement;\n if (isInline) {\n containsBlockElement = Array.prototype.some.call(node.childNodes, isBlockElement);\n }\n\n // Block elements should not be nested (e.g.

  • ...); if\n // they are, we want to unwrap the inner block element.\n var isNotTopContainer = !! parentNode.parentNode;\n var isNestedBlockElement =\n isBlockElement(parentNode) &&\n isBlockElement(node) &&\n isNotTopContainer;\n\n var nodeName = node.nodeName.toLowerCase();\n\n var allowedAttrs = getAllowedAttrs(this.config, nodeName, node);\n\n var isInvalid = isInline && containsBlockElement;\n\n // Drop tag entirely according to the whitelist *and* if the markup\n // is invalid.\n if (isInvalid || shouldRejectNode(node, allowedAttrs)\n || (!this.config.keepNestedBlockElements && isNestedBlockElement)) {\n // Do not keep the inner text of SCRIPT/STYLE elements.\n if (! (node.nodeName === 'SCRIPT' || node.nodeName === 'STYLE')) {\n while (node.childNodes.length > 0) {\n parentNode.insertBefore(node.childNodes[0], node);\n }\n }\n parentNode.removeChild(node);\n\n this._sanitize(parentNode);\n break;\n }\n\n // Sanitize attributes\n for (var a = 0; a < node.attributes.length; a += 1) {\n var attr = node.attributes[a];\n\n if (shouldRejectAttr(attr, allowedAttrs, node)) {\n node.removeAttribute(attr.name);\n // Shift the array to continue looping.\n a = a - 1;\n }\n }\n\n // Sanitize children\n this._sanitize(node);\n\n // Mark node as sanitized so it's ignored in future runs\n node._sanitized = true;\n } while ((node = treeWalker.nextSibling()));\n };\n\n function createTreeWalker(node) {\n return document.createTreeWalker(node,\n NodeFilter.SHOW_TEXT | NodeFilter.SHOW_ELEMENT | NodeFilter.SHOW_COMMENT,\n null, false);\n }\n\n function getAllowedAttrs(config, nodeName, node){\n if (typeof config.tags[nodeName] === 'function') {\n return config.tags[nodeName](node);\n } else {\n return config.tags[nodeName];\n }\n }\n\n function shouldRejectNode(node, allowedAttrs){\n if (typeof allowedAttrs === 'undefined') {\n return true;\n } else if (typeof allowedAttrs === 'boolean') {\n return !allowedAttrs;\n }\n\n return false;\n }\n\n function shouldRejectAttr(attr, allowedAttrs, node){\n var attrName = attr.name.toLowerCase();\n\n if (allowedAttrs === true){\n return false;\n } else if (typeof allowedAttrs[attrName] === 'function'){\n return !allowedAttrs[attrName](attr.value, node);\n } else if (typeof allowedAttrs[attrName] === 'undefined'){\n return true;\n } else if (allowedAttrs[attrName] === false) {\n return true;\n } else if (typeof allowedAttrs[attrName] === 'string') {\n return (allowedAttrs[attrName] !== attr.value);\n }\n\n return false;\n }\n\n return HTMLJanitor;\n\n}));\n\n\n\n//////////////////\n// WEBPACK FOOTER\n// ./~/html-janitor/src/html-janitor.js\n// module id = 18\n// module chunks = 0","/**\r\n * Codex Editor Listeners module\r\n *\r\n * @author Codex Team\r\n * @version 1.0\r\n */\r\n\r\n/**\r\n * Module-decorator for event listeners assignment\r\n */\r\nmodule.exports = function (listeners) {\r\n\r\n var allListeners = [];\r\n\r\n /**\r\n * Search methods\r\n *\r\n * byElement, byType and byHandler returns array of suitable listeners\r\n * one and all takes element, eventType, and handler and returns first (all) suitable listener\r\n *\r\n */\r\n listeners.search = function () {\r\n\r\n var byElement = function (element, context) {\r\n\r\n var listenersOnElement = [];\r\n\r\n context = context || allListeners;\r\n\r\n for (var i = 0; i < context.length; i++) {\r\n\r\n var listener = context[i];\r\n\r\n if (listener.element === element) {\r\n\r\n listenersOnElement.push(listener);\r\n\r\n }\r\n\r\n }\r\n\r\n return listenersOnElement;\r\n\r\n };\r\n\r\n var byType = function (eventType, context) {\r\n\r\n var listenersWithType = [];\r\n\r\n context = context || allListeners;\r\n\r\n for (var i = 0; i < context.length; i++) {\r\n\r\n var listener = context[i];\r\n\r\n if (listener.type === eventType) {\r\n\r\n listenersWithType.push(listener);\r\n\r\n }\r\n\r\n }\r\n\r\n return listenersWithType;\r\n\r\n };\r\n\r\n var byHandler = function (handler, context) {\r\n\r\n var listenersWithHandler = [];\r\n\r\n context = context || allListeners;\r\n\r\n for (var i = 0; i < context.length; i++) {\r\n\r\n var listener = context[i];\r\n\r\n if (listener.handler === handler) {\r\n\r\n listenersWithHandler.push(listener);\r\n\r\n }\r\n\r\n }\r\n\r\n return listenersWithHandler;\r\n\r\n };\r\n\r\n var one = function (element, eventType, handler) {\r\n\r\n var result = allListeners;\r\n\r\n if (element)\r\n result = byElement(element, result);\r\n\r\n if (eventType)\r\n result = byType(eventType, result);\r\n\r\n if (handler)\r\n result = byHandler(handler, result);\r\n\r\n return result[0];\r\n\r\n };\r\n\r\n var all = function (element, eventType, handler) {\r\n\r\n var result = allListeners;\r\n\r\n if (element)\r\n result = byElement(element, result);\r\n\r\n if (eventType)\r\n result = byType(eventType, result);\r\n\r\n if (handler)\r\n result = byHandler(handler, result);\r\n\r\n return result;\r\n\r\n };\r\n\r\n return {\r\n byElement : byElement,\r\n byType : byType,\r\n byHandler : byHandler,\r\n one : one,\r\n all : all\r\n };\r\n\r\n }();\r\n\r\n listeners.add = function (element, eventType, handler, isCapture) {\r\n\r\n element.addEventListener(eventType, handler, isCapture);\r\n\r\n var data = {\r\n element: element,\r\n type: eventType,\r\n handler: handler\r\n };\r\n\r\n var alreadyAddedListener = listeners.search.one(element, eventType, handler);\r\n\r\n if (!alreadyAddedListener) {\r\n\r\n allListeners.push(data);\r\n\r\n }\r\n\r\n };\r\n\r\n listeners.remove = function (element, eventType, handler) {\r\n\r\n element.removeEventListener(eventType, handler);\r\n\r\n var existingListeners = listeners.search.all(element, eventType, handler);\r\n\r\n for (var i = 0; i < existingListeners.length; i++) {\r\n\r\n var index = allListeners.indexOf(existingListeners[i]);\r\n\r\n if (index > 0) {\r\n\r\n allListeners.splice(index, 1);\r\n\r\n }\r\n\r\n }\r\n\r\n };\r\n\r\n listeners.removeAll = function () {\r\n\r\n allListeners.map(function (current) {\r\n\r\n listeners.remove(current.element, current.type, current.handler);\r\n\r\n });\r\n\r\n };\r\n\r\n listeners.get = function (element, eventType, handler) {\r\n\r\n return listeners.search.all(element, eventType, handler);\r\n\r\n };\r\n\r\n return listeners;\r\n\r\n}({});\n\n\n// WEBPACK FOOTER //\n// ./modules/listeners.js","/**\r\n * Codex Editor Destroyer module\r\n *\r\n * @auhor Codex Team\r\n * @version 1.0\r\n */\r\n\r\nmodule.exports = function (destroyer) {\r\n\r\n let editor = codex.editor;\r\n\r\n destroyer.removeNodes = function () {\r\n\r\n editor.nodes.wrapper.remove();\r\n editor.nodes.notifications.remove();\r\n\r\n };\r\n\r\n destroyer.destroyPlugins = function () {\r\n\r\n for (var tool in editor.tools) {\r\n\r\n if (typeof editor.tools[tool].destroy === 'function') {\r\n\r\n editor.tools[tool].destroy();\r\n\r\n }\r\n\r\n }\r\n\r\n };\r\n\r\n destroyer.destroyScripts = function () {\r\n\r\n var scripts = document.getElementsByTagName('SCRIPT');\r\n\r\n for (var i = 0; i < scripts.length; i++) {\r\n\r\n if (scripts[i].id.indexOf(editor.scriptPrefix) + 1) {\r\n\r\n scripts[i].remove();\r\n i--;\r\n\r\n }\r\n\r\n }\r\n\r\n };\r\n\r\n\r\n /**\r\n * Delete editor data from webpage.\r\n * You should send settings argument with boolean flags:\r\n * @param settings.ui- remove redactor event listeners and DOM nodes\r\n * @param settings.scripts - remove redactor scripts from DOM\r\n * @param settings.plugins - remove plugin's objects\r\n * @param settings.core - remove editor core. You can remove core only if UI and scripts flags is true\r\n * }\r\n *\r\n */\r\n destroyer.destroy = function (settings) {\r\n\r\n if (!settings || typeof settings !== 'object') {\r\n\r\n return;\r\n\r\n }\r\n\r\n if (settings.ui) {\r\n\r\n destroyer.removeNodes();\r\n editor.listeners.removeAll();\r\n\r\n }\r\n\r\n if (settings.scripts) {\r\n\r\n destroyer.destroyScripts();\r\n\r\n }\r\n\r\n if (settings.plugins) {\r\n\r\n destroyer.destroyPlugins();\r\n\r\n }\r\n\r\n if (settings.ui && settings.scripts && settings.core) {\r\n\r\n delete codex.editor;\r\n\r\n }\r\n\r\n };\r\n\r\n return destroyer;\r\n\r\n}({});\n\n\n// WEBPACK FOOTER //\n// ./modules/destroyer.js","/**\r\n * Codex Editor Paste module\r\n *\r\n * @author Codex Team\r\n * @version 1.1.1\r\n */\r\n\r\nmodule.exports = function (paste) {\r\n\r\n let editor = codex.editor;\r\n\r\n var patterns = [];\r\n\r\n paste.prepare = function () {\r\n\r\n var tools = editor.tools;\r\n\r\n for (var tool in tools) {\r\n\r\n if (!tools[tool].renderOnPastePatterns || !Array.isArray(tools[tool].renderOnPastePatterns)) {\r\n\r\n continue;\r\n\r\n }\r\n\r\n tools[tool].renderOnPastePatterns.map(function (pattern) {\r\n\r\n\r\n patterns.push(pattern);\r\n\r\n });\r\n\r\n }\r\n\r\n return Promise.resolve();\r\n\r\n };\r\n\r\n /**\r\n * Saves data\r\n * @param event\r\n */\r\n paste.pasted = function (event) {\r\n\r\n var clipBoardData = event.clipboardData || window.clipboardData,\r\n content = clipBoardData.getData('Text');\r\n\r\n var result = analize(content);\r\n\r\n if (result) {\r\n\r\n event.preventDefault();\r\n event.stopImmediatePropagation();\r\n\r\n }\r\n\r\n return result;\r\n\r\n };\r\n\r\n /**\r\n * Analizes pated string and calls necessary method\r\n */\r\n\r\n var analize = function (string) {\r\n\r\n var result = false,\r\n content = editor.content.currentNode,\r\n plugin = content.dataset.tool;\r\n\r\n patterns.map( function (pattern) {\r\n\r\n var execArray = pattern.regex.exec(string),\r\n match = execArray && execArray[0];\r\n\r\n if ( match && match === string.trim()) {\r\n\r\n /** current block is not empty */\r\n if ( content.textContent.trim() && plugin == editor.settings.initialBlockPlugin ) {\r\n\r\n pasteToNewBlock_();\r\n\r\n }\r\n\r\n pattern.callback(string, pattern);\r\n result = true;\r\n\r\n }\r\n\r\n });\r\n\r\n return result;\r\n\r\n };\r\n\r\n var pasteToNewBlock_ = function () {\r\n\r\n /** Create new initial block */\r\n editor.content.insertBlock({\r\n\r\n type : editor.settings.initialBlockPlugin,\r\n block : editor.tools[editor.settings.initialBlockPlugin].render({\r\n text : ''\r\n })\r\n\r\n }, false);\r\n\r\n };\r\n\r\n /**\r\n * This method prevents default behaviour.\r\n *\r\n * @param {Object} event\r\n * @protected\r\n *\r\n * @description We get from clipboard pasted data, sanitize, make a fragment that contains of this sanitized nodes.\r\n * Firstly, we need to memorize the caret position. We can do that by getting the range of selection.\r\n * After all, we insert clear fragment into caret placed position. Then, we should move the caret to the last node\r\n */\r\n paste.blockPasteCallback = function (event) {\r\n\r\n\r\n if (!needsToHandlePasteEvent(event.target)) {\r\n\r\n return;\r\n\r\n }\r\n\r\n /** get html pasted data - dirty data */\r\n var htmlData = event.clipboardData.getData('text/html'),\r\n plainData = event.clipboardData.getData('text/plain');\r\n\r\n /** Temporary DIV that is used to work with text's paragraphs as DOM-elements*/\r\n var paragraphs = editor.draw.node('DIV', '', {}),\r\n cleanData,\r\n wrappedData;\r\n\r\n /** Create fragment, that we paste to range after proccesing */\r\n cleanData = editor.sanitizer.clean(htmlData);\r\n\r\n /**\r\n * We wrap pasted text with

    tags to split it logically\r\n * @type {string}\r\n */\r\n wrappedData = editor.content.wrapTextWithParagraphs(cleanData, plainData);\r\n paragraphs.innerHTML = wrappedData;\r\n\r\n /**\r\n * If there only one paragraph, lets user agent paste it\r\n */\r\n if (paragraphs.childNodes.length == 1) {\r\n\r\n return;\r\n\r\n }\r\n\r\n /** Prevent default behaviour */\r\n event.preventDefault();\r\n\r\n insertPastedParagraphs(paragraphs.childNodes);\r\n\r\n };\r\n\r\n /**\r\n * Checks if we should handle paste event on block\r\n * @param block\r\n *\r\n * @return {boolean}\r\n */\r\n var needsToHandlePasteEvent = function (block) {\r\n\r\n /** If area is input or textarea then allow default behaviour */\r\n if ( editor.core.isNativeInput(block) ) {\r\n\r\n return false;\r\n\r\n }\r\n\r\n var editableParent = editor.content.getEditableParent(block);\r\n\r\n /** Allow paste when event target placed in Editable element */\r\n if (!editableParent) {\r\n\r\n return false;\r\n\r\n }\r\n\r\n return true;\r\n\r\n };\r\n\r\n var insertPastedParagraphs = function (paragraphs) {\r\n\r\n var NEW_BLOCK_TYPE = editor.settings.initialBlockPlugin,\r\n currentBlockContent = editor.content.currentNode.firstChild.firstChild;\r\n\r\n paragraphs.forEach(function (paragraph, index) {\r\n\r\n /**\r\n * If there was no data in working node, replace it with first paragraph of pasted text\r\n */\r\n if (index == 0 && currentBlockContent.innerHTML.trim() === '') {\r\n\r\n editor.content.switchBlock(editor.content.currentNode, editor.tools[NEW_BLOCK_TYPE].render({\r\n text : paragraph.innerHTML\r\n }), NEW_BLOCK_TYPE);\r\n\r\n return;\r\n\r\n }\r\n\r\n editor.content.insertBlock({\r\n type : NEW_BLOCK_TYPE,\r\n block : editor.tools[NEW_BLOCK_TYPE].render({\r\n text : paragraph.innerHTML\r\n })\r\n });\r\n\r\n editor.caret.inputIndex++;\r\n\r\n });\r\n\r\n editor.caret.setToPreviousBlock(editor.caret.getCurrentInputIndex() + 1);\r\n\r\n\r\n };\r\n\r\n\r\n return paste;\r\n\r\n}({});\n\n\n// WEBPACK FOOTER //\n// ./modules/paste.js"],"sourceRoot":""} \ No newline at end of file +{"version":3,"sources":["webpack:///codex-editor.js","webpack:///webpack/bootstrap b9ba5e0c4242473ccc41","webpack:///./codex.js","webpack:///./modules/core.js","webpack:///./modules/tools.js","webpack:///./modules/ui.js","webpack:///./modules/transport.js","webpack:///./modules/renderer.js","webpack:///./modules/saver.js","webpack:///./modules/content.js","webpack:///./modules/toolbar/toolbar.js","webpack:///./modules/toolbar/settings.js","webpack:///./modules/toolbar/inline.js","webpack:///./modules/toolbar/toolbox.js","webpack:///./modules/callbacks.js","webpack:///./modules/draw.js","webpack:///./modules/caret.js","webpack:///./modules/notifications.js","webpack:///./modules/parser.js","webpack:///./modules/sanitizer.js","webpack:///./~/html-janitor/src/html-janitor.js","webpack:///./modules/listeners.js","webpack:///./modules/destroyer.js","webpack:///./modules/paste.js"],"names":["codex","modules","__webpack_require__","moduleId","installedModules","exports","module","id","loaded","call","m","c","p","editor","version","scriptPrefix","init","core","tools","ui","transport","renderer","saver","content","toolbar","callback","draw","caret","notifications","parser","sanitizer","listeners","destroyer","paste","settings","holderId","initialBlockPlugin","nodes","holder","wrapper","inlineToolbar","buttons","actions","toolbox","plusButton","showSettingsButton","showTrashButton","blockSettings","pluginSettings","defaultSettings","toolbarButtons","redactor","state","jsonOutput","blocks","inputs","start","userSettings","prepare","then","makeBlocksFromData","saveInputs","catch","error","log","_typeof","Symbol","iterator","obj","constructor","prototype","Promise","resolve","reject","data","hideToolbar","document","getElementById","undefined","Error","msg","type","arg","window","console","e","insertAfter","target","element","parentNode","insertBefore","nextSibling","nodeTypes","TAG","TEXT","COMMENT","keys","BACKSPACE","TAB","ENTER","SHIFT","CTRL","ALT","ESC","SPACE","LEFT","UP","DOWN","RIGHT","DELETE","META","isDomNode","el","nodeType","this","isEmpty","Object","length","ajax","url","encodedString","isFormData","prop","XMLHTTP","XMLHttpRequest","ActiveXObject","async","test","encodeURIComponent","withCredentials","responseContext","beforeSendResult","beforeSend","open","isFormData_","setRequestHeader","progress","upload","onprogress","bind","onreadystatechange","readyState","status","success","responseText","send","importScript","scriptPath","instanceName","script","createElement","defer","onload","onerror","src","head","appendChild","object","FormData","isNativeInput","nativeInputAreas","indexOf","tagName","resolve_","reject_","pluginsRequiresPreparation","allPlugins","pluginName","plugin","push","waitAllPluginsPreparation_","plugins","allPluginsProcessed__","reduce","previousValue","iteration","pluginIsReady__","callPluginsPrepareMethod_","available","loadingMessage","config","className","BLOCK_CLASSNAME","BLOCK_CONTENT","BLOCK_STRETCHED","BLOCK_HIGHLIGHTED","SETTINGS_ITEM","makeToolBar_","addTools_","makeInlineToolbar_","addInlineToolbarTools_","makeNotificationHolder_","bindEvents_","container","inlineToolbarButtons","inlineToolbarActions","blockButtons","makeToolbarSettings_","toolbarContent","makeToolbarContent_","settingsButton","makeRemoveBlockButton","pluginsSettings","createHolder","tool","toolName","toolButton","iconClassname","displayInToolbox","render","toolbarButton","bold","icon","command","italic","underline","link","name","toolbarButtonInline","setInlineToolbarButtonBehaviour","add","globalKeydown","redactorKeyDown","globalKeyup","redactorClicked","plusButtonClicked","showSettingsButtonClicked","button","toolbarButtonClicked","addBlockHandlers","block","blockKeydown","blockPasteCallback","inline","show","querySelectorAll","Array","map","current","addInitialBlock","initialBlock","initialBlockType","setAttribute","insertBlock","workingNodeChanged","event","toolClicked","currentRequest","input","arguments","node","fileSelected","clearInput","i","files","formData","multiple","append","selectAndUpload","args","accept","click","abort","items","appendBlocks","nodeSequence","index","appendNodeAtIndex","getNodeAsync","createBlockFromData","blockData","blocksList","position","toolData","unavailableBlock","innerHTML","dataset","inputPosition","stretched","isStretched","saveBlocks","html","childNodes","makeQueue","queue","getBlockData","makeFormDataFromBlocks","save","savedData","output","blockContent","pluginsContent","validate","result","currentNode","editorAreaHightlighted","sync","markBlock","classList","clearMark","remove","getFirstLevelBlock","body","contains","targetNode","replaceBlock","targetBlock","newBlock","replaceChild","needPlaceCaret","workingBlock","newBlockContent","blockType","composeNewBlock_","currentInputIndex","getCurrentInputIndex","editableElement","querySelector","emptyText","createTextNode","set","move","showPlusButton","setTimeout","setToNextBlock","switchBlock","blockToReplace","newBlockComposed","getDeepestTextNodeFromPosition","text","blockChilds","textContent","trim","removeChild","lookingFromStart","getRange","selection","getSelection","getRangeAt","splitBlock","inputIndex","textBeforeCaret","textNodeBeforeCaret","textAfterCaret","textNodeAfterCaret","anchorNode","anchorNodeText","caretOffset","anchorOffset","currentBlock","substring","previousChilds","nextChilds","reachedCurrent","child","previousChildsLength","nextChildsLength","newNode","NEW_BLOCK_TYPE","mergeBlocks","targetInputIndex","targetInput","currentInputContent","isLastNode","allChecked","allSiblingsEmpty_","sibling","wrapTextWithParagraphs","htmlData","plainData","wrapPlainTextWithParagraphs","paragraph","blockTyped","newWrapper","firstLevelBlocks","cloneNode","plainText","split","join","getEditableParent","contentEditable","defaultToolbarHeight","defaultOffset","opened","close","toggle","hidePlusButton","newYCoordinate","offsetTop","style","transform","Math","floor","hideRemoveActions","setting","toolType","makeSettings","settingsBlock","removeBlockWrapper","settingButton","actionWrapper","confirmAction","cancelAction","removeButtonClicked","confirmRemovingRequest","cancelRemovingRequest","action","showRemoveActions","firstLevelBlocksCount","buttonsOpened","actionsOpened","wrappersOffset","storedSelection","showInlineToolbar","selectedText","getSelectionText","showButtons","getWrappersOffset","newCoordinateX","newCoordinateY","coords","getSelectionCoords","offsetHeight","x","left","y","scrollY","top","closeButtons","closeAction","createLinkAction","defaultToolAction","forEach","hightlight","offset","getOffset","_x","_y","isNaN","offsetLeft","clientLeft","clientTop","offsetParent","range","sel","createRange","collapse","boundingLeft","boundingTop","rangeCount","cloneRange","getClientRects","rect","toString","showActions","inlineToolbarAnchorInputKeydown_","keyCode","editable","restoreSelection","setAnchor","value","preventDefault","stopImmediatePropagation","clearRange","isActive","isLinkActive","saveSelection","inputForLink","focus","dataType","execCommand","containerEl","preSelectionRange","selectNodeContents","setEnd","startContainer","startOffset","end","savedSel","charIndex","setStart","nextCharIndex","nodeStack","foundStart","stop","pop","removeAllRanges","addRange","queryCommandState","setButtonHighlighted","removeButtonsHighLight","tag","leaf","currentTool","barButtons","nextToolIndex","toolToSelect","visibleTool","appendCallback","UNREPLACEBLE_TOOLS","workingNode","setToBlock","callbacks","enterKeyPressed_","tabKeyPressedOnRedactorsZone_","enterKeyPressedOnRedactorsZone_","escapeKeyPressedOnRedactorsZone_","defaultKeyPressedOnRedactorsZone_","arrowKeyPressed_","nativeInputs","nativeInputsAreEmpty","textContentIsEmpty","blockIsEmpty","enterPressedOnBlock_","saveCurrentInputIndex","isEnterPressedOnToolbar","enableLineBreaks","stopPropagation","shiftKey","currentSelection","currentSelectedNode","caretAtTheEndOfText","atTheEnd","isTextNodeHasParentBetweenContenteditable","enterPressedOnBlock","islastNode","detectWhenClickedOnFirstLevelBlockArea_","firstLevelBlock","indexOfLastInput","inputIsEmpty","currentNodeType","isInitialType","flag","blockRightOrDownArrowPressed_","backspacePressed_","blockLeftOrUpArrowPressed_","focusedNodeHolder","focusedNode","editableElementIndex","lastChild","deepestTextnode","caretInLastChild","firstChild","caretInFirstChild","caretAtTheBeginning","setToPreviousBlock","selectionLength","endOffset","atStart","currentToolType","ceBlock","bar","placeholder","div","toggler","classname","toolIcon","toolTitle","properties","focusedNodeIndex","nodeToSet","childs","nextInput","emptyTextElement","lastChildNode","lengthOfLastChildNode","previousInput","pluginsRender","isFirstNode","isOffsetZero","insertNode","deleteContents","setStartAfter","addToQueue","splice","errorThrown","errorMsg","notification","message","constructorSettings","create","time","okBtn","cancelBtn","okMsg","cancelMsg","confirmHandler","cancelHandler","confirm","cancel","inputField","clear","insertPastedContent","isFirstLevelBlock","janitor","Config","CUSTOM","BASIC","tags","a","href","rel","b","strong","em","span","init_","userCustomConfig","configuration","clean","dirtyString","customConfig","janitorInstance","__WEBPACK_AMD_DEFINE_FACTORY__","__WEBPACK_AMD_DEFINE_RESULT__","root","factory","HTMLJanitor","tagDefinitions","validConfigValues","k","every","isBlockElement","blockElementNames","nodeName","isInlineElement","inlineElementNames","createTreeWalker","NodeFilter","SHOW_TEXT","SHOW_ELEMENT","SHOW_COMMENT","getAllowedAttrs","shouldRejectNode","allowedAttrs","shouldRejectAttr","attr","attrName","toLowerCase","sandbox","_sanitize","treeWalker","_sanitized","Node","TEXT_NODE","COMMENT_NODE","containsBlockElement","isInline","some","isNotTopContainer","isNestedBlockElement","isInvalid","keepNestedBlockElements","attributes","removeAttribute","previousElementSibling","nextElementSibling","allListeners","search","byElement","context","listenersOnElement","listener","byType","eventType","listenersWithType","byHandler","handler","listenersWithHandler","one","all","isCapture","addEventListener","alreadyAddedListener","removeEventListener","existingListeners","removeAll","get","removeNodes","destroyPlugins","destroy","destroyScripts","scripts","getElementsByTagName","patterns","renderOnPastePatterns","isArray","pattern","pasted","clipBoardData","clipboardData","getData","analize","string","execArray","regex","exec","match","pasteToNewBlock_","needsToHandlePasteEvent","cleanData","wrappedData","paragraphs","childElementCount","insertPastedParagraphs","editableParent","currentBlockContent"],"mappings":"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","file":"codex-editor.js","sourcesContent":["var codex = codex || {}; codex[\"editor\"] =\n/******/ (function(modules) { // webpackBootstrap\n/******/ \t// The module cache\n/******/ \tvar installedModules = {};\n/******/\n/******/ \t// The require function\n/******/ \tfunction __webpack_require__(moduleId) {\n/******/\n/******/ \t\t// Check if module is in cache\n/******/ \t\tif(installedModules[moduleId])\n/******/ \t\t\treturn installedModules[moduleId].exports;\n/******/\n/******/ \t\t// Create a new module (and put it into the cache)\n/******/ \t\tvar module = installedModules[moduleId] = {\n/******/ \t\t\texports: {},\n/******/ \t\t\tid: moduleId,\n/******/ \t\t\tloaded: false\n/******/ \t\t};\n/******/\n/******/ \t\t// Execute the module function\n/******/ \t\tmodules[moduleId].call(module.exports, module, module.exports, __webpack_require__);\n/******/\n/******/ \t\t// Flag the module as loaded\n/******/ \t\tmodule.loaded = true;\n/******/\n/******/ \t\t// Return the exports of the module\n/******/ \t\treturn module.exports;\n/******/ \t}\n/******/\n/******/\n/******/ \t// expose the modules object (__webpack_modules__)\n/******/ \t__webpack_require__.m = modules;\n/******/\n/******/ \t// expose the module cache\n/******/ \t__webpack_require__.c = installedModules;\n/******/\n/******/ \t// __webpack_public_path__\n/******/ \t__webpack_require__.p = \"\";\n/******/\n/******/ \t// Load entry module and return exports\n/******/ \treturn __webpack_require__(0);\n/******/ })\n/************************************************************************/\n/******/ ([\n/* 0 */\n/***/ function(module, exports, __webpack_require__) {\n\n\t'use strict';\n\t\n\t/**\r\n\t *\r\n\t * Codex Editor\r\n\t *\r\n\t * @author Codex Team\r\n\t */\n\t\n\tmodule.exports = function (editor) {\n\t\n\t 'use strict';\n\t\n\t editor.version = (\"1.6.4\");\n\t editor.scriptPrefix = 'cdx-script-';\n\t\n\t var init = function init() {\n\t\n\t editor.core = __webpack_require__(1);\n\t editor.tools = __webpack_require__(2);\n\t editor.ui = __webpack_require__(3);\n\t editor.transport = __webpack_require__(4);\n\t editor.renderer = __webpack_require__(5);\n\t editor.saver = __webpack_require__(6);\n\t editor.content = __webpack_require__(7);\n\t editor.toolbar = __webpack_require__(8);\n\t editor.callback = __webpack_require__(12);\n\t editor.draw = __webpack_require__(13);\n\t editor.caret = __webpack_require__(14);\n\t editor.notifications = __webpack_require__(15);\n\t editor.parser = __webpack_require__(16);\n\t editor.sanitizer = __webpack_require__(17);\n\t editor.listeners = __webpack_require__(19);\n\t editor.destroyer = __webpack_require__(20);\n\t editor.paste = __webpack_require__(21);\n\t };\n\t\n\t /**\r\n\t * @public\r\n\t * holds initial settings\r\n\t */\n\t editor.settings = {\n\t tools: ['paragraph', 'header', 'picture', 'list', 'quote', 'code', 'twitter', 'instagram', 'smile'],\n\t holderId: 'codex-editor',\n\t\n\t // Type of block showing on empty editor\n\t initialBlockPlugin: 'paragraph'\n\t };\n\t\n\t /**\r\n\t * public\r\n\t *\r\n\t * Static nodes\r\n\t */\n\t editor.nodes = {\n\t holder: null,\n\t wrapper: null,\n\t toolbar: null,\n\t inlineToolbar: {\n\t wrapper: null,\n\t buttons: null,\n\t actions: null\n\t },\n\t toolbox: null,\n\t notifications: null,\n\t plusButton: null,\n\t showSettingsButton: null,\n\t showTrashButton: null,\n\t blockSettings: null,\n\t pluginSettings: null,\n\t defaultSettings: null,\n\t toolbarButtons: {}, // { type : DomEl, ... }\n\t redactor: null\n\t };\n\t\n\t /**\r\n\t * @public\r\n\t *\r\n\t * Output state\r\n\t */\n\t editor.state = {\n\t jsonOutput: [],\n\t blocks: [],\n\t inputs: []\n\t };\n\t\n\t /**\r\n\t * @public\r\n\t * Editor plugins\r\n\t */\n\t editor.tools = {};\n\t\n\t /**\r\n\t * Initialization\r\n\t * @uses Promise cEditor.core.prepare\r\n\t * @param {Object} userSettings\r\n\t * @param {Array} userSettings.tools list of plugins\r\n\t * @param {String} userSettings.holderId Element's id to append editor\r\n\t *\r\n\t * Load user defined tools\r\n\t * Tools must contain this important objects :\r\n\t * @param {String} type - this is a type of plugin. It can be used as plugin name\r\n\t * @param {String} iconClassname - this a icon in toolbar\r\n\t * @param {Object} make - what should plugin do, when it is clicked\r\n\t * @param {Object} appendCallback - callback after clicking\r\n\t * @param {Element} settings - what settings does it have\r\n\t * @param {Object} render - plugin get JSON, and should return HTML\r\n\t * @param {Object} save - plugin gets HTML content, returns JSON\r\n\t * @param {Boolean} displayInToolbox - will be displayed in toolbox. Default value is TRUE\r\n\t * @param {Boolean} enableLineBreaks - inserts new block or break lines. Default value is FALSE\r\n\t *\r\n\t * @example\r\n\t * - type : 'header',\r\n\t * - iconClassname : 'ce-icon-header',\r\n\t * - make : headerTool.make,\r\n\t * - appendCallback : headerTool.appendCallback,\r\n\t * - settings : headerTool.makeSettings(),\r\n\t * - render : headerTool.render,\r\n\t * - save : headerTool.save,\r\n\t * - displayInToolbox : true,\r\n\t * - enableLineBreaks : false\r\n\t */\n\t editor.start = function (userSettings) {\n\t\n\t init();\n\t\n\t editor.core.prepare(userSettings)\n\t\n\t // If all ok, make UI, bind events and parse initial-content\n\t .then(editor.ui.prepare).then(editor.tools.prepare).then(editor.sanitizer.prepare).then(editor.paste.prepare).then(editor.transport.prepare).then(editor.renderer.makeBlocksFromData).then(editor.ui.saveInputs).catch(function (error) {\n\t\n\t editor.core.log('Initialization failed with error: %o', 'warn', error);\n\t });\n\t };\n\t\n\t return editor;\n\t}({});\n\n/***/ },\n/* 1 */\n/***/ function(module, exports) {\n\n\t'use strict';\n\t\n\tvar _typeof = typeof Symbol === \"function\" && typeof Symbol.iterator === \"symbol\" ? function (obj) { return typeof obj; } : function (obj) { return obj && typeof Symbol === \"function\" && obj.constructor === Symbol && obj !== Symbol.prototype ? \"symbol\" : typeof obj; };\n\t\n\t/**\r\n\t * Codex Editor Core\r\n\t *\r\n\t * @author Codex Team\r\n\t * @version 1.1.3\r\n\t */\n\t\n\tmodule.exports = function (core) {\n\t\n\t var editor = codex.editor;\n\t\n\t /**\r\n\t * @public\r\n\t *\r\n\t * Editor preparing method\r\n\t * @return Promise\r\n\t */\n\t core.prepare = function (userSettings) {\n\t\n\t return new Promise(function (resolve, reject) {\n\t\n\t if (userSettings) {\n\t\n\t editor.settings.tools = userSettings.tools || editor.settings.tools;\n\t }\n\t\n\t if (userSettings.data) {\n\t\n\t editor.state.blocks = userSettings.data;\n\t }\n\t\n\t if (userSettings.initialBlockPlugin) {\n\t\n\t editor.settings.initialBlockPlugin = userSettings.initialBlockPlugin;\n\t }\n\t\n\t if (userSettings.sanitizer) {\n\t\n\t editor.settings.sanitizer = userSettings.sanitizer;\n\t }\n\t\n\t editor.hideToolbar = userSettings.hideToolbar;\n\t\n\t editor.nodes.holder = document.getElementById(userSettings.holderId || editor.settings.holderId);\n\t\n\t if (_typeof(editor.nodes.holder) === undefined || editor.nodes.holder === null) {\n\t\n\t reject(Error(\"Holder wasn't found by ID: #\" + userSettings.holderId));\n\t } else {\n\t\n\t resolve();\n\t }\n\t });\n\t };\n\t\n\t /**\r\n\t * Logging method\r\n\t * @param type = ['log', 'info', 'warn']\r\n\t */\n\t core.log = function (msg, type, arg) {\n\t\n\t type = type || 'log';\n\t\n\t if (!arg) {\n\t\n\t arg = msg || 'undefined';\n\t msg = '[codex-editor]: %o';\n\t } else {\n\t\n\t msg = '[codex-editor]: ' + msg;\n\t }\n\t\n\t try {\n\t\n\t if ('console' in window && window.console[type]) {\n\t\n\t if (arg) window.console[type](msg, arg);else window.console[type](msg);\n\t }\n\t } catch (e) {}\n\t };\n\t\n\t /**\r\n\t * @protected\r\n\t *\r\n\t * Helper for insert one element after another\r\n\t */\n\t core.insertAfter = function (target, element) {\n\t\n\t target.parentNode.insertBefore(element, target.nextSibling);\n\t };\n\t\n\t /**\r\n\t * @const\r\n\t *\r\n\t * Readable DOM-node types map\r\n\t */\n\t core.nodeTypes = {\n\t TAG: 1,\n\t TEXT: 3,\n\t COMMENT: 8\n\t };\n\t\n\t /**\r\n\t * @const\r\n\t * Readable keys map\r\n\t */\n\t core.keys = { BACKSPACE: 8, TAB: 9, ENTER: 13, SHIFT: 16, CTRL: 17, ALT: 18, ESC: 27, SPACE: 32, LEFT: 37, UP: 38, DOWN: 40, RIGHT: 39, DELETE: 46, META: 91 };\n\t\n\t /**\r\n\t * @protected\r\n\t *\r\n\t * Check object for DOM node\r\n\t */\n\t core.isDomNode = function (el) {\n\t\n\t return el && (typeof el === 'undefined' ? 'undefined' : _typeof(el)) === 'object' && el.nodeType && el.nodeType == this.nodeTypes.TAG;\n\t };\n\t\n\t /**\r\n\t * Checks passed object for emptiness\r\n\t * @require ES5 - Object.keys\r\n\t * @param {object}\r\n\t */\n\t core.isEmpty = function (obj) {\n\t\n\t return Object.keys(obj).length === 0;\n\t };\n\t\n\t /**\r\n\t * Native Ajax\r\n\t * @param {String} settings.url - request URL\r\n\t * @param {function} settings.beforeSend - returned value will be passed as context to the Success, Error and Progress callbacks\r\n\t * @param {function} settings.success\r\n\t * @param {function} settings.progress\r\n\t */\n\t core.ajax = function (settings) {\n\t\n\t if (!settings || !settings.url) {\n\t\n\t return;\n\t }\n\t\n\t var XMLHTTP = window.XMLHttpRequest ? new XMLHttpRequest() : new ActiveXObject('Microsoft.XMLHTTP'),\n\t encodedString,\n\t isFormData,\n\t prop;\n\t\n\t settings.async = true;\n\t settings.type = settings.type || 'GET';\n\t settings.data = settings.data || '';\n\t settings['content-type'] = settings['content-type'] || 'application/json; charset=utf-8';\n\t\n\t if (settings.type == 'GET' && settings.data) {\n\t\n\t settings.url = /\\?/.test(settings.url) ? settings.url + '&' + settings.data : settings.url + '?' + settings.data;\n\t } else {\n\t\n\t encodedString = '';\n\t for (prop in settings.data) {\n\t\n\t encodedString += prop + '=' + encodeURIComponent(settings.data[prop]) + '&';\n\t }\n\t }\n\t\n\t if (settings.withCredentials) {\n\t\n\t XMLHTTP.withCredentials = true;\n\t }\n\t\n\t /**\r\n\t * Value returned in beforeSend funtion will be passed as context to the other response callbacks\r\n\t * If beforeSend returns false, AJAX will be blocked\r\n\t */\n\t var responseContext = void 0,\n\t beforeSendResult = void 0;\n\t\n\t if (typeof settings.beforeSend === 'function') {\n\t\n\t beforeSendResult = settings.beforeSend.call();\n\t\n\t if (beforeSendResult === false) {\n\t\n\t return;\n\t }\n\t }\n\t\n\t XMLHTTP.open(settings.type, settings.url, settings.async);\n\t\n\t /**\r\n\t * If we send FormData, we need no content-type header\r\n\t */\n\t isFormData = isFormData_(settings.data);\n\t\n\t if (!isFormData) {\n\t\n\t if (settings.type !== 'POST') {\n\t\n\t XMLHTTP.setRequestHeader('Content-type', settings['content-type']);\n\t } else {\n\t\n\t XMLHTTP.setRequestHeader('Content-type', 'application/x-www-form-urlencoded');\n\t }\n\t }\n\t\n\t XMLHTTP.setRequestHeader('X-Requested-With', 'XMLHttpRequest');\n\t\n\t responseContext = beforeSendResult || XMLHTTP;\n\t\n\t if (typeof settings.progress === 'function') {\n\t\n\t XMLHTTP.upload.onprogress = settings.progress.bind(responseContext);\n\t }\n\t\n\t XMLHTTP.onreadystatechange = function () {\n\t\n\t if (XMLHTTP.readyState === 4) {\n\t\n\t if (XMLHTTP.status === 200) {\n\t\n\t if (typeof settings.success === 'function') {\n\t\n\t settings.success.call(responseContext, XMLHTTP.responseText);\n\t }\n\t } else {\n\t\n\t if (typeof settings.error === 'function') {\n\t\n\t settings.error.call(responseContext, XMLHTTP.responseText, XMLHTTP.status);\n\t }\n\t }\n\t }\n\t };\n\t\n\t if (isFormData) {\n\t\n\t // Sending FormData\n\t XMLHTTP.send(settings.data);\n\t } else {\n\t\n\t // POST requests\n\t XMLHTTP.send(encodedString);\n\t }\n\t\n\t return XMLHTTP;\n\t };\n\t\n\t /**\r\n\t * Appends script to head of document\r\n\t * @return Promise\r\n\t */\n\t core.importScript = function (scriptPath, instanceName) {\n\t\n\t return new Promise(function (resolve, reject) {\n\t\n\t var script = void 0;\n\t\n\t /** Script is already loaded */\n\t if (!instanceName) {\n\t\n\t reject('Instance name is missed');\n\t } else if (document.getElementById(editor.scriptPrefix + instanceName)) {\n\t\n\t resolve(scriptPath);\n\t }\n\t\n\t script = document.createElement('SCRIPT');\n\t script.async = true;\n\t script.defer = true;\n\t script.id = editor.scriptPrefix + instanceName;\n\t\n\t script.onload = function () {\n\t\n\t resolve(scriptPath);\n\t };\n\t\n\t script.onerror = function () {\n\t\n\t reject(scriptPath);\n\t };\n\t\n\t script.src = scriptPath;\n\t document.head.appendChild(script);\n\t });\n\t };\n\t\n\t /**\r\n\t * Function for checking is it FormData object to send.\r\n\t * @param {Object} object to check\r\n\t * @return boolean\r\n\t */\n\t var isFormData_ = function isFormData_(object) {\n\t\n\t return object instanceof FormData;\n\t };\n\t\n\t /**\r\n\t * Check block\r\n\t * @param target\r\n\t * @description Checks target is it native input\r\n\t */\n\t core.isNativeInput = function (target) {\n\t\n\t var nativeInputAreas = ['INPUT', 'TEXTAREA'];\n\t\n\t return nativeInputAreas.indexOf(target.tagName) != -1;\n\t };\n\t\n\t return core;\n\t}({});\n\n/***/ },\n/* 2 */\n/***/ function(module, exports) {\n\n\t'use strict';\n\t\n\t/**\r\n\t* Module working with plugins\r\n\t*/\n\tmodule.exports = function () {\n\t\n\t var editor = codex.editor;\n\t\n\t /**\r\n\t * Initialize plugins before using\r\n\t * Ex. Load scripts or call some internal methods\r\n\t * @return Promise\r\n\t */\n\t function prepare() {\n\t\n\t return new Promise(function (resolve_, reject_) {\n\t\n\t Promise.resolve()\n\t\n\t /**\r\n\t * Compose a sequence of plugins that requires preparation\r\n\t */\n\t .then(function () {\n\t\n\t var pluginsRequiresPreparation = [],\n\t allPlugins = editor.tools;\n\t\n\t for (var pluginName in allPlugins) {\n\t\n\t var plugin = allPlugins[pluginName];\n\t\n\t if (plugin.prepare && typeof plugin.prepare != 'function' || !plugin.prepare) {\n\t\n\t continue;\n\t }\n\t\n\t pluginsRequiresPreparation.push(plugin);\n\t }\n\t\n\t /**\r\n\t * If no one passed plugins requires preparation, finish prepare() and go ahead\r\n\t */\n\t if (!pluginsRequiresPreparation.length) {\n\t\n\t resolve_();\n\t }\n\t\n\t return pluginsRequiresPreparation;\n\t })\n\t\n\t /** Wait plugins while they prepares */\n\t .then(waitAllPluginsPreparation_).then(function () {\n\t\n\t editor.core.log('Plugins loaded', 'info');\n\t resolve_();\n\t }).catch(function (error) {\n\t\n\t reject_(error);\n\t });\n\t });\n\t }\n\t\n\t /**\r\n\t * @param {array} plugins - list of tools that requires preparation\r\n\t * @return {Promise} resolved while all plugins will be ready or failed\r\n\t */\n\t function waitAllPluginsPreparation_(plugins) {\n\t\n\t /**\r\n\t * @calls allPluginsProcessed__ when all plugins prepared or failed\r\n\t */\n\t return new Promise(function (allPluginsProcessed__) {\n\t\n\t /**\r\n\t * pluck each element from queue\r\n\t * First, send resolved Promise as previous value\r\n\t * Each plugins \"prepare\" method returns a Promise, that's why\r\n\t * reduce current element will not be able to continue while can't get\r\n\t * a resolved Promise\r\n\t *\r\n\t * If last plugin is \"prepared\" then go to the next stage of initialization\r\n\t */\n\t plugins.reduce(function (previousValue, plugin, iteration) {\n\t\n\t return previousValue.then(function () {\n\t\n\t /**\r\n\t * Wait till plugins prepared\r\n\t * @calls pluginIsReady__ when plugin is ready or failed\r\n\t */\n\t return new Promise(function (pluginIsReady__) {\n\t\n\t callPluginsPrepareMethod_(plugin).then(pluginIsReady__).then(function () {\n\t\n\t plugin.available = true;\n\t }).catch(function (error) {\n\t\n\t editor.core.log('Plugin \\xAB' + plugin.type + '\\xBB was not loaded. Preparation failed because %o', 'warn', error);\n\t plugin.available = false;\n\t plugin.loadingMessage = error;\n\t\n\t /** Go ahead even some plugin has problems */\n\t pluginIsReady__();\n\t }).then(function () {\n\t\n\t /** If last plugin has problems then just ignore and continue */\n\t if (iteration == plugins.length - 1) {\n\t\n\t allPluginsProcessed__();\n\t }\n\t });\n\t });\n\t });\n\t }, Promise.resolve());\n\t });\n\t }\n\t\n\t var callPluginsPrepareMethod_ = function callPluginsPrepareMethod_(plugin) {\n\t\n\t return plugin.prepare(plugin.config || {});\n\t };\n\t\n\t return {\n\t prepare: prepare\n\t };\n\t}();\n\n/***/ },\n/* 3 */\n/***/ function(module, exports) {\n\n\t'use strict';\n\t\n\t/**\r\n\t * Codex Editor UI module\r\n\t *\r\n\t * @author Codex Team\r\n\t * @version 1.2.0\r\n\t */\n\t\n\tmodule.exports = function (ui) {\n\t\n\t var editor = codex.editor;\n\t\n\t /**\r\n\t * Basic editor classnames\r\n\t */\n\t ui.className = {\n\t\n\t /**\r\n\t * @const {string} BLOCK_CLASSNAME - redactor blocks name\r\n\t */\n\t BLOCK_CLASSNAME: 'ce-block',\n\t\n\t /**\r\n\t * @const {String} wrapper for plugins content\r\n\t */\n\t BLOCK_CONTENT: 'ce-block__content',\n\t\n\t /**\r\n\t * @const {String} BLOCK_STRETCHED - makes block stretched\r\n\t */\n\t BLOCK_STRETCHED: 'ce-block--stretched',\n\t\n\t /**\r\n\t * @const {String} BLOCK_HIGHLIGHTED - adds background\r\n\t */\n\t BLOCK_HIGHLIGHTED: 'ce-block--focused',\n\t\n\t /**\r\n\t * @const {String} - for all default settings\r\n\t */\n\t SETTINGS_ITEM: 'ce-settings__item'\n\t\n\t };\n\t\n\t /**\r\n\t * @protected\r\n\t *\r\n\t * Making main interface\r\n\t */\n\t ui.prepare = function () {\n\t\n\t return new Promise(function (resolve) {\n\t\n\t var wrapper = editor.draw.wrapper(),\n\t redactor = editor.draw.redactor(),\n\t toolbar = makeToolBar_();\n\t\n\t wrapper.appendChild(toolbar);\n\t wrapper.appendChild(redactor);\n\t\n\t /** Save created ui-elements to static nodes state */\n\t editor.nodes.wrapper = wrapper;\n\t editor.nodes.redactor = redactor;\n\t\n\t /** Append editor wrapper with redactor zone into holder */\n\t editor.nodes.holder.appendChild(wrapper);\n\t\n\t resolve();\n\t })\n\t\n\t /** Add toolbox tools */\n\t .then(addTools_)\n\t\n\t /** Make container for inline toolbar */\n\t .then(makeInlineToolbar_)\n\t\n\t /** Add inline toolbar tools */\n\t .then(addInlineToolbarTools_)\n\t\n\t /** Draw wrapper for notifications */\n\t .then(makeNotificationHolder_)\n\t\n\t /** Add eventlisteners to redactor elements */\n\t .then(bindEvents_).catch(function () {\n\t\n\t editor.core.log(\"Can't draw editor interface\");\n\t });\n\t };\n\t\n\t /**\r\n\t * @private\r\n\t * Draws inline toolbar zone\r\n\t */\n\t var makeInlineToolbar_ = function makeInlineToolbar_() {\n\t\n\t var container = editor.draw.inlineToolbar();\n\t\n\t /** Append to redactor new inline block */\n\t editor.nodes.inlineToolbar.wrapper = container;\n\t\n\t /** Draw toolbar buttons */\n\t editor.nodes.inlineToolbar.buttons = editor.draw.inlineToolbarButtons();\n\t\n\t /** Buttons action or settings */\n\t editor.nodes.inlineToolbar.actions = editor.draw.inlineToolbarActions();\n\t\n\t /** Append to inline toolbar buttons as part of it */\n\t editor.nodes.inlineToolbar.wrapper.appendChild(editor.nodes.inlineToolbar.buttons);\n\t editor.nodes.inlineToolbar.wrapper.appendChild(editor.nodes.inlineToolbar.actions);\n\t\n\t editor.nodes.wrapper.appendChild(editor.nodes.inlineToolbar.wrapper);\n\t };\n\t\n\t var makeToolBar_ = function makeToolBar_() {\n\t\n\t var toolbar = editor.draw.toolbar(),\n\t blockButtons = makeToolbarSettings_(),\n\t toolbarContent = makeToolbarContent_();\n\t\n\t /** Appending first-level block buttons */\n\t toolbar.appendChild(blockButtons);\n\t\n\t /** Append toolbarContent to toolbar */\n\t toolbar.appendChild(toolbarContent);\n\t\n\t /** Make toolbar global */\n\t editor.nodes.toolbar = toolbar;\n\t\n\t return toolbar;\n\t };\n\t\n\t var makeToolbarContent_ = function makeToolbarContent_() {\n\t\n\t var toolbarContent = editor.draw.toolbarContent(),\n\t toolbox = editor.draw.toolbox(),\n\t plusButton = editor.draw.plusButton();\n\t\n\t /** Append plus button */\n\t toolbarContent.appendChild(plusButton);\n\t\n\t /** Appending toolbar tools */\n\t toolbarContent.appendChild(toolbox);\n\t\n\t /** Make Toolbox and plusButton global */\n\t editor.nodes.toolbox = toolbox;\n\t editor.nodes.plusButton = plusButton;\n\t\n\t return toolbarContent;\n\t };\n\t\n\t var makeToolbarSettings_ = function makeToolbarSettings_() {\n\t\n\t var blockSettings = editor.draw.blockSettings(),\n\t blockButtons = editor.draw.blockButtons(),\n\t defaultSettings = editor.draw.defaultSettings(),\n\t showSettingsButton = editor.draw.settingsButton(),\n\t showTrashButton = editor.toolbar.settings.makeRemoveBlockButton(),\n\t pluginSettings = editor.draw.pluginsSettings();\n\t\n\t /** Add default and plugins settings */\n\t blockSettings.appendChild(pluginSettings);\n\t blockSettings.appendChild(defaultSettings);\n\t\n\t /**\r\n\t * Make blocks buttons\r\n\t * This block contains settings button and remove block button\r\n\t */\n\t blockButtons.appendChild(showSettingsButton);\n\t blockButtons.appendChild(showTrashButton);\n\t blockButtons.appendChild(blockSettings);\n\t\n\t /** Make BlockSettings, PluginSettings, DefaultSettings global */\n\t editor.nodes.blockSettings = blockSettings;\n\t editor.nodes.pluginSettings = pluginSettings;\n\t editor.nodes.defaultSettings = defaultSettings;\n\t editor.nodes.showSettingsButton = showSettingsButton;\n\t editor.nodes.showTrashButton = showTrashButton;\n\t\n\t return blockButtons;\n\t };\n\t\n\t /** Draw notifications holder */\n\t var makeNotificationHolder_ = function makeNotificationHolder_() {\n\t\n\t /** Append block with notifications to the document */\n\t editor.nodes.notifications = editor.notifications.createHolder();\n\t };\n\t\n\t /**\r\n\t * @private\r\n\t * Append tools passed in editor.tools\r\n\t */\n\t var addTools_ = function addTools_() {\n\t\n\t var tool, toolName, toolButton;\n\t\n\t for (toolName in editor.settings.tools) {\n\t\n\t tool = editor.settings.tools[toolName];\n\t\n\t editor.tools[toolName] = tool;\n\t\n\t if (!tool.iconClassname && tool.displayInToolbox) {\n\t\n\t editor.core.log('Toolbar icon classname missed. Tool %o skipped', 'warn', toolName);\n\t continue;\n\t }\n\t\n\t if (typeof tool.render != 'function') {\n\t\n\t editor.core.log('render method missed. Tool %o skipped', 'warn', toolName);\n\t continue;\n\t }\n\t\n\t if (!tool.displayInToolbox) {\n\t\n\t continue;\n\t } else {\n\t\n\t /** if tools is for toolbox */\n\t toolButton = editor.draw.toolbarButton(toolName, tool.iconClassname);\n\t\n\t editor.nodes.toolbox.appendChild(toolButton);\n\t\n\t editor.nodes.toolbarButtons[toolName] = toolButton;\n\t }\n\t }\n\t };\n\t\n\t var addInlineToolbarTools_ = function addInlineToolbarTools_() {\n\t\n\t var tools = {\n\t\n\t bold: {\n\t icon: 'ce-icon-bold',\n\t command: 'bold'\n\t },\n\t\n\t italic: {\n\t icon: 'ce-icon-italic',\n\t command: 'italic'\n\t },\n\t\n\t underline: {\n\t icon: 'ce-icon-underline',\n\t command: 'underline'\n\t },\n\t\n\t link: {\n\t icon: 'ce-icon-link',\n\t command: 'createLink'\n\t }\n\t };\n\t\n\t var toolButton, tool;\n\t\n\t for (var name in tools) {\n\t\n\t tool = tools[name];\n\t\n\t toolButton = editor.draw.toolbarButtonInline(name, tool.icon);\n\t\n\t editor.nodes.inlineToolbar.buttons.appendChild(toolButton);\n\t /**\r\n\t * Add callbacks to this buttons\r\n\t */\n\t editor.ui.setInlineToolbarButtonBehaviour(toolButton, tool.command);\n\t }\n\t };\n\t\n\t /**\r\n\t * @private\r\n\t * Bind editor UI events\r\n\t */\n\t var bindEvents_ = function bindEvents_() {\n\t\n\t editor.core.log('ui.bindEvents fired', 'info');\n\t\n\t // window.addEventListener('error', function (errorMsg, url, lineNumber) {\n\t // editor.notifications.errorThrown(errorMsg, event);\n\t // }, false );\n\t\n\t /** All keydowns on Document */\n\t editor.listeners.add(document, 'keydown', editor.callback.globalKeydown, false);\n\t\n\t /** All keydowns on Redactor zone */\n\t editor.listeners.add(editor.nodes.redactor, 'keydown', editor.callback.redactorKeyDown, false);\n\t\n\t /** All keydowns on Document */\n\t editor.listeners.add(document, 'keyup', editor.callback.globalKeyup, false);\n\t\n\t /**\r\n\t * Mouse click to radactor\r\n\t */\n\t editor.listeners.add(editor.nodes.redactor, 'click', editor.callback.redactorClicked, false);\n\t\n\t /**\r\n\t * Clicks to the Plus button\r\n\t */\n\t editor.listeners.add(editor.nodes.plusButton, 'click', editor.callback.plusButtonClicked, false);\n\t\n\t /**\r\n\t * Clicks to SETTINGS button in toolbar\r\n\t */\n\t editor.listeners.add(editor.nodes.showSettingsButton, 'click', editor.callback.showSettingsButtonClicked, false);\n\t\n\t /** Bind click listeners on toolbar buttons */\n\t for (var button in editor.nodes.toolbarButtons) {\n\t\n\t editor.listeners.add(editor.nodes.toolbarButtons[button], 'click', editor.callback.toolbarButtonClicked, false);\n\t }\n\t };\n\t\n\t ui.addBlockHandlers = function (block) {\n\t\n\t if (!block) return;\n\t\n\t /**\r\n\t * Block keydowns\r\n\t */\n\t editor.listeners.add(block, 'keydown', editor.callback.blockKeydown, false);\n\t\n\t /**\r\n\t * Pasting content from another source\r\n\t * We have two type of sanitization\r\n\t * First - uses deep-first search algorithm to get sub nodes,\r\n\t * sanitizes whole Block_content and replaces cleared nodes\r\n\t * This method is deprecated\r\n\t * Method is used in editor.callback.blockPaste(event)\r\n\t *\r\n\t * Secont - uses Mutation observer.\r\n\t * Observer \"observe\" DOM changes and send changings to callback.\r\n\t * Callback gets changed node, not whole Block_content.\r\n\t * Inserted or changed node, which we've gotten have been cleared and replaced with diry node\r\n\t *\r\n\t * Method is used in editor.callback.blockPasteViaSanitize(event)\r\n\t *\r\n\t * @uses html-janitor\r\n\t * @example editor.callback.blockPasteViaSanitize(event), the second method.\r\n\t *\r\n\t */\n\t editor.listeners.add(block, 'paste', editor.paste.blockPasteCallback, false);\n\t\n\t editor.listeners.add(block, 'mouseup', editor.toolbar.inline.show, false);\n\t };\n\t\n\t /** getting all contenteditable elements */\n\t ui.saveInputs = function () {\n\t\n\t var redactor = editor.nodes.redactor;\n\t\n\t editor.state.inputs = [];\n\t\n\t /** Save all inputs in global variable state */\n\t var inputs = redactor.querySelectorAll('[contenteditable], input, textarea');\n\t\n\t Array.prototype.map.call(inputs, function (current) {\n\t\n\t if (!current.type || current.type == 'text' || current.type == 'textarea') {\n\t\n\t editor.state.inputs.push(current);\n\t }\n\t });\n\t };\n\t\n\t /**\r\n\t * Adds first initial block on empty redactor\r\n\t */\n\t ui.addInitialBlock = function () {\n\t\n\t var initialBlockType = editor.settings.initialBlockPlugin,\n\t initialBlock;\n\t\n\t if (!editor.tools[initialBlockType]) {\n\t\n\t editor.core.log('Plugin %o was not implemented and can\\'t be used as initial block', 'warn', initialBlockType);\n\t return;\n\t }\n\t\n\t initialBlock = editor.tools[initialBlockType].render();\n\t\n\t initialBlock.setAttribute('data-placeholder', 'Расскажите свою историю...');\n\t\n\t editor.content.insertBlock({\n\t type: initialBlockType,\n\t block: initialBlock\n\t });\n\t\n\t editor.content.workingNodeChanged(initialBlock);\n\t };\n\t\n\t ui.setInlineToolbarButtonBehaviour = function (button, type) {\n\t\n\t editor.listeners.add(button, 'mousedown', function (event) {\n\t\n\t editor.toolbar.inline.toolClicked(event, type);\n\t }, false);\n\t };\n\t\n\t return ui;\n\t}({});\n\n/***/ },\n/* 4 */\n/***/ function(module, exports) {\n\n\t'use strict';\n\t\n\t/**\r\n\t *\r\n\t * Codex.Editor Transport Module\r\n\t *\r\n\t * @copyright 2017 Codex-Team\r\n\t * @version 1.2.0\r\n\t */\n\t\n\tmodule.exports = function (transport) {\n\t\n\t var editor = codex.editor;\n\t\n\t /**\r\n\t * @private {Object} current XmlHttpRequest instance\r\n\t */\n\t var currentRequest = null;\n\t\n\t /**\r\n\t * @type {null} | {DOMElement} input - keeps input element in memory\r\n\t */\n\t transport.input = null;\n\t\n\t /**\r\n\t * @property {Object} arguments - keep plugin settings and defined callbacks\r\n\t */\n\t transport.arguments = null;\n\t\n\t /**\r\n\t * Prepares input element where will be files\r\n\t */\n\t transport.prepare = function () {\n\t\n\t var input = editor.draw.node('INPUT', '', { type: 'file' });\n\t\n\t editor.listeners.add(input, 'change', editor.transport.fileSelected);\n\t editor.transport.input = input;\n\t };\n\t\n\t /** Clear input when files is uploaded */\n\t transport.clearInput = function () {\n\t\n\t /** Remove old input */\n\t transport.input = null;\n\t\n\t /** Prepare new one */\n\t transport.prepare();\n\t };\n\t\n\t /**\r\n\t * Callback for file selection\r\n\t * @param {Event} event\r\n\t */\n\t transport.fileSelected = function () {\n\t\n\t var input = this,\n\t i,\n\t files = input.files,\n\t formData = new FormData();\n\t\n\t if (editor.transport.arguments.multiple === true) {\n\t\n\t for (i = 0; i < files.length; i++) {\n\t\n\t formData.append('files[]', files[i], files[i].name);\n\t }\n\t } else {\n\t\n\t formData.append('files', files[0], files[0].name);\n\t }\n\t\n\t currentRequest = editor.core.ajax({\n\t type: 'POST',\n\t data: formData,\n\t url: editor.transport.arguments.url,\n\t beforeSend: editor.transport.arguments.beforeSend,\n\t success: editor.transport.arguments.success,\n\t error: editor.transport.arguments.error,\n\t progress: editor.transport.arguments.progress\n\t });\n\t\n\t /** Clear input */\n\t transport.clearInput();\n\t };\n\t\n\t /**\r\n\t * Use plugin callbacks\r\n\t * @protected\r\n\t *\r\n\t * @param {Object} args - can have :\r\n\t * @param {String} args.url - fetch URL\r\n\t * @param {Function} args.beforeSend - function calls before sending ajax\r\n\t * @param {Function} args.success - success callback\r\n\t * @param {Function} args.error - on error handler\r\n\t * @param {Function} args.progress - xhr onprogress handler\r\n\t * @param {Boolean} args.multiple - allow select several files\r\n\t * @param {String} args.accept - adds accept attribute\r\n\t */\n\t transport.selectAndUpload = function (args) {\n\t\n\t transport.arguments = args;\n\t\n\t if (args.multiple === true) {\n\t\n\t transport.input.setAttribute('multiple', 'multiple');\n\t }\n\t\n\t if (args.accept) {\n\t\n\t transport.input.setAttribute('accept', args.accept);\n\t }\n\t\n\t transport.input.click();\n\t };\n\t\n\t transport.abort = function () {\n\t\n\t currentRequest.abort();\n\t\n\t currentRequest = null;\n\t };\n\t\n\t return transport;\n\t}({});\n\n/***/ },\n/* 5 */\n/***/ function(module, exports) {\n\n\t'use strict';\n\t\n\t/**\r\n\t * Codex Editor Renderer Module\r\n\t *\r\n\t * @author Codex Team\r\n\t * @version 1.0\r\n\t */\n\t\n\tmodule.exports = function (renderer) {\n\t\n\t var editor = codex.editor;\n\t\n\t /**\r\n\t * Asyncronously parses input JSON to redactor blocks\r\n\t */\n\t renderer.makeBlocksFromData = function () {\n\t\n\t /**\r\n\t * If redactor is empty, add first paragraph to start writing\r\n\t */\n\t if (editor.core.isEmpty(editor.state.blocks) || !editor.state.blocks.items.length) {\n\t\n\t editor.ui.addInitialBlock();\n\t return;\n\t }\n\t\n\t Promise.resolve()\n\t\n\t /** First, get JSON from state */\n\t .then(function () {\n\t\n\t return editor.state.blocks;\n\t })\n\t\n\t /** Then, start to iterate they */\n\t .then(editor.renderer.appendBlocks)\n\t\n\t /** Write log if something goes wrong */\n\t .catch(function (error) {\n\t\n\t editor.core.log('Error while parsing JSON: %o', 'error', error);\n\t });\n\t };\n\t\n\t /**\r\n\t * Parses JSON to blocks\r\n\t * @param {object} data\r\n\t * @return Primise -> nodeList\r\n\t */\n\t renderer.appendBlocks = function (data) {\n\t\n\t var blocks = data.items;\n\t\n\t /**\r\n\t * Sequence of one-by-one blocks appending\r\n\t * Uses to save blocks order after async-handler\r\n\t */\n\t var nodeSequence = Promise.resolve();\n\t\n\t for (var index = 0; index < blocks.length; index++) {\n\t\n\t /** Add node to sequence at specified index */\n\t editor.renderer.appendNodeAtIndex(nodeSequence, blocks, index);\n\t }\n\t };\n\t\n\t /**\r\n\t * Append node at specified index\r\n\t */\n\t renderer.appendNodeAtIndex = function (nodeSequence, blocks, index) {\n\t\n\t /** We need to append node to sequence */\n\t nodeSequence\n\t\n\t /** first, get node async-aware */\n\t .then(function () {\n\t\n\t return editor.renderer.getNodeAsync(blocks, index);\n\t })\n\t\n\t /**\r\n\t * second, compose editor-block from JSON object\r\n\t */\n\t .then(editor.renderer.createBlockFromData)\n\t\n\t /**\r\n\t * now insert block to redactor\r\n\t */\n\t .then(function (blockData) {\n\t\n\t /**\r\n\t * blockData has 'block', 'type' and 'stretched' information\r\n\t */\n\t editor.content.insertBlock(blockData);\n\t\n\t /** Pass created block to next step */\n\t return blockData.block;\n\t })\n\t\n\t /** Log if something wrong with node */\n\t .catch(function (error) {\n\t\n\t editor.core.log('Node skipped while parsing because %o', 'error', error);\n\t });\n\t };\n\t\n\t /**\r\n\t * Asynchronously returns block data from blocksList by index\r\n\t * @return Promise to node\r\n\t */\n\t renderer.getNodeAsync = function (blocksList, index) {\n\t\n\t return Promise.resolve().then(function () {\n\t\n\t return {\n\t tool: blocksList[index],\n\t position: index\n\t };\n\t });\n\t };\n\t\n\t /**\r\n\t * Creates editor block by JSON-data\r\n\t *\r\n\t * @uses render method of each plugin\r\n\t *\r\n\t * @param {Object} toolData.tool\r\n\t * { header : {\r\n\t * text: '',\r\n\t * type: 'H3', ...\r\n\t * }\r\n\t * }\r\n\t * @param {Number} toolData.position - index in input-blocks array\r\n\t * @return {Object} with type and Element\r\n\t */\n\t renderer.createBlockFromData = function (toolData) {\n\t\n\t /** New parser */\n\t var block,\n\t tool = toolData.tool,\n\t pluginName = tool.type;\n\t\n\t /** Get first key of object that stores plugin name */\n\t // for (var pluginName in blockData) break;\n\t\n\t /** Check for plugin existance */\n\t if (!editor.tools[pluginName]) {\n\t\n\t throw Error('Plugin \\xAB' + pluginName + '\\xBB not found');\n\t }\n\t\n\t /** Check for plugin having render method */\n\t if (typeof editor.tools[pluginName].render != 'function') {\n\t\n\t throw Error('Plugin \\xAB' + pluginName + '\\xBB must have \\xABrender\\xBB method');\n\t }\n\t\n\t if (editor.tools[pluginName].available === false) {\n\t\n\t block = editor.draw.unavailableBlock();\n\t\n\t block.innerHTML = editor.tools[pluginName].loadingMessage;\n\t\n\t /**\r\n\t * Saver will extract data from initial block data by position in array\r\n\t */\n\t block.dataset.inputPosition = toolData.position;\n\t } else {\n\t\n\t /** New Parser */\n\t block = editor.tools[pluginName].render(tool.data);\n\t }\n\t\n\t /** is first-level block stretched */\n\t var stretched = editor.tools[pluginName].isStretched || false;\n\t\n\t /** Retrun type and block */\n\t return {\n\t type: pluginName,\n\t block: block,\n\t stretched: stretched\n\t };\n\t };\n\t\n\t return renderer;\n\t}({});\n\n/***/ },\n/* 6 */\n/***/ function(module, exports) {\n\n\t'use strict';\n\t\n\t/**\r\n\t * Codex Editor Saver\r\n\t *\r\n\t * @author Codex Team\r\n\t * @version 1.0.2\r\n\t */\n\t\n\tmodule.exports = function (saver) {\n\t\n\t var editor = codex.editor;\n\t\n\t /**\r\n\t * Saves blocks\r\n\t * @private\r\n\t */\n\t saver.saveBlocks = function () {\n\t\n\t /** Save html content of redactor to memory */\n\t editor.state.html = editor.nodes.redactor.innerHTML;\n\t\n\t /** Empty jsonOutput state */\n\t editor.state.jsonOutput = [];\n\t\n\t Promise.resolve().then(function () {\n\t\n\t return editor.nodes.redactor.childNodes;\n\t })\n\t /** Making a sequence from separate blocks */\n\t .then(editor.saver.makeQueue).then(function () {\n\t // editor.nodes.textarea.innerHTML = editor.state.html;\n\t }).catch(function (error) {\n\t\n\t editor.core.log(error);\n\t });\n\t };\n\t\n\t saver.makeQueue = function (blocks) {\n\t\n\t var queue = Promise.resolve();\n\t\n\t for (var index = 0; index < blocks.length; index++) {\n\t\n\t /** Add node to sequence at specified index */\n\t editor.saver.getBlockData(queue, blocks, index);\n\t }\n\t };\n\t\n\t /** Gets every block and makes From Data */\n\t saver.getBlockData = function (queue, blocks, index) {\n\t\n\t queue.then(function () {\n\t\n\t return editor.saver.getNodeAsync(blocks, index);\n\t }).then(editor.saver.makeFormDataFromBlocks);\n\t };\n\t\n\t /**\r\n\t * Asynchronously returns block data from blocksList by index\r\n\t * @return Promise to node\r\n\t */\n\t saver.getNodeAsync = function (blocksList, index) {\n\t\n\t return Promise.resolve().then(function () {\n\t\n\t return blocksList[index];\n\t });\n\t };\n\t\n\t saver.makeFormDataFromBlocks = function (block) {\n\t\n\t var pluginName = block.dataset.tool;\n\t\n\t /** Check for plugin existance */\n\t if (!editor.tools[pluginName]) {\n\t\n\t throw Error('Plugin \\xAB' + pluginName + '\\xBB not found');\n\t }\n\t\n\t /** Check for plugin having render method */\n\t if (typeof editor.tools[pluginName].save != 'function') {\n\t\n\t throw Error('Plugin \\xAB' + pluginName + '\\xBB must have save method');\n\t }\n\t\n\t /** Result saver */\n\t var blockContent = block.childNodes[0],\n\t pluginsContent = blockContent.childNodes[0],\n\t savedData,\n\t position,\n\t output;\n\t\n\t /** If plugin wasn't available then return data from cache */\n\t if (editor.tools[pluginName].available === false) {\n\t\n\t position = pluginsContent.dataset.inputPosition;\n\t savedData = codex.editor.state.blocks.items[position].data;\n\t } else {\n\t\n\t savedData = editor.tools[pluginName].save(pluginsContent);\n\t\n\t if (editor.tools[pluginName].validate) {\n\t\n\t var result = editor.tools[pluginName].validate(savedData);\n\t\n\t /**\r\n\t * Do not allow invalid data\r\n\t */\n\t if (!result) return;\n\t }\n\t }\n\t\n\t output = {\n\t type: pluginName,\n\t data: savedData\n\t };\n\t\n\t editor.state.jsonOutput.push(output);\n\t };\n\t\n\t return saver;\n\t}({});\n\n/***/ },\n/* 7 */\n/***/ function(module, exports) {\n\n\t'use strict';\n\t\n\t/**\r\n\t * Codex Editor Content Module\r\n\t * Works with DOM\r\n\t *\r\n\t * @module Codex Editor content module\r\n\t *\r\n\t * @author Codex Team\r\n\t * @version 1.3.13\r\n\t *\r\n\t * @description Module works with Elements that have been appended to the main DOM\r\n\t */\n\t\n\tmodule.exports = function (content) {\n\t\n\t var editor = codex.editor;\n\t\n\t /**\r\n\t * Links to current active block\r\n\t * @type {null | Element}\r\n\t */\n\t content.currentNode = null;\n\t\n\t /**\r\n\t * clicked in redactor area\r\n\t * @type {null | Boolean}\r\n\t */\n\t content.editorAreaHightlighted = null;\n\t\n\t /**\r\n\t * @deprecated\r\n\t * Synchronizes redactor with original textarea\r\n\t */\n\t content.sync = function () {\n\t\n\t editor.core.log('syncing...');\n\t\n\t /**\r\n\t * Save redactor content to editor.state\r\n\t */\n\t editor.state.html = editor.nodes.redactor.innerHTML;\n\t };\n\t\n\t /**\r\n\t * Appends background to the block\r\n\t *\r\n\t * @description add CSS class to highlight visually first-level block area\r\n\t */\n\t content.markBlock = function () {\n\t\n\t editor.content.currentNode.classList.add(editor.ui.className.BLOCK_HIGHLIGHTED);\n\t };\n\t\n\t /**\r\n\t * Clear background\r\n\t *\r\n\t * @description clears styles that highlights block\r\n\t */\n\t content.clearMark = function () {\n\t\n\t if (editor.content.currentNode) {\n\t\n\t editor.content.currentNode.classList.remove(editor.ui.className.BLOCK_HIGHLIGHTED);\n\t }\n\t };\n\t\n\t /**\r\n\t * Finds first-level block\r\n\t *\r\n\t * @param {Element} node - selected or clicked in redactors area node\r\n\t * @protected\r\n\t *\r\n\t * @description looks for first-level block.\r\n\t * gets parent while node is not first-level\r\n\t */\n\t content.getFirstLevelBlock = function (node) {\n\t\n\t if (!editor.core.isDomNode(node)) {\n\t\n\t node = node.parentNode;\n\t }\n\t\n\t if (node === editor.nodes.redactor || node === document.body) {\n\t\n\t return null;\n\t } else {\n\t\n\t while (!node.classList.contains(editor.ui.className.BLOCK_CLASSNAME)) {\n\t\n\t node = node.parentNode;\n\t }\n\t\n\t return node;\n\t }\n\t };\n\t\n\t /**\r\n\t * Trigger this event when working node changed\r\n\t * @param {Element} targetNode - first-level of this node will be current\r\n\t * @protected\r\n\t *\r\n\t * @description If targetNode is first-level then we set it as current else we look for parents to find first-level\r\n\t */\n\t content.workingNodeChanged = function (targetNode) {\n\t\n\t /** Clear background from previous marked block before we change */\n\t editor.content.clearMark();\n\t\n\t if (!targetNode) {\n\t\n\t return;\n\t }\n\t\n\t content.currentNode = content.getFirstLevelBlock(targetNode);\n\t };\n\t\n\t /**\r\n\t * Replaces one redactor block with another\r\n\t * @protected\r\n\t * @param {Element} targetBlock - block to replace. Mostly currentNode.\r\n\t * @param {Element} newBlock\r\n\t * @param {string} newBlockType - type of new block; we need to store it to data-attribute\r\n\t *\r\n\t * [!] Function does not saves old block content.\r\n\t * You can get it manually and pass with newBlock.innerHTML\r\n\t */\n\t content.replaceBlock = function (targetBlock, newBlock) {\n\t\n\t if (!targetBlock || !newBlock) {\n\t\n\t editor.core.log('replaceBlock: missed params');\n\t return;\n\t }\n\t\n\t /** If target-block is not a frist-level block, then we iterate parents to find it */\n\t while (!targetBlock.classList.contains(editor.ui.className.BLOCK_CLASSNAME)) {\n\t\n\t targetBlock = targetBlock.parentNode;\n\t }\n\t\n\t /** Replacing */\n\t editor.nodes.redactor.replaceChild(newBlock, targetBlock);\n\t\n\t /**\r\n\t * Set new node as current\r\n\t */\n\t editor.content.workingNodeChanged(newBlock);\n\t\n\t /**\r\n\t * Add block handlers\r\n\t */\n\t editor.ui.addBlockHandlers(newBlock);\n\t\n\t /**\r\n\t * Save changes\r\n\t */\n\t editor.ui.saveInputs();\n\t };\n\t\n\t /**\r\n\t * @protected\r\n\t *\r\n\t * Inserts new block to redactor\r\n\t * Wrapps block into a DIV with BLOCK_CLASSNAME class\r\n\t *\r\n\t * @param blockData {object}\r\n\t * @param blockData.block {Element} element with block content\r\n\t * @param blockData.type {string} block plugin\r\n\t * @param needPlaceCaret {bool} pass true to set caret in new block\r\n\t *\r\n\t */\n\t content.insertBlock = function (blockData, needPlaceCaret) {\n\t\n\t var workingBlock = editor.content.currentNode,\n\t newBlockContent = blockData.block,\n\t blockType = blockData.type,\n\t isStretched = blockData.stretched;\n\t\n\t var newBlock = composeNewBlock_(newBlockContent, blockType, isStretched);\n\t\n\t if (workingBlock) {\n\t\n\t editor.core.insertAfter(workingBlock, newBlock);\n\t } else {\n\t\n\t /**\r\n\t * If redactor is empty, append as first child\r\n\t */\n\t editor.nodes.redactor.appendChild(newBlock);\n\t }\n\t\n\t /**\r\n\t * Block handler\r\n\t */\n\t editor.ui.addBlockHandlers(newBlock);\n\t\n\t /**\r\n\t * Set new node as current\r\n\t */\n\t editor.content.workingNodeChanged(newBlock);\n\t\n\t /**\r\n\t * Save changes\r\n\t */\n\t editor.ui.saveInputs();\n\t\n\t if (needPlaceCaret) {\n\t\n\t /**\r\n\t * If we don't know input index then we set default value -1\r\n\t */\n\t var currentInputIndex = editor.caret.getCurrentInputIndex() || -1;\n\t\n\t if (currentInputIndex == -1) {\n\t\n\t var editableElement = newBlock.querySelector('[contenteditable]'),\n\t emptyText = document.createTextNode('');\n\t\n\t editableElement.appendChild(emptyText);\n\t editor.caret.set(editableElement, 0, 0);\n\t\n\t editor.toolbar.move();\n\t editor.toolbar.showPlusButton();\n\t } else {\n\t\n\t if (currentInputIndex === editor.state.inputs.length - 1) return;\n\t\n\t /** Timeout for browsers execution */\n\t window.setTimeout(function () {\n\t\n\t /** Setting to the new input */\n\t editor.caret.setToNextBlock(currentInputIndex);\n\t editor.toolbar.move();\n\t editor.toolbar.open();\n\t }, 10);\n\t }\n\t }\n\t\n\t /**\r\n\t * Block is inserted, wait for new click that defined focusing on editors area\r\n\t * @type {boolean}\r\n\t */\n\t content.editorAreaHightlighted = false;\n\t };\n\t\n\t /**\r\n\t * Replaces blocks with saving content\r\n\t * @protected\r\n\t * @param {Element} noteToReplace\r\n\t * @param {Element} newNode\r\n\t * @param {Element} blockType\r\n\t */\n\t content.switchBlock = function (blockToReplace, newBlock, tool) {\n\t\n\t tool = tool || editor.content.currentNode.dataset.tool;\n\t var newBlockComposed = composeNewBlock_(newBlock, tool);\n\t\n\t /** Replacing */\n\t editor.content.replaceBlock(blockToReplace, newBlockComposed);\n\t\n\t /** Save new Inputs when block is changed */\n\t editor.ui.saveInputs();\n\t };\n\t\n\t /**\r\n\t * Iterates between child noted and looking for #text node on deepest level\r\n\t * @protected\r\n\t *\r\n\t * @param {Element} block - node where find\r\n\t * @param {int} postiton - starting postion\r\n\t * Example: childNodex.length to find from the end\r\n\t * or 0 to find from the start\r\n\t * @return {Text} block\r\n\t * @uses DFS\r\n\t */\n\t content.getDeepestTextNodeFromPosition = function (block, position) {\n\t\n\t /**\r\n\t * Clear Block from empty and useless spaces with trim.\r\n\t * Such nodes we should remove\r\n\t */\n\t var blockChilds = block.childNodes,\n\t index,\n\t node,\n\t text;\n\t\n\t for (index = 0; index < blockChilds.length; index++) {\n\t\n\t node = blockChilds[index];\n\t\n\t if (node.nodeType == editor.core.nodeTypes.TEXT) {\n\t\n\t text = node.textContent.trim();\n\t\n\t /** Text is empty. We should remove this child from node before we start DFS\r\n\t * decrease the quantity of childs.\r\n\t */\n\t if (text === '') {\n\t\n\t block.removeChild(node);\n\t position--;\n\t }\n\t }\n\t }\n\t\n\t if (block.childNodes.length === 0) {\n\t\n\t return document.createTextNode('');\n\t }\n\t\n\t /** Setting default position when we deleted all empty nodes */\n\t if (position < 0) position = 1;\n\t\n\t var lookingFromStart = false;\n\t\n\t /** For looking from START */\n\t if (position === 0) {\n\t\n\t lookingFromStart = true;\n\t position = 1;\n\t }\n\t\n\t while (position) {\n\t\n\t /** initial verticle of node. */\n\t if (lookingFromStart) {\n\t\n\t block = block.childNodes[0];\n\t } else {\n\t\n\t block = block.childNodes[position - 1];\n\t }\n\t\n\t if (block.nodeType == editor.core.nodeTypes.TAG) {\n\t\n\t position = block.childNodes.length;\n\t } else if (block.nodeType == editor.core.nodeTypes.TEXT) {\n\t\n\t position = 0;\n\t }\n\t }\n\t\n\t return block;\n\t };\n\t\n\t /**\r\n\t * @private\r\n\t * @param {Element} block - current plugins render\r\n\t * @param {String} tool - plugins name\r\n\t * @param {Boolean} isStretched - make stretched block or not\r\n\t *\r\n\t * @description adds necessary information to wrap new created block by first-level holder\r\n\t */\n\t var composeNewBlock_ = function composeNewBlock_(block, tool, isStretched) {\n\t\n\t var newBlock = editor.draw.node('DIV', editor.ui.className.BLOCK_CLASSNAME, {}),\n\t blockContent = editor.draw.node('DIV', editor.ui.className.BLOCK_CONTENT, {});\n\t\n\t blockContent.appendChild(block);\n\t newBlock.appendChild(blockContent);\n\t\n\t if (isStretched) {\n\t\n\t blockContent.classList.add(editor.ui.className.BLOCK_STRETCHED);\n\t }\n\t\n\t newBlock.dataset.tool = tool;\n\t return newBlock;\n\t };\n\t\n\t /**\r\n\t * Returns Range object of current selection\r\n\t * @protected\r\n\t */\n\t content.getRange = function () {\n\t\n\t var selection = window.getSelection().getRangeAt(0);\n\t\n\t return selection;\n\t };\n\t\n\t /**\r\n\t * Divides block in two blocks (after and before caret)\r\n\t *\r\n\t * @protected\r\n\t * @param {int} inputIndex - target input index\r\n\t *\r\n\t * @description splits current input content to the separate blocks\r\n\t * When enter is pressed among the words, that text will be splited.\r\n\t */\n\t content.splitBlock = function (inputIndex) {\n\t\n\t var selection = window.getSelection(),\n\t anchorNode = selection.anchorNode,\n\t anchorNodeText = anchorNode.textContent,\n\t caretOffset = selection.anchorOffset,\n\t textBeforeCaret,\n\t textNodeBeforeCaret,\n\t textAfterCaret,\n\t textNodeAfterCaret;\n\t\n\t var currentBlock = editor.content.currentNode.querySelector('[contentEditable]');\n\t\n\t textBeforeCaret = anchorNodeText.substring(0, caretOffset);\n\t textAfterCaret = anchorNodeText.substring(caretOffset);\n\t\n\t textNodeBeforeCaret = document.createTextNode(textBeforeCaret);\n\t\n\t if (textAfterCaret) {\n\t\n\t textNodeAfterCaret = document.createTextNode(textAfterCaret);\n\t }\n\t\n\t var previousChilds = [],\n\t nextChilds = [],\n\t reachedCurrent = false;\n\t\n\t if (textNodeAfterCaret) {\n\t\n\t nextChilds.push(textNodeAfterCaret);\n\t }\n\t\n\t for (var i = 0, child; !!(child = currentBlock.childNodes[i]); i++) {\n\t\n\t if (child != anchorNode) {\n\t\n\t if (!reachedCurrent) {\n\t\n\t previousChilds.push(child);\n\t } else {\n\t\n\t nextChilds.push(child);\n\t }\n\t } else {\n\t\n\t reachedCurrent = true;\n\t }\n\t }\n\t\n\t /** Clear current input */\n\t editor.state.inputs[inputIndex].innerHTML = '';\n\t\n\t /**\r\n\t * Append all childs founded before anchorNode\r\n\t */\n\t var previousChildsLength = previousChilds.length;\n\t\n\t for (i = 0; i < previousChildsLength; i++) {\n\t\n\t editor.state.inputs[inputIndex].appendChild(previousChilds[i]);\n\t }\n\t\n\t editor.state.inputs[inputIndex].appendChild(textNodeBeforeCaret);\n\t\n\t /**\r\n\t * Append text node which is after caret\r\n\t */\n\t var nextChildsLength = nextChilds.length,\n\t newNode = document.createElement('div');\n\t\n\t for (i = 0; i < nextChildsLength; i++) {\n\t\n\t newNode.appendChild(nextChilds[i]);\n\t }\n\t\n\t newNode = newNode.innerHTML;\n\t\n\t /** This type of block creates when enter is pressed */\n\t var NEW_BLOCK_TYPE = editor.settings.initialBlockPlugin;\n\t\n\t /**\r\n\t * Make new paragraph with text after caret\r\n\t */\n\t editor.content.insertBlock({\n\t type: NEW_BLOCK_TYPE,\n\t block: editor.tools[NEW_BLOCK_TYPE].render({\n\t text: newNode\n\t })\n\t }, true);\n\t };\n\t\n\t /**\r\n\t * Merges two blocks — current and target\r\n\t * If target index is not exist, then previous will be as target\r\n\t *\r\n\t * @protected\r\n\t * @param {int} currentInputIndex\r\n\t * @param {int} targetInputIndex\r\n\t *\r\n\t * @description gets two inputs indexes and merges into one\r\n\t */\n\t content.mergeBlocks = function (currentInputIndex, targetInputIndex) {\n\t\n\t /** If current input index is zero, then prevent method execution */\n\t if (currentInputIndex === 0) {\n\t\n\t return;\n\t }\n\t\n\t var targetInput,\n\t currentInputContent = editor.state.inputs[currentInputIndex].innerHTML;\n\t\n\t if (!targetInputIndex) {\n\t\n\t targetInput = editor.state.inputs[currentInputIndex - 1];\n\t } else {\n\t\n\t targetInput = editor.state.inputs[targetInputIndex];\n\t }\n\t\n\t targetInput.innerHTML += currentInputContent;\n\t };\n\t\n\t /**\r\n\t * Iterates all right siblings and parents, which has right siblings\r\n\t * while it does not reached the first-level block\r\n\t *\r\n\t * @param {Element} node\r\n\t * @return {boolean}\r\n\t */\n\t content.isLastNode = function (node) {\n\t\n\t // console.log('погнали перебор родителей');\n\t\n\t var allChecked = false;\n\t\n\t while (!allChecked) {\n\t\n\t // console.log('Смотрим на %o', node);\n\t // console.log('Проверим, пустые ли соседи справа');\n\t\n\t if (!allSiblingsEmpty_(node)) {\n\t\n\t // console.log('Есть непустые соседи. Узел не последний. Выходим.');\n\t return false;\n\t }\n\t\n\t node = node.parentNode;\n\t\n\t /**\r\n\t * Проверяем родителей до тех пор, пока не найдем блок первого уровня\r\n\t */\n\t if (node.classList.contains(editor.ui.className.BLOCK_CONTENT)) {\n\t\n\t allChecked = true;\n\t }\n\t }\n\t\n\t return true;\n\t };\n\t\n\t /**\r\n\t * Checks if all element right siblings is empty\r\n\t * @param node\r\n\t */\n\t var allSiblingsEmpty_ = function allSiblingsEmpty_(node) {\n\t\n\t /**\r\n\t * Нужно убедиться, что после пустого соседа ничего нет\r\n\t */\n\t var sibling = node.nextSibling;\n\t\n\t while (sibling) {\n\t\n\t if (sibling.textContent.length) {\n\t\n\t return false;\n\t }\n\t\n\t sibling = sibling.nextSibling;\n\t }\n\t\n\t return true;\n\t };\n\t\n\t /**\r\n\t * @public\r\n\t *\r\n\t * @param {string} htmlData - html content as string\r\n\t * @param {string} plainData - plain text\r\n\t * @return {string} - html content as string\r\n\t */\n\t content.wrapTextWithParagraphs = function (htmlData, plainData) {\n\t\n\t if (!htmlData) {\n\t\n\t return wrapPlainTextWithParagraphs(plainData);\n\t }\n\t\n\t var wrapper = document.createElement('DIV'),\n\t newWrapper = document.createElement('DIV'),\n\t i,\n\t paragraph,\n\t firstLevelBlocks = ['DIV', 'P'],\n\t blockTyped,\n\t node;\n\t\n\t /**\r\n\t * Make HTML Element to Wrap Text\r\n\t * It allows us to work with input data as HTML content\r\n\t */\n\t wrapper.innerHTML = htmlData;\n\t paragraph = document.createElement('P');\n\t\n\t for (i = 0; i < wrapper.childNodes.length; i++) {\n\t\n\t node = wrapper.childNodes[i];\n\t\n\t blockTyped = firstLevelBlocks.indexOf(node.tagName) != -1;\n\t\n\t /**\r\n\t * If node is first-levet\r\n\t * we add this node to our new wrapper\r\n\t */\n\t if (blockTyped) {\n\t\n\t /**\r\n\t * If we had splitted inline nodes to paragraph before\r\n\t */\n\t if (paragraph.childNodes.length) {\n\t\n\t newWrapper.appendChild(paragraph.cloneNode(true));\n\t\n\t /** empty paragraph */\n\t paragraph = null;\n\t paragraph = document.createElement('P');\n\t }\n\t\n\t newWrapper.appendChild(node.cloneNode(true));\n\t } else {\n\t\n\t /** Collect all inline nodes to one as paragraph */\n\t paragraph.appendChild(node.cloneNode(true));\n\t\n\t /** if node is last we should append this node to paragraph and paragraph to new wrapper */\n\t if (i == wrapper.childNodes.length - 1) {\n\t\n\t newWrapper.appendChild(paragraph.cloneNode(true));\n\t }\n\t }\n\t }\n\t\n\t return newWrapper.innerHTML;\n\t };\n\t\n\t /**\r\n\t * Splits strings on new line and wraps paragraphs with

    tag\r\n\t * @param plainText\r\n\t * @returns {string}\r\n\t */\n\t var wrapPlainTextWithParagraphs = function wrapPlainTextWithParagraphs(plainText) {\n\t\n\t return '

    ' + plainText.split('\\n\\n').join('

    ') + '

    ';\n\t };\n\t\n\t /**\r\n\t * Finds closest Contenteditable parent from Element\r\n\t * @param {Element} node element looking from\r\n\t * @return {Element} node contenteditable\r\n\t */\n\t content.getEditableParent = function (node) {\n\t\n\t while (node && node.contentEditable != 'true') {\n\t\n\t node = node.parentNode;\n\t }\n\t\n\t return node;\n\t };\n\t\n\t return content;\n\t}({});\n\n/***/ },\n/* 8 */\n/***/ function(module, exports, __webpack_require__) {\n\n\t'use strict';\n\t\n\t/**\r\n\t * Codex Editor toolbar module\r\n\t *\r\n\t * Contains:\r\n\t * - Inline toolbox\r\n\t * - Toolbox within plus button\r\n\t * - Settings section\r\n\t *\r\n\t * @author Codex Team\r\n\t * @version 1.0\r\n\t */\n\t\n\tmodule.exports = function (toolbar) {\n\t\n\t var editor = codex.editor;\n\t\n\t toolbar.settings = __webpack_require__(9);\n\t toolbar.inline = __webpack_require__(10);\n\t toolbar.toolbox = __webpack_require__(11);\n\t\n\t /**\r\n\t * Margin between focused node and toolbar\r\n\t */\n\t toolbar.defaultToolbarHeight = 49;\n\t\n\t toolbar.defaultOffset = 34;\n\t\n\t toolbar.opened = false;\n\t\n\t toolbar.current = null;\n\t\n\t /**\r\n\t * @protected\r\n\t */\n\t toolbar.open = function () {\n\t\n\t if (editor.hideToolbar) {\n\t\n\t return;\n\t }\n\t\n\t editor.nodes.toolbar.classList.add('opened');\n\t this.opened = true;\n\t };\n\t\n\t /**\r\n\t * @protected\r\n\t */\n\t toolbar.close = function () {\n\t\n\t editor.nodes.toolbar.classList.remove('opened');\n\t\n\t toolbar.opened = false;\n\t toolbar.current = null;\n\t\n\t for (var button in editor.nodes.toolbarButtons) {\n\t\n\t editor.nodes.toolbarButtons[button].classList.remove('selected');\n\t }\n\t\n\t /** Close toolbox when toolbar is not displayed */\n\t editor.toolbar.toolbox.close();\n\t editor.toolbar.settings.close();\n\t };\n\t\n\t toolbar.toggle = function () {\n\t\n\t if (!this.opened) {\n\t\n\t this.open();\n\t } else {\n\t\n\t this.close();\n\t }\n\t };\n\t\n\t toolbar.hidePlusButton = function () {\n\t\n\t editor.nodes.plusButton.classList.add('hide');\n\t };\n\t\n\t toolbar.showPlusButton = function () {\n\t\n\t editor.nodes.plusButton.classList.remove('hide');\n\t };\n\t\n\t /**\r\n\t * Moving toolbar to the specified node\r\n\t */\n\t toolbar.move = function () {\n\t\n\t /** Close Toolbox when we move toolbar */\n\t editor.toolbar.toolbox.close();\n\t\n\t if (!editor.content.currentNode) {\n\t\n\t return;\n\t }\n\t\n\t var newYCoordinate = editor.content.currentNode.offsetTop - editor.toolbar.defaultToolbarHeight / 2 + editor.toolbar.defaultOffset;\n\t\n\t editor.nodes.toolbar.style.transform = 'translate3D(0, ' + Math.floor(newYCoordinate) + 'px, 0)';\n\t\n\t /** Close trash actions */\n\t editor.toolbar.settings.hideRemoveActions();\n\t };\n\t\n\t return toolbar;\n\t}({});\n\n/***/ },\n/* 9 */\n/***/ function(module, exports) {\n\n\t'use strict';\n\t\n\t/**\r\n\t * Toolbar settings\r\n\t *\r\n\t * @version 1.0.4\r\n\t */\n\t\n\tmodule.exports = function (settings) {\n\t\n\t var editor = codex.editor;\n\t\n\t settings.opened = false;\n\t\n\t settings.setting = null;\n\t settings.actions = null;\n\t\n\t /**\r\n\t * Append and open settings\r\n\t */\n\t settings.open = function (toolType) {\n\t\n\t /**\r\n\t * Append settings content\r\n\t * It's stored in tool.settings\r\n\t */\n\t if (!editor.tools[toolType] || !editor.tools[toolType].makeSettings) {\n\t\n\t editor.core.log('Plugin \\xAB' + toolType + '\\xBB has no settings', 'warn');\n\t editor.nodes.pluginSettings.innerHTML = '\\u041F\\u043B\\u0430\\u0433\\u0438\\u043D \\xAB' + toolType + '\\xBB \\u043D\\u0435 \\u0438\\u043C\\u0435\\u0435\\u0442 \\u043D\\u0430\\u0441\\u0442\\u0440\\u043E\\u0435\\u043A';\n\t } else {\n\t\n\t /**\r\n\t * Draw settings block\r\n\t */\n\t var settingsBlock = editor.tools[toolType].makeSettings();\n\t\n\t editor.nodes.pluginSettings.appendChild(settingsBlock);\n\t }\n\t\n\t /** Open settings block */\n\t editor.nodes.blockSettings.classList.add('opened');\n\t this.opened = true;\n\t };\n\t\n\t /**\r\n\t * Close and clear settings\r\n\t */\n\t settings.close = function () {\n\t\n\t editor.nodes.blockSettings.classList.remove('opened');\n\t editor.nodes.pluginSettings.innerHTML = '';\n\t\n\t this.opened = false;\n\t };\n\t\n\t /**\r\n\t * @param {string} toolType - plugin type\r\n\t */\n\t settings.toggle = function (toolType) {\n\t\n\t if (!this.opened) {\n\t\n\t this.open(toolType);\n\t } else {\n\t\n\t this.close();\n\t }\n\t };\n\t\n\t /**\r\n\t * Here we will draw buttons and add listeners to components\r\n\t */\n\t settings.makeRemoveBlockButton = function () {\n\t\n\t var removeBlockWrapper = editor.draw.node('SPAN', 'ce-toolbar__remove-btn', {}),\n\t settingButton = editor.draw.node('SPAN', 'ce-toolbar__remove-setting', { innerHTML: '' }),\n\t actionWrapper = editor.draw.node('DIV', 'ce-toolbar__remove-confirmation', {}),\n\t confirmAction = editor.draw.node('DIV', 'ce-toolbar__remove-confirm', { textContent: 'Удалить блок' }),\n\t cancelAction = editor.draw.node('DIV', 'ce-toolbar__remove-cancel', { textContent: 'Отмена' });\n\t\n\t editor.listeners.add(settingButton, 'click', editor.toolbar.settings.removeButtonClicked, false);\n\t\n\t editor.listeners.add(confirmAction, 'click', editor.toolbar.settings.confirmRemovingRequest, false);\n\t\n\t editor.listeners.add(cancelAction, 'click', editor.toolbar.settings.cancelRemovingRequest, false);\n\t\n\t actionWrapper.appendChild(confirmAction);\n\t actionWrapper.appendChild(cancelAction);\n\t\n\t removeBlockWrapper.appendChild(settingButton);\n\t removeBlockWrapper.appendChild(actionWrapper);\n\t\n\t /** Save setting */\n\t editor.toolbar.settings.setting = settingButton;\n\t editor.toolbar.settings.actions = actionWrapper;\n\t\n\t return removeBlockWrapper;\n\t };\n\t\n\t settings.removeButtonClicked = function () {\n\t\n\t var action = editor.toolbar.settings.actions;\n\t\n\t if (action.classList.contains('opened')) {\n\t\n\t editor.toolbar.settings.hideRemoveActions();\n\t } else {\n\t\n\t editor.toolbar.settings.showRemoveActions();\n\t }\n\t\n\t editor.toolbar.toolbox.close();\n\t editor.toolbar.settings.close();\n\t };\n\t\n\t settings.cancelRemovingRequest = function () {\n\t\n\t editor.toolbar.settings.actions.classList.remove('opened');\n\t };\n\t\n\t settings.confirmRemovingRequest = function () {\n\t\n\t var currentBlock = editor.content.currentNode,\n\t firstLevelBlocksCount;\n\t\n\t currentBlock.remove();\n\t\n\t firstLevelBlocksCount = editor.nodes.redactor.childNodes.length;\n\t\n\t /**\r\n\t * If all blocks are removed\r\n\t */\n\t if (firstLevelBlocksCount === 0) {\n\t\n\t /** update currentNode variable */\n\t editor.content.currentNode = null;\n\t\n\t /** Inserting new empty initial block */\n\t editor.ui.addInitialBlock();\n\t }\n\t\n\t editor.ui.saveInputs();\n\t\n\t editor.toolbar.close();\n\t };\n\t\n\t settings.showRemoveActions = function () {\n\t\n\t editor.toolbar.settings.actions.classList.add('opened');\n\t };\n\t\n\t settings.hideRemoveActions = function () {\n\t\n\t editor.toolbar.settings.actions.classList.remove('opened');\n\t };\n\t\n\t return settings;\n\t}({});\n\n/***/ },\n/* 10 */\n/***/ function(module, exports) {\n\n\t'use strict';\n\t\n\t/**\r\n\t * Inline toolbar\r\n\t *\r\n\t * Contains from tools:\r\n\t * Bold, Italic, Underline and Anchor\r\n\t *\r\n\t * @author Codex Team\r\n\t * @version 1.0\r\n\t */\n\t\n\tmodule.exports = function (inline) {\n\t\n\t var editor = codex.editor;\n\t\n\t inline.buttonsOpened = null;\n\t inline.actionsOpened = null;\n\t inline.wrappersOffset = null;\n\t\n\t /**\r\n\t * saving selection that need for execCommand for styling\r\n\t *\r\n\t */\n\t inline.storedSelection = null;\n\t\n\t /**\r\n\t * @protected\r\n\t *\r\n\t * Open inline toobar\r\n\t */\n\t inline.show = function () {\n\t\n\t var currentNode = editor.content.currentNode,\n\t tool = currentNode.dataset.tool,\n\t plugin;\n\t\n\t /**\r\n\t * tool allowed to open inline toolbar\r\n\t */\n\t plugin = editor.tools[tool];\n\t\n\t if (!plugin.showInlineToolbar) return;\n\t\n\t var selectedText = inline.getSelectionText(),\n\t toolbar = editor.nodes.inlineToolbar.wrapper;\n\t\n\t if (selectedText.length > 0) {\n\t\n\t /** Move toolbar and open */\n\t editor.toolbar.inline.move();\n\t\n\t /** Open inline toolbar */\n\t toolbar.classList.add('opened');\n\t\n\t /** show buttons of inline toolbar */\n\t editor.toolbar.inline.showButtons();\n\t }\n\t };\n\t\n\t /**\r\n\t * @protected\r\n\t *\r\n\t * Closes inline toolbar\r\n\t */\n\t inline.close = function () {\n\t\n\t var toolbar = editor.nodes.inlineToolbar.wrapper;\n\t\n\t toolbar.classList.remove('opened');\n\t };\n\t\n\t /**\r\n\t * @private\r\n\t *\r\n\t * Moving toolbar\r\n\t */\n\t inline.move = function () {\n\t\n\t if (!this.wrappersOffset) {\n\t\n\t this.wrappersOffset = this.getWrappersOffset();\n\t }\n\t\n\t var coords = this.getSelectionCoords(),\n\t defaultOffset = 0,\n\t toolbar = editor.nodes.inlineToolbar.wrapper,\n\t newCoordinateX,\n\t newCoordinateY;\n\t\n\t if (toolbar.offsetHeight === 0) {\n\t\n\t defaultOffset = 40;\n\t }\n\t\n\t newCoordinateX = coords.x - this.wrappersOffset.left;\n\t newCoordinateY = coords.y + window.scrollY - this.wrappersOffset.top - defaultOffset - toolbar.offsetHeight;\n\t\n\t toolbar.style.transform = 'translate3D(' + Math.floor(newCoordinateX) + 'px, ' + Math.floor(newCoordinateY) + 'px, 0)';\n\t\n\t /** Close everything */\n\t editor.toolbar.inline.closeButtons();\n\t editor.toolbar.inline.closeAction();\n\t };\n\t\n\t /**\r\n\t * @private\r\n\t *\r\n\t * Tool Clicked\r\n\t */\n\t\n\t inline.toolClicked = function (event, type) {\n\t\n\t /**\r\n\t * For simple tools we use default browser function\r\n\t * For more complicated tools, we should write our own behavior\r\n\t */\n\t switch (type) {\n\t case 'createLink':\n\t editor.toolbar.inline.createLinkAction(event, type);break;\n\t default:\n\t editor.toolbar.inline.defaultToolAction(type);break;\n\t }\n\t\n\t /**\r\n\t * highlight buttons\r\n\t * after making some action\r\n\t */\n\t editor.nodes.inlineToolbar.buttons.childNodes.forEach(editor.toolbar.inline.hightlight);\n\t };\n\t\n\t /**\r\n\t * @private\r\n\t *\r\n\t * Saving wrappers offset in DOM\r\n\t */\n\t inline.getWrappersOffset = function () {\n\t\n\t var wrapper = editor.nodes.wrapper,\n\t offset = this.getOffset(wrapper);\n\t\n\t this.wrappersOffset = offset;\n\t return offset;\n\t };\n\t\n\t /**\r\n\t * @private\r\n\t *\r\n\t * Calculates offset of DOM element\r\n\t *\r\n\t * @param el\r\n\t * @returns {{top: number, left: number}}\r\n\t */\n\t inline.getOffset = function (el) {\n\t\n\t var _x = 0;\n\t var _y = 0;\n\t\n\t while (el && !isNaN(el.offsetLeft) && !isNaN(el.offsetTop)) {\n\t\n\t _x += el.offsetLeft + el.clientLeft;\n\t _y += el.offsetTop + el.clientTop;\n\t el = el.offsetParent;\n\t }\n\t return { top: _y, left: _x };\n\t };\n\t\n\t /**\r\n\t * @private\r\n\t *\r\n\t * Calculates position of selected text\r\n\t * @returns {{x: number, y: number}}\r\n\t */\n\t inline.getSelectionCoords = function () {\n\t\n\t var sel = document.selection,\n\t range;\n\t var x = 0,\n\t y = 0;\n\t\n\t if (sel) {\n\t\n\t if (sel.type != 'Control') {\n\t\n\t range = sel.createRange();\n\t range.collapse(true);\n\t x = range.boundingLeft;\n\t y = range.boundingTop;\n\t }\n\t } else if (window.getSelection) {\n\t\n\t sel = window.getSelection();\n\t\n\t if (sel.rangeCount) {\n\t\n\t range = sel.getRangeAt(0).cloneRange();\n\t if (range.getClientRects) {\n\t\n\t range.collapse(true);\n\t var rect = range.getClientRects()[0];\n\t\n\t if (!rect) {\n\t\n\t return;\n\t }\n\t\n\t x = rect.left;\n\t y = rect.top;\n\t }\n\t }\n\t }\n\t return { x: x, y: y };\n\t };\n\t\n\t /**\r\n\t * @private\r\n\t *\r\n\t * Returns selected text as String\r\n\t * @returns {string}\r\n\t */\n\t inline.getSelectionText = function () {\n\t\n\t var selectedText = '';\n\t\n\t // all modern browsers and IE9+\n\t if (window.getSelection) {\n\t\n\t selectedText = window.getSelection().toString();\n\t }\n\t\n\t return selectedText;\n\t };\n\t\n\t /** Opens buttons block */\n\t inline.showButtons = function () {\n\t\n\t var buttons = editor.nodes.inlineToolbar.buttons;\n\t\n\t buttons.classList.add('opened');\n\t\n\t editor.toolbar.inline.buttonsOpened = true;\n\t\n\t /** highlight buttons */\n\t editor.nodes.inlineToolbar.buttons.childNodes.forEach(editor.toolbar.inline.hightlight);\n\t };\n\t\n\t /** Makes buttons disappear */\n\t inline.closeButtons = function () {\n\t\n\t var buttons = editor.nodes.inlineToolbar.buttons;\n\t\n\t buttons.classList.remove('opened');\n\t\n\t editor.toolbar.inline.buttonsOpened = false;\n\t };\n\t\n\t /** Open buttons defined action if exist */\n\t inline.showActions = function () {\n\t\n\t var action = editor.nodes.inlineToolbar.actions;\n\t\n\t action.classList.add('opened');\n\t\n\t editor.toolbar.inline.actionsOpened = true;\n\t };\n\t\n\t /** Close actions block */\n\t inline.closeAction = function () {\n\t\n\t var action = editor.nodes.inlineToolbar.actions;\n\t\n\t action.innerHTML = '';\n\t action.classList.remove('opened');\n\t editor.toolbar.inline.actionsOpened = false;\n\t };\n\t\n\t /**\r\n\t * Callback for keydowns in inline toolbar \"Insert link...\" input\r\n\t */\n\t var inlineToolbarAnchorInputKeydown_ = function inlineToolbarAnchorInputKeydown_(event) {\n\t\n\t if (event.keyCode != editor.core.keys.ENTER) {\n\t\n\t return;\n\t }\n\t\n\t var editable = editor.content.currentNode,\n\t storedSelection = editor.toolbar.inline.storedSelection;\n\t\n\t editor.toolbar.inline.restoreSelection(editable, storedSelection);\n\t editor.toolbar.inline.setAnchor(this.value);\n\t\n\t /**\r\n\t * Preventing events that will be able to happen\r\n\t */\n\t event.preventDefault();\n\t event.stopImmediatePropagation();\n\t\n\t editor.toolbar.inline.clearRange();\n\t };\n\t\n\t /** Action for link creation or for setting anchor */\n\t inline.createLinkAction = function (event) {\n\t\n\t var isActive = this.isLinkActive();\n\t\n\t var editable = editor.content.currentNode,\n\t storedSelection = editor.toolbar.inline.saveSelection(editable);\n\t\n\t /** Save globally selection */\n\t editor.toolbar.inline.storedSelection = storedSelection;\n\t\n\t if (isActive) {\n\t\n\t /**\r\n\t * Changing stored selection. if we want to remove anchor from word\r\n\t * we should remove anchor from whole word, not only selected part.\r\n\t * The solution is than we get the length of current link\r\n\t * Change start position to - end of selection minus length of anchor\r\n\t */\n\t editor.toolbar.inline.restoreSelection(editable, storedSelection);\n\t\n\t editor.toolbar.inline.defaultToolAction('unlink');\n\t } else {\n\t\n\t /** Create input and close buttons */\n\t var action = editor.draw.inputForLink();\n\t\n\t editor.nodes.inlineToolbar.actions.appendChild(action);\n\t\n\t editor.toolbar.inline.closeButtons();\n\t editor.toolbar.inline.showActions();\n\t\n\t /**\r\n\t * focus to input\r\n\t * Solution: https://developer.mozilla.org/ru/docs/Web/API/HTMLElement/focus\r\n\t * Prevents event after showing input and when we need to focus an input which is in unexisted form\r\n\t */\n\t action.focus();\n\t event.preventDefault();\n\t\n\t /** Callback to link action */\n\t editor.listeners.add(action, 'keydown', inlineToolbarAnchorInputKeydown_, false);\n\t }\n\t };\n\t\n\t inline.isLinkActive = function () {\n\t\n\t var isActive = false;\n\t\n\t editor.nodes.inlineToolbar.buttons.childNodes.forEach(function (tool) {\n\t\n\t var dataType = tool.dataset.type;\n\t\n\t if (dataType == 'link' && tool.classList.contains('hightlighted')) {\n\t\n\t isActive = true;\n\t }\n\t });\n\t\n\t return isActive;\n\t };\n\t\n\t /** default action behavior of tool */\n\t inline.defaultToolAction = function (type) {\n\t\n\t document.execCommand(type, false, null);\n\t };\n\t\n\t /**\r\n\t * @private\r\n\t *\r\n\t * Sets URL\r\n\t *\r\n\t * @param {String} url - URL\r\n\t */\n\t inline.setAnchor = function (url) {\n\t\n\t document.execCommand('createLink', false, url);\n\t\n\t /** Close after URL inserting */\n\t editor.toolbar.inline.closeAction();\n\t };\n\t\n\t /**\r\n\t * @private\r\n\t *\r\n\t * Saves selection\r\n\t */\n\t inline.saveSelection = function (containerEl) {\n\t\n\t var range = window.getSelection().getRangeAt(0),\n\t preSelectionRange = range.cloneRange(),\n\t start;\n\t\n\t preSelectionRange.selectNodeContents(containerEl);\n\t preSelectionRange.setEnd(range.startContainer, range.startOffset);\n\t\n\t start = preSelectionRange.toString().length;\n\t\n\t return {\n\t start: start,\n\t end: start + range.toString().length\n\t };\n\t };\n\t\n\t /**\r\n\t * @private\r\n\t *\r\n\t * Sets to previous selection (Range)\r\n\t *\r\n\t * @param {Element} containerEl - editable element where we restore range\r\n\t * @param {Object} savedSel - range basic information to restore\r\n\t */\n\t inline.restoreSelection = function (containerEl, savedSel) {\n\t\n\t var range = document.createRange(),\n\t charIndex = 0;\n\t\n\t range.setStart(containerEl, 0);\n\t range.collapse(true);\n\t\n\t var nodeStack = [containerEl],\n\t node,\n\t foundStart = false,\n\t stop = false,\n\t nextCharIndex;\n\t\n\t while (!stop && (node = nodeStack.pop())) {\n\t\n\t if (node.nodeType == 3) {\n\t\n\t nextCharIndex = charIndex + node.length;\n\t\n\t if (!foundStart && savedSel.start >= charIndex && savedSel.start <= nextCharIndex) {\n\t\n\t range.setStart(node, savedSel.start - charIndex);\n\t foundStart = true;\n\t }\n\t if (foundStart && savedSel.end >= charIndex && savedSel.end <= nextCharIndex) {\n\t\n\t range.setEnd(node, savedSel.end - charIndex);\n\t stop = true;\n\t }\n\t charIndex = nextCharIndex;\n\t } else {\n\t\n\t var i = node.childNodes.length;\n\t\n\t while (i--) {\n\t\n\t nodeStack.push(node.childNodes[i]);\n\t }\n\t }\n\t }\n\t\n\t var sel = window.getSelection();\n\t\n\t sel.removeAllRanges();\n\t sel.addRange(range);\n\t };\n\t\n\t /**\r\n\t * @private\r\n\t *\r\n\t * Removes all ranges from window selection\r\n\t */\n\t inline.clearRange = function () {\n\t\n\t var selection = window.getSelection();\n\t\n\t selection.removeAllRanges();\n\t };\n\t\n\t /**\r\n\t * @private\r\n\t *\r\n\t * sets or removes hightlight\r\n\t */\n\t inline.hightlight = function (tool) {\n\t\n\t var dataType = tool.dataset.type;\n\t\n\t if (document.queryCommandState(dataType)) {\n\t\n\t editor.toolbar.inline.setButtonHighlighted(tool);\n\t } else {\n\t\n\t editor.toolbar.inline.removeButtonsHighLight(tool);\n\t }\n\t\n\t /**\r\n\t *\r\n\t * hightlight for anchors\r\n\t */\n\t var selection = window.getSelection(),\n\t tag = selection.anchorNode.parentNode;\n\t\n\t if (tag.tagName == 'A' && dataType == 'link') {\n\t\n\t editor.toolbar.inline.setButtonHighlighted(tool);\n\t }\n\t };\n\t\n\t /**\r\n\t * @private\r\n\t *\r\n\t * Mark button if text is already executed\r\n\t */\n\t inline.setButtonHighlighted = function (button) {\n\t\n\t button.classList.add('hightlighted');\n\t\n\t /** At link tool we also change icon */\n\t if (button.dataset.type == 'link') {\n\t\n\t var icon = button.childNodes[0];\n\t\n\t icon.classList.remove('ce-icon-link');\n\t icon.classList.add('ce-icon-unlink');\n\t }\n\t };\n\t\n\t /**\r\n\t * @private\r\n\t *\r\n\t * Removes hightlight\r\n\t */\n\t inline.removeButtonsHighLight = function (button) {\n\t\n\t button.classList.remove('hightlighted');\n\t\n\t /** At link tool we also change icon */\n\t if (button.dataset.type == 'link') {\n\t\n\t var icon = button.childNodes[0];\n\t\n\t icon.classList.remove('ce-icon-unlink');\n\t icon.classList.add('ce-icon-link');\n\t }\n\t };\n\t\n\t return inline;\n\t}({});\n\n/***/ },\n/* 11 */\n/***/ function(module, exports) {\n\n\t'use strict';\n\t\n\t/**\r\n\t * Codex Editor toolbox\r\n\t *\r\n\t * All tools be able to appended here\r\n\t *\r\n\t * @author Codex Team\r\n\t * @version 1.0\r\n\t */\n\t\n\tmodule.exports = function (toolbox) {\n\t\n\t var editor = codex.editor;\n\t\n\t toolbox.opened = false;\n\t\n\t /** Shows toolbox */\n\t toolbox.open = function () {\n\t\n\t /** Close setting if toolbox is opened */\n\t if (editor.toolbar.settings.opened) {\n\t\n\t editor.toolbar.settings.close();\n\t }\n\t\n\t /** display toolbox */\n\t editor.nodes.toolbox.classList.add('opened');\n\t\n\t /** Animate plus button */\n\t editor.nodes.plusButton.classList.add('clicked');\n\t\n\t /** toolbox state */\n\t editor.toolbar.toolbox.opened = true;\n\t };\n\t\n\t /** Closes toolbox */\n\t toolbox.close = function () {\n\t\n\t /** Makes toolbox disapear */\n\t editor.nodes.toolbox.classList.remove('opened');\n\t\n\t /** Rotate plus button */\n\t editor.nodes.plusButton.classList.remove('clicked');\n\t\n\t /** toolbox state */\n\t editor.toolbar.toolbox.opened = false;\n\t\n\t editor.toolbar.current = null;\n\t };\n\t\n\t toolbox.leaf = function () {\n\t\n\t var currentTool = editor.toolbar.current,\n\t tools = Object.keys(editor.tools),\n\t barButtons = editor.nodes.toolbarButtons,\n\t nextToolIndex = 0,\n\t toolToSelect = void 0,\n\t visibleTool = void 0,\n\t tool = void 0;\n\t\n\t if (!currentTool) {\n\t\n\t /** Get first tool from object*/\n\t for (tool in editor.tools) {\n\t\n\t if (editor.tools[tool].displayInToolbox) {\n\t\n\t break;\n\t }\n\t\n\t nextToolIndex++;\n\t }\n\t } else {\n\t\n\t nextToolIndex = (tools.indexOf(currentTool) + 1) % tools.length;\n\t visibleTool = tools[nextToolIndex];\n\t\n\t while (!editor.tools[visibleTool].displayInToolbox) {\n\t\n\t nextToolIndex = (nextToolIndex + 1) % tools.length;\n\t visibleTool = tools[nextToolIndex];\n\t }\n\t }\n\t\n\t toolToSelect = tools[nextToolIndex];\n\t\n\t for (var button in barButtons) {\n\t\n\t barButtons[button].classList.remove('selected');\n\t }\n\t\n\t barButtons[toolToSelect].classList.add('selected');\n\t editor.toolbar.current = toolToSelect;\n\t };\n\t\n\t /**\r\n\t * Transforming selected node type into selected toolbar element type\r\n\t * @param {event} event\r\n\t */\n\t toolbox.toolClicked = function (event) {\n\t\n\t /**\r\n\t * UNREPLACEBLE_TOOLS this types of tools are forbidden to replace even they are empty\r\n\t */\n\t var UNREPLACEBLE_TOOLS = ['image', 'link', 'list', 'instagram', 'twitter', 'embed'],\n\t tool = editor.tools[editor.toolbar.current],\n\t workingNode = editor.content.currentNode,\n\t currentInputIndex = editor.caret.inputIndex,\n\t newBlockContent,\n\t appendCallback,\n\t blockData;\n\t\n\t /** Make block from plugin */\n\t newBlockContent = tool.render();\n\t\n\t /** information about block */\n\t blockData = {\n\t block: newBlockContent,\n\t type: tool.type,\n\t stretched: false\n\t };\n\t\n\t if (workingNode && UNREPLACEBLE_TOOLS.indexOf(workingNode.dataset.tool) === -1 && workingNode.textContent.trim() === '') {\n\t\n\t /** Replace current block */\n\t editor.content.switchBlock(workingNode, newBlockContent, tool.type);\n\t } else {\n\t\n\t /** Insert new Block from plugin */\n\t editor.content.insertBlock(blockData);\n\t\n\t /** increase input index */\n\t currentInputIndex++;\n\t }\n\t\n\t /** Fire tool append callback */\n\t appendCallback = tool.appendCallback;\n\t\n\t if (appendCallback && typeof appendCallback == 'function') {\n\t\n\t appendCallback.call(event);\n\t }\n\t\n\t window.setTimeout(function () {\n\t\n\t /** Set caret to current block */\n\t editor.caret.setToBlock(currentInputIndex);\n\t }, 10);\n\t\n\t /**\r\n\t * Changing current Node\r\n\t */\n\t editor.content.workingNodeChanged();\n\t\n\t /**\r\n\t * Move toolbar when node is changed\r\n\t */\n\t editor.toolbar.move();\n\t };\n\t\n\t return toolbox;\n\t}({});\n\n/***/ },\n/* 12 */\n/***/ function(module, exports) {\n\n\t'use strict';\n\t\n\t/**\r\n\t * @module Codex Editor Callbacks module\r\n\t * @description Module works with editor added Elements\r\n\t *\r\n\t * @author Codex Team\r\n\t * @version 1.4.0\r\n\t */\n\t\n\tmodule.exports = function (callbacks) {\n\t\n\t var editor = codex.editor;\n\t\n\t /**\r\n\t * used by UI module\r\n\t * @description Routes all keydowns on document\r\n\t * @param {Object} event\r\n\t */\n\t callbacks.globalKeydown = function (event) {\n\t\n\t switch (event.keyCode) {\n\t case editor.core.keys.ENTER:\n\t enterKeyPressed_(event);break;\n\t }\n\t };\n\t\n\t /**\r\n\t * used by UI module\r\n\t * @description Routes all keydowns on redactors area\r\n\t * @param {Object} event\r\n\t */\n\t callbacks.redactorKeyDown = function (event) {\n\t\n\t switch (event.keyCode) {\n\t case editor.core.keys.TAB:\n\t tabKeyPressedOnRedactorsZone_(event);break;\n\t case editor.core.keys.ENTER:\n\t enterKeyPressedOnRedactorsZone_(event);break;\n\t case editor.core.keys.ESC:\n\t escapeKeyPressedOnRedactorsZone_(event);break;\n\t default:\n\t defaultKeyPressedOnRedactorsZone_(event);break;\n\t }\n\t };\n\t\n\t /**\r\n\t * used by UI module\r\n\t * @description Routes all keyup events\r\n\t * @param {Object} event\r\n\t */\n\t callbacks.globalKeyup = function (event) {\n\t\n\t switch (event.keyCode) {\n\t case editor.core.keys.UP:\n\t case editor.core.keys.LEFT:\n\t case editor.core.keys.RIGHT:\n\t case editor.core.keys.DOWN:\n\t arrowKeyPressed_(event);break;\n\t }\n\t };\n\t\n\t /**\r\n\t * @param {Object} event\r\n\t * @private\r\n\t *\r\n\t * Handles behaviour when tab pressed\r\n\t * @description if Content is empty show toolbox (if it is closed) or leaf tools\r\n\t * uses Toolbars toolbox module to handle the situation\r\n\t */\n\t var tabKeyPressedOnRedactorsZone_ = function tabKeyPressedOnRedactorsZone_(event) {\n\t\n\t /**\r\n\t * Wait for solution. Would like to know the behaviour\r\n\t * @todo Add spaces\r\n\t */\n\t event.preventDefault();\n\t\n\t var nativeInputs = editor.content.currentNode.querySelectorAll('textarea, input'),\n\t nativeInputsAreEmpty = true,\n\t textContentIsEmpty = !editor.content.currentNode.textContent.trim();\n\t\n\t Array.prototype.map.call(nativeInputs, function (input) {\n\t\n\t if (input.type == 'textarea' || input.type == 'text') {\n\t\n\t nativeInputsAreEmpty = nativeInputsAreEmpty && !input.value.trim();\n\t }\n\t });\n\t\n\t var blockIsEmpty = textContentIsEmpty && nativeInputsAreEmpty;\n\t\n\t if (!blockIsEmpty) {\n\t\n\t return;\n\t }\n\t\n\t if (!editor.toolbar.opened) {\n\t\n\t editor.toolbar.open();\n\t }\n\t\n\t if (editor.toolbar.opened && !editor.toolbar.toolbox.opened) {\n\t\n\t editor.toolbar.toolbox.open();\n\t } else {\n\t\n\t editor.toolbar.toolbox.leaf();\n\t }\n\t };\n\t\n\t /**\r\n\t * Handles global EnterKey Press\r\n\t * @see enterPressedOnBlock_\r\n\t * @param {Object} event\r\n\t */\n\t var enterKeyPressed_ = function enterKeyPressed_() {\n\t\n\t if (editor.content.editorAreaHightlighted) {\n\t\n\t /**\r\n\t * it means that we lose input index, saved index before is not correct\r\n\t * therefore we need to set caret when we insert new block\r\n\t */\n\t editor.caret.inputIndex = -1;\n\t\n\t enterPressedOnBlock_();\n\t }\n\t };\n\t\n\t /**\r\n\t * Callback for enter key pressing in first-level block area\r\n\t *\r\n\t * @param {Event} event\r\n\t * @private\r\n\t *\r\n\t * @description Inserts new block with initial type from settings\r\n\t */\n\t var enterPressedOnBlock_ = function enterPressedOnBlock_() {\n\t\n\t var NEW_BLOCK_TYPE = editor.settings.initialBlockPlugin;\n\t\n\t editor.content.insertBlock({\n\t type: NEW_BLOCK_TYPE,\n\t block: editor.tools[NEW_BLOCK_TYPE].render()\n\t }, true);\n\t\n\t editor.toolbar.move();\n\t editor.toolbar.open();\n\t };\n\t\n\t /**\r\n\t * ENTER key handler\r\n\t *\r\n\t * @param {Object} event\r\n\t * @private\r\n\t *\r\n\t * @description Makes new block with initial type from settings\r\n\t */\n\t var enterKeyPressedOnRedactorsZone_ = function enterKeyPressedOnRedactorsZone_(event) {\n\t\n\t if (event.target.contentEditable == 'true') {\n\t\n\t /** Update input index */\n\t editor.caret.saveCurrentInputIndex();\n\t }\n\t\n\t var currentInputIndex = editor.caret.getCurrentInputIndex() || 0,\n\t workingNode = editor.content.currentNode,\n\t tool = workingNode.dataset.tool,\n\t isEnterPressedOnToolbar = editor.toolbar.opened && editor.toolbar.current && event.target == editor.state.inputs[currentInputIndex];\n\t\n\t /** The list of tools which needs the default browser behaviour */\n\t var enableLineBreaks = editor.tools[tool].enableLineBreaks;\n\t\n\t /** This type of block creates when enter is pressed */\n\t var NEW_BLOCK_TYPE = editor.settings.initialBlockPlugin;\n\t\n\t /**\r\n\t * When toolbar is opened, select tool instead of making new paragraph\r\n\t */\n\t if (isEnterPressedOnToolbar) {\n\t\n\t event.preventDefault();\n\t\n\t editor.toolbar.toolbox.toolClicked(event);\n\t\n\t editor.toolbar.close();\n\t\n\t /**\r\n\t * Stop other listeners callback executions\r\n\t */\n\t event.stopPropagation();\n\t event.stopImmediatePropagation();\n\t\n\t return;\n\t }\n\t\n\t /**\r\n\t * Allow paragraph lineBreaks with shift enter\r\n\t * Or if shiftkey pressed and enter and enabledLineBreaks, the let new block creation\r\n\t */\n\t if (event.shiftKey || enableLineBreaks) {\n\t\n\t event.stopPropagation();\n\t event.stopImmediatePropagation();\n\t return;\n\t }\n\t\n\t var currentSelection = window.getSelection(),\n\t currentSelectedNode = currentSelection.anchorNode,\n\t caretAtTheEndOfText = editor.caret.position.atTheEnd(),\n\t isTextNodeHasParentBetweenContenteditable = false;\n\t\n\t /**\r\n\t * Allow making new

    in same block by SHIFT+ENTER and forbids to prevent default browser behaviour\r\n\t */\n\t if (event.shiftKey && !enableLineBreaks) {\n\t\n\t editor.callback.enterPressedOnBlock(editor.content.currentBlock, event);\n\t event.preventDefault();\n\t return;\n\t }\n\t\n\t /**\r\n\t * Workaround situation when caret at the Text node that has some wrapper Elements\r\n\t * Split block cant handle this.\r\n\t * We need to save default behavior\r\n\t */\n\t isTextNodeHasParentBetweenContenteditable = currentSelectedNode && currentSelectedNode.parentNode.contentEditable != 'true';\n\t\n\t /**\r\n\t * Split blocks when input has several nodes and caret placed in textNode\r\n\t */\n\t if (currentSelectedNode.nodeType == editor.core.nodeTypes.TEXT && !isTextNodeHasParentBetweenContenteditable && !caretAtTheEndOfText) {\n\t\n\t event.preventDefault();\n\t\n\t editor.core.log('Splitting Text node...');\n\t\n\t editor.content.splitBlock(currentInputIndex);\n\t\n\t /** Show plus button when next input after split is empty*/\n\t if (!editor.state.inputs[currentInputIndex + 1].textContent.trim()) {\n\t\n\t editor.toolbar.showPlusButton();\n\t }\n\t } else {\n\t\n\t var islastNode = editor.content.isLastNode(currentSelectedNode);\n\t\n\t if (islastNode && caretAtTheEndOfText) {\n\t\n\t event.preventDefault();\n\t event.stopPropagation();\n\t event.stopImmediatePropagation();\n\t\n\t editor.core.log('ENTER clicked in last textNode. Create new BLOCK');\n\t\n\t editor.content.insertBlock({\n\t type: NEW_BLOCK_TYPE,\n\t block: editor.tools[NEW_BLOCK_TYPE].render()\n\t }, true);\n\t\n\t editor.toolbar.move();\n\t editor.toolbar.open();\n\t\n\t /** Show plus button with empty block */\n\t editor.toolbar.showPlusButton();\n\t }\n\t }\n\t\n\t /** get all inputs after new appending block */\n\t editor.ui.saveInputs();\n\t };\n\t\n\t /**\r\n\t * Escape behaviour\r\n\t * @param event\r\n\t * @private\r\n\t *\r\n\t * @description Closes toolbox and toolbar. Prevents default behaviour\r\n\t */\n\t var escapeKeyPressedOnRedactorsZone_ = function escapeKeyPressedOnRedactorsZone_(event) {\n\t\n\t /** Close all toolbar */\n\t editor.toolbar.close();\n\t\n\t /** Close toolbox */\n\t editor.toolbar.toolbox.close();\n\t\n\t event.preventDefault();\n\t };\n\t\n\t /**\r\n\t * @param {Event} event\r\n\t * @private\r\n\t *\r\n\t * closes and moves toolbar\r\n\t */\n\t var arrowKeyPressed_ = function arrowKeyPressed_(event) {\n\t\n\t editor.content.workingNodeChanged();\n\t\n\t /* Closing toolbar */\n\t editor.toolbar.close();\n\t editor.toolbar.move();\n\t };\n\t\n\t /**\r\n\t * @private\r\n\t * @param {Event} event\r\n\t *\r\n\t * @description Closes all opened bars from toolbar.\r\n\t * If block is mark, clears highlightning\r\n\t */\n\t var defaultKeyPressedOnRedactorsZone_ = function defaultKeyPressedOnRedactorsZone_() {\n\t\n\t editor.toolbar.close();\n\t\n\t if (!editor.toolbar.inline.actionsOpened) {\n\t\n\t editor.toolbar.inline.close();\n\t editor.content.clearMark();\n\t }\n\t };\n\t\n\t /**\r\n\t * Handler when clicked on redactors area\r\n\t *\r\n\t * @protected\r\n\t * @param event\r\n\t *\r\n\t * @description Detects clicked area. If it is first-level block area, marks as detected and\r\n\t * on next enter press will be inserted new block\r\n\t * Otherwise, save carets position (input index) and put caret to the editable zone.\r\n\t *\r\n\t * @see detectWhenClickedOnFirstLevelBlockArea_\r\n\t *\r\n\t */\n\t callbacks.redactorClicked = function (event) {\n\t\n\t detectWhenClickedOnFirstLevelBlockArea_();\n\t\n\t editor.content.workingNodeChanged(event.target);\n\t editor.ui.saveInputs();\n\t\n\t var selectedText = editor.toolbar.inline.getSelectionText(),\n\t firstLevelBlock;\n\t\n\t /** If selection range took off, then we hide inline toolbar */\n\t if (selectedText.length === 0) {\n\t\n\t editor.toolbar.inline.close();\n\t }\n\t\n\t /** Update current input index in memory when caret focused into existed input */\n\t if (event.target.contentEditable == 'true') {\n\t\n\t editor.caret.saveCurrentInputIndex();\n\t }\n\t\n\t if (editor.content.currentNode === null) {\n\t\n\t /**\r\n\t * If inputs in redactor does not exits, then we put input index 0 not -1\r\n\t */\n\t var indexOfLastInput = editor.state.inputs.length > 0 ? editor.state.inputs.length - 1 : 0;\n\t\n\t /** If we have any inputs */\n\t if (editor.state.inputs.length) {\n\t\n\t /** getting firstlevel parent of input */\n\t firstLevelBlock = editor.content.getFirstLevelBlock(editor.state.inputs[indexOfLastInput]);\n\t }\n\t\n\t /** If input is empty, then we set caret to the last input */\n\t if (editor.state.inputs.length && editor.state.inputs[indexOfLastInput].textContent === '' && firstLevelBlock.dataset.tool == editor.settings.initialBlockPlugin) {\n\t\n\t editor.caret.setToBlock(indexOfLastInput);\n\t } else {\n\t\n\t /** Create new input when caret clicked in redactors area */\n\t var NEW_BLOCK_TYPE = editor.settings.initialBlockPlugin;\n\t\n\t editor.content.insertBlock({\n\t type: NEW_BLOCK_TYPE,\n\t block: editor.tools[NEW_BLOCK_TYPE].render()\n\t });\n\t\n\t /** If there is no inputs except inserted */\n\t if (editor.state.inputs.length === 1) {\n\t\n\t editor.caret.setToBlock(indexOfLastInput);\n\t } else {\n\t\n\t /** Set caret to this appended input */\n\t editor.caret.setToNextBlock(indexOfLastInput);\n\t }\n\t }\n\t } else {\n\t\n\t /** Close all panels */\n\t editor.toolbar.settings.close();\n\t editor.toolbar.toolbox.close();\n\t }\n\t\n\t /**\r\n\t * Move toolbar and open\r\n\t */\n\t editor.toolbar.move();\n\t editor.toolbar.open();\n\t\n\t var inputIsEmpty = !editor.content.currentNode.textContent.trim(),\n\t currentNodeType = editor.content.currentNode.dataset.tool,\n\t isInitialType = currentNodeType == editor.settings.initialBlockPlugin;\n\t\n\t /** Hide plus buttons */\n\t editor.toolbar.hidePlusButton();\n\t\n\t if (!inputIsEmpty) {\n\t\n\t /** Mark current block */\n\t editor.content.markBlock();\n\t }\n\t\n\t if (isInitialType && inputIsEmpty) {\n\t\n\t /** Show plus button */\n\t editor.toolbar.showPlusButton();\n\t }\n\t };\n\t\n\t /**\r\n\t * This method allows to define, is caret in contenteditable element or not.\r\n\t *\r\n\t * @private\r\n\t *\r\n\t * @description Otherwise, if we get TEXT node from range container, that will means we have input index.\r\n\t * In this case we use default browsers behaviour (if plugin allows that) or overwritten action.\r\n\t * Therefore, to be sure that we've clicked first-level block area, we should have currentNode, which always\r\n\t * specifies to the first-level block. Other cases we just ignore.\r\n\t */\n\t var detectWhenClickedOnFirstLevelBlockArea_ = function detectWhenClickedOnFirstLevelBlockArea_() {\n\t\n\t var selection = window.getSelection(),\n\t anchorNode = selection.anchorNode,\n\t flag = false;\n\t\n\t if (selection.rangeCount === 0) {\n\t\n\t editor.content.editorAreaHightlighted = true;\n\t } else {\n\t\n\t if (!editor.core.isDomNode(anchorNode)) {\n\t\n\t anchorNode = anchorNode.parentNode;\n\t }\n\t\n\t /** Already founded, without loop */\n\t if (anchorNode.contentEditable == 'true') {\n\t\n\t flag = true;\n\t }\n\t\n\t while (anchorNode.contentEditable != 'true') {\n\t\n\t anchorNode = anchorNode.parentNode;\n\t\n\t if (anchorNode.contentEditable == 'true') {\n\t\n\t flag = true;\n\t }\n\t\n\t if (anchorNode == document.body) {\n\t\n\t break;\n\t }\n\t }\n\t\n\t /** If editable element founded, flag is \"TRUE\", Therefore we return \"FALSE\" */\n\t editor.content.editorAreaHightlighted = !flag;\n\t }\n\t };\n\t\n\t /**\r\n\t * Toolbar button click handler\r\n\t *\r\n\t * @param {Object} event - cursor to the button\r\n\t * @protected\r\n\t *\r\n\t * @description gets current tool and calls render method\r\n\t */\n\t callbacks.toolbarButtonClicked = function (event) {\n\t\n\t var button = this;\n\t\n\t editor.toolbar.current = button.dataset.type;\n\t\n\t editor.toolbar.toolbox.toolClicked(event);\n\t editor.toolbar.close();\n\t };\n\t\n\t /**\r\n\t * Show or Hide toolbox when plus button is clicked\r\n\t */\n\t callbacks.plusButtonClicked = function () {\n\t\n\t if (!editor.nodes.toolbox.classList.contains('opened')) {\n\t\n\t editor.toolbar.toolbox.open();\n\t } else {\n\t\n\t editor.toolbar.toolbox.close();\n\t }\n\t };\n\t\n\t /**\r\n\t * Block handlers for KeyDown events\r\n\t *\r\n\t * @protected\r\n\t * @param {Object} event\r\n\t *\r\n\t * Handles keydowns on block\r\n\t * @see blockRightOrDownArrowPressed_\r\n\t * @see backspacePressed_\r\n\t * @see blockLeftOrUpArrowPressed_\r\n\t */\n\t callbacks.blockKeydown = function (event) {\n\t\n\t var block = event.target; // event.target is input\n\t\n\t switch (event.keyCode) {\n\t\n\t case editor.core.keys.DOWN:\n\t case editor.core.keys.RIGHT:\n\t blockRightOrDownArrowPressed_(event);\n\t break;\n\t\n\t case editor.core.keys.BACKSPACE:\n\t backspacePressed_(block, event);\n\t break;\n\t\n\t case editor.core.keys.UP:\n\t case editor.core.keys.LEFT:\n\t blockLeftOrUpArrowPressed_(event);\n\t break;\n\t\n\t }\n\t };\n\t\n\t /**\r\n\t * RIGHT or DOWN keydowns on block\r\n\t *\r\n\t * @param {Object} event\r\n\t * @private\r\n\t *\r\n\t * @description watches the selection and gets closest editable element.\r\n\t * Uses method getDeepestTextNodeFromPosition to get the last node of next block\r\n\t * Sets caret if it is contenteditable\r\n\t */\n\t var blockRightOrDownArrowPressed_ = function blockRightOrDownArrowPressed_(event) {\n\t\n\t var selection = window.getSelection(),\n\t inputs = editor.state.inputs,\n\t focusedNode = selection.anchorNode,\n\t focusedNodeHolder;\n\t\n\t /** Check for caret existance */\n\t if (!focusedNode) {\n\t\n\t return false;\n\t }\n\t\n\t /** Looking for closest (parent) contentEditable element of focused node */\n\t while (focusedNode.contentEditable != 'true') {\n\t\n\t focusedNodeHolder = focusedNode.parentNode;\n\t focusedNode = focusedNodeHolder;\n\t }\n\t\n\t /** Input index in DOM level */\n\t var editableElementIndex = 0;\n\t\n\t while (focusedNode != inputs[editableElementIndex]) {\n\t\n\t editableElementIndex++;\n\t }\n\t\n\t /**\r\n\t * Founded contentEditable element doesn't have childs\r\n\t * Or maybe New created block\r\n\t */\n\t if (!focusedNode.textContent) {\n\t\n\t editor.caret.setToNextBlock(editableElementIndex);\n\t return;\n\t }\n\t\n\t /**\r\n\t * Do nothing when caret doesn not reaches the end of last child\r\n\t */\n\t var caretInLastChild = false,\n\t caretAtTheEndOfText = false;\n\t\n\t var lastChild, deepestTextnode;\n\t\n\t lastChild = focusedNode.childNodes[focusedNode.childNodes.length - 1];\n\t\n\t if (editor.core.isDomNode(lastChild)) {\n\t\n\t deepestTextnode = editor.content.getDeepestTextNodeFromPosition(lastChild, lastChild.childNodes.length);\n\t } else {\n\t\n\t deepestTextnode = lastChild;\n\t }\n\t\n\t caretInLastChild = selection.anchorNode == deepestTextnode;\n\t caretAtTheEndOfText = deepestTextnode.length == selection.anchorOffset;\n\t\n\t if (!caretInLastChild || !caretAtTheEndOfText) {\n\t\n\t editor.core.log('arrow [down|right] : caret does not reached the end');\n\t return false;\n\t }\n\t\n\t editor.caret.setToNextBlock(editableElementIndex);\n\t };\n\t\n\t /**\r\n\t * LEFT or UP keydowns on block\r\n\t *\r\n\t * @param {Object} event\r\n\t * @private\r\n\t *\r\n\t * watches the selection and gets closest editable element.\r\n\t * Uses method getDeepestTextNodeFromPosition to get the last node of previous block\r\n\t * Sets caret if it is contenteditable\r\n\t *\r\n\t */\n\t var blockLeftOrUpArrowPressed_ = function blockLeftOrUpArrowPressed_(event) {\n\t\n\t var selection = window.getSelection(),\n\t inputs = editor.state.inputs,\n\t focusedNode = selection.anchorNode,\n\t focusedNodeHolder;\n\t\n\t /** Check for caret existance */\n\t if (!focusedNode) {\n\t\n\t return false;\n\t }\n\t\n\t /**\r\n\t * LEFT or UP not at the beginning\r\n\t */\n\t if (selection.anchorOffset !== 0) {\n\t\n\t return false;\n\t }\n\t\n\t /** Looking for parent contentEditable block */\n\t while (focusedNode.contentEditable != 'true') {\n\t\n\t focusedNodeHolder = focusedNode.parentNode;\n\t focusedNode = focusedNodeHolder;\n\t }\n\t\n\t /** Input index in DOM level */\n\t var editableElementIndex = 0;\n\t\n\t while (focusedNode != inputs[editableElementIndex]) {\n\t\n\t editableElementIndex++;\n\t }\n\t\n\t /**\r\n\t * Do nothing if caret is not at the beginning of first child\r\n\t */\n\t var caretInFirstChild = false,\n\t caretAtTheBeginning = false;\n\t\n\t var firstChild, deepestTextnode;\n\t\n\t /**\r\n\t * Founded contentEditable element doesn't have childs\r\n\t * Or maybe New created block\r\n\t */\n\t if (!focusedNode.textContent) {\n\t\n\t editor.caret.setToPreviousBlock(editableElementIndex);\n\t return;\n\t }\n\t\n\t firstChild = focusedNode.childNodes[0];\n\t\n\t if (editor.core.isDomNode(firstChild)) {\n\t\n\t deepestTextnode = editor.content.getDeepestTextNodeFromPosition(firstChild, 0);\n\t } else {\n\t\n\t deepestTextnode = firstChild;\n\t }\n\t\n\t caretInFirstChild = selection.anchorNode == deepestTextnode;\n\t caretAtTheBeginning = selection.anchorOffset === 0;\n\t\n\t if (caretInFirstChild && caretAtTheBeginning) {\n\t\n\t editor.caret.setToPreviousBlock(editableElementIndex);\n\t }\n\t };\n\t\n\t /**\r\n\t * Handles backspace keydown\r\n\t *\r\n\t * @param {Element} block\r\n\t * @param {Object} event\r\n\t * @private\r\n\t *\r\n\t * @description if block is empty, delete the block and set caret to the previous block\r\n\t * If block is not empty, try to merge two blocks - current and previous\r\n\t * But it we try'n to remove first block, then we should set caret to the next block, not previous.\r\n\t * If we removed the last block, create new one\r\n\t */\n\t var backspacePressed_ = function backspacePressed_(block, event) {\n\t\n\t var currentInputIndex = editor.caret.getCurrentInputIndex(),\n\t range,\n\t selectionLength,\n\t firstLevelBlocksCount;\n\t\n\t if (editor.core.isNativeInput(event.target)) {\n\t\n\t /** If input value is empty - remove block */\n\t if (event.target.value.trim() == '') {\n\t\n\t block.remove();\n\t } else {\n\t\n\t return;\n\t }\n\t }\n\t\n\t if (block.textContent.trim()) {\n\t\n\t range = editor.content.getRange();\n\t selectionLength = range.endOffset - range.startOffset;\n\t\n\t if (editor.caret.position.atStart() && !selectionLength && editor.state.inputs[currentInputIndex - 1]) {\n\t\n\t editor.content.mergeBlocks(currentInputIndex);\n\t } else {\n\t\n\t return;\n\t }\n\t }\n\t\n\t if (!selectionLength) {\n\t\n\t block.remove();\n\t }\n\t\n\t firstLevelBlocksCount = editor.nodes.redactor.childNodes.length;\n\t\n\t /**\r\n\t * If all blocks are removed\r\n\t */\n\t if (firstLevelBlocksCount === 0) {\n\t\n\t /** update currentNode variable */\n\t editor.content.currentNode = null;\n\t\n\t /** Inserting new empty initial block */\n\t editor.ui.addInitialBlock();\n\t\n\t /** Updating inputs state after deleting last block */\n\t editor.ui.saveInputs();\n\t\n\t /** Set to current appended block */\n\t window.setTimeout(function () {\n\t\n\t editor.caret.setToPreviousBlock(1);\n\t }, 10);\n\t } else {\n\t\n\t if (editor.caret.inputIndex !== 0) {\n\t\n\t /** Target block is not first */\n\t editor.caret.setToPreviousBlock(editor.caret.inputIndex);\n\t } else {\n\t\n\t /** If we try to delete first block */\n\t editor.caret.setToNextBlock(editor.caret.inputIndex);\n\t }\n\t }\n\t\n\t editor.toolbar.move();\n\t\n\t if (!editor.toolbar.opened) {\n\t\n\t editor.toolbar.open();\n\t }\n\t\n\t /** Updating inputs state */\n\t editor.ui.saveInputs();\n\t\n\t /** Prevent default browser behaviour */\n\t event.preventDefault();\n\t };\n\t\n\t /**\r\n\t * used by UI module\r\n\t * Clicks on block settings button\r\n\t *\r\n\t * @param {Object} event\r\n\t * @protected\r\n\t * @description Opens toolbar settings\r\n\t */\n\t callbacks.showSettingsButtonClicked = function (event) {\n\t\n\t /**\r\n\t * Get type of current block\r\n\t * It uses to append settings from tool.settings property.\r\n\t * ...\r\n\t * Type is stored in data-type attribute on block\r\n\t */\n\t var currentToolType = editor.content.currentNode.dataset.tool;\n\t\n\t editor.toolbar.settings.toggle(currentToolType);\n\t\n\t /** Close toolbox when settings button is active */\n\t editor.toolbar.toolbox.close();\n\t editor.toolbar.settings.hideRemoveActions();\n\t };\n\t\n\t return callbacks;\n\t}({});\n\n/***/ },\n/* 13 */\n/***/ function(module, exports) {\n\n\t'use strict';\n\t\n\t/**\r\n\t * Codex Editor Draw module\r\n\t *\r\n\t * @author Codex Team\r\n\t * @version 1.0.\r\n\t */\n\t\n\tmodule.exports = function (draw) {\n\t\n\t /**\r\n\t * Base editor wrapper\r\n\t */\n\t draw.wrapper = function () {\n\t\n\t var wrapper = document.createElement('div');\n\t\n\t wrapper.className += 'codex-editor';\n\t\n\t return wrapper;\n\t };\n\t\n\t /**\r\n\t * Content-editable holder\r\n\t */\n\t draw.redactor = function () {\n\t\n\t var redactor = document.createElement('div');\n\t\n\t redactor.className += 'ce-redactor';\n\t\n\t return redactor;\n\t };\n\t\n\t draw.ceBlock = function () {\n\t\n\t var block = document.createElement('DIV');\n\t\n\t block.className += 'ce_block';\n\t\n\t return block;\n\t };\n\t\n\t /**\r\n\t * Empty toolbar with toggler\r\n\t */\n\t draw.toolbar = function () {\n\t\n\t var bar = document.createElement('div');\n\t\n\t bar.className += 'ce-toolbar';\n\t\n\t return bar;\n\t };\n\t\n\t draw.toolbarContent = function () {\n\t\n\t var wrapper = document.createElement('DIV');\n\t\n\t wrapper.classList.add('ce-toolbar__content');\n\t\n\t return wrapper;\n\t };\n\t\n\t /**\r\n\t * Inline toolbar\r\n\t */\n\t draw.inlineToolbar = function () {\n\t\n\t var bar = document.createElement('DIV');\n\t\n\t bar.className += 'ce-toolbar-inline';\n\t\n\t return bar;\n\t };\n\t\n\t /**\r\n\t * Wrapper for inline toobar buttons\r\n\t */\n\t draw.inlineToolbarButtons = function () {\n\t\n\t var wrapper = document.createElement('DIV');\n\t\n\t wrapper.className += 'ce-toolbar-inline__buttons';\n\t\n\t return wrapper;\n\t };\n\t\n\t /**\r\n\t * For some actions\r\n\t */\n\t draw.inlineToolbarActions = function () {\n\t\n\t var wrapper = document.createElement('DIV');\n\t\n\t wrapper.className += 'ce-toolbar-inline__actions';\n\t\n\t return wrapper;\n\t };\n\t\n\t draw.inputForLink = function () {\n\t\n\t var input = document.createElement('INPUT');\n\t\n\t input.type = 'input';\n\t input.className += 'inputForLink';\n\t input.placeholder = 'Вставьте ссылку ...';\n\t input.setAttribute('form', 'defaultForm');\n\t\n\t input.setAttribute('autofocus', 'autofocus');\n\t\n\t return input;\n\t };\n\t\n\t /**\r\n\t * @todo Desc\r\n\t */\n\t draw.blockButtons = function () {\n\t\n\t var block = document.createElement('div');\n\t\n\t block.className += 'ce-toolbar__actions';\n\t\n\t return block;\n\t };\n\t\n\t /**\r\n\t * Block settings panel\r\n\t */\n\t draw.blockSettings = function () {\n\t\n\t var settings = document.createElement('div');\n\t\n\t settings.className += 'ce-settings';\n\t\n\t return settings;\n\t };\n\t\n\t draw.defaultSettings = function () {\n\t\n\t var div = document.createElement('div');\n\t\n\t div.classList.add('ce-settings_default');\n\t\n\t return div;\n\t };\n\t\n\t draw.pluginsSettings = function () {\n\t\n\t var div = document.createElement('div');\n\t\n\t div.classList.add('ce-settings_plugin');\n\t\n\t return div;\n\t };\n\t\n\t draw.plusButton = function () {\n\t\n\t var button = document.createElement('span');\n\t\n\t button.className = 'ce-toolbar__plus';\n\t // button.innerHTML = '';\n\t\n\t return button;\n\t };\n\t\n\t /**\r\n\t * Settings button in toolbar\r\n\t */\n\t draw.settingsButton = function () {\n\t\n\t var toggler = document.createElement('span');\n\t\n\t toggler.className = 'ce-toolbar__settings-btn';\n\t\n\t /** Toggler button*/\n\t toggler.innerHTML = '';\n\t\n\t return toggler;\n\t };\n\t\n\t /**\r\n\t * Redactor tools wrapper\r\n\t */\n\t\n\t draw.toolbox = function () {\n\t\n\t var wrapper = document.createElement('div');\n\t\n\t wrapper.className = 'ce-toolbar__tools';\n\t\n\t return wrapper;\n\t };\n\t\n\t /**\r\n\t * @protected\r\n\t *\r\n\t * Draws tool buttons for toolbox\r\n\t *\r\n\t * @param {String} type\r\n\t * @param {String} classname\r\n\t * @returns {Element}\r\n\t */\n\t draw.toolbarButton = function (type, classname) {\n\t\n\t var button = document.createElement('li'),\n\t toolIcon = document.createElement('i'),\n\t toolTitle = document.createElement('span');\n\t\n\t button.dataset.type = type;\n\t button.setAttribute('title', type);\n\t\n\t toolIcon.classList.add(classname);\n\t toolTitle.classList.add('ce_toolbar_tools--title');\n\t\n\t button.appendChild(toolIcon);\n\t button.appendChild(toolTitle);\n\t\n\t return button;\n\t };\n\t\n\t /**\r\n\t * @protected\r\n\t *\r\n\t * Draws tools for inline toolbar\r\n\t *\r\n\t * @param {String} type\r\n\t * @param {String} classname\r\n\t */\n\t draw.toolbarButtonInline = function (type, classname) {\n\t\n\t var button = document.createElement('BUTTON'),\n\t toolIcon = document.createElement('I');\n\t\n\t button.type = 'button';\n\t button.dataset.type = type;\n\t toolIcon.classList.add(classname);\n\t\n\t button.appendChild(toolIcon);\n\t\n\t return button;\n\t };\n\t\n\t /**\r\n\t * Redactor block\r\n\t */\n\t draw.block = function (tagName, content) {\n\t\n\t var node = document.createElement(tagName);\n\t\n\t node.innerHTML = content || '';\n\t\n\t return node;\n\t };\n\t\n\t /**\r\n\t * Creates Node with passed tagName and className\r\n\t * @param {string} tagName\r\n\t * @param {string} className\r\n\t * @param {object} properties - allow to assign properties\r\n\t */\n\t draw.node = function (tagName, className, properties) {\n\t\n\t var el = document.createElement(tagName);\n\t\n\t if (className) el.className = className;\n\t\n\t if (properties) {\n\t\n\t for (var name in properties) {\n\t\n\t el[name] = properties[name];\n\t }\n\t }\n\t\n\t return el;\n\t };\n\t\n\t /**\r\n\t * Unavailable plugin block\r\n\t */\n\t draw.unavailableBlock = function () {\n\t\n\t var wrapper = document.createElement('DIV');\n\t\n\t wrapper.classList.add('cdx-unavailable-block');\n\t\n\t return wrapper;\n\t };\n\t\n\t return draw;\n\t}({});\n\n/***/ },\n/* 14 */\n/***/ function(module, exports) {\n\n\t'use strict';\n\t\n\t/**\r\n\t * Codex Editor Caret Module\r\n\t *\r\n\t * @author Codex Team\r\n\t * @version 1.0\r\n\t */\n\t\n\tmodule.exports = function (caret) {\n\t\n\t var editor = codex.editor;\n\t\n\t /**\r\n\t * @var {int} InputIndex - editable element in DOM\r\n\t */\n\t caret.inputIndex = null;\n\t\n\t /**\r\n\t * @var {int} offset - caret position in a text node.\r\n\t */\n\t caret.offset = null;\n\t\n\t /**\r\n\t * @var {int} focusedNodeIndex - we get index of child node from first-level block\r\n\t */\n\t caret.focusedNodeIndex = null;\n\t\n\t /**\r\n\t * Creates Document Range and sets caret to the element.\r\n\t * @protected\r\n\t * @uses caret.save — if you need to save caret position\r\n\t * @param {Element} el - Changed Node.\r\n\t */\n\t caret.set = function (el, index, offset) {\n\t\n\t offset = offset || caret.offset || 0;\n\t index = index || caret.focusedNodeIndex || 0;\n\t\n\t var childs = el.childNodes,\n\t nodeToSet;\n\t\n\t if (childs.length === 0) {\n\t\n\t nodeToSet = el;\n\t } else {\n\t\n\t nodeToSet = childs[index];\n\t }\n\t\n\t /** If Element is INPUT */\n\t if (el.contentEditable != 'true') {\n\t\n\t el.focus();\n\t return;\n\t }\n\t\n\t if (editor.core.isDomNode(nodeToSet)) {\n\t\n\t nodeToSet = editor.content.getDeepestTextNodeFromPosition(nodeToSet, nodeToSet.childNodes.length);\n\t }\n\t\n\t var range = document.createRange(),\n\t selection = window.getSelection();\n\t\n\t window.setTimeout(function () {\n\t\n\t range.setStart(nodeToSet, offset);\n\t range.setEnd(nodeToSet, offset);\n\t\n\t selection.removeAllRanges();\n\t selection.addRange(range);\n\t\n\t editor.caret.saveCurrentInputIndex();\n\t }, 20);\n\t };\n\t\n\t /**\r\n\t * @protected\r\n\t * Updates index of input and saves it in caret object\r\n\t */\n\t caret.saveCurrentInputIndex = function () {\n\t\n\t /** Index of Input that we paste sanitized content */\n\t var selection = window.getSelection(),\n\t inputs = editor.state.inputs,\n\t focusedNode = selection.anchorNode,\n\t focusedNodeHolder;\n\t\n\t if (!focusedNode) {\n\t\n\t return;\n\t }\n\t\n\t /** Looking for parent contentEditable block */\n\t while (focusedNode.contentEditable != 'true') {\n\t\n\t focusedNodeHolder = focusedNode.parentNode;\n\t focusedNode = focusedNodeHolder;\n\t }\n\t\n\t /** Input index in DOM level */\n\t var editableElementIndex = 0;\n\t\n\t while (focusedNode != inputs[editableElementIndex]) {\n\t\n\t editableElementIndex++;\n\t }\n\t\n\t caret.inputIndex = editableElementIndex;\n\t };\n\t\n\t /**\r\n\t * Returns current input index (caret object)\r\n\t */\n\t caret.getCurrentInputIndex = function () {\n\t\n\t return caret.inputIndex;\n\t };\n\t\n\t /**\r\n\t * @param {int} index - index of first-level block after that we set caret into next input\r\n\t */\n\t caret.setToNextBlock = function (index) {\n\t\n\t var inputs = editor.state.inputs,\n\t nextInput = inputs[index + 1];\n\t\n\t if (!nextInput) {\n\t\n\t editor.core.log('We are reached the end');\n\t return;\n\t }\n\t\n\t /**\r\n\t * When new Block created or deleted content of input\r\n\t * We should add some text node to set caret\r\n\t */\n\t if (!nextInput.childNodes.length) {\n\t\n\t var emptyTextElement = document.createTextNode('');\n\t\n\t nextInput.appendChild(emptyTextElement);\n\t }\n\t\n\t editor.caret.inputIndex = index + 1;\n\t editor.caret.set(nextInput, 0, 0);\n\t editor.content.workingNodeChanged(nextInput);\n\t };\n\t\n\t /**\r\n\t * @param {int} index - index of target input.\r\n\t * Sets caret to input with this index\r\n\t */\n\t caret.setToBlock = function (index) {\n\t\n\t var inputs = editor.state.inputs,\n\t targetInput = inputs[index];\n\t\n\t if (!targetInput) {\n\t\n\t return;\n\t }\n\t\n\t /**\r\n\t * When new Block created or deleted content of input\r\n\t * We should add some text node to set caret\r\n\t */\n\t if (!targetInput.childNodes.length) {\n\t\n\t var emptyTextElement = document.createTextNode('');\n\t\n\t targetInput.appendChild(emptyTextElement);\n\t }\n\t\n\t editor.caret.inputIndex = index;\n\t editor.caret.set(targetInput, 0, 0);\n\t editor.content.workingNodeChanged(targetInput);\n\t };\n\t\n\t /**\r\n\t * @param {int} index - index of input\r\n\t */\n\t caret.setToPreviousBlock = function (index) {\n\t\n\t index = index || 0;\n\t\n\t var inputs = editor.state.inputs,\n\t previousInput = inputs[index - 1],\n\t lastChildNode,\n\t lengthOfLastChildNode,\n\t emptyTextElement;\n\t\n\t if (!previousInput) {\n\t\n\t editor.core.log('We are reached first node');\n\t return;\n\t }\n\t\n\t lastChildNode = editor.content.getDeepestTextNodeFromPosition(previousInput, previousInput.childNodes.length);\n\t lengthOfLastChildNode = lastChildNode.length;\n\t\n\t /**\r\n\t * When new Block created or deleted content of input\r\n\t * We should add some text node to set caret\r\n\t */\n\t if (!previousInput.childNodes.length) {\n\t\n\t emptyTextElement = document.createTextNode('');\n\t previousInput.appendChild(emptyTextElement);\n\t }\n\t editor.caret.inputIndex = index - 1;\n\t editor.caret.set(previousInput, previousInput.childNodes.length - 1, lengthOfLastChildNode);\n\t editor.content.workingNodeChanged(inputs[index - 1]);\n\t };\n\t\n\t caret.position = {\n\t\n\t atStart: function atStart() {\n\t\n\t var selection = window.getSelection(),\n\t anchorOffset = selection.anchorOffset,\n\t anchorNode = selection.anchorNode,\n\t firstLevelBlock = editor.content.getFirstLevelBlock(anchorNode),\n\t pluginsRender = firstLevelBlock.childNodes[0];\n\t\n\t if (!editor.core.isDomNode(anchorNode)) {\n\t\n\t anchorNode = anchorNode.parentNode;\n\t }\n\t\n\t var isFirstNode = anchorNode === pluginsRender.childNodes[0],\n\t isOffsetZero = anchorOffset === 0;\n\t\n\t return isFirstNode && isOffsetZero;\n\t },\n\t\n\t atTheEnd: function atTheEnd() {\n\t\n\t var selection = window.getSelection(),\n\t anchorOffset = selection.anchorOffset,\n\t anchorNode = selection.anchorNode;\n\t\n\t /** Caret is at the end of input */\n\t return !anchorNode || !anchorNode.length || anchorOffset === anchorNode.length;\n\t }\n\t };\n\t\n\t /**\r\n\t * Inserts node at the caret location\r\n\t * @param node\r\n\t */\n\t caret.insertNode = function (node) {\n\t\n\t var selection, range;\n\t\n\t selection = window.getSelection();\n\t\n\t range = selection.getRangeAt(0);\n\t range.deleteContents();\n\t\n\t range.insertNode(node);\n\t\n\t range.setStartAfter(node);\n\t range.collapse(true);\n\t\n\t selection.removeAllRanges();\n\t selection.addRange(range);\n\t };\n\t\n\t return caret;\n\t}({});\n\n/***/ },\n/* 15 */\n/***/ function(module, exports) {\n\n\t'use strict';\n\t\n\t/**\r\n\t * Codex Editor Notification Module\r\n\t *\r\n\t * @author Codex Team\r\n\t * @version 1.0\r\n\t */\n\t\n\tmodule.exports = function (notifications) {\n\t\n\t var editor = codex.editor;\n\t\n\t var queue = [];\n\t\n\t var addToQueue = function addToQueue(settings) {\n\t\n\t queue.push(settings);\n\t\n\t var index = 0;\n\t\n\t while (index < queue.length && queue.length > 5) {\n\t\n\t if (queue[index].type == 'confirm' || queue[index].type == 'prompt') {\n\t\n\t index++;\n\t continue;\n\t }\n\t\n\t queue[index].close();\n\t queue.splice(index, 1);\n\t }\n\t };\n\t\n\t notifications.createHolder = function () {\n\t\n\t var holder = editor.draw.node('DIV', 'cdx-notifications-block');\n\t\n\t editor.nodes.notifications = document.body.appendChild(holder);\n\t\n\t return holder;\n\t };\n\t\n\t /**\r\n\t * Error notificator. Shows block with message\r\n\t * @protected\r\n\t */\n\t notifications.errorThrown = function (errorMsg, event) {\n\t\n\t editor.notifications.notification({ message: 'This action is not available currently', type: event.type });\n\t };\n\t\n\t /**\r\n\t *\r\n\t * Appends notification\r\n\t *\r\n\t * settings = {\r\n\t * type - notification type (reserved types: alert, confirm, prompt). Just add class 'cdx-notification-'+type\r\n\t * message - notification message\r\n\t * okMsg - confirm button text (default - 'Ok')\r\n\t * cancelBtn - cancel button text (default - 'Cancel'). Only for confirm and prompt types\r\n\t * confirm - function-handler for ok button click\r\n\t * cancel - function-handler for cancel button click. Only for confirm and prompt types\r\n\t * time - time (in seconds) after which notification will close (default - 10s)\r\n\t * }\r\n\t *\r\n\t * @param settings\r\n\t */\n\t notifications.notification = function (constructorSettings) {\n\t\n\t /** Private vars and methods */\n\t var notification = null,\n\t cancel = null,\n\t type = null,\n\t confirm = null,\n\t inputField = null;\n\t\n\t var confirmHandler = function confirmHandler() {\n\t\n\t close();\n\t\n\t if (typeof confirm !== 'function') {\n\t\n\t return;\n\t }\n\t\n\t if (type == 'prompt') {\n\t\n\t confirm(inputField.value);\n\t return;\n\t }\n\t\n\t confirm();\n\t };\n\t\n\t var cancelHandler = function cancelHandler() {\n\t\n\t close();\n\t\n\t if (typeof cancel !== 'function') {\n\t\n\t return;\n\t }\n\t\n\t cancel();\n\t };\n\t\n\t /** Public methods */\n\t function create(settings) {\n\t\n\t if (!(settings && settings.message)) {\n\t\n\t editor.core.log('Can\\'t create notification. Message is missed');\n\t return;\n\t }\n\t\n\t settings.type = settings.type || 'alert';\n\t settings.time = settings.time * 1000 || 10000;\n\t\n\t var wrapper = editor.draw.node('DIV', 'cdx-notification'),\n\t message = editor.draw.node('DIV', 'cdx-notification__message'),\n\t input = editor.draw.node('INPUT', 'cdx-notification__input'),\n\t okBtn = editor.draw.node('SPAN', 'cdx-notification__ok-btn'),\n\t cancelBtn = editor.draw.node('SPAN', 'cdx-notification__cancel-btn');\n\t\n\t message.textContent = settings.message;\n\t okBtn.textContent = settings.okMsg || 'ОК';\n\t cancelBtn.textContent = settings.cancelMsg || 'Отмена';\n\t\n\t editor.listeners.add(okBtn, 'click', confirmHandler);\n\t editor.listeners.add(cancelBtn, 'click', cancelHandler);\n\t\n\t wrapper.appendChild(message);\n\t\n\t if (settings.type == 'prompt') {\n\t\n\t wrapper.appendChild(input);\n\t }\n\t\n\t wrapper.appendChild(okBtn);\n\t\n\t if (settings.type == 'prompt' || settings.type == 'confirm') {\n\t\n\t wrapper.appendChild(cancelBtn);\n\t }\n\t\n\t wrapper.classList.add('cdx-notification-' + settings.type);\n\t wrapper.dataset.type = settings.type;\n\t\n\t notification = wrapper;\n\t type = settings.type;\n\t confirm = settings.confirm;\n\t cancel = settings.cancel;\n\t inputField = input;\n\t\n\t if (settings.type != 'prompt' && settings.type != 'confirm') {\n\t\n\t window.setTimeout(close, settings.time);\n\t }\n\t };\n\t\n\t function send() {\n\t\n\t editor.nodes.notifications.appendChild(notification);\n\t inputField.focus();\n\t\n\t editor.nodes.notifications.classList.add('cdx-notification__notification-appending');\n\t\n\t window.setTimeout(function () {\n\t\n\t editor.nodes.notifications.classList.remove('cdx-notification__notification-appending');\n\t }, 100);\n\t\n\t addToQueue({ type: type, close: close });\n\t };\n\t\n\t function close() {\n\t\n\t notification.remove();\n\t };\n\t\n\t if (constructorSettings) {\n\t\n\t create(constructorSettings);\n\t send();\n\t }\n\t\n\t return {\n\t create: create,\n\t send: send,\n\t close: close\n\t };\n\t };\n\t\n\t notifications.clear = function () {\n\t\n\t editor.nodes.notifications.innerHTML = '';\n\t queue = [];\n\t };\n\t\n\t return notifications;\n\t}({});\n\n/***/ },\n/* 16 */\n/***/ function(module, exports) {\n\n\t\"use strict\";\n\t\n\t/**\r\n\t * Codex Editor Parser Module\r\n\t *\r\n\t * @author Codex Team\r\n\t * @version 1.1\r\n\t */\n\t\n\tmodule.exports = function (parser) {\n\t\n\t var editor = codex.editor;\n\t\n\t /** inserting text */\n\t parser.insertPastedContent = function (blockType, tag) {\n\t\n\t editor.content.insertBlock({\n\t type: blockType.type,\n\t block: blockType.render({\n\t text: tag.innerHTML\n\t })\n\t });\n\t };\n\t\n\t /**\r\n\t * Check DOM node for display style: separated block or child-view\r\n\t */\n\t parser.isFirstLevelBlock = function (node) {\n\t\n\t return node.nodeType == editor.core.nodeTypes.TAG && node.classList.contains(editor.ui.className.BLOCK_CLASSNAME);\n\t };\n\t\n\t return parser;\n\t}({});\n\n/***/ },\n/* 17 */\n/***/ function(module, exports, __webpack_require__) {\n\n\t'use strict';\n\t\n\t/**\r\n\t * Codex Sanitizer\r\n\t */\n\t\n\tmodule.exports = function (sanitizer) {\n\t\n\t /** HTML Janitor library */\n\t var janitor = __webpack_require__(18);\n\t\n\t /** Codex Editor */\n\t var editor = codex.editor;\n\t\n\t sanitizer.prepare = function () {\n\t\n\t if (editor.settings.sanitizer && !editor.core.isEmpty(editor.settings.sanitizer)) {\n\t\n\t Config.CUSTOM = editor.settings.sanitizer;\n\t }\n\t };\n\t\n\t /**\r\n\t * Basic config\r\n\t */\n\t var Config = {\n\t\n\t /** User configuration */\n\t CUSTOM: null,\n\t\n\t BASIC: {\n\t\n\t tags: {\n\t p: {},\n\t a: {\n\t href: true,\n\t target: '_blank',\n\t rel: 'nofollow'\n\t },\n\t i: {},\n\t b: {},\n\t strong: {},\n\t em: {},\n\t span: {}\n\t }\n\t }\n\t };\n\t\n\t sanitizer.Config = Config;\n\t\n\t /**\r\n\t *\r\n\t * @param userCustomConfig\r\n\t * @returns {*}\r\n\t * @private\r\n\t *\r\n\t * @description If developer uses editor's API, then he can customize sane restrictions.\r\n\t * Or, sane config can be defined globally in editors initialization. That config will be used everywhere\r\n\t * At least, if there is no config overrides, that API uses BASIC Default configation\r\n\t */\n\t var init_ = function init_(userCustomConfig) {\n\t\n\t var configuration = userCustomConfig || Config.CUSTOM || Config.BASIC;\n\t\n\t return new janitor(configuration);\n\t };\n\t\n\t /**\r\n\t * Cleans string from unwanted tags\r\n\t * @protected\r\n\t * @param {String} dirtyString - taint string\r\n\t * @param {Object} customConfig - allowed tags\r\n\t */\n\t sanitizer.clean = function (dirtyString, customConfig) {\n\t\n\t var janitorInstance = init_(customConfig);\n\t\n\t return janitorInstance.clean(dirtyString);\n\t };\n\t\n\t return sanitizer;\n\t}({});\n\n/***/ },\n/* 18 */\n/***/ function(module, exports, __webpack_require__) {\n\n\tvar __WEBPACK_AMD_DEFINE_FACTORY__, __WEBPACK_AMD_DEFINE_RESULT__;(function (root, factory) {\n\t if (true) {\n\t !(__WEBPACK_AMD_DEFINE_FACTORY__ = (factory), __WEBPACK_AMD_DEFINE_RESULT__ = (typeof __WEBPACK_AMD_DEFINE_FACTORY__ === 'function' ? (__WEBPACK_AMD_DEFINE_FACTORY__.call(exports, __webpack_require__, exports, module)) : __WEBPACK_AMD_DEFINE_FACTORY__), __WEBPACK_AMD_DEFINE_RESULT__ !== undefined && (module.exports = __WEBPACK_AMD_DEFINE_RESULT__));\n\t } else if (typeof exports === 'object') {\n\t module.exports = factory();\n\t } else {\n\t root.HTMLJanitor = factory();\n\t }\n\t}(this, function () {\n\t\n\t /**\n\t * @param {Object} config.tags Dictionary of allowed tags.\n\t * @param {boolean} config.keepNestedBlockElements Default false.\n\t */\n\t function HTMLJanitor(config) {\n\t\n\t var tagDefinitions = config['tags'];\n\t var tags = Object.keys(tagDefinitions);\n\t\n\t var validConfigValues = tags\n\t .map(function(k) { return typeof tagDefinitions[k]; })\n\t .every(function(type) { return type === 'object' || type === 'boolean' || type === 'function'; });\n\t\n\t if(!validConfigValues) {\n\t throw new Error(\"The configuration was invalid\");\n\t }\n\t\n\t this.config = config;\n\t }\n\t\n\t // TODO: not exhaustive?\n\t var blockElementNames = ['P', 'LI', 'TD', 'TH', 'DIV', 'H1', 'H2', 'H3', 'H4', 'H5', 'H6', 'PRE'];\n\t function isBlockElement(node) {\n\t return blockElementNames.indexOf(node.nodeName) !== -1;\n\t }\n\t\n\t var inlineElementNames = ['A', 'B', 'STRONG', 'I', 'EM', 'SUB', 'SUP', 'U', 'STRIKE'];\n\t function isInlineElement(node) {\n\t return inlineElementNames.indexOf(node.nodeName) !== -1;\n\t }\n\t\n\t HTMLJanitor.prototype.clean = function (html) {\n\t var sandbox = document.createElement('div');\n\t sandbox.innerHTML = html;\n\t\n\t this._sanitize(sandbox);\n\t\n\t return sandbox.innerHTML;\n\t };\n\t\n\t HTMLJanitor.prototype._sanitize = function (parentNode) {\n\t var treeWalker = createTreeWalker(parentNode);\n\t var node = treeWalker.firstChild();\n\t if (!node) { return; }\n\t\n\t do {\n\t // Ignore nodes that have already been sanitized\n\t if (node._sanitized) {\n\t continue;\n\t }\n\t\n\t if (node.nodeType === Node.TEXT_NODE) {\n\t // If this text node is just whitespace and the previous or next element\n\t // sibling is a block element, remove it\n\t // N.B.: This heuristic could change. Very specific to a bug with\n\t // `contenteditable` in Firefox: http://jsbin.com/EyuKase/1/edit?js,output\n\t // FIXME: make this an option?\n\t if (node.data.trim() === ''\n\t && ((node.previousElementSibling && isBlockElement(node.previousElementSibling))\n\t || (node.nextElementSibling && isBlockElement(node.nextElementSibling)))) {\n\t parentNode.removeChild(node);\n\t this._sanitize(parentNode);\n\t break;\n\t } else {\n\t continue;\n\t }\n\t }\n\t\n\t // Remove all comments\n\t if (node.nodeType === Node.COMMENT_NODE) {\n\t parentNode.removeChild(node);\n\t this._sanitize(parentNode);\n\t break;\n\t }\n\t\n\t var isInline = isInlineElement(node);\n\t var containsBlockElement;\n\t if (isInline) {\n\t containsBlockElement = Array.prototype.some.call(node.childNodes, isBlockElement);\n\t }\n\t\n\t // Block elements should not be nested (e.g.

  • ...); if\n\t // they are, we want to unwrap the inner block element.\n\t var isNotTopContainer = !! parentNode.parentNode;\n\t var isNestedBlockElement =\n\t isBlockElement(parentNode) &&\n\t isBlockElement(node) &&\n\t isNotTopContainer;\n\t\n\t var nodeName = node.nodeName.toLowerCase();\n\t\n\t var allowedAttrs = getAllowedAttrs(this.config, nodeName, node);\n\t\n\t var isInvalid = isInline && containsBlockElement;\n\t\n\t // Drop tag entirely according to the whitelist *and* if the markup\n\t // is invalid.\n\t if (isInvalid || shouldRejectNode(node, allowedAttrs)\n\t || (!this.config.keepNestedBlockElements && isNestedBlockElement)) {\n\t // Do not keep the inner text of SCRIPT/STYLE elements.\n\t if (! (node.nodeName === 'SCRIPT' || node.nodeName === 'STYLE')) {\n\t while (node.childNodes.length > 0) {\n\t parentNode.insertBefore(node.childNodes[0], node);\n\t }\n\t }\n\t parentNode.removeChild(node);\n\t\n\t this._sanitize(parentNode);\n\t break;\n\t }\n\t\n\t // Sanitize attributes\n\t for (var a = 0; a < node.attributes.length; a += 1) {\n\t var attr = node.attributes[a];\n\t\n\t if (shouldRejectAttr(attr, allowedAttrs, node)) {\n\t node.removeAttribute(attr.name);\n\t // Shift the array to continue looping.\n\t a = a - 1;\n\t }\n\t }\n\t\n\t // Sanitize children\n\t this._sanitize(node);\n\t\n\t // Mark node as sanitized so it's ignored in future runs\n\t node._sanitized = true;\n\t } while ((node = treeWalker.nextSibling()));\n\t };\n\t\n\t function createTreeWalker(node) {\n\t return document.createTreeWalker(node,\n\t NodeFilter.SHOW_TEXT | NodeFilter.SHOW_ELEMENT | NodeFilter.SHOW_COMMENT,\n\t null, false);\n\t }\n\t\n\t function getAllowedAttrs(config, nodeName, node){\n\t if (typeof config.tags[nodeName] === 'function') {\n\t return config.tags[nodeName](node);\n\t } else {\n\t return config.tags[nodeName];\n\t }\n\t }\n\t\n\t function shouldRejectNode(node, allowedAttrs){\n\t if (typeof allowedAttrs === 'undefined') {\n\t return true;\n\t } else if (typeof allowedAttrs === 'boolean') {\n\t return !allowedAttrs;\n\t }\n\t\n\t return false;\n\t }\n\t\n\t function shouldRejectAttr(attr, allowedAttrs, node){\n\t var attrName = attr.name.toLowerCase();\n\t\n\t if (allowedAttrs === true){\n\t return false;\n\t } else if (typeof allowedAttrs[attrName] === 'function'){\n\t return !allowedAttrs[attrName](attr.value, node);\n\t } else if (typeof allowedAttrs[attrName] === 'undefined'){\n\t return true;\n\t } else if (allowedAttrs[attrName] === false) {\n\t return true;\n\t } else if (typeof allowedAttrs[attrName] === 'string') {\n\t return (allowedAttrs[attrName] !== attr.value);\n\t }\n\t\n\t return false;\n\t }\n\t\n\t return HTMLJanitor;\n\t\n\t}));\n\n\n/***/ },\n/* 19 */\n/***/ function(module, exports) {\n\n\t\"use strict\";\n\t\n\t/**\r\n\t * Codex Editor Listeners module\r\n\t *\r\n\t * @author Codex Team\r\n\t * @version 1.0\r\n\t */\n\t\n\t/**\r\n\t * Module-decorator for event listeners assignment\r\n\t */\n\tmodule.exports = function (listeners) {\n\t\n\t var allListeners = [];\n\t\n\t /**\r\n\t * Search methods\r\n\t *\r\n\t * byElement, byType and byHandler returns array of suitable listeners\r\n\t * one and all takes element, eventType, and handler and returns first (all) suitable listener\r\n\t *\r\n\t */\n\t listeners.search = function () {\n\t\n\t var byElement = function byElement(element, context) {\n\t\n\t var listenersOnElement = [];\n\t\n\t context = context || allListeners;\n\t\n\t for (var i = 0; i < context.length; i++) {\n\t\n\t var listener = context[i];\n\t\n\t if (listener.element === element) {\n\t\n\t listenersOnElement.push(listener);\n\t }\n\t }\n\t\n\t return listenersOnElement;\n\t };\n\t\n\t var byType = function byType(eventType, context) {\n\t\n\t var listenersWithType = [];\n\t\n\t context = context || allListeners;\n\t\n\t for (var i = 0; i < context.length; i++) {\n\t\n\t var listener = context[i];\n\t\n\t if (listener.type === eventType) {\n\t\n\t listenersWithType.push(listener);\n\t }\n\t }\n\t\n\t return listenersWithType;\n\t };\n\t\n\t var byHandler = function byHandler(handler, context) {\n\t\n\t var listenersWithHandler = [];\n\t\n\t context = context || allListeners;\n\t\n\t for (var i = 0; i < context.length; i++) {\n\t\n\t var listener = context[i];\n\t\n\t if (listener.handler === handler) {\n\t\n\t listenersWithHandler.push(listener);\n\t }\n\t }\n\t\n\t return listenersWithHandler;\n\t };\n\t\n\t var one = function one(element, eventType, handler) {\n\t\n\t var result = allListeners;\n\t\n\t if (element) result = byElement(element, result);\n\t\n\t if (eventType) result = byType(eventType, result);\n\t\n\t if (handler) result = byHandler(handler, result);\n\t\n\t return result[0];\n\t };\n\t\n\t var all = function all(element, eventType, handler) {\n\t\n\t var result = allListeners;\n\t\n\t if (element) result = byElement(element, result);\n\t\n\t if (eventType) result = byType(eventType, result);\n\t\n\t if (handler) result = byHandler(handler, result);\n\t\n\t return result;\n\t };\n\t\n\t return {\n\t byElement: byElement,\n\t byType: byType,\n\t byHandler: byHandler,\n\t one: one,\n\t all: all\n\t };\n\t }();\n\t\n\t listeners.add = function (element, eventType, handler, isCapture) {\n\t\n\t element.addEventListener(eventType, handler, isCapture);\n\t\n\t var data = {\n\t element: element,\n\t type: eventType,\n\t handler: handler\n\t };\n\t\n\t var alreadyAddedListener = listeners.search.one(element, eventType, handler);\n\t\n\t if (!alreadyAddedListener) {\n\t\n\t allListeners.push(data);\n\t }\n\t };\n\t\n\t listeners.remove = function (element, eventType, handler) {\n\t\n\t element.removeEventListener(eventType, handler);\n\t\n\t var existingListeners = listeners.search.all(element, eventType, handler);\n\t\n\t for (var i = 0; i < existingListeners.length; i++) {\n\t\n\t var index = allListeners.indexOf(existingListeners[i]);\n\t\n\t if (index > 0) {\n\t\n\t allListeners.splice(index, 1);\n\t }\n\t }\n\t };\n\t\n\t listeners.removeAll = function () {\n\t\n\t allListeners.map(function (current) {\n\t\n\t listeners.remove(current.element, current.type, current.handler);\n\t });\n\t };\n\t\n\t listeners.get = function (element, eventType, handler) {\n\t\n\t return listeners.search.all(element, eventType, handler);\n\t };\n\t\n\t return listeners;\n\t}({});\n\n/***/ },\n/* 20 */\n/***/ function(module, exports) {\n\n\t'use strict';\n\t\n\tvar _typeof = typeof Symbol === \"function\" && typeof Symbol.iterator === \"symbol\" ? function (obj) { return typeof obj; } : function (obj) { return obj && typeof Symbol === \"function\" && obj.constructor === Symbol && obj !== Symbol.prototype ? \"symbol\" : typeof obj; };\n\t\n\t/**\r\n\t * Codex Editor Destroyer module\r\n\t *\r\n\t * @auhor Codex Team\r\n\t * @version 1.0\r\n\t */\n\t\n\tmodule.exports = function (destroyer) {\n\t\n\t var editor = codex.editor;\n\t\n\t destroyer.removeNodes = function () {\n\t\n\t editor.nodes.wrapper.remove();\n\t editor.nodes.notifications.remove();\n\t };\n\t\n\t destroyer.destroyPlugins = function () {\n\t\n\t for (var tool in editor.tools) {\n\t\n\t if (typeof editor.tools[tool].destroy === 'function') {\n\t\n\t editor.tools[tool].destroy();\n\t }\n\t }\n\t };\n\t\n\t destroyer.destroyScripts = function () {\n\t\n\t var scripts = document.getElementsByTagName('SCRIPT');\n\t\n\t for (var i = 0; i < scripts.length; i++) {\n\t\n\t if (scripts[i].id.indexOf(editor.scriptPrefix) + 1) {\n\t\n\t scripts[i].remove();\n\t i--;\n\t }\n\t }\n\t };\n\t\n\t /**\r\n\t * Delete editor data from webpage.\r\n\t * You should send settings argument with boolean flags:\r\n\t * @param settings.ui- remove redactor event listeners and DOM nodes\r\n\t * @param settings.scripts - remove redactor scripts from DOM\r\n\t * @param settings.plugins - remove plugin's objects\r\n\t * @param settings.core - remove editor core. You can remove core only if UI and scripts flags is true\r\n\t * }\r\n\t *\r\n\t */\n\t destroyer.destroy = function (settings) {\n\t\n\t if (!settings || (typeof settings === 'undefined' ? 'undefined' : _typeof(settings)) !== 'object') {\n\t\n\t return;\n\t }\n\t\n\t if (settings.ui) {\n\t\n\t destroyer.removeNodes();\n\t editor.listeners.removeAll();\n\t }\n\t\n\t if (settings.scripts) {\n\t\n\t destroyer.destroyScripts();\n\t }\n\t\n\t if (settings.plugins) {\n\t\n\t destroyer.destroyPlugins();\n\t }\n\t\n\t if (settings.ui && settings.scripts && settings.core) {\n\t\n\t delete codex.editor;\n\t }\n\t };\n\t\n\t return destroyer;\n\t}({});\n\n/***/ },\n/* 21 */\n/***/ function(module, exports) {\n\n\t'use strict';\n\t\n\t/**\r\n\t * Codex Editor Paste module\r\n\t *\r\n\t * @author Codex Team\r\n\t * @version 1.1.1\r\n\t */\n\t\n\tmodule.exports = function (paste) {\n\t\n\t var editor = codex.editor;\n\t\n\t var patterns = [];\n\t\n\t paste.prepare = function () {\n\t\n\t var tools = editor.tools;\n\t\n\t for (var tool in tools) {\n\t\n\t if (!tools[tool].renderOnPastePatterns || !Array.isArray(tools[tool].renderOnPastePatterns)) {\n\t\n\t continue;\n\t }\n\t\n\t tools[tool].renderOnPastePatterns.map(function (pattern) {\n\t\n\t patterns.push(pattern);\n\t });\n\t }\n\t\n\t return Promise.resolve();\n\t };\n\t\n\t /**\r\n\t * Saves data\r\n\t * @param event\r\n\t */\n\t paste.pasted = function (event) {\n\t\n\t var clipBoardData = event.clipboardData || window.clipboardData,\n\t content = clipBoardData.getData('Text');\n\t\n\t var result = analize(content);\n\t\n\t if (result) {\n\t\n\t event.preventDefault();\n\t event.stopImmediatePropagation();\n\t }\n\t\n\t return result;\n\t };\n\t\n\t /**\r\n\t * Analizes pated string and calls necessary method\r\n\t */\n\t\n\t var analize = function analize(string) {\n\t\n\t var result = false,\n\t content = editor.content.currentNode,\n\t plugin = content.dataset.tool;\n\t\n\t patterns.map(function (pattern) {\n\t\n\t var execArray = pattern.regex.exec(string),\n\t match = execArray && execArray[0];\n\t\n\t if (match && match === string.trim()) {\n\t\n\t /** current block is not empty */\n\t if (content.textContent.trim() && plugin == editor.settings.initialBlockPlugin) {\n\t\n\t pasteToNewBlock_();\n\t }\n\t\n\t pattern.callback(string, pattern);\n\t result = true;\n\t }\n\t });\n\t\n\t return result;\n\t };\n\t\n\t var pasteToNewBlock_ = function pasteToNewBlock_() {\n\t\n\t /** Create new initial block */\n\t editor.content.insertBlock({\n\t\n\t type: editor.settings.initialBlockPlugin,\n\t block: editor.tools[editor.settings.initialBlockPlugin].render({\n\t text: ''\n\t })\n\t\n\t }, false);\n\t };\n\t\n\t /**\r\n\t * This method prevents default behaviour.\r\n\t *\r\n\t * @param {Object} event\r\n\t * @protected\r\n\t *\r\n\t * @description We get from clipboard pasted data, sanitize, make a fragment that contains of this sanitized nodes.\r\n\t * Firstly, we need to memorize the caret position. We can do that by getting the range of selection.\r\n\t * After all, we insert clear fragment into caret placed position. Then, we should move the caret to the last node\r\n\t */\n\t paste.blockPasteCallback = function (event) {\n\t\n\t if (!needsToHandlePasteEvent(event.target)) {\n\t\n\t return;\n\t }\n\t\n\t /** Prevent default behaviour */\n\t event.preventDefault();\n\t\n\t /** get html pasted data - dirty data */\n\t var htmlData = event.clipboardData.getData('text/html'),\n\t plainData = event.clipboardData.getData('text/plain');\n\t\n\t /** Temporary DIV that is used to work with text's paragraphs as DOM-elements*/\n\t var paragraphs = editor.draw.node('DIV', '', {}),\n\t cleanData,\n\t wrappedData;\n\t\n\t /** Create fragment, that we paste to range after proccesing */\n\t cleanData = editor.sanitizer.clean(htmlData);\n\t\n\t /**\r\n\t * We wrap pasted text with

    tags to split it logically\r\n\t * @type {string}\r\n\t */\n\t wrappedData = editor.content.wrapTextWithParagraphs(cleanData, plainData);\n\t paragraphs.innerHTML = wrappedData;\n\t\n\t /**\r\n\t * If there only one paragraph, lets user agent paste it\r\n\t */\n\t if (paragraphs.childNodes.length == 1) {\n\t\n\t var newNode,\n\t paragraph = paragraphs.firstChild;\n\t\n\t if (paragraph.childElementCount) {\n\t\n\t newNode = editor.draw.node('SPAN', '', { innerHTML: paragraph.innerHTML.trim() });\n\t } else {\n\t\n\t newNode = document.createTextNode(paragraph.textContent);\n\t }\n\t\n\t editor.caret.insertNode(newNode);\n\t return;\n\t }\n\t\n\t insertPastedParagraphs(paragraphs.childNodes);\n\t };\n\t\n\t /**\r\n\t * Checks if we should handle paste event on block\r\n\t * @param block\r\n\t *\r\n\t * @return {boolean}\r\n\t */\n\t var needsToHandlePasteEvent = function needsToHandlePasteEvent(block) {\n\t\n\t /** If area is input or textarea then allow default behaviour */\n\t if (editor.core.isNativeInput(block)) {\n\t\n\t return false;\n\t }\n\t\n\t var editableParent = editor.content.getEditableParent(block);\n\t\n\t /** Allow paste when event target placed in Editable element */\n\t if (!editableParent) {\n\t\n\t return false;\n\t }\n\t\n\t return true;\n\t };\n\t\n\t var insertPastedParagraphs = function insertPastedParagraphs(paragraphs) {\n\t\n\t var NEW_BLOCK_TYPE = editor.settings.initialBlockPlugin,\n\t currentBlockContent = editor.content.currentNode.firstChild.firstChild;\n\t\n\t paragraphs.forEach(function (paragraph, index) {\n\t\n\t /** Don't allow empty paragraphs */\n\t if (paragraph.innerHTML.trim() === '') {\n\t\n\t return;\n\t }\n\t\n\t /**\r\n\t * If there was no data in working node, replace it with first paragraph of pasted text\r\n\t */\n\t if (index == 0 && currentBlockContent.innerHTML.trim() === '') {\n\t\n\t editor.content.switchBlock(editor.content.currentNode, editor.tools[NEW_BLOCK_TYPE].render({\n\t text: paragraph.innerHTML\n\t }), NEW_BLOCK_TYPE);\n\t\n\t return;\n\t }\n\t\n\t editor.content.insertBlock({\n\t type: NEW_BLOCK_TYPE,\n\t block: editor.tools[NEW_BLOCK_TYPE].render({\n\t text: paragraph.innerHTML\n\t })\n\t });\n\t\n\t editor.caret.inputIndex++;\n\t });\n\t\n\t editor.caret.setToPreviousBlock(editor.caret.getCurrentInputIndex() + 1);\n\t };\n\t\n\t return paste;\n\t}({});\n\n/***/ }\n/******/ ]);\n\n\n// WEBPACK FOOTER //\n// codex-editor.js"," \t// The module cache\n \tvar installedModules = {};\n\n \t// The require function\n \tfunction __webpack_require__(moduleId) {\n\n \t\t// Check if module is in cache\n \t\tif(installedModules[moduleId])\n \t\t\treturn installedModules[moduleId].exports;\n\n \t\t// Create a new module (and put it into the cache)\n \t\tvar module = installedModules[moduleId] = {\n \t\t\texports: {},\n \t\t\tid: moduleId,\n \t\t\tloaded: false\n \t\t};\n\n \t\t// Execute the module function\n \t\tmodules[moduleId].call(module.exports, module, module.exports, __webpack_require__);\n\n \t\t// Flag the module as loaded\n \t\tmodule.loaded = true;\n\n \t\t// Return the exports of the module\n \t\treturn module.exports;\n \t}\n\n\n \t// expose the modules object (__webpack_modules__)\n \t__webpack_require__.m = modules;\n\n \t// expose the module cache\n \t__webpack_require__.c = installedModules;\n\n \t// __webpack_public_path__\n \t__webpack_require__.p = \"\";\n\n \t// Load entry module and return exports\n \treturn __webpack_require__(0);\n\n\n\n// WEBPACK FOOTER //\n// webpack/bootstrap b9ba5e0c4242473ccc41","/**\r\n *\r\n * Codex Editor\r\n *\r\n * @author Codex Team\r\n */\r\n\r\nmodule.exports = (function (editor) {\r\n\r\n 'use strict';\r\n\r\n editor.version = VERSION;\r\n editor.scriptPrefix = 'cdx-script-';\r\n\r\n var init = function () {\r\n\r\n editor.core = require('./modules/core');\r\n editor.tools = require('./modules/tools');\r\n editor.ui = require('./modules/ui');\r\n editor.transport = require('./modules/transport');\r\n editor.renderer = require('./modules/renderer');\r\n editor.saver = require('./modules/saver');\r\n editor.content = require('./modules/content');\r\n editor.toolbar = require('./modules/toolbar/toolbar');\r\n editor.callback = require('./modules/callbacks');\r\n editor.draw = require('./modules/draw');\r\n editor.caret = require('./modules/caret');\r\n editor.notifications = require('./modules/notifications');\r\n editor.parser = require('./modules/parser');\r\n editor.sanitizer = require('./modules/sanitizer');\r\n editor.listeners = require('./modules/listeners');\r\n editor.destroyer = require('./modules/destroyer');\r\n editor.paste = require('./modules/paste');\r\n\r\n };\r\n\r\n /**\r\n * @public\r\n * holds initial settings\r\n */\r\n editor.settings = {\r\n tools : ['paragraph', 'header', 'picture', 'list', 'quote', 'code', 'twitter', 'instagram', 'smile'],\r\n holderId : 'codex-editor',\r\n\r\n // Type of block showing on empty editor\r\n initialBlockPlugin: 'paragraph'\r\n };\r\n\r\n /**\r\n * public\r\n *\r\n * Static nodes\r\n */\r\n editor.nodes = {\r\n holder : null,\r\n wrapper : null,\r\n toolbar : null,\r\n inlineToolbar : {\r\n wrapper : null,\r\n buttons : null,\r\n actions : null\r\n },\r\n toolbox : null,\r\n notifications : null,\r\n plusButton : null,\r\n showSettingsButton: null,\r\n showTrashButton : null,\r\n blockSettings : null,\r\n pluginSettings : null,\r\n defaultSettings : null,\r\n toolbarButtons : {}, // { type : DomEl, ... }\r\n redactor : null\r\n };\r\n\r\n /**\r\n * @public\r\n *\r\n * Output state\r\n */\r\n editor.state = {\r\n jsonOutput : [],\r\n blocks : [],\r\n inputs : []\r\n };\r\n\r\n /**\r\n * @public\r\n * Editor plugins\r\n */\r\n editor.tools = {};\r\n\r\n /**\r\n * Initialization\r\n * @uses Promise cEditor.core.prepare\r\n * @param {Object} userSettings\r\n * @param {Array} userSettings.tools list of plugins\r\n * @param {String} userSettings.holderId Element's id to append editor\r\n *\r\n * Load user defined tools\r\n * Tools must contain this important objects :\r\n * @param {String} type - this is a type of plugin. It can be used as plugin name\r\n * @param {String} iconClassname - this a icon in toolbar\r\n * @param {Object} make - what should plugin do, when it is clicked\r\n * @param {Object} appendCallback - callback after clicking\r\n * @param {Element} settings - what settings does it have\r\n * @param {Object} render - plugin get JSON, and should return HTML\r\n * @param {Object} save - plugin gets HTML content, returns JSON\r\n * @param {Boolean} displayInToolbox - will be displayed in toolbox. Default value is TRUE\r\n * @param {Boolean} enableLineBreaks - inserts new block or break lines. Default value is FALSE\r\n *\r\n * @example\r\n * - type : 'header',\r\n * - iconClassname : 'ce-icon-header',\r\n * - make : headerTool.make,\r\n * - appendCallback : headerTool.appendCallback,\r\n * - settings : headerTool.makeSettings(),\r\n * - render : headerTool.render,\r\n * - save : headerTool.save,\r\n * - displayInToolbox : true,\r\n * - enableLineBreaks : false\r\n */\r\n editor.start = function (userSettings) {\r\n\r\n init();\r\n\r\n editor.core.prepare(userSettings)\r\n\r\n // If all ok, make UI, bind events and parse initial-content\r\n .then(editor.ui.prepare)\r\n .then(editor.tools.prepare)\r\n .then(editor.sanitizer.prepare)\r\n .then(editor.paste.prepare)\r\n .then(editor.transport.prepare)\r\n .then(editor.renderer.makeBlocksFromData)\r\n .then(editor.ui.saveInputs)\r\n .catch(function (error) {\r\n\r\n editor.core.log('Initialization failed with error: %o', 'warn', error);\r\n\r\n });\r\n\r\n };\r\n\r\n return editor;\r\n\r\n})({});\r\n\n\n\n// WEBPACK FOOTER //\n// ./codex.js","/**\r\n * Codex Editor Core\r\n *\r\n * @author Codex Team\r\n * @version 1.1.3\r\n */\r\n\r\nmodule.exports = (function (core) {\r\n\r\n let editor = codex.editor;\r\n\r\n /**\r\n * @public\r\n *\r\n * Editor preparing method\r\n * @return Promise\r\n */\r\n core.prepare = function (userSettings) {\r\n\r\n return new Promise(function (resolve, reject) {\r\n\r\n if ( userSettings ) {\r\n\r\n editor.settings.tools = userSettings.tools || editor.settings.tools;\r\n\r\n }\r\n\r\n if (userSettings.data) {\r\n\r\n editor.state.blocks = userSettings.data;\r\n\r\n }\r\n\r\n if (userSettings.initialBlockPlugin) {\r\n\r\n editor.settings.initialBlockPlugin = userSettings.initialBlockPlugin;\r\n\r\n }\r\n\r\n if (userSettings.sanitizer) {\r\n\r\n editor.settings.sanitizer = userSettings.sanitizer;\r\n\r\n }\r\n\r\n editor.hideToolbar = userSettings.hideToolbar;\r\n\r\n editor.nodes.holder = document.getElementById(userSettings.holderId || editor.settings.holderId);\r\n\r\n if (typeof editor.nodes.holder === undefined || editor.nodes.holder === null) {\r\n\r\n reject(Error(\"Holder wasn't found by ID: #\" + userSettings.holderId));\r\n\r\n } else {\r\n\r\n resolve();\r\n\r\n }\r\n\r\n });\r\n\r\n };\r\n\r\n /**\r\n * Logging method\r\n * @param type = ['log', 'info', 'warn']\r\n */\r\n core.log = function (msg, type, arg) {\r\n\r\n type = type || 'log';\r\n\r\n if (!arg) {\r\n\r\n arg = msg || 'undefined';\r\n msg = '[codex-editor]: %o';\r\n\r\n } else {\r\n\r\n msg = '[codex-editor]: ' + msg;\r\n\r\n }\r\n\r\n try{\r\n\r\n if ( 'console' in window && window.console[ type ] ) {\r\n\r\n if ( arg ) window.console[ type ]( msg, arg );\r\n else window.console[ type ]( msg );\r\n\r\n }\r\n\r\n }catch(e) {}\r\n\r\n };\r\n\r\n /**\r\n * @protected\r\n *\r\n * Helper for insert one element after another\r\n */\r\n core.insertAfter = function (target, element) {\r\n\r\n target.parentNode.insertBefore(element, target.nextSibling);\r\n\r\n };\r\n\r\n /**\r\n * @const\r\n *\r\n * Readable DOM-node types map\r\n */\r\n core.nodeTypes = {\r\n TAG : 1,\r\n TEXT : 3,\r\n COMMENT : 8\r\n };\r\n\r\n /**\r\n * @const\r\n * Readable keys map\r\n */\r\n core.keys = { BACKSPACE: 8, TAB: 9, ENTER: 13, SHIFT: 16, CTRL: 17, ALT: 18, ESC: 27, SPACE: 32, LEFT: 37, UP: 38, DOWN: 40, RIGHT: 39, DELETE: 46, META: 91 };\r\n\r\n /**\r\n * @protected\r\n *\r\n * Check object for DOM node\r\n */\r\n core.isDomNode = function (el) {\r\n\r\n return el && typeof el === 'object' && el.nodeType && el.nodeType == this.nodeTypes.TAG;\r\n\r\n };\r\n\r\n /**\r\n * Checks passed object for emptiness\r\n * @require ES5 - Object.keys\r\n * @param {object}\r\n */\r\n core.isEmpty = function ( obj ) {\r\n\r\n return Object.keys(obj).length === 0;\r\n\r\n };\r\n\r\n /**\r\n * Native Ajax\r\n * @param {String} settings.url - request URL\r\n * @param {function} settings.beforeSend - returned value will be passed as context to the Success, Error and Progress callbacks\r\n * @param {function} settings.success\r\n * @param {function} settings.progress\r\n */\r\n core.ajax = function (settings) {\r\n\r\n if (!settings || !settings.url) {\r\n\r\n return;\r\n\r\n }\r\n\r\n var XMLHTTP = window.XMLHttpRequest ? new XMLHttpRequest() : new ActiveXObject('Microsoft.XMLHTTP'),\r\n encodedString,\r\n isFormData,\r\n prop;\r\n\r\n\r\n settings.async = true;\r\n settings.type = settings.type || 'GET';\r\n settings.data = settings.data || '';\r\n settings['content-type'] = settings['content-type'] || 'application/json; charset=utf-8';\r\n\r\n if (settings.type == 'GET' && settings.data) {\r\n\r\n settings.url = /\\?/.test(settings.url) ? settings.url + '&' + settings.data : settings.url + '?' + settings.data;\r\n\r\n } else {\r\n\r\n encodedString = '';\r\n for(prop in settings.data) {\r\n\r\n encodedString += (prop + '=' + encodeURIComponent(settings.data[prop]) + '&');\r\n\r\n }\r\n\r\n }\r\n\r\n if (settings.withCredentials) {\r\n\r\n XMLHTTP.withCredentials = true;\r\n\r\n }\r\n\r\n /**\r\n * Value returned in beforeSend funtion will be passed as context to the other response callbacks\r\n * If beforeSend returns false, AJAX will be blocked\r\n */\r\n let responseContext,\r\n beforeSendResult;\r\n\r\n if (typeof settings.beforeSend === 'function') {\r\n\r\n beforeSendResult = settings.beforeSend.call();\r\n\r\n if (beforeSendResult === false) {\r\n\r\n return;\r\n\r\n }\r\n\r\n }\r\n\r\n XMLHTTP.open( settings.type, settings.url, settings.async );\r\n\r\n /**\r\n * If we send FormData, we need no content-type header\r\n */\r\n isFormData = isFormData_(settings.data);\r\n\r\n if (!isFormData) {\r\n\r\n if (settings.type !== 'POST') {\r\n\r\n XMLHTTP.setRequestHeader('Content-type', settings['content-type']);\r\n\r\n } else {\r\n\r\n XMLHTTP.setRequestHeader('Content-type', 'application/x-www-form-urlencoded');\r\n\r\n }\r\n\r\n }\r\n\r\n XMLHTTP.setRequestHeader('X-Requested-With', 'XMLHttpRequest');\r\n\r\n responseContext = beforeSendResult || XMLHTTP;\r\n\r\n if (typeof settings.progress === 'function') {\r\n\r\n XMLHTTP.upload.onprogress = settings.progress.bind(responseContext);\r\n\r\n }\r\n\r\n XMLHTTP.onreadystatechange = function () {\r\n\r\n if (XMLHTTP.readyState === 4) {\r\n\r\n if (XMLHTTP.status === 200) {\r\n\r\n if (typeof settings.success === 'function') {\r\n\r\n settings.success.call(responseContext, XMLHTTP.responseText);\r\n\r\n }\r\n\r\n } else {\r\n\r\n if (typeof settings.error === 'function') {\r\n\r\n settings.error.call(responseContext, XMLHTTP.responseText, XMLHTTP.status);\r\n\r\n }\r\n\r\n }\r\n\r\n }\r\n\r\n };\r\n\r\n if (isFormData) {\r\n\r\n // Sending FormData\r\n XMLHTTP.send(settings.data);\r\n\r\n } else {\r\n\r\n // POST requests\r\n XMLHTTP.send(encodedString);\r\n\r\n }\r\n\r\n return XMLHTTP;\r\n\r\n };\r\n\r\n /**\r\n * Appends script to head of document\r\n * @return Promise\r\n */\r\n core.importScript = function (scriptPath, instanceName) {\r\n\r\n return new Promise(function (resolve, reject) {\r\n\r\n let script;\r\n\r\n /** Script is already loaded */\r\n if ( !instanceName ) {\r\n\r\n reject('Instance name is missed');\r\n\r\n } else if ( document.getElementById(editor.scriptPrefix + instanceName) ) {\r\n\r\n resolve(scriptPath);\r\n\r\n }\r\n\r\n script = document.createElement('SCRIPT');\r\n script.async = true;\r\n script.defer = true;\r\n script.id = editor.scriptPrefix + instanceName;\r\n\r\n script.onload = function () {\r\n\r\n resolve(scriptPath);\r\n\r\n };\r\n\r\n script.onerror = function () {\r\n\r\n reject(scriptPath);\r\n\r\n };\r\n\r\n script.src = scriptPath;\r\n document.head.appendChild(script);\r\n\r\n });\r\n\r\n };\r\n\r\n /**\r\n * Function for checking is it FormData object to send.\r\n * @param {Object} object to check\r\n * @return boolean\r\n */\r\n var isFormData_ = function (object) {\r\n\r\n return object instanceof FormData;\r\n\r\n };\r\n\r\n /**\r\n * Check block\r\n * @param target\r\n * @description Checks target is it native input\r\n */\r\n core.isNativeInput = function (target) {\r\n\r\n var nativeInputAreas = ['INPUT', 'TEXTAREA'];\r\n\r\n return (nativeInputAreas.indexOf(target.tagName) != -1);\r\n\r\n };\r\n\r\n\r\n return core;\r\n\r\n})({});\r\n\n\n\n// WEBPACK FOOTER //\n// ./modules/core.js","/**\r\n* Module working with plugins\r\n*/\r\nmodule.exports = (function () {\r\n\r\n let editor = codex.editor;\r\n\r\n /**\r\n * Initialize plugins before using\r\n * Ex. Load scripts or call some internal methods\r\n * @return Promise\r\n */\r\n function prepare() {\r\n\r\n return new Promise(function (resolve_, reject_) {\r\n\r\n Promise.resolve()\r\n\r\n /**\r\n * Compose a sequence of plugins that requires preparation\r\n */\r\n .then(function () {\r\n\r\n let pluginsRequiresPreparation = [],\r\n allPlugins = editor.tools;\r\n\r\n for ( let pluginName in allPlugins ) {\r\n\r\n let plugin = allPlugins[pluginName];\r\n\r\n if (plugin.prepare && typeof plugin.prepare != 'function' || !plugin.prepare) {\r\n\r\n continue;\r\n\r\n }\r\n\r\n pluginsRequiresPreparation.push(plugin);\r\n\r\n }\r\n\r\n /**\r\n * If no one passed plugins requires preparation, finish prepare() and go ahead\r\n */\r\n if (!pluginsRequiresPreparation.length) {\r\n\r\n resolve_();\r\n\r\n }\r\n\r\n return pluginsRequiresPreparation;\r\n\r\n })\r\n\r\n /** Wait plugins while they prepares */\r\n .then(waitAllPluginsPreparation_)\r\n\r\n .then(function () {\r\n\r\n editor.core.log('Plugins loaded', 'info');\r\n resolve_();\r\n\r\n }).catch(function (error) {\r\n\r\n reject_(error);\r\n\r\n });\r\n\r\n });\r\n\r\n }\r\n\r\n /**\r\n * @param {array} plugins - list of tools that requires preparation\r\n * @return {Promise} resolved while all plugins will be ready or failed\r\n */\r\n function waitAllPluginsPreparation_(plugins) {\r\n\r\n /**\r\n * @calls allPluginsProcessed__ when all plugins prepared or failed\r\n */\r\n return new Promise (function (allPluginsProcessed__) {\r\n\r\n /**\r\n * pluck each element from queue\r\n * First, send resolved Promise as previous value\r\n * Each plugins \"prepare\" method returns a Promise, that's why\r\n * reduce current element will not be able to continue while can't get\r\n * a resolved Promise\r\n *\r\n * If last plugin is \"prepared\" then go to the next stage of initialization\r\n */\r\n plugins.reduce(function (previousValue, plugin, iteration) {\r\n\r\n return previousValue.then(function () {\r\n\r\n /**\r\n * Wait till plugins prepared\r\n * @calls pluginIsReady__ when plugin is ready or failed\r\n */\r\n return new Promise ( function (pluginIsReady__) {\r\n\r\n callPluginsPrepareMethod_( plugin )\r\n\r\n .then( pluginIsReady__ )\r\n .then( function () {\r\n\r\n plugin.available = true;\r\n\r\n })\r\n\r\n .catch(function (error) {\r\n\r\n editor.core.log(`Plugin «${plugin.type}» was not loaded. Preparation failed because %o`, 'warn', error);\r\n plugin.available = false;\r\n plugin.loadingMessage = error;\r\n\r\n /** Go ahead even some plugin has problems */\r\n pluginIsReady__();\r\n\r\n })\r\n\r\n .then(function () {\r\n\r\n /** If last plugin has problems then just ignore and continue */\r\n if (iteration == plugins.length - 1) {\r\n\r\n allPluginsProcessed__();\r\n\r\n }\r\n\r\n });\r\n\r\n });\r\n\r\n });\r\n\r\n }, Promise.resolve() );\r\n\r\n });\r\n\r\n }\r\n\r\n var callPluginsPrepareMethod_ = function (plugin) {\r\n\r\n return plugin.prepare( plugin.config || {} );\r\n\r\n };\r\n\r\n return {\r\n prepare: prepare\r\n };\r\n\r\n}());\n\n\n// WEBPACK FOOTER //\n// ./modules/tools.js","/**\r\n * Codex Editor UI module\r\n *\r\n * @author Codex Team\r\n * @version 1.2.0\r\n */\r\n\r\nmodule.exports = (function (ui) {\r\n\r\n let editor = codex.editor;\r\n\r\n /**\r\n * Basic editor classnames\r\n */\r\n ui.className = {\r\n\r\n /**\r\n * @const {string} BLOCK_CLASSNAME - redactor blocks name\r\n */\r\n BLOCK_CLASSNAME : 'ce-block',\r\n\r\n /**\r\n * @const {String} wrapper for plugins content\r\n */\r\n BLOCK_CONTENT : 'ce-block__content',\r\n\r\n /**\r\n * @const {String} BLOCK_STRETCHED - makes block stretched\r\n */\r\n BLOCK_STRETCHED : 'ce-block--stretched',\r\n\r\n /**\r\n * @const {String} BLOCK_HIGHLIGHTED - adds background\r\n */\r\n BLOCK_HIGHLIGHTED : 'ce-block--focused',\r\n\r\n /**\r\n * @const {String} - for all default settings\r\n */\r\n SETTINGS_ITEM : 'ce-settings__item'\r\n\r\n };\r\n\r\n /**\r\n * @protected\r\n *\r\n * Making main interface\r\n */\r\n ui.prepare = function () {\r\n\r\n return new Promise(function (resolve) {\r\n\r\n let wrapper = editor.draw.wrapper(),\r\n redactor = editor.draw.redactor(),\r\n toolbar = makeToolBar_();\r\n\r\n wrapper.appendChild(toolbar);\r\n wrapper.appendChild(redactor);\r\n\r\n /** Save created ui-elements to static nodes state */\r\n editor.nodes.wrapper = wrapper;\r\n editor.nodes.redactor = redactor;\r\n\r\n /** Append editor wrapper with redactor zone into holder */\r\n editor.nodes.holder.appendChild(wrapper);\r\n\r\n resolve();\r\n\r\n })\r\n\r\n /** Add toolbox tools */\r\n .then(addTools_)\r\n\r\n /** Make container for inline toolbar */\r\n .then(makeInlineToolbar_)\r\n\r\n /** Add inline toolbar tools */\r\n .then(addInlineToolbarTools_)\r\n\r\n /** Draw wrapper for notifications */\r\n .then(makeNotificationHolder_)\r\n\r\n /** Add eventlisteners to redactor elements */\r\n .then(bindEvents_)\r\n\r\n .catch( function () {\r\n\r\n editor.core.log(\"Can't draw editor interface\");\r\n\r\n });\r\n\r\n };\r\n\r\n /**\r\n * @private\r\n * Draws inline toolbar zone\r\n */\r\n var makeInlineToolbar_ = function () {\r\n\r\n var container = editor.draw.inlineToolbar();\r\n\r\n /** Append to redactor new inline block */\r\n editor.nodes.inlineToolbar.wrapper = container;\r\n\r\n /** Draw toolbar buttons */\r\n editor.nodes.inlineToolbar.buttons = editor.draw.inlineToolbarButtons();\r\n\r\n /** Buttons action or settings */\r\n editor.nodes.inlineToolbar.actions = editor.draw.inlineToolbarActions();\r\n\r\n /** Append to inline toolbar buttons as part of it */\r\n editor.nodes.inlineToolbar.wrapper.appendChild(editor.nodes.inlineToolbar.buttons);\r\n editor.nodes.inlineToolbar.wrapper.appendChild(editor.nodes.inlineToolbar.actions);\r\n\r\n editor.nodes.wrapper.appendChild(editor.nodes.inlineToolbar.wrapper);\r\n\r\n };\r\n\r\n var makeToolBar_ = function () {\r\n\r\n let toolbar = editor.draw.toolbar(),\r\n blockButtons = makeToolbarSettings_(),\r\n toolbarContent = makeToolbarContent_();\r\n\r\n /** Appending first-level block buttons */\r\n toolbar.appendChild(blockButtons);\r\n\r\n /** Append toolbarContent to toolbar */\r\n toolbar.appendChild(toolbarContent);\r\n\r\n /** Make toolbar global */\r\n editor.nodes.toolbar = toolbar;\r\n\r\n return toolbar;\r\n\r\n };\r\n\r\n var makeToolbarContent_ = function () {\r\n\r\n let toolbarContent = editor.draw.toolbarContent(),\r\n toolbox = editor.draw.toolbox(),\r\n plusButton = editor.draw.plusButton();\r\n\r\n /** Append plus button */\r\n toolbarContent.appendChild(plusButton);\r\n\r\n /** Appending toolbar tools */\r\n toolbarContent.appendChild(toolbox);\r\n\r\n /** Make Toolbox and plusButton global */\r\n editor.nodes.toolbox = toolbox;\r\n editor.nodes.plusButton = plusButton;\r\n\r\n return toolbarContent;\r\n\r\n };\r\n\r\n var makeToolbarSettings_ = function () {\r\n\r\n let blockSettings = editor.draw.blockSettings(),\r\n blockButtons = editor.draw.blockButtons(),\r\n defaultSettings = editor.draw.defaultSettings(),\r\n showSettingsButton = editor.draw.settingsButton(),\r\n showTrashButton = editor.toolbar.settings.makeRemoveBlockButton(),\r\n pluginSettings = editor.draw.pluginsSettings();\r\n\r\n /** Add default and plugins settings */\r\n blockSettings.appendChild(pluginSettings);\r\n blockSettings.appendChild(defaultSettings);\r\n\r\n /**\r\n * Make blocks buttons\r\n * This block contains settings button and remove block button\r\n */\r\n blockButtons.appendChild(showSettingsButton);\r\n blockButtons.appendChild(showTrashButton);\r\n blockButtons.appendChild(blockSettings);\r\n\r\n /** Make BlockSettings, PluginSettings, DefaultSettings global */\r\n editor.nodes.blockSettings = blockSettings;\r\n editor.nodes.pluginSettings = pluginSettings;\r\n editor.nodes.defaultSettings = defaultSettings;\r\n editor.nodes.showSettingsButton = showSettingsButton;\r\n editor.nodes.showTrashButton = showTrashButton;\r\n\r\n return blockButtons;\r\n\r\n };\r\n\r\n /** Draw notifications holder */\r\n var makeNotificationHolder_ = function () {\r\n\r\n /** Append block with notifications to the document */\r\n editor.nodes.notifications = editor.notifications.createHolder();\r\n\r\n };\r\n\r\n /**\r\n * @private\r\n * Append tools passed in editor.tools\r\n */\r\n var addTools_ = function () {\r\n\r\n var tool,\r\n toolName,\r\n toolButton;\r\n\r\n for ( toolName in editor.settings.tools ) {\r\n\r\n tool = editor.settings.tools[toolName];\r\n\r\n editor.tools[toolName] = tool;\r\n\r\n if (!tool.iconClassname && tool.displayInToolbox) {\r\n\r\n editor.core.log('Toolbar icon classname missed. Tool %o skipped', 'warn', toolName);\r\n continue;\r\n\r\n }\r\n\r\n if (typeof tool.render != 'function') {\r\n\r\n editor.core.log('render method missed. Tool %o skipped', 'warn', toolName);\r\n continue;\r\n\r\n }\r\n\r\n if (!tool.displayInToolbox) {\r\n\r\n continue;\r\n\r\n } else {\r\n\r\n /** if tools is for toolbox */\r\n toolButton = editor.draw.toolbarButton(toolName, tool.iconClassname);\r\n\r\n editor.nodes.toolbox.appendChild(toolButton);\r\n\r\n editor.nodes.toolbarButtons[toolName] = toolButton;\r\n\r\n }\r\n\r\n }\r\n\r\n };\r\n\r\n var addInlineToolbarTools_ = function () {\r\n\r\n var tools = {\r\n\r\n bold: {\r\n icon : 'ce-icon-bold',\r\n command : 'bold'\r\n },\r\n\r\n italic: {\r\n icon : 'ce-icon-italic',\r\n command : 'italic'\r\n },\r\n\r\n underline: {\r\n icon : 'ce-icon-underline',\r\n command : 'underline'\r\n },\r\n\r\n link: {\r\n icon : 'ce-icon-link',\r\n command : 'createLink'\r\n }\r\n };\r\n\r\n var toolButton,\r\n tool;\r\n\r\n for(var name in tools) {\r\n\r\n tool = tools[name];\r\n\r\n toolButton = editor.draw.toolbarButtonInline(name, tool.icon);\r\n\r\n editor.nodes.inlineToolbar.buttons.appendChild(toolButton);\r\n /**\r\n * Add callbacks to this buttons\r\n */\r\n editor.ui.setInlineToolbarButtonBehaviour(toolButton, tool.command);\r\n\r\n }\r\n\r\n };\r\n\r\n /**\r\n * @private\r\n * Bind editor UI events\r\n */\r\n var bindEvents_ = function () {\r\n\r\n editor.core.log('ui.bindEvents fired', 'info');\r\n\r\n // window.addEventListener('error', function (errorMsg, url, lineNumber) {\r\n // editor.notifications.errorThrown(errorMsg, event);\r\n // }, false );\r\n\r\n /** All keydowns on Document */\r\n editor.listeners.add(document, 'keydown', editor.callback.globalKeydown, false);\r\n\r\n /** All keydowns on Redactor zone */\r\n editor.listeners.add(editor.nodes.redactor, 'keydown', editor.callback.redactorKeyDown, false);\r\n\r\n /** All keydowns on Document */\r\n editor.listeners.add(document, 'keyup', editor.callback.globalKeyup, false );\r\n\r\n /**\r\n * Mouse click to radactor\r\n */\r\n editor.listeners.add(editor.nodes.redactor, 'click', editor.callback.redactorClicked, false );\r\n\r\n /**\r\n * Clicks to the Plus button\r\n */\r\n editor.listeners.add(editor.nodes.plusButton, 'click', editor.callback.plusButtonClicked, false);\r\n\r\n /**\r\n * Clicks to SETTINGS button in toolbar\r\n */\r\n editor.listeners.add(editor.nodes.showSettingsButton, 'click', editor.callback.showSettingsButtonClicked, false );\r\n\r\n /** Bind click listeners on toolbar buttons */\r\n for (var button in editor.nodes.toolbarButtons) {\r\n\r\n editor.listeners.add(editor.nodes.toolbarButtons[button], 'click', editor.callback.toolbarButtonClicked, false);\r\n\r\n }\r\n\r\n };\r\n\r\n ui.addBlockHandlers = function (block) {\r\n\r\n if (!block) return;\r\n\r\n /**\r\n * Block keydowns\r\n */\r\n editor.listeners.add(block, 'keydown', editor.callback.blockKeydown, false);\r\n\r\n /**\r\n * Pasting content from another source\r\n * We have two type of sanitization\r\n * First - uses deep-first search algorithm to get sub nodes,\r\n * sanitizes whole Block_content and replaces cleared nodes\r\n * This method is deprecated\r\n * Method is used in editor.callback.blockPaste(event)\r\n *\r\n * Secont - uses Mutation observer.\r\n * Observer \"observe\" DOM changes and send changings to callback.\r\n * Callback gets changed node, not whole Block_content.\r\n * Inserted or changed node, which we've gotten have been cleared and replaced with diry node\r\n *\r\n * Method is used in editor.callback.blockPasteViaSanitize(event)\r\n *\r\n * @uses html-janitor\r\n * @example editor.callback.blockPasteViaSanitize(event), the second method.\r\n *\r\n */\r\n editor.listeners.add(block, 'paste', editor.paste.blockPasteCallback, false);\r\n\r\n editor.listeners.add(block, 'mouseup', editor.toolbar.inline.show, false);\r\n\r\n };\r\n\r\n /** getting all contenteditable elements */\r\n ui.saveInputs = function () {\r\n\r\n var redactor = editor.nodes.redactor;\r\n\r\n editor.state.inputs = [];\r\n\r\n /** Save all inputs in global variable state */\r\n var inputs = redactor.querySelectorAll('[contenteditable], input, textarea');\r\n\r\n Array.prototype.map.call(inputs, function (current) {\r\n\r\n if (!current.type || current.type == 'text' || current.type == 'textarea') {\r\n\r\n editor.state.inputs.push(current);\r\n\r\n }\r\n\r\n });\r\n\r\n };\r\n\r\n /**\r\n * Adds first initial block on empty redactor\r\n */\r\n ui.addInitialBlock = function () {\r\n\r\n var initialBlockType = editor.settings.initialBlockPlugin,\r\n initialBlock;\r\n\r\n if ( !editor.tools[initialBlockType] ) {\r\n\r\n editor.core.log('Plugin %o was not implemented and can\\'t be used as initial block', 'warn', initialBlockType);\r\n return;\r\n\r\n }\r\n\r\n initialBlock = editor.tools[initialBlockType].render();\r\n\r\n initialBlock.setAttribute('data-placeholder', 'Расскажите свою историю...');\r\n\r\n editor.content.insertBlock({\r\n type : initialBlockType,\r\n block : initialBlock\r\n });\r\n\r\n editor.content.workingNodeChanged(initialBlock);\r\n\r\n };\r\n\r\n ui.setInlineToolbarButtonBehaviour = function (button, type) {\r\n\r\n editor.listeners.add(button, 'mousedown', function (event) {\r\n\r\n editor.toolbar.inline.toolClicked(event, type);\r\n\r\n }, false);\r\n\r\n };\r\n\r\n return ui;\r\n\r\n})({});\r\n\n\n\n// WEBPACK FOOTER //\n// ./modules/ui.js","/**\r\n *\r\n * Codex.Editor Transport Module\r\n *\r\n * @copyright 2017 Codex-Team\r\n * @version 1.2.0\r\n */\r\n\r\nmodule.exports = (function (transport) {\r\n\r\n let editor = codex.editor;\r\n\r\n\r\n /**\r\n * @private {Object} current XmlHttpRequest instance\r\n */\r\n var currentRequest = null;\r\n\r\n\r\n /**\r\n * @type {null} | {DOMElement} input - keeps input element in memory\r\n */\r\n transport.input = null;\r\n\r\n /**\r\n * @property {Object} arguments - keep plugin settings and defined callbacks\r\n */\r\n transport.arguments = null;\r\n\r\n /**\r\n * Prepares input element where will be files\r\n */\r\n transport.prepare = function () {\r\n\r\n let input = editor.draw.node( 'INPUT', '', { type : 'file' } );\r\n\r\n editor.listeners.add(input, 'change', editor.transport.fileSelected);\r\n editor.transport.input = input;\r\n\r\n };\r\n\r\n /** Clear input when files is uploaded */\r\n transport.clearInput = function () {\r\n\r\n /** Remove old input */\r\n transport.input = null;\r\n\r\n /** Prepare new one */\r\n transport.prepare();\r\n\r\n };\r\n\r\n /**\r\n * Callback for file selection\r\n * @param {Event} event\r\n */\r\n transport.fileSelected = function () {\r\n\r\n var input = this,\r\n i,\r\n files = input.files,\r\n formData = new FormData();\r\n\r\n if (editor.transport.arguments.multiple === true) {\r\n\r\n for ( i = 0; i < files.length; i++) {\r\n\r\n formData.append('files[]', files[i], files[i].name);\r\n\r\n }\r\n\r\n } else {\r\n\r\n formData.append('files', files[0], files[0].name);\r\n\r\n }\r\n\r\n currentRequest = editor.core.ajax({\r\n type : 'POST',\r\n data : formData,\r\n url : editor.transport.arguments.url,\r\n beforeSend : editor.transport.arguments.beforeSend,\r\n success : editor.transport.arguments.success,\r\n error : editor.transport.arguments.error,\r\n progress : editor.transport.arguments.progress\r\n });\r\n\r\n /** Clear input */\r\n transport.clearInput();\r\n\r\n };\r\n\r\n /**\r\n * Use plugin callbacks\r\n * @protected\r\n *\r\n * @param {Object} args - can have :\r\n * @param {String} args.url - fetch URL\r\n * @param {Function} args.beforeSend - function calls before sending ajax\r\n * @param {Function} args.success - success callback\r\n * @param {Function} args.error - on error handler\r\n * @param {Function} args.progress - xhr onprogress handler\r\n * @param {Boolean} args.multiple - allow select several files\r\n * @param {String} args.accept - adds accept attribute\r\n */\r\n transport.selectAndUpload = function (args) {\r\n\r\n transport.arguments = args;\r\n\r\n if ( args.multiple === true) {\r\n\r\n transport.input.setAttribute('multiple', 'multiple');\r\n\r\n }\r\n\r\n if ( args.accept ) {\r\n\r\n transport.input.setAttribute('accept', args.accept);\r\n\r\n }\r\n\r\n transport.input.click();\r\n\r\n };\r\n\r\n transport.abort = function () {\r\n\r\n currentRequest.abort();\r\n\r\n currentRequest = null;\r\n\r\n };\r\n\r\n return transport;\r\n\r\n})({});\n\n\n// WEBPACK FOOTER //\n// ./modules/transport.js","/**\r\n * Codex Editor Renderer Module\r\n *\r\n * @author Codex Team\r\n * @version 1.0\r\n */\r\n\r\nmodule.exports = (function (renderer) {\r\n\r\n let editor = codex.editor;\r\n\r\n /**\r\n * Asyncronously parses input JSON to redactor blocks\r\n */\r\n renderer.makeBlocksFromData = function () {\r\n\r\n /**\r\n * If redactor is empty, add first paragraph to start writing\r\n */\r\n if (editor.core.isEmpty(editor.state.blocks) || !editor.state.blocks.items.length) {\r\n\r\n editor.ui.addInitialBlock();\r\n return;\r\n\r\n }\r\n\r\n Promise.resolve()\r\n\r\n /** First, get JSON from state */\r\n .then(function () {\r\n\r\n return editor.state.blocks;\r\n\r\n })\r\n\r\n /** Then, start to iterate they */\r\n .then(editor.renderer.appendBlocks)\r\n\r\n /** Write log if something goes wrong */\r\n .catch(function (error) {\r\n\r\n editor.core.log('Error while parsing JSON: %o', 'error', error);\r\n\r\n });\r\n\r\n };\r\n\r\n /**\r\n * Parses JSON to blocks\r\n * @param {object} data\r\n * @return Primise -> nodeList\r\n */\r\n renderer.appendBlocks = function (data) {\r\n\r\n var blocks = data.items;\r\n\r\n /**\r\n * Sequence of one-by-one blocks appending\r\n * Uses to save blocks order after async-handler\r\n */\r\n var nodeSequence = Promise.resolve();\r\n\r\n for (var index = 0; index < blocks.length ; index++ ) {\r\n\r\n /** Add node to sequence at specified index */\r\n editor.renderer.appendNodeAtIndex(nodeSequence, blocks, index);\r\n\r\n }\r\n\r\n };\r\n\r\n /**\r\n * Append node at specified index\r\n */\r\n renderer.appendNodeAtIndex = function (nodeSequence, blocks, index) {\r\n\r\n /** We need to append node to sequence */\r\n nodeSequence\r\n\r\n /** first, get node async-aware */\r\n .then(function () {\r\n\r\n return editor.renderer.getNodeAsync(blocks, index);\r\n\r\n })\r\n\r\n /**\r\n * second, compose editor-block from JSON object\r\n */\r\n .then(editor.renderer.createBlockFromData)\r\n\r\n /**\r\n * now insert block to redactor\r\n */\r\n .then(function (blockData) {\r\n\r\n /**\r\n * blockData has 'block', 'type' and 'stretched' information\r\n */\r\n editor.content.insertBlock(blockData);\r\n\r\n /** Pass created block to next step */\r\n return blockData.block;\r\n\r\n })\r\n\r\n /** Log if something wrong with node */\r\n .catch(function (error) {\r\n\r\n editor.core.log('Node skipped while parsing because %o', 'error', error);\r\n\r\n });\r\n\r\n };\r\n\r\n /**\r\n * Asynchronously returns block data from blocksList by index\r\n * @return Promise to node\r\n */\r\n renderer.getNodeAsync = function (blocksList, index) {\r\n\r\n return Promise.resolve().then(function () {\r\n\r\n return {\r\n tool : blocksList[index],\r\n position : index\r\n };\r\n\r\n });\r\n\r\n };\r\n\r\n /**\r\n * Creates editor block by JSON-data\r\n *\r\n * @uses render method of each plugin\r\n *\r\n * @param {Object} toolData.tool\r\n * { header : {\r\n * text: '',\r\n * type: 'H3', ...\r\n * }\r\n * }\r\n * @param {Number} toolData.position - index in input-blocks array\r\n * @return {Object} with type and Element\r\n */\r\n renderer.createBlockFromData = function ( toolData ) {\r\n\r\n /** New parser */\r\n var block,\r\n tool = toolData.tool,\r\n pluginName = tool.type;\r\n\r\n /** Get first key of object that stores plugin name */\r\n // for (var pluginName in blockData) break;\r\n\r\n /** Check for plugin existance */\r\n if (!editor.tools[pluginName]) {\r\n\r\n throw Error(`Plugin «${pluginName}» not found`);\r\n\r\n }\r\n\r\n /** Check for plugin having render method */\r\n if (typeof editor.tools[pluginName].render != 'function') {\r\n\r\n throw Error(`Plugin «${pluginName}» must have «render» method`);\r\n\r\n }\r\n\r\n if ( editor.tools[pluginName].available === false ) {\r\n\r\n block = editor.draw.unavailableBlock();\r\n\r\n block.innerHTML = editor.tools[pluginName].loadingMessage;\r\n\r\n /**\r\n * Saver will extract data from initial block data by position in array\r\n */\r\n block.dataset.inputPosition = toolData.position;\r\n\r\n } else {\r\n\r\n /** New Parser */\r\n block = editor.tools[pluginName].render(tool.data);\r\n\r\n }\r\n\r\n /** is first-level block stretched */\r\n var stretched = editor.tools[pluginName].isStretched || false;\r\n\r\n /** Retrun type and block */\r\n return {\r\n type : pluginName,\r\n block : block,\r\n stretched : stretched\r\n };\r\n\r\n };\r\n\r\n return renderer;\r\n\r\n})({});\n\n\n// WEBPACK FOOTER //\n// ./modules/renderer.js","/**\r\n * Codex Editor Saver\r\n *\r\n * @author Codex Team\r\n * @version 1.0.2\r\n */\r\n\r\nmodule.exports = (function (saver) {\r\n\r\n let editor = codex.editor;\r\n\r\n /**\r\n * Saves blocks\r\n * @private\r\n */\r\n saver.saveBlocks = function () {\r\n\r\n /** Save html content of redactor to memory */\r\n editor.state.html = editor.nodes.redactor.innerHTML;\r\n\r\n /** Empty jsonOutput state */\r\n editor.state.jsonOutput = [];\r\n\r\n Promise.resolve()\r\n\r\n .then(function () {\r\n\r\n return editor.nodes.redactor.childNodes;\r\n\r\n })\r\n /** Making a sequence from separate blocks */\r\n .then(editor.saver.makeQueue)\r\n\r\n .then(function () {\r\n // editor.nodes.textarea.innerHTML = editor.state.html;\r\n })\r\n\r\n .catch( function (error) {\r\n\r\n editor.core.log(error);\r\n\r\n });\r\n\r\n };\r\n\r\n saver.makeQueue = function (blocks) {\r\n\r\n var queue = Promise.resolve();\r\n\r\n for(var index = 0; index < blocks.length; index++) {\r\n\r\n /** Add node to sequence at specified index */\r\n editor.saver.getBlockData(queue, blocks, index);\r\n\r\n }\r\n\r\n };\r\n\r\n /** Gets every block and makes From Data */\r\n saver.getBlockData = function (queue, blocks, index) {\r\n\r\n queue.then(function () {\r\n\r\n return editor.saver.getNodeAsync(blocks, index);\r\n\r\n })\r\n\r\n .then(editor.saver.makeFormDataFromBlocks);\r\n\r\n };\r\n\r\n\r\n /**\r\n * Asynchronously returns block data from blocksList by index\r\n * @return Promise to node\r\n */\r\n saver.getNodeAsync = function (blocksList, index) {\r\n\r\n return Promise.resolve().then(function () {\r\n\r\n return blocksList[index];\r\n\r\n });\r\n\r\n };\r\n\r\n saver.makeFormDataFromBlocks = function (block) {\r\n\r\n var pluginName = block.dataset.tool;\r\n\r\n /** Check for plugin existance */\r\n if (!editor.tools[pluginName]) {\r\n\r\n throw Error(`Plugin «${pluginName}» not found`);\r\n\r\n }\r\n\r\n /** Check for plugin having render method */\r\n if (typeof editor.tools[pluginName].save != 'function') {\r\n\r\n throw Error(`Plugin «${pluginName}» must have save method`);\r\n\r\n }\r\n\r\n /** Result saver */\r\n var blockContent = block.childNodes[0],\r\n pluginsContent = blockContent.childNodes[0],\r\n savedData,\r\n position,\r\n output;\r\n\r\n /** If plugin wasn't available then return data from cache */\r\n if ( editor.tools[pluginName].available === false ) {\r\n\r\n position = pluginsContent.dataset.inputPosition;\r\n savedData = codex.editor.state.blocks.items[position].data;\r\n\r\n } else {\r\n\r\n savedData = editor.tools[pluginName].save(pluginsContent);\r\n\r\n if (editor.tools[pluginName].validate) {\r\n\r\n var result = editor.tools[pluginName].validate(savedData);\r\n\r\n /**\r\n * Do not allow invalid data\r\n */\r\n if (!result)\r\n return;\r\n\r\n }\r\n\r\n }\r\n\r\n output = {\r\n type : pluginName,\r\n data : savedData\r\n };\r\n\r\n editor.state.jsonOutput.push(output);\r\n\r\n };\r\n\r\n return saver;\r\n\r\n})({});\n\n\n// WEBPACK FOOTER //\n// ./modules/saver.js","/**\r\n * Codex Editor Content Module\r\n * Works with DOM\r\n *\r\n * @module Codex Editor content module\r\n *\r\n * @author Codex Team\r\n * @version 1.3.13\r\n *\r\n * @description Module works with Elements that have been appended to the main DOM\r\n */\r\n\r\nmodule.exports = (function (content) {\r\n\r\n let editor = codex.editor;\r\n\r\n /**\r\n * Links to current active block\r\n * @type {null | Element}\r\n */\r\n content.currentNode = null;\r\n\r\n /**\r\n * clicked in redactor area\r\n * @type {null | Boolean}\r\n */\r\n content.editorAreaHightlighted = null;\r\n\r\n /**\r\n * @deprecated\r\n * Synchronizes redactor with original textarea\r\n */\r\n content.sync = function () {\r\n\r\n editor.core.log('syncing...');\r\n\r\n /**\r\n * Save redactor content to editor.state\r\n */\r\n editor.state.html = editor.nodes.redactor.innerHTML;\r\n\r\n };\r\n\r\n /**\r\n * Appends background to the block\r\n *\r\n * @description add CSS class to highlight visually first-level block area\r\n */\r\n content.markBlock = function () {\r\n\r\n editor.content.currentNode.classList.add(editor.ui.className.BLOCK_HIGHLIGHTED);\r\n\r\n };\r\n\r\n /**\r\n * Clear background\r\n *\r\n * @description clears styles that highlights block\r\n */\r\n content.clearMark = function () {\r\n\r\n if (editor.content.currentNode) {\r\n\r\n editor.content.currentNode.classList.remove(editor.ui.className.BLOCK_HIGHLIGHTED);\r\n\r\n }\r\n\r\n };\r\n\r\n /**\r\n * Finds first-level block\r\n *\r\n * @param {Element} node - selected or clicked in redactors area node\r\n * @protected\r\n *\r\n * @description looks for first-level block.\r\n * gets parent while node is not first-level\r\n */\r\n content.getFirstLevelBlock = function (node) {\r\n\r\n if (!editor.core.isDomNode(node)) {\r\n\r\n node = node.parentNode;\r\n\r\n }\r\n\r\n if (node === editor.nodes.redactor || node === document.body) {\r\n\r\n return null;\r\n\r\n } else {\r\n\r\n while(!node.classList.contains(editor.ui.className.BLOCK_CLASSNAME)) {\r\n\r\n node = node.parentNode;\r\n\r\n }\r\n\r\n return node;\r\n\r\n }\r\n\r\n };\r\n\r\n /**\r\n * Trigger this event when working node changed\r\n * @param {Element} targetNode - first-level of this node will be current\r\n * @protected\r\n *\r\n * @description If targetNode is first-level then we set it as current else we look for parents to find first-level\r\n */\r\n content.workingNodeChanged = function (targetNode) {\r\n\r\n /** Clear background from previous marked block before we change */\r\n editor.content.clearMark();\r\n\r\n if (!targetNode) {\r\n\r\n return;\r\n\r\n }\r\n\r\n content.currentNode = content.getFirstLevelBlock(targetNode);\r\n\r\n };\r\n\r\n /**\r\n * Replaces one redactor block with another\r\n * @protected\r\n * @param {Element} targetBlock - block to replace. Mostly currentNode.\r\n * @param {Element} newBlock\r\n * @param {string} newBlockType - type of new block; we need to store it to data-attribute\r\n *\r\n * [!] Function does not saves old block content.\r\n * You can get it manually and pass with newBlock.innerHTML\r\n */\r\n content.replaceBlock = function (targetBlock, newBlock) {\r\n\r\n if (!targetBlock || !newBlock) {\r\n\r\n editor.core.log('replaceBlock: missed params');\r\n return;\r\n\r\n }\r\n\r\n /** If target-block is not a frist-level block, then we iterate parents to find it */\r\n while(!targetBlock.classList.contains(editor.ui.className.BLOCK_CLASSNAME)) {\r\n\r\n targetBlock = targetBlock.parentNode;\r\n\r\n }\r\n\r\n /** Replacing */\r\n editor.nodes.redactor.replaceChild(newBlock, targetBlock);\r\n\r\n /**\r\n * Set new node as current\r\n */\r\n editor.content.workingNodeChanged(newBlock);\r\n\r\n /**\r\n * Add block handlers\r\n */\r\n editor.ui.addBlockHandlers(newBlock);\r\n\r\n /**\r\n * Save changes\r\n */\r\n editor.ui.saveInputs();\r\n\r\n };\r\n\r\n /**\r\n * @protected\r\n *\r\n * Inserts new block to redactor\r\n * Wrapps block into a DIV with BLOCK_CLASSNAME class\r\n *\r\n * @param blockData {object}\r\n * @param blockData.block {Element} element with block content\r\n * @param blockData.type {string} block plugin\r\n * @param needPlaceCaret {bool} pass true to set caret in new block\r\n *\r\n */\r\n content.insertBlock = function ( blockData, needPlaceCaret ) {\r\n\r\n var workingBlock = editor.content.currentNode,\r\n newBlockContent = blockData.block,\r\n blockType = blockData.type,\r\n isStretched = blockData.stretched;\r\n\r\n var newBlock = composeNewBlock_(newBlockContent, blockType, isStretched);\r\n\r\n if (workingBlock) {\r\n\r\n editor.core.insertAfter(workingBlock, newBlock);\r\n\r\n } else {\r\n\r\n /**\r\n * If redactor is empty, append as first child\r\n */\r\n editor.nodes.redactor.appendChild(newBlock);\r\n\r\n }\r\n\r\n /**\r\n * Block handler\r\n */\r\n editor.ui.addBlockHandlers(newBlock);\r\n\r\n /**\r\n * Set new node as current\r\n */\r\n editor.content.workingNodeChanged(newBlock);\r\n\r\n /**\r\n * Save changes\r\n */\r\n editor.ui.saveInputs();\r\n\r\n\r\n if ( needPlaceCaret ) {\r\n\r\n /**\r\n * If we don't know input index then we set default value -1\r\n */\r\n var currentInputIndex = editor.caret.getCurrentInputIndex() || -1;\r\n\r\n\r\n if (currentInputIndex == -1) {\r\n\r\n\r\n var editableElement = newBlock.querySelector('[contenteditable]'),\r\n emptyText = document.createTextNode('');\r\n\r\n editableElement.appendChild(emptyText);\r\n editor.caret.set(editableElement, 0, 0);\r\n\r\n editor.toolbar.move();\r\n editor.toolbar.showPlusButton();\r\n\r\n\r\n } else {\r\n\r\n if (currentInputIndex === editor.state.inputs.length - 1)\r\n return;\r\n\r\n /** Timeout for browsers execution */\r\n window.setTimeout(function () {\r\n\r\n /** Setting to the new input */\r\n editor.caret.setToNextBlock(currentInputIndex);\r\n editor.toolbar.move();\r\n editor.toolbar.open();\r\n\r\n }, 10);\r\n\r\n }\r\n\r\n }\r\n\r\n /**\r\n * Block is inserted, wait for new click that defined focusing on editors area\r\n * @type {boolean}\r\n */\r\n content.editorAreaHightlighted = false;\r\n\r\n };\r\n\r\n /**\r\n * Replaces blocks with saving content\r\n * @protected\r\n * @param {Element} noteToReplace\r\n * @param {Element} newNode\r\n * @param {Element} blockType\r\n */\r\n content.switchBlock = function (blockToReplace, newBlock, tool) {\r\n\r\n tool = tool || editor.content.currentNode.dataset.tool;\r\n var newBlockComposed = composeNewBlock_(newBlock, tool);\r\n\r\n /** Replacing */\r\n editor.content.replaceBlock(blockToReplace, newBlockComposed);\r\n\r\n /** Save new Inputs when block is changed */\r\n editor.ui.saveInputs();\r\n\r\n };\r\n\r\n /**\r\n * Iterates between child noted and looking for #text node on deepest level\r\n * @protected\r\n *\r\n * @param {Element} block - node where find\r\n * @param {int} postiton - starting postion\r\n * Example: childNodex.length to find from the end\r\n * or 0 to find from the start\r\n * @return {Text} block\r\n * @uses DFS\r\n */\r\n content.getDeepestTextNodeFromPosition = function (block, position) {\r\n\r\n /**\r\n * Clear Block from empty and useless spaces with trim.\r\n * Such nodes we should remove\r\n */\r\n var blockChilds = block.childNodes,\r\n index,\r\n node,\r\n text;\r\n\r\n for(index = 0; index < blockChilds.length; index++) {\r\n\r\n node = blockChilds[index];\r\n\r\n if (node.nodeType == editor.core.nodeTypes.TEXT) {\r\n\r\n text = node.textContent.trim();\r\n\r\n /** Text is empty. We should remove this child from node before we start DFS\r\n * decrease the quantity of childs.\r\n */\r\n if (text === '') {\r\n\r\n block.removeChild(node);\r\n position--;\r\n\r\n }\r\n\r\n }\r\n\r\n }\r\n\r\n if (block.childNodes.length === 0) {\r\n\r\n return document.createTextNode('');\r\n\r\n }\r\n\r\n /** Setting default position when we deleted all empty nodes */\r\n if ( position < 0 )\r\n position = 1;\r\n\r\n var lookingFromStart = false;\r\n\r\n /** For looking from START */\r\n if (position === 0) {\r\n\r\n lookingFromStart = true;\r\n position = 1;\r\n\r\n }\r\n\r\n while ( position ) {\r\n\r\n /** initial verticle of node. */\r\n if ( lookingFromStart ) {\r\n\r\n block = block.childNodes[0];\r\n\r\n } else {\r\n\r\n block = block.childNodes[position - 1];\r\n\r\n }\r\n\r\n if ( block.nodeType == editor.core.nodeTypes.TAG ) {\r\n\r\n position = block.childNodes.length;\r\n\r\n } else if (block.nodeType == editor.core.nodeTypes.TEXT ) {\r\n\r\n position = 0;\r\n\r\n }\r\n\r\n }\r\n\r\n return block;\r\n\r\n };\r\n\r\n /**\r\n * @private\r\n * @param {Element} block - current plugins render\r\n * @param {String} tool - plugins name\r\n * @param {Boolean} isStretched - make stretched block or not\r\n *\r\n * @description adds necessary information to wrap new created block by first-level holder\r\n */\r\n var composeNewBlock_ = function (block, tool, isStretched) {\r\n\r\n var newBlock = editor.draw.node('DIV', editor.ui.className.BLOCK_CLASSNAME, {}),\r\n blockContent = editor.draw.node('DIV', editor.ui.className.BLOCK_CONTENT, {});\r\n\r\n blockContent.appendChild(block);\r\n newBlock.appendChild(blockContent);\r\n\r\n if (isStretched) {\r\n\r\n blockContent.classList.add(editor.ui.className.BLOCK_STRETCHED);\r\n\r\n }\r\n\r\n newBlock.dataset.tool = tool;\r\n return newBlock;\r\n\r\n };\r\n\r\n /**\r\n * Returns Range object of current selection\r\n * @protected\r\n */\r\n content.getRange = function () {\r\n\r\n var selection = window.getSelection().getRangeAt(0);\r\n\r\n return selection;\r\n\r\n };\r\n\r\n /**\r\n * Divides block in two blocks (after and before caret)\r\n *\r\n * @protected\r\n * @param {int} inputIndex - target input index\r\n *\r\n * @description splits current input content to the separate blocks\r\n * When enter is pressed among the words, that text will be splited.\r\n */\r\n content.splitBlock = function (inputIndex) {\r\n\r\n var selection = window.getSelection(),\r\n anchorNode = selection.anchorNode,\r\n anchorNodeText = anchorNode.textContent,\r\n caretOffset = selection.anchorOffset,\r\n textBeforeCaret,\r\n textNodeBeforeCaret,\r\n textAfterCaret,\r\n textNodeAfterCaret;\r\n\r\n var currentBlock = editor.content.currentNode.querySelector('[contentEditable]');\r\n\r\n\r\n textBeforeCaret = anchorNodeText.substring(0, caretOffset);\r\n textAfterCaret = anchorNodeText.substring(caretOffset);\r\n\r\n textNodeBeforeCaret = document.createTextNode(textBeforeCaret);\r\n\r\n if (textAfterCaret) {\r\n\r\n textNodeAfterCaret = document.createTextNode(textAfterCaret);\r\n\r\n }\r\n\r\n var previousChilds = [],\r\n nextChilds = [],\r\n reachedCurrent = false;\r\n\r\n if (textNodeAfterCaret) {\r\n\r\n nextChilds.push(textNodeAfterCaret);\r\n\r\n }\r\n\r\n for ( var i = 0, child; !!(child = currentBlock.childNodes[i]); i++) {\r\n\r\n if ( child != anchorNode ) {\r\n\r\n if ( !reachedCurrent ) {\r\n\r\n previousChilds.push(child);\r\n\r\n } else {\r\n\r\n nextChilds.push(child);\r\n\r\n }\r\n\r\n } else {\r\n\r\n reachedCurrent = true;\r\n\r\n }\r\n\r\n }\r\n\r\n /** Clear current input */\r\n editor.state.inputs[inputIndex].innerHTML = '';\r\n\r\n /**\r\n * Append all childs founded before anchorNode\r\n */\r\n var previousChildsLength = previousChilds.length;\r\n\r\n for(i = 0; i < previousChildsLength; i++) {\r\n\r\n editor.state.inputs[inputIndex].appendChild(previousChilds[i]);\r\n\r\n }\r\n\r\n editor.state.inputs[inputIndex].appendChild(textNodeBeforeCaret);\r\n\r\n /**\r\n * Append text node which is after caret\r\n */\r\n var nextChildsLength = nextChilds.length,\r\n newNode = document.createElement('div');\r\n\r\n for(i = 0; i < nextChildsLength; i++) {\r\n\r\n newNode.appendChild(nextChilds[i]);\r\n\r\n }\r\n\r\n newNode = newNode.innerHTML;\r\n\r\n /** This type of block creates when enter is pressed */\r\n var NEW_BLOCK_TYPE = editor.settings.initialBlockPlugin;\r\n\r\n /**\r\n * Make new paragraph with text after caret\r\n */\r\n editor.content.insertBlock({\r\n type : NEW_BLOCK_TYPE,\r\n block : editor.tools[NEW_BLOCK_TYPE].render({\r\n text : newNode\r\n })\r\n }, true );\r\n\r\n };\r\n\r\n /**\r\n * Merges two blocks — current and target\r\n * If target index is not exist, then previous will be as target\r\n *\r\n * @protected\r\n * @param {int} currentInputIndex\r\n * @param {int} targetInputIndex\r\n *\r\n * @description gets two inputs indexes and merges into one\r\n */\r\n content.mergeBlocks = function (currentInputIndex, targetInputIndex) {\r\n\r\n /** If current input index is zero, then prevent method execution */\r\n if (currentInputIndex === 0) {\r\n\r\n return;\r\n\r\n }\r\n\r\n var targetInput,\r\n currentInputContent = editor.state.inputs[currentInputIndex].innerHTML;\r\n\r\n if (!targetInputIndex) {\r\n\r\n targetInput = editor.state.inputs[currentInputIndex - 1];\r\n\r\n } else {\r\n\r\n targetInput = editor.state.inputs[targetInputIndex];\r\n\r\n }\r\n\r\n targetInput.innerHTML += currentInputContent;\r\n\r\n };\r\n\r\n /**\r\n * Iterates all right siblings and parents, which has right siblings\r\n * while it does not reached the first-level block\r\n *\r\n * @param {Element} node\r\n * @return {boolean}\r\n */\r\n content.isLastNode = function (node) {\r\n\r\n // console.log('погнали перебор родителей');\r\n\r\n var allChecked = false;\r\n\r\n while ( !allChecked ) {\r\n\r\n // console.log('Смотрим на %o', node);\r\n // console.log('Проверим, пустые ли соседи справа');\r\n\r\n if ( !allSiblingsEmpty_(node) ) {\r\n\r\n // console.log('Есть непустые соседи. Узел не последний. Выходим.');\r\n return false;\r\n\r\n }\r\n\r\n node = node.parentNode;\r\n\r\n /**\r\n * Проверяем родителей до тех пор, пока не найдем блок первого уровня\r\n */\r\n if ( node.classList.contains(editor.ui.className.BLOCK_CONTENT) ) {\r\n\r\n allChecked = true;\r\n\r\n }\r\n\r\n }\r\n\r\n return true;\r\n\r\n };\r\n\r\n /**\r\n * Checks if all element right siblings is empty\r\n * @param node\r\n */\r\n var allSiblingsEmpty_ = function (node) {\r\n\r\n /**\r\n * Нужно убедиться, что после пустого соседа ничего нет\r\n */\r\n var sibling = node.nextSibling;\r\n\r\n while ( sibling ) {\r\n\r\n if (sibling.textContent.length) {\r\n\r\n return false;\r\n\r\n }\r\n\r\n sibling = sibling.nextSibling;\r\n\r\n }\r\n\r\n return true;\r\n\r\n };\r\n\r\n /**\r\n * @public\r\n *\r\n * @param {string} htmlData - html content as string\r\n * @param {string} plainData - plain text\r\n * @return {string} - html content as string\r\n */\r\n content.wrapTextWithParagraphs = function (htmlData, plainData) {\r\n\r\n if (!htmlData) {\r\n\r\n return wrapPlainTextWithParagraphs(plainData);\r\n\r\n }\r\n\r\n var wrapper = document.createElement('DIV'),\r\n newWrapper = document.createElement('DIV'),\r\n i,\r\n paragraph,\r\n firstLevelBlocks = ['DIV', 'P'],\r\n blockTyped,\r\n node;\r\n\r\n /**\r\n * Make HTML Element to Wrap Text\r\n * It allows us to work with input data as HTML content\r\n */\r\n wrapper.innerHTML = htmlData;\r\n paragraph = document.createElement('P');\r\n\r\n for (i = 0; i < wrapper.childNodes.length; i++) {\r\n\r\n node = wrapper.childNodes[i];\r\n\r\n blockTyped = firstLevelBlocks.indexOf(node.tagName) != -1;\r\n\r\n /**\r\n * If node is first-levet\r\n * we add this node to our new wrapper\r\n */\r\n if ( blockTyped ) {\r\n\r\n /**\r\n * If we had splitted inline nodes to paragraph before\r\n */\r\n if ( paragraph.childNodes.length ) {\r\n\r\n newWrapper.appendChild(paragraph.cloneNode(true));\r\n\r\n /** empty paragraph */\r\n paragraph = null;\r\n paragraph = document.createElement('P');\r\n\r\n }\r\n\r\n newWrapper.appendChild(node.cloneNode(true));\r\n\r\n } else {\r\n\r\n /** Collect all inline nodes to one as paragraph */\r\n paragraph.appendChild(node.cloneNode(true));\r\n\r\n /** if node is last we should append this node to paragraph and paragraph to new wrapper */\r\n if ( i == wrapper.childNodes.length - 1 ) {\r\n\r\n newWrapper.appendChild(paragraph.cloneNode(true));\r\n\r\n }\r\n\r\n }\r\n\r\n }\r\n\r\n return newWrapper.innerHTML;\r\n\r\n };\r\n\r\n /**\r\n * Splits strings on new line and wraps paragraphs with

    tag\r\n * @param plainText\r\n * @returns {string}\r\n */\r\n var wrapPlainTextWithParagraphs = function (plainText) {\r\n\r\n return '

    ' + plainText.split('\\n\\n').join('

    ') + '

    ';\r\n\r\n };\r\n\r\n /**\r\n * Finds closest Contenteditable parent from Element\r\n * @param {Element} node element looking from\r\n * @return {Element} node contenteditable\r\n */\r\n content.getEditableParent = function (node) {\r\n\r\n while (node && node.contentEditable != 'true') {\r\n\r\n node = node.parentNode;\r\n\r\n }\r\n\r\n return node;\r\n\r\n };\r\n\r\n return content;\r\n\r\n})({});\n\n\n// WEBPACK FOOTER //\n// ./modules/content.js","/**\r\n * Codex Editor toolbar module\r\n *\r\n * Contains:\r\n * - Inline toolbox\r\n * - Toolbox within plus button\r\n * - Settings section\r\n *\r\n * @author Codex Team\r\n * @version 1.0\r\n */\r\n\r\nmodule.exports = (function (toolbar) {\r\n\r\n let editor = codex.editor;\r\n\r\n toolbar.settings = require('./settings');\r\n toolbar.inline = require('./inline');\r\n toolbar.toolbox = require('./toolbox');\r\n\r\n /**\r\n * Margin between focused node and toolbar\r\n */\r\n toolbar.defaultToolbarHeight = 49;\r\n\r\n toolbar.defaultOffset = 34;\r\n\r\n toolbar.opened = false;\r\n\r\n toolbar.current = null;\r\n\r\n /**\r\n * @protected\r\n */\r\n toolbar.open = function () {\r\n\r\n if (editor.hideToolbar) {\r\n\r\n return;\r\n\r\n }\r\n\r\n editor.nodes.toolbar.classList.add('opened');\r\n this.opened = true;\r\n\r\n };\r\n\r\n /**\r\n * @protected\r\n */\r\n toolbar.close = function () {\r\n\r\n editor.nodes.toolbar.classList.remove('opened');\r\n\r\n toolbar.opened = false;\r\n toolbar.current = null;\r\n\r\n for (var button in editor.nodes.toolbarButtons) {\r\n\r\n editor.nodes.toolbarButtons[button].classList.remove('selected');\r\n\r\n }\r\n\r\n /** Close toolbox when toolbar is not displayed */\r\n editor.toolbar.toolbox.close();\r\n editor.toolbar.settings.close();\r\n\r\n };\r\n\r\n toolbar.toggle = function () {\r\n\r\n if ( !this.opened ) {\r\n\r\n this.open();\r\n\r\n } else {\r\n\r\n this.close();\r\n\r\n }\r\n\r\n };\r\n\r\n toolbar.hidePlusButton = function () {\r\n\r\n editor.nodes.plusButton.classList.add('hide');\r\n\r\n };\r\n\r\n toolbar.showPlusButton = function () {\r\n\r\n editor.nodes.plusButton.classList.remove('hide');\r\n\r\n };\r\n\r\n /**\r\n * Moving toolbar to the specified node\r\n */\r\n toolbar.move = function () {\r\n\r\n /** Close Toolbox when we move toolbar */\r\n editor.toolbar.toolbox.close();\r\n\r\n if (!editor.content.currentNode) {\r\n\r\n return;\r\n\r\n }\r\n\r\n var newYCoordinate = editor.content.currentNode.offsetTop - (editor.toolbar.defaultToolbarHeight / 2) + editor.toolbar.defaultOffset;\r\n\r\n editor.nodes.toolbar.style.transform = `translate3D(0, ${Math.floor(newYCoordinate)}px, 0)`;\r\n\r\n /** Close trash actions */\r\n editor.toolbar.settings.hideRemoveActions();\r\n\r\n };\r\n\r\n return toolbar;\r\n\r\n})({});\n\n\n// WEBPACK FOOTER //\n// ./modules/toolbar/toolbar.js","/**\r\n * Toolbar settings\r\n *\r\n * @version 1.0.4\r\n */\r\n\r\nmodule.exports = (function (settings) {\r\n\r\n let editor = codex.editor;\r\n\r\n settings.opened = false;\r\n\r\n settings.setting = null;\r\n settings.actions = null;\r\n\r\n /**\r\n * Append and open settings\r\n */\r\n settings.open = function (toolType) {\r\n\r\n /**\r\n * Append settings content\r\n * It's stored in tool.settings\r\n */\r\n if (!editor.tools[toolType] || !editor.tools[toolType].makeSettings ) {\r\n\r\n editor.core.log(`Plugin «${toolType}» has no settings`, 'warn');\r\n editor.nodes.pluginSettings.innerHTML = `Плагин «${toolType}» не имеет настроек`;\r\n\r\n } else {\r\n\r\n /**\r\n * Draw settings block\r\n */\r\n var settingsBlock = editor.tools[toolType].makeSettings();\r\n\r\n editor.nodes.pluginSettings.appendChild(settingsBlock);\r\n\r\n }\r\n\r\n /** Open settings block */\r\n editor.nodes.blockSettings.classList.add('opened');\r\n this.opened = true;\r\n\r\n };\r\n\r\n /**\r\n * Close and clear settings\r\n */\r\n settings.close = function () {\r\n\r\n editor.nodes.blockSettings.classList.remove('opened');\r\n editor.nodes.pluginSettings.innerHTML = '';\r\n\r\n this.opened = false;\r\n\r\n };\r\n\r\n /**\r\n * @param {string} toolType - plugin type\r\n */\r\n settings.toggle = function ( toolType ) {\r\n\r\n if ( !this.opened ) {\r\n\r\n this.open(toolType);\r\n\r\n } else {\r\n\r\n this.close();\r\n\r\n }\r\n\r\n };\r\n\r\n /**\r\n * Here we will draw buttons and add listeners to components\r\n */\r\n settings.makeRemoveBlockButton = function () {\r\n\r\n var removeBlockWrapper = editor.draw.node('SPAN', 'ce-toolbar__remove-btn', {}),\r\n settingButton = editor.draw.node('SPAN', 'ce-toolbar__remove-setting', { innerHTML : '' }),\r\n actionWrapper = editor.draw.node('DIV', 'ce-toolbar__remove-confirmation', {}),\r\n confirmAction = editor.draw.node('DIV', 'ce-toolbar__remove-confirm', { textContent : 'Удалить блок' }),\r\n cancelAction = editor.draw.node('DIV', 'ce-toolbar__remove-cancel', { textContent : 'Отмена' });\r\n\r\n editor.listeners.add(settingButton, 'click', editor.toolbar.settings.removeButtonClicked, false);\r\n\r\n editor.listeners.add(confirmAction, 'click', editor.toolbar.settings.confirmRemovingRequest, false);\r\n\r\n editor.listeners.add(cancelAction, 'click', editor.toolbar.settings.cancelRemovingRequest, false);\r\n\r\n actionWrapper.appendChild(confirmAction);\r\n actionWrapper.appendChild(cancelAction);\r\n\r\n removeBlockWrapper.appendChild(settingButton);\r\n removeBlockWrapper.appendChild(actionWrapper);\r\n\r\n /** Save setting */\r\n editor.toolbar.settings.setting = settingButton;\r\n editor.toolbar.settings.actions = actionWrapper;\r\n\r\n return removeBlockWrapper;\r\n\r\n };\r\n\r\n settings.removeButtonClicked = function () {\r\n\r\n var action = editor.toolbar.settings.actions;\r\n\r\n if (action.classList.contains('opened')) {\r\n\r\n editor.toolbar.settings.hideRemoveActions();\r\n\r\n } else {\r\n\r\n editor.toolbar.settings.showRemoveActions();\r\n\r\n }\r\n\r\n editor.toolbar.toolbox.close();\r\n editor.toolbar.settings.close();\r\n\r\n };\r\n\r\n settings.cancelRemovingRequest = function () {\r\n\r\n editor.toolbar.settings.actions.classList.remove('opened');\r\n\r\n };\r\n\r\n settings.confirmRemovingRequest = function () {\r\n\r\n var currentBlock = editor.content.currentNode,\r\n firstLevelBlocksCount;\r\n\r\n currentBlock.remove();\r\n\r\n firstLevelBlocksCount = editor.nodes.redactor.childNodes.length;\r\n\r\n /**\r\n * If all blocks are removed\r\n */\r\n if (firstLevelBlocksCount === 0) {\r\n\r\n /** update currentNode variable */\r\n editor.content.currentNode = null;\r\n\r\n /** Inserting new empty initial block */\r\n editor.ui.addInitialBlock();\r\n\r\n }\r\n\r\n editor.ui.saveInputs();\r\n\r\n editor.toolbar.close();\r\n\r\n };\r\n\r\n settings.showRemoveActions = function () {\r\n\r\n editor.toolbar.settings.actions.classList.add('opened');\r\n\r\n };\r\n\r\n settings.hideRemoveActions = function () {\r\n\r\n editor.toolbar.settings.actions.classList.remove('opened');\r\n\r\n };\r\n\r\n return settings;\r\n\r\n})({});\n\n\n// WEBPACK FOOTER //\n// ./modules/toolbar/settings.js","/**\r\n * Inline toolbar\r\n *\r\n * Contains from tools:\r\n * Bold, Italic, Underline and Anchor\r\n *\r\n * @author Codex Team\r\n * @version 1.0\r\n */\r\n\r\nmodule.exports = (function (inline) {\r\n\r\n let editor = codex.editor;\r\n\r\n inline.buttonsOpened = null;\r\n inline.actionsOpened = null;\r\n inline.wrappersOffset = null;\r\n\r\n /**\r\n * saving selection that need for execCommand for styling\r\n *\r\n */\r\n inline.storedSelection = null;\r\n\r\n /**\r\n * @protected\r\n *\r\n * Open inline toobar\r\n */\r\n inline.show = function () {\r\n\r\n var currentNode = editor.content.currentNode,\r\n tool = currentNode.dataset.tool,\r\n plugin;\r\n\r\n /**\r\n * tool allowed to open inline toolbar\r\n */\r\n plugin = editor.tools[tool];\r\n\r\n if (!plugin.showInlineToolbar)\r\n return;\r\n\r\n var selectedText = inline.getSelectionText(),\r\n toolbar = editor.nodes.inlineToolbar.wrapper;\r\n\r\n if (selectedText.length > 0) {\r\n\r\n /** Move toolbar and open */\r\n editor.toolbar.inline.move();\r\n\r\n /** Open inline toolbar */\r\n toolbar.classList.add('opened');\r\n\r\n /** show buttons of inline toolbar */\r\n editor.toolbar.inline.showButtons();\r\n\r\n }\r\n\r\n };\r\n\r\n /**\r\n * @protected\r\n *\r\n * Closes inline toolbar\r\n */\r\n inline.close = function () {\r\n\r\n var toolbar = editor.nodes.inlineToolbar.wrapper;\r\n\r\n toolbar.classList.remove('opened');\r\n\r\n };\r\n\r\n /**\r\n * @private\r\n *\r\n * Moving toolbar\r\n */\r\n inline.move = function () {\r\n\r\n if (!this.wrappersOffset) {\r\n\r\n this.wrappersOffset = this.getWrappersOffset();\r\n\r\n }\r\n\r\n var coords = this.getSelectionCoords(),\r\n defaultOffset = 0,\r\n toolbar = editor.nodes.inlineToolbar.wrapper,\r\n newCoordinateX,\r\n newCoordinateY;\r\n\r\n if (toolbar.offsetHeight === 0) {\r\n\r\n defaultOffset = 40;\r\n\r\n }\r\n\r\n newCoordinateX = coords.x - this.wrappersOffset.left;\r\n newCoordinateY = coords.y + window.scrollY - this.wrappersOffset.top - defaultOffset - toolbar.offsetHeight;\r\n\r\n toolbar.style.transform = `translate3D(${Math.floor(newCoordinateX)}px, ${Math.floor(newCoordinateY)}px, 0)`;\r\n\r\n /** Close everything */\r\n editor.toolbar.inline.closeButtons();\r\n editor.toolbar.inline.closeAction();\r\n\r\n };\r\n\r\n /**\r\n * @private\r\n *\r\n * Tool Clicked\r\n */\r\n\r\n inline.toolClicked = function (event, type) {\r\n\r\n /**\r\n * For simple tools we use default browser function\r\n * For more complicated tools, we should write our own behavior\r\n */\r\n switch (type) {\r\n case 'createLink' : editor.toolbar.inline.createLinkAction(event, type); break;\r\n default : editor.toolbar.inline.defaultToolAction(type); break;\r\n }\r\n\r\n /**\r\n * highlight buttons\r\n * after making some action\r\n */\r\n editor.nodes.inlineToolbar.buttons.childNodes.forEach(editor.toolbar.inline.hightlight);\r\n\r\n };\r\n\r\n /**\r\n * @private\r\n *\r\n * Saving wrappers offset in DOM\r\n */\r\n inline.getWrappersOffset = function () {\r\n\r\n var wrapper = editor.nodes.wrapper,\r\n offset = this.getOffset(wrapper);\r\n\r\n this.wrappersOffset = offset;\r\n return offset;\r\n\r\n };\r\n\r\n /**\r\n * @private\r\n *\r\n * Calculates offset of DOM element\r\n *\r\n * @param el\r\n * @returns {{top: number, left: number}}\r\n */\r\n inline.getOffset = function ( el ) {\r\n\r\n var _x = 0;\r\n var _y = 0;\r\n\r\n while( el && !isNaN( el.offsetLeft ) && !isNaN( el.offsetTop ) ) {\r\n\r\n _x += (el.offsetLeft + el.clientLeft);\r\n _y += (el.offsetTop + el.clientTop);\r\n el = el.offsetParent;\r\n\r\n }\r\n return { top: _y, left: _x };\r\n\r\n };\r\n\r\n /**\r\n * @private\r\n *\r\n * Calculates position of selected text\r\n * @returns {{x: number, y: number}}\r\n */\r\n inline.getSelectionCoords = function () {\r\n\r\n var sel = document.selection, range;\r\n var x = 0, y = 0;\r\n\r\n if (sel) {\r\n\r\n if (sel.type != 'Control') {\r\n\r\n range = sel.createRange();\r\n range.collapse(true);\r\n x = range.boundingLeft;\r\n y = range.boundingTop;\r\n\r\n }\r\n\r\n } else if (window.getSelection) {\r\n\r\n sel = window.getSelection();\r\n\r\n if (sel.rangeCount) {\r\n\r\n range = sel.getRangeAt(0).cloneRange();\r\n if (range.getClientRects) {\r\n\r\n range.collapse(true);\r\n var rect = range.getClientRects()[0];\r\n\r\n if (!rect) {\r\n\r\n return;\r\n\r\n }\r\n\r\n x = rect.left;\r\n y = rect.top;\r\n\r\n }\r\n\r\n }\r\n\r\n }\r\n return { x: x, y: y };\r\n\r\n };\r\n\r\n /**\r\n * @private\r\n *\r\n * Returns selected text as String\r\n * @returns {string}\r\n */\r\n inline.getSelectionText = function () {\r\n\r\n var selectedText = '';\r\n\r\n // all modern browsers and IE9+\r\n if (window.getSelection) {\r\n\r\n selectedText = window.getSelection().toString();\r\n\r\n }\r\n\r\n return selectedText;\r\n\r\n };\r\n\r\n /** Opens buttons block */\r\n inline.showButtons = function () {\r\n\r\n var buttons = editor.nodes.inlineToolbar.buttons;\r\n\r\n buttons.classList.add('opened');\r\n\r\n editor.toolbar.inline.buttonsOpened = true;\r\n\r\n /** highlight buttons */\r\n editor.nodes.inlineToolbar.buttons.childNodes.forEach(editor.toolbar.inline.hightlight);\r\n\r\n };\r\n\r\n /** Makes buttons disappear */\r\n inline.closeButtons = function () {\r\n\r\n var buttons = editor.nodes.inlineToolbar.buttons;\r\n\r\n buttons.classList.remove('opened');\r\n\r\n editor.toolbar.inline.buttonsOpened = false;\r\n\r\n };\r\n\r\n /** Open buttons defined action if exist */\r\n inline.showActions = function () {\r\n\r\n var action = editor.nodes.inlineToolbar.actions;\r\n\r\n action.classList.add('opened');\r\n\r\n editor.toolbar.inline.actionsOpened = true;\r\n\r\n };\r\n\r\n /** Close actions block */\r\n inline.closeAction = function () {\r\n\r\n var action = editor.nodes.inlineToolbar.actions;\r\n\r\n action.innerHTML = '';\r\n action.classList.remove('opened');\r\n editor.toolbar.inline.actionsOpened = false;\r\n\r\n };\r\n\r\n\r\n /**\r\n * Callback for keydowns in inline toolbar \"Insert link...\" input\r\n */\r\n let inlineToolbarAnchorInputKeydown_ = function (event) {\r\n\r\n if (event.keyCode != editor.core.keys.ENTER) {\r\n\r\n return;\r\n\r\n }\r\n\r\n let editable = editor.content.currentNode,\r\n storedSelection = editor.toolbar.inline.storedSelection;\r\n\r\n editor.toolbar.inline.restoreSelection(editable, storedSelection);\r\n editor.toolbar.inline.setAnchor(this.value);\r\n\r\n /**\r\n * Preventing events that will be able to happen\r\n */\r\n event.preventDefault();\r\n event.stopImmediatePropagation();\r\n\r\n editor.toolbar.inline.clearRange();\r\n\r\n };\r\n\r\n /** Action for link creation or for setting anchor */\r\n inline.createLinkAction = function (event) {\r\n\r\n var isActive = this.isLinkActive();\r\n\r\n var editable = editor.content.currentNode,\r\n storedSelection = editor.toolbar.inline.saveSelection(editable);\r\n\r\n /** Save globally selection */\r\n editor.toolbar.inline.storedSelection = storedSelection;\r\n\r\n if (isActive) {\r\n\r\n\r\n /**\r\n * Changing stored selection. if we want to remove anchor from word\r\n * we should remove anchor from whole word, not only selected part.\r\n * The solution is than we get the length of current link\r\n * Change start position to - end of selection minus length of anchor\r\n */\r\n editor.toolbar.inline.restoreSelection(editable, storedSelection);\r\n\r\n editor.toolbar.inline.defaultToolAction('unlink');\r\n\r\n } else {\r\n\r\n /** Create input and close buttons */\r\n var action = editor.draw.inputForLink();\r\n\r\n editor.nodes.inlineToolbar.actions.appendChild(action);\r\n\r\n editor.toolbar.inline.closeButtons();\r\n editor.toolbar.inline.showActions();\r\n\r\n /**\r\n * focus to input\r\n * Solution: https://developer.mozilla.org/ru/docs/Web/API/HTMLElement/focus\r\n * Prevents event after showing input and when we need to focus an input which is in unexisted form\r\n */\r\n action.focus();\r\n event.preventDefault();\r\n\r\n /** Callback to link action */\r\n editor.listeners.add(action, 'keydown', inlineToolbarAnchorInputKeydown_, false);\r\n\r\n }\r\n\r\n };\r\n\r\n inline.isLinkActive = function () {\r\n\r\n var isActive = false;\r\n\r\n editor.nodes.inlineToolbar.buttons.childNodes.forEach(function (tool) {\r\n\r\n var dataType = tool.dataset.type;\r\n\r\n if (dataType == 'link' && tool.classList.contains('hightlighted')) {\r\n\r\n isActive = true;\r\n\r\n }\r\n\r\n });\r\n\r\n return isActive;\r\n\r\n };\r\n\r\n /** default action behavior of tool */\r\n inline.defaultToolAction = function (type) {\r\n\r\n document.execCommand(type, false, null);\r\n\r\n };\r\n\r\n /**\r\n * @private\r\n *\r\n * Sets URL\r\n *\r\n * @param {String} url - URL\r\n */\r\n inline.setAnchor = function (url) {\r\n\r\n document.execCommand('createLink', false, url);\r\n\r\n /** Close after URL inserting */\r\n editor.toolbar.inline.closeAction();\r\n\r\n };\r\n\r\n /**\r\n * @private\r\n *\r\n * Saves selection\r\n */\r\n inline.saveSelection = function (containerEl) {\r\n\r\n var range = window.getSelection().getRangeAt(0),\r\n preSelectionRange = range.cloneRange(),\r\n start;\r\n\r\n preSelectionRange.selectNodeContents(containerEl);\r\n preSelectionRange.setEnd(range.startContainer, range.startOffset);\r\n\r\n start = preSelectionRange.toString().length;\r\n\r\n return {\r\n start: start,\r\n end: start + range.toString().length\r\n };\r\n\r\n };\r\n\r\n /**\r\n * @private\r\n *\r\n * Sets to previous selection (Range)\r\n *\r\n * @param {Element} containerEl - editable element where we restore range\r\n * @param {Object} savedSel - range basic information to restore\r\n */\r\n inline.restoreSelection = function (containerEl, savedSel) {\r\n\r\n var range = document.createRange(),\r\n charIndex = 0;\r\n\r\n range.setStart(containerEl, 0);\r\n range.collapse(true);\r\n\r\n var nodeStack = [ containerEl ],\r\n node,\r\n foundStart = false,\r\n stop = false,\r\n nextCharIndex;\r\n\r\n while (!stop && (node = nodeStack.pop())) {\r\n\r\n if (node.nodeType == 3) {\r\n\r\n nextCharIndex = charIndex + node.length;\r\n\r\n if (!foundStart && savedSel.start >= charIndex && savedSel.start <= nextCharIndex) {\r\n\r\n range.setStart(node, savedSel.start - charIndex);\r\n foundStart = true;\r\n\r\n }\r\n if (foundStart && savedSel.end >= charIndex && savedSel.end <= nextCharIndex) {\r\n\r\n range.setEnd(node, savedSel.end - charIndex);\r\n stop = true;\r\n\r\n }\r\n charIndex = nextCharIndex;\r\n\r\n } else {\r\n\r\n var i = node.childNodes.length;\r\n\r\n while (i--) {\r\n\r\n nodeStack.push(node.childNodes[i]);\r\n\r\n }\r\n\r\n }\r\n\r\n }\r\n\r\n var sel = window.getSelection();\r\n\r\n sel.removeAllRanges();\r\n sel.addRange(range);\r\n\r\n };\r\n\r\n /**\r\n * @private\r\n *\r\n * Removes all ranges from window selection\r\n */\r\n inline.clearRange = function () {\r\n\r\n var selection = window.getSelection();\r\n\r\n selection.removeAllRanges();\r\n\r\n };\r\n\r\n /**\r\n * @private\r\n *\r\n * sets or removes hightlight\r\n */\r\n inline.hightlight = function (tool) {\r\n\r\n var dataType = tool.dataset.type;\r\n\r\n if (document.queryCommandState(dataType)) {\r\n\r\n editor.toolbar.inline.setButtonHighlighted(tool);\r\n\r\n } else {\r\n\r\n editor.toolbar.inline.removeButtonsHighLight(tool);\r\n\r\n }\r\n\r\n /**\r\n *\r\n * hightlight for anchors\r\n */\r\n var selection = window.getSelection(),\r\n tag = selection.anchorNode.parentNode;\r\n\r\n if (tag.tagName == 'A' && dataType == 'link') {\r\n\r\n editor.toolbar.inline.setButtonHighlighted(tool);\r\n\r\n }\r\n\r\n };\r\n\r\n /**\r\n * @private\r\n *\r\n * Mark button if text is already executed\r\n */\r\n inline.setButtonHighlighted = function (button) {\r\n\r\n button.classList.add('hightlighted');\r\n\r\n /** At link tool we also change icon */\r\n if (button.dataset.type == 'link') {\r\n\r\n var icon = button.childNodes[0];\r\n\r\n icon.classList.remove('ce-icon-link');\r\n icon.classList.add('ce-icon-unlink');\r\n\r\n }\r\n\r\n };\r\n\r\n /**\r\n * @private\r\n *\r\n * Removes hightlight\r\n */\r\n inline.removeButtonsHighLight = function (button) {\r\n\r\n button.classList.remove('hightlighted');\r\n\r\n /** At link tool we also change icon */\r\n if (button.dataset.type == 'link') {\r\n\r\n var icon = button.childNodes[0];\r\n\r\n icon.classList.remove('ce-icon-unlink');\r\n icon.classList.add('ce-icon-link');\r\n\r\n }\r\n\r\n };\r\n\r\n\r\n return inline;\r\n\r\n})({});\n\n\n// WEBPACK FOOTER //\n// ./modules/toolbar/inline.js","/**\r\n * Codex Editor toolbox\r\n *\r\n * All tools be able to appended here\r\n *\r\n * @author Codex Team\r\n * @version 1.0\r\n */\r\n\r\nmodule.exports = (function (toolbox) {\r\n\r\n let editor = codex.editor;\r\n\r\n toolbox.opened = false;\r\n\r\n /** Shows toolbox */\r\n toolbox.open = function () {\r\n\r\n /** Close setting if toolbox is opened */\r\n if (editor.toolbar.settings.opened) {\r\n\r\n editor.toolbar.settings.close();\r\n\r\n }\r\n\r\n /** display toolbox */\r\n editor.nodes.toolbox.classList.add('opened');\r\n\r\n /** Animate plus button */\r\n editor.nodes.plusButton.classList.add('clicked');\r\n\r\n /** toolbox state */\r\n editor.toolbar.toolbox.opened = true;\r\n\r\n };\r\n\r\n /** Closes toolbox */\r\n toolbox.close = function () {\r\n\r\n /** Makes toolbox disapear */\r\n editor.nodes.toolbox.classList.remove('opened');\r\n\r\n /** Rotate plus button */\r\n editor.nodes.plusButton.classList.remove('clicked');\r\n\r\n /** toolbox state */\r\n editor.toolbar.toolbox.opened = false;\r\n\r\n editor.toolbar.current = null;\r\n\r\n };\r\n\r\n toolbox.leaf = function () {\r\n\r\n let currentTool = editor.toolbar.current,\r\n tools = Object.keys(editor.tools),\r\n barButtons = editor.nodes.toolbarButtons,\r\n nextToolIndex = 0,\r\n toolToSelect,\r\n visibleTool,\r\n tool;\r\n\r\n if ( !currentTool ) {\r\n\r\n /** Get first tool from object*/\r\n for(tool in editor.tools) {\r\n\r\n if (editor.tools[tool].displayInToolbox) {\r\n\r\n break;\r\n\r\n }\r\n\r\n nextToolIndex ++;\r\n\r\n }\r\n\r\n } else {\r\n\r\n nextToolIndex = (tools.indexOf(currentTool) + 1) % tools.length;\r\n visibleTool = tools[nextToolIndex];\r\n\r\n while (!editor.tools[visibleTool].displayInToolbox) {\r\n\r\n nextToolIndex = (nextToolIndex + 1) % tools.length;\r\n visibleTool = tools[nextToolIndex];\r\n\r\n }\r\n\r\n }\r\n\r\n toolToSelect = tools[nextToolIndex];\r\n\r\n for ( var button in barButtons ) {\r\n\r\n barButtons[button].classList.remove('selected');\r\n\r\n }\r\n\r\n barButtons[toolToSelect].classList.add('selected');\r\n editor.toolbar.current = toolToSelect;\r\n\r\n };\r\n\r\n /**\r\n * Transforming selected node type into selected toolbar element type\r\n * @param {event} event\r\n */\r\n toolbox.toolClicked = function (event) {\r\n\r\n /**\r\n * UNREPLACEBLE_TOOLS this types of tools are forbidden to replace even they are empty\r\n */\r\n var UNREPLACEBLE_TOOLS = ['image', 'link', 'list', 'instagram', 'twitter', 'embed'],\r\n tool = editor.tools[editor.toolbar.current],\r\n workingNode = editor.content.currentNode,\r\n currentInputIndex = editor.caret.inputIndex,\r\n newBlockContent,\r\n appendCallback,\r\n blockData;\r\n\r\n /** Make block from plugin */\r\n newBlockContent = tool.render();\r\n\r\n /** information about block */\r\n blockData = {\r\n block : newBlockContent,\r\n type : tool.type,\r\n stretched : false\r\n };\r\n\r\n if (\r\n workingNode &&\r\n UNREPLACEBLE_TOOLS.indexOf(workingNode.dataset.tool) === -1 &&\r\n workingNode.textContent.trim() === ''\r\n ) {\r\n\r\n /** Replace current block */\r\n editor.content.switchBlock(workingNode, newBlockContent, tool.type);\r\n\r\n } else {\r\n\r\n /** Insert new Block from plugin */\r\n editor.content.insertBlock(blockData);\r\n\r\n /** increase input index */\r\n currentInputIndex++;\r\n\r\n }\r\n\r\n /** Fire tool append callback */\r\n appendCallback = tool.appendCallback;\r\n\r\n if (appendCallback && typeof appendCallback == 'function') {\r\n\r\n appendCallback.call(event);\r\n\r\n }\r\n\r\n window.setTimeout(function () {\r\n\r\n /** Set caret to current block */\r\n editor.caret.setToBlock(currentInputIndex);\r\n\r\n }, 10);\r\n\r\n\r\n /**\r\n * Changing current Node\r\n */\r\n editor.content.workingNodeChanged();\r\n\r\n /**\r\n * Move toolbar when node is changed\r\n */\r\n editor.toolbar.move();\r\n\r\n };\r\n\r\n return toolbox;\r\n\r\n})({});\n\n\n// WEBPACK FOOTER //\n// ./modules/toolbar/toolbox.js","/**\r\n * @module Codex Editor Callbacks module\r\n * @description Module works with editor added Elements\r\n *\r\n * @author Codex Team\r\n * @version 1.4.0\r\n */\r\n\r\nmodule.exports = (function (callbacks) {\r\n\r\n let editor = codex.editor;\r\n\r\n /**\r\n * used by UI module\r\n * @description Routes all keydowns on document\r\n * @param {Object} event\r\n */\r\n callbacks.globalKeydown = function (event) {\r\n\r\n switch (event.keyCode) {\r\n case editor.core.keys.ENTER : enterKeyPressed_(event); break;\r\n }\r\n\r\n };\r\n\r\n /**\r\n * used by UI module\r\n * @description Routes all keydowns on redactors area\r\n * @param {Object} event\r\n */\r\n callbacks.redactorKeyDown = function (event) {\r\n\r\n switch (event.keyCode) {\r\n case editor.core.keys.TAB : tabKeyPressedOnRedactorsZone_(event); break;\r\n case editor.core.keys.ENTER : enterKeyPressedOnRedactorsZone_(event); break;\r\n case editor.core.keys.ESC : escapeKeyPressedOnRedactorsZone_(event); break;\r\n default : defaultKeyPressedOnRedactorsZone_(event); break;\r\n }\r\n\r\n };\r\n\r\n /**\r\n * used by UI module\r\n * @description Routes all keyup events\r\n * @param {Object} event\r\n */\r\n callbacks.globalKeyup = function (event) {\r\n\r\n switch (event.keyCode) {\r\n case editor.core.keys.UP :\r\n case editor.core.keys.LEFT :\r\n case editor.core.keys.RIGHT :\r\n case editor.core.keys.DOWN : arrowKeyPressed_(event); break;\r\n }\r\n\r\n };\r\n\r\n /**\r\n * @param {Object} event\r\n * @private\r\n *\r\n * Handles behaviour when tab pressed\r\n * @description if Content is empty show toolbox (if it is closed) or leaf tools\r\n * uses Toolbars toolbox module to handle the situation\r\n */\r\n var tabKeyPressedOnRedactorsZone_ = function (event) {\r\n\r\n /**\r\n * Wait for solution. Would like to know the behaviour\r\n * @todo Add spaces\r\n */\r\n event.preventDefault();\r\n\r\n var nativeInputs = editor.content.currentNode.querySelectorAll('textarea, input'),\r\n nativeInputsAreEmpty = true,\r\n textContentIsEmpty = !editor.content.currentNode.textContent.trim();\r\n\r\n Array.prototype.map.call(nativeInputs, function (input) {\r\n\r\n if (input.type == 'textarea' || input.type == 'text') {\r\n\r\n nativeInputsAreEmpty = nativeInputsAreEmpty && !input.value.trim();\r\n\r\n }\r\n\r\n });\r\n\r\n var blockIsEmpty = textContentIsEmpty && nativeInputsAreEmpty;\r\n\r\n if (!blockIsEmpty) {\r\n\r\n return;\r\n\r\n }\r\n\r\n if ( !editor.toolbar.opened ) {\r\n\r\n editor.toolbar.open();\r\n\r\n }\r\n\r\n if (editor.toolbar.opened && !editor.toolbar.toolbox.opened) {\r\n\r\n editor.toolbar.toolbox.open();\r\n\r\n } else {\r\n\r\n editor.toolbar.toolbox.leaf();\r\n\r\n }\r\n\r\n };\r\n\r\n /**\r\n * Handles global EnterKey Press\r\n * @see enterPressedOnBlock_\r\n * @param {Object} event\r\n */\r\n var enterKeyPressed_ = function () {\r\n\r\n if (editor.content.editorAreaHightlighted) {\r\n\r\n /**\r\n * it means that we lose input index, saved index before is not correct\r\n * therefore we need to set caret when we insert new block\r\n */\r\n editor.caret.inputIndex = -1;\r\n\r\n enterPressedOnBlock_();\r\n\r\n }\r\n\r\n };\r\n\r\n /**\r\n * Callback for enter key pressing in first-level block area\r\n *\r\n * @param {Event} event\r\n * @private\r\n *\r\n * @description Inserts new block with initial type from settings\r\n */\r\n var enterPressedOnBlock_ = function () {\r\n\r\n var NEW_BLOCK_TYPE = editor.settings.initialBlockPlugin;\r\n\r\n editor.content.insertBlock({\r\n type : NEW_BLOCK_TYPE,\r\n block : editor.tools[NEW_BLOCK_TYPE].render()\r\n }, true );\r\n\r\n editor.toolbar.move();\r\n editor.toolbar.open();\r\n\r\n };\r\n\r\n\r\n /**\r\n * ENTER key handler\r\n *\r\n * @param {Object} event\r\n * @private\r\n *\r\n * @description Makes new block with initial type from settings\r\n */\r\n var enterKeyPressedOnRedactorsZone_ = function (event) {\r\n\r\n if (event.target.contentEditable == 'true') {\r\n\r\n /** Update input index */\r\n editor.caret.saveCurrentInputIndex();\r\n\r\n }\r\n\r\n var currentInputIndex = editor.caret.getCurrentInputIndex() || 0,\r\n workingNode = editor.content.currentNode,\r\n tool = workingNode.dataset.tool,\r\n isEnterPressedOnToolbar = editor.toolbar.opened &&\r\n editor.toolbar.current &&\r\n event.target == editor.state.inputs[currentInputIndex];\r\n\r\n /** The list of tools which needs the default browser behaviour */\r\n var enableLineBreaks = editor.tools[tool].enableLineBreaks;\r\n\r\n /** This type of block creates when enter is pressed */\r\n var NEW_BLOCK_TYPE = editor.settings.initialBlockPlugin;\r\n\r\n /**\r\n * When toolbar is opened, select tool instead of making new paragraph\r\n */\r\n if ( isEnterPressedOnToolbar ) {\r\n\r\n event.preventDefault();\r\n\r\n editor.toolbar.toolbox.toolClicked(event);\r\n\r\n editor.toolbar.close();\r\n\r\n /**\r\n * Stop other listeners callback executions\r\n */\r\n event.stopPropagation();\r\n event.stopImmediatePropagation();\r\n\r\n return;\r\n\r\n }\r\n\r\n /**\r\n * Allow paragraph lineBreaks with shift enter\r\n * Or if shiftkey pressed and enter and enabledLineBreaks, the let new block creation\r\n */\r\n if ( event.shiftKey || enableLineBreaks ) {\r\n\r\n event.stopPropagation();\r\n event.stopImmediatePropagation();\r\n return;\r\n\r\n }\r\n\r\n var currentSelection = window.getSelection(),\r\n currentSelectedNode = currentSelection.anchorNode,\r\n caretAtTheEndOfText = editor.caret.position.atTheEnd(),\r\n isTextNodeHasParentBetweenContenteditable = false;\r\n\r\n /**\r\n * Allow making new

    in same block by SHIFT+ENTER and forbids to prevent default browser behaviour\r\n */\r\n if ( event.shiftKey && !enableLineBreaks ) {\r\n\r\n editor.callback.enterPressedOnBlock(editor.content.currentBlock, event);\r\n event.preventDefault();\r\n return;\r\n\r\n }\r\n\r\n /**\r\n * Workaround situation when caret at the Text node that has some wrapper Elements\r\n * Split block cant handle this.\r\n * We need to save default behavior\r\n */\r\n isTextNodeHasParentBetweenContenteditable = currentSelectedNode && currentSelectedNode.parentNode.contentEditable != 'true';\r\n\r\n /**\r\n * Split blocks when input has several nodes and caret placed in textNode\r\n */\r\n if (\r\n currentSelectedNode.nodeType == editor.core.nodeTypes.TEXT &&\r\n !isTextNodeHasParentBetweenContenteditable &&\r\n !caretAtTheEndOfText\r\n ) {\r\n\r\n event.preventDefault();\r\n\r\n editor.core.log('Splitting Text node...');\r\n\r\n editor.content.splitBlock(currentInputIndex);\r\n\r\n /** Show plus button when next input after split is empty*/\r\n if (!editor.state.inputs[currentInputIndex + 1].textContent.trim()) {\r\n\r\n editor.toolbar.showPlusButton();\r\n\r\n }\r\n\r\n } else {\r\n\r\n var islastNode = editor.content.isLastNode(currentSelectedNode);\r\n\r\n if ( islastNode && caretAtTheEndOfText ) {\r\n\r\n event.preventDefault();\r\n event.stopPropagation();\r\n event.stopImmediatePropagation();\r\n\r\n editor.core.log('ENTER clicked in last textNode. Create new BLOCK');\r\n\r\n editor.content.insertBlock({\r\n type: NEW_BLOCK_TYPE,\r\n block: editor.tools[NEW_BLOCK_TYPE].render()\r\n }, true);\r\n\r\n editor.toolbar.move();\r\n editor.toolbar.open();\r\n\r\n /** Show plus button with empty block */\r\n editor.toolbar.showPlusButton();\r\n\r\n }\r\n\r\n }\r\n\r\n /** get all inputs after new appending block */\r\n editor.ui.saveInputs();\r\n\r\n };\r\n\r\n /**\r\n * Escape behaviour\r\n * @param event\r\n * @private\r\n *\r\n * @description Closes toolbox and toolbar. Prevents default behaviour\r\n */\r\n var escapeKeyPressedOnRedactorsZone_ = function (event) {\r\n\r\n /** Close all toolbar */\r\n editor.toolbar.close();\r\n\r\n /** Close toolbox */\r\n editor.toolbar.toolbox.close();\r\n\r\n event.preventDefault();\r\n\r\n };\r\n\r\n /**\r\n * @param {Event} event\r\n * @private\r\n *\r\n * closes and moves toolbar\r\n */\r\n var arrowKeyPressed_ = function (event) {\r\n\r\n editor.content.workingNodeChanged();\r\n\r\n /* Closing toolbar */\r\n editor.toolbar.close();\r\n editor.toolbar.move();\r\n\r\n };\r\n\r\n /**\r\n * @private\r\n * @param {Event} event\r\n *\r\n * @description Closes all opened bars from toolbar.\r\n * If block is mark, clears highlightning\r\n */\r\n var defaultKeyPressedOnRedactorsZone_ = function () {\r\n\r\n editor.toolbar.close();\r\n\r\n if (!editor.toolbar.inline.actionsOpened) {\r\n\r\n editor.toolbar.inline.close();\r\n editor.content.clearMark();\r\n\r\n }\r\n\r\n };\r\n\r\n /**\r\n * Handler when clicked on redactors area\r\n *\r\n * @protected\r\n * @param event\r\n *\r\n * @description Detects clicked area. If it is first-level block area, marks as detected and\r\n * on next enter press will be inserted new block\r\n * Otherwise, save carets position (input index) and put caret to the editable zone.\r\n *\r\n * @see detectWhenClickedOnFirstLevelBlockArea_\r\n *\r\n */\r\n callbacks.redactorClicked = function (event) {\r\n\r\n detectWhenClickedOnFirstLevelBlockArea_();\r\n\r\n editor.content.workingNodeChanged(event.target);\r\n editor.ui.saveInputs();\r\n\r\n var selectedText = editor.toolbar.inline.getSelectionText(),\r\n firstLevelBlock;\r\n\r\n /** If selection range took off, then we hide inline toolbar */\r\n if (selectedText.length === 0) {\r\n\r\n editor.toolbar.inline.close();\r\n\r\n }\r\n\r\n /** Update current input index in memory when caret focused into existed input */\r\n if (event.target.contentEditable == 'true') {\r\n\r\n editor.caret.saveCurrentInputIndex();\r\n\r\n }\r\n\r\n if (editor.content.currentNode === null) {\r\n\r\n /**\r\n * If inputs in redactor does not exits, then we put input index 0 not -1\r\n */\r\n var indexOfLastInput = editor.state.inputs.length > 0 ? editor.state.inputs.length - 1 : 0;\r\n\r\n /** If we have any inputs */\r\n if (editor.state.inputs.length) {\r\n\r\n /** getting firstlevel parent of input */\r\n firstLevelBlock = editor.content.getFirstLevelBlock(editor.state.inputs[indexOfLastInput]);\r\n\r\n }\r\n\r\n /** If input is empty, then we set caret to the last input */\r\n if (editor.state.inputs.length && editor.state.inputs[indexOfLastInput].textContent === '' && firstLevelBlock.dataset.tool == editor.settings.initialBlockPlugin) {\r\n\r\n editor.caret.setToBlock(indexOfLastInput);\r\n\r\n } else {\r\n\r\n /** Create new input when caret clicked in redactors area */\r\n var NEW_BLOCK_TYPE = editor.settings.initialBlockPlugin;\r\n\r\n editor.content.insertBlock({\r\n type : NEW_BLOCK_TYPE,\r\n block : editor.tools[NEW_BLOCK_TYPE].render()\r\n });\r\n\r\n /** If there is no inputs except inserted */\r\n if (editor.state.inputs.length === 1) {\r\n\r\n editor.caret.setToBlock(indexOfLastInput);\r\n\r\n } else {\r\n\r\n /** Set caret to this appended input */\r\n editor.caret.setToNextBlock(indexOfLastInput);\r\n\r\n }\r\n\r\n }\r\n\r\n } else {\r\n\r\n /** Close all panels */\r\n editor.toolbar.settings.close();\r\n editor.toolbar.toolbox.close();\r\n\r\n }\r\n\r\n /**\r\n * Move toolbar and open\r\n */\r\n editor.toolbar.move();\r\n editor.toolbar.open();\r\n\r\n var inputIsEmpty = !editor.content.currentNode.textContent.trim(),\r\n currentNodeType = editor.content.currentNode.dataset.tool,\r\n isInitialType = currentNodeType == editor.settings.initialBlockPlugin;\r\n\r\n\r\n /** Hide plus buttons */\r\n editor.toolbar.hidePlusButton();\r\n\r\n if (!inputIsEmpty) {\r\n\r\n /** Mark current block */\r\n editor.content.markBlock();\r\n\r\n }\r\n\r\n if ( isInitialType && inputIsEmpty ) {\r\n\r\n /** Show plus button */\r\n editor.toolbar.showPlusButton();\r\n\r\n }\r\n\r\n\r\n };\r\n\r\n /**\r\n * This method allows to define, is caret in contenteditable element or not.\r\n *\r\n * @private\r\n *\r\n * @description Otherwise, if we get TEXT node from range container, that will means we have input index.\r\n * In this case we use default browsers behaviour (if plugin allows that) or overwritten action.\r\n * Therefore, to be sure that we've clicked first-level block area, we should have currentNode, which always\r\n * specifies to the first-level block. Other cases we just ignore.\r\n */\r\n var detectWhenClickedOnFirstLevelBlockArea_ = function () {\r\n\r\n var selection = window.getSelection(),\r\n anchorNode = selection.anchorNode,\r\n flag = false;\r\n\r\n if (selection.rangeCount === 0) {\r\n\r\n editor.content.editorAreaHightlighted = true;\r\n\r\n } else {\r\n\r\n if (!editor.core.isDomNode(anchorNode)) {\r\n\r\n anchorNode = anchorNode.parentNode;\r\n\r\n }\r\n\r\n /** Already founded, without loop */\r\n if (anchorNode.contentEditable == 'true') {\r\n\r\n flag = true;\r\n\r\n }\r\n\r\n while (anchorNode.contentEditable != 'true') {\r\n\r\n anchorNode = anchorNode.parentNode;\r\n\r\n if (anchorNode.contentEditable == 'true') {\r\n\r\n flag = true;\r\n\r\n }\r\n\r\n if (anchorNode == document.body) {\r\n\r\n break;\r\n\r\n }\r\n\r\n }\r\n\r\n /** If editable element founded, flag is \"TRUE\", Therefore we return \"FALSE\" */\r\n editor.content.editorAreaHightlighted = !flag;\r\n\r\n }\r\n\r\n };\r\n\r\n /**\r\n * Toolbar button click handler\r\n *\r\n * @param {Object} event - cursor to the button\r\n * @protected\r\n *\r\n * @description gets current tool and calls render method\r\n */\r\n callbacks.toolbarButtonClicked = function (event) {\r\n\r\n var button = this;\r\n\r\n editor.toolbar.current = button.dataset.type;\r\n\r\n editor.toolbar.toolbox.toolClicked(event);\r\n editor.toolbar.close();\r\n\r\n };\r\n\r\n /**\r\n * Show or Hide toolbox when plus button is clicked\r\n */\r\n callbacks.plusButtonClicked = function () {\r\n\r\n if (!editor.nodes.toolbox.classList.contains('opened')) {\r\n\r\n editor.toolbar.toolbox.open();\r\n\r\n } else {\r\n\r\n editor.toolbar.toolbox.close();\r\n\r\n }\r\n\r\n };\r\n\r\n /**\r\n * Block handlers for KeyDown events\r\n *\r\n * @protected\r\n * @param {Object} event\r\n *\r\n * Handles keydowns on block\r\n * @see blockRightOrDownArrowPressed_\r\n * @see backspacePressed_\r\n * @see blockLeftOrUpArrowPressed_\r\n */\r\n callbacks.blockKeydown = function (event) {\r\n\r\n let block = event.target; // event.target is input\r\n\r\n switch (event.keyCode) {\r\n\r\n case editor.core.keys.DOWN:\r\n case editor.core.keys.RIGHT:\r\n blockRightOrDownArrowPressed_(event);\r\n break;\r\n\r\n case editor.core.keys.BACKSPACE:\r\n backspacePressed_(block, event);\r\n break;\r\n\r\n case editor.core.keys.UP:\r\n case editor.core.keys.LEFT:\r\n blockLeftOrUpArrowPressed_(event);\r\n break;\r\n\r\n }\r\n\r\n };\r\n\r\n /**\r\n * RIGHT or DOWN keydowns on block\r\n *\r\n * @param {Object} event\r\n * @private\r\n *\r\n * @description watches the selection and gets closest editable element.\r\n * Uses method getDeepestTextNodeFromPosition to get the last node of next block\r\n * Sets caret if it is contenteditable\r\n */\r\n var blockRightOrDownArrowPressed_ = function (event) {\r\n\r\n var selection = window.getSelection(),\r\n inputs = editor.state.inputs,\r\n focusedNode = selection.anchorNode,\r\n focusedNodeHolder;\r\n\r\n /** Check for caret existance */\r\n if (!focusedNode) {\r\n\r\n return false;\r\n\r\n }\r\n\r\n /** Looking for closest (parent) contentEditable element of focused node */\r\n while (focusedNode.contentEditable != 'true') {\r\n\r\n focusedNodeHolder = focusedNode.parentNode;\r\n focusedNode = focusedNodeHolder;\r\n\r\n }\r\n\r\n /** Input index in DOM level */\r\n var editableElementIndex = 0;\r\n\r\n while (focusedNode != inputs[editableElementIndex]) {\r\n\r\n editableElementIndex ++;\r\n\r\n }\r\n\r\n /**\r\n * Founded contentEditable element doesn't have childs\r\n * Or maybe New created block\r\n */\r\n if (!focusedNode.textContent) {\r\n\r\n editor.caret.setToNextBlock(editableElementIndex);\r\n return;\r\n\r\n }\r\n\r\n /**\r\n * Do nothing when caret doesn not reaches the end of last child\r\n */\r\n var caretInLastChild = false,\r\n caretAtTheEndOfText = false;\r\n\r\n var lastChild,\r\n deepestTextnode;\r\n\r\n lastChild = focusedNode.childNodes[focusedNode.childNodes.length - 1 ];\r\n\r\n if (editor.core.isDomNode(lastChild)) {\r\n\r\n deepestTextnode = editor.content.getDeepestTextNodeFromPosition(lastChild, lastChild.childNodes.length);\r\n\r\n } else {\r\n\r\n deepestTextnode = lastChild;\r\n\r\n }\r\n\r\n caretInLastChild = selection.anchorNode == deepestTextnode;\r\n caretAtTheEndOfText = deepestTextnode.length == selection.anchorOffset;\r\n\r\n if ( !caretInLastChild || !caretAtTheEndOfText ) {\r\n\r\n editor.core.log('arrow [down|right] : caret does not reached the end');\r\n return false;\r\n\r\n }\r\n\r\n editor.caret.setToNextBlock(editableElementIndex);\r\n\r\n };\r\n\r\n /**\r\n * LEFT or UP keydowns on block\r\n *\r\n * @param {Object} event\r\n * @private\r\n *\r\n * watches the selection and gets closest editable element.\r\n * Uses method getDeepestTextNodeFromPosition to get the last node of previous block\r\n * Sets caret if it is contenteditable\r\n *\r\n */\r\n var blockLeftOrUpArrowPressed_ = function (event) {\r\n\r\n var selection = window.getSelection(),\r\n inputs = editor.state.inputs,\r\n focusedNode = selection.anchorNode,\r\n focusedNodeHolder;\r\n\r\n /** Check for caret existance */\r\n if (!focusedNode) {\r\n\r\n return false;\r\n\r\n }\r\n\r\n /**\r\n * LEFT or UP not at the beginning\r\n */\r\n if ( selection.anchorOffset !== 0) {\r\n\r\n return false;\r\n\r\n }\r\n\r\n /** Looking for parent contentEditable block */\r\n while (focusedNode.contentEditable != 'true') {\r\n\r\n focusedNodeHolder = focusedNode.parentNode;\r\n focusedNode = focusedNodeHolder;\r\n\r\n }\r\n\r\n /** Input index in DOM level */\r\n var editableElementIndex = 0;\r\n\r\n while (focusedNode != inputs[editableElementIndex]) {\r\n\r\n editableElementIndex ++;\r\n\r\n }\r\n\r\n /**\r\n * Do nothing if caret is not at the beginning of first child\r\n */\r\n var caretInFirstChild = false,\r\n caretAtTheBeginning = false;\r\n\r\n var firstChild,\r\n deepestTextnode;\r\n\r\n /**\r\n * Founded contentEditable element doesn't have childs\r\n * Or maybe New created block\r\n */\r\n if (!focusedNode.textContent) {\r\n\r\n editor.caret.setToPreviousBlock(editableElementIndex);\r\n return;\r\n\r\n }\r\n\r\n firstChild = focusedNode.childNodes[0];\r\n\r\n if (editor.core.isDomNode(firstChild)) {\r\n\r\n deepestTextnode = editor.content.getDeepestTextNodeFromPosition(firstChild, 0);\r\n\r\n } else {\r\n\r\n deepestTextnode = firstChild;\r\n\r\n }\r\n\r\n caretInFirstChild = selection.anchorNode == deepestTextnode;\r\n caretAtTheBeginning = selection.anchorOffset === 0;\r\n\r\n if ( caretInFirstChild && caretAtTheBeginning ) {\r\n\r\n editor.caret.setToPreviousBlock(editableElementIndex);\r\n\r\n }\r\n\r\n };\r\n\r\n /**\r\n * Handles backspace keydown\r\n *\r\n * @param {Element} block\r\n * @param {Object} event\r\n * @private\r\n *\r\n * @description if block is empty, delete the block and set caret to the previous block\r\n * If block is not empty, try to merge two blocks - current and previous\r\n * But it we try'n to remove first block, then we should set caret to the next block, not previous.\r\n * If we removed the last block, create new one\r\n */\r\n var backspacePressed_ = function (block, event) {\r\n\r\n var currentInputIndex = editor.caret.getCurrentInputIndex(),\r\n range,\r\n selectionLength,\r\n firstLevelBlocksCount;\r\n\r\n if (editor.core.isNativeInput(event.target)) {\r\n\r\n /** If input value is empty - remove block */\r\n if (event.target.value.trim() == '') {\r\n\r\n block.remove();\r\n\r\n } else {\r\n\r\n return;\r\n\r\n }\r\n\r\n }\r\n\r\n if (block.textContent.trim()) {\r\n\r\n range = editor.content.getRange();\r\n selectionLength = range.endOffset - range.startOffset;\r\n\r\n if (editor.caret.position.atStart() && !selectionLength && editor.state.inputs[currentInputIndex - 1]) {\r\n\r\n editor.content.mergeBlocks(currentInputIndex);\r\n\r\n } else {\r\n\r\n return;\r\n\r\n }\r\n\r\n }\r\n\r\n if (!selectionLength) {\r\n\r\n block.remove();\r\n\r\n }\r\n\r\n\r\n firstLevelBlocksCount = editor.nodes.redactor.childNodes.length;\r\n\r\n /**\r\n * If all blocks are removed\r\n */\r\n if (firstLevelBlocksCount === 0) {\r\n\r\n /** update currentNode variable */\r\n editor.content.currentNode = null;\r\n\r\n /** Inserting new empty initial block */\r\n editor.ui.addInitialBlock();\r\n\r\n /** Updating inputs state after deleting last block */\r\n editor.ui.saveInputs();\r\n\r\n /** Set to current appended block */\r\n window.setTimeout(function () {\r\n\r\n editor.caret.setToPreviousBlock(1);\r\n\r\n }, 10);\r\n\r\n } else {\r\n\r\n if (editor.caret.inputIndex !== 0) {\r\n\r\n /** Target block is not first */\r\n editor.caret.setToPreviousBlock(editor.caret.inputIndex);\r\n\r\n } else {\r\n\r\n /** If we try to delete first block */\r\n editor.caret.setToNextBlock(editor.caret.inputIndex);\r\n\r\n }\r\n\r\n }\r\n\r\n editor.toolbar.move();\r\n\r\n if (!editor.toolbar.opened) {\r\n\r\n editor.toolbar.open();\r\n\r\n }\r\n\r\n /** Updating inputs state */\r\n editor.ui.saveInputs();\r\n\r\n /** Prevent default browser behaviour */\r\n event.preventDefault();\r\n\r\n };\r\n\r\n /**\r\n * used by UI module\r\n * Clicks on block settings button\r\n *\r\n * @param {Object} event\r\n * @protected\r\n * @description Opens toolbar settings\r\n */\r\n callbacks.showSettingsButtonClicked = function (event) {\r\n\r\n /**\r\n * Get type of current block\r\n * It uses to append settings from tool.settings property.\r\n * ...\r\n * Type is stored in data-type attribute on block\r\n */\r\n var currentToolType = editor.content.currentNode.dataset.tool;\r\n\r\n editor.toolbar.settings.toggle(currentToolType);\r\n\r\n /** Close toolbox when settings button is active */\r\n editor.toolbar.toolbox.close();\r\n editor.toolbar.settings.hideRemoveActions();\r\n\r\n };\r\n\r\n return callbacks;\r\n\r\n})({});\n\n\n// WEBPACK FOOTER //\n// ./modules/callbacks.js","/**\r\n * Codex Editor Draw module\r\n *\r\n * @author Codex Team\r\n * @version 1.0.\r\n */\r\n\r\nmodule.exports = (function (draw) {\r\n\r\n /**\r\n * Base editor wrapper\r\n */\r\n draw.wrapper = function () {\r\n\r\n var wrapper = document.createElement('div');\r\n\r\n wrapper.className += 'codex-editor';\r\n\r\n return wrapper;\r\n\r\n };\r\n\r\n /**\r\n * Content-editable holder\r\n */\r\n draw.redactor = function () {\r\n\r\n var redactor = document.createElement('div');\r\n\r\n redactor.className += 'ce-redactor';\r\n\r\n return redactor;\r\n\r\n };\r\n\r\n draw.ceBlock = function () {\r\n\r\n var block = document.createElement('DIV');\r\n\r\n block.className += 'ce_block';\r\n\r\n return block;\r\n\r\n };\r\n\r\n /**\r\n * Empty toolbar with toggler\r\n */\r\n draw.toolbar = function () {\r\n\r\n var bar = document.createElement('div');\r\n\r\n bar.className += 'ce-toolbar';\r\n\r\n return bar;\r\n\r\n };\r\n\r\n draw.toolbarContent = function () {\r\n\r\n var wrapper = document.createElement('DIV');\r\n\r\n wrapper.classList.add('ce-toolbar__content');\r\n\r\n return wrapper;\r\n\r\n };\r\n\r\n /**\r\n * Inline toolbar\r\n */\r\n draw.inlineToolbar = function () {\r\n\r\n var bar = document.createElement('DIV');\r\n\r\n bar.className += 'ce-toolbar-inline';\r\n\r\n return bar;\r\n\r\n };\r\n\r\n /**\r\n * Wrapper for inline toobar buttons\r\n */\r\n draw.inlineToolbarButtons = function () {\r\n\r\n var wrapper = document.createElement('DIV');\r\n\r\n wrapper.className += 'ce-toolbar-inline__buttons';\r\n\r\n return wrapper;\r\n\r\n };\r\n\r\n /**\r\n * For some actions\r\n */\r\n draw.inlineToolbarActions = function () {\r\n\r\n var wrapper = document.createElement('DIV');\r\n\r\n wrapper.className += 'ce-toolbar-inline__actions';\r\n\r\n return wrapper;\r\n\r\n };\r\n\r\n draw.inputForLink = function () {\r\n\r\n var input = document.createElement('INPUT');\r\n\r\n input.type = 'input';\r\n input.className += 'inputForLink';\r\n input.placeholder = 'Вставьте ссылку ...';\r\n input.setAttribute('form', 'defaultForm');\r\n\r\n input.setAttribute('autofocus', 'autofocus');\r\n\r\n return input;\r\n\r\n };\r\n\r\n /**\r\n * @todo Desc\r\n */\r\n draw.blockButtons = function () {\r\n\r\n var block = document.createElement('div');\r\n\r\n block.className += 'ce-toolbar__actions';\r\n\r\n return block;\r\n\r\n };\r\n\r\n /**\r\n * Block settings panel\r\n */\r\n draw.blockSettings = function () {\r\n\r\n var settings = document.createElement('div');\r\n\r\n settings.className += 'ce-settings';\r\n\r\n return settings;\r\n\r\n };\r\n\r\n draw.defaultSettings = function () {\r\n\r\n var div = document.createElement('div');\r\n\r\n div.classList.add('ce-settings_default');\r\n\r\n return div;\r\n\r\n };\r\n\r\n draw.pluginsSettings = function () {\r\n\r\n var div = document.createElement('div');\r\n\r\n div.classList.add('ce-settings_plugin');\r\n\r\n return div;\r\n\r\n };\r\n\r\n draw.plusButton = function () {\r\n\r\n var button = document.createElement('span');\r\n\r\n button.className = 'ce-toolbar__plus';\r\n // button.innerHTML = '';\r\n\r\n return button;\r\n\r\n };\r\n\r\n /**\r\n * Settings button in toolbar\r\n */\r\n draw.settingsButton = function () {\r\n\r\n var toggler = document.createElement('span');\r\n\r\n toggler.className = 'ce-toolbar__settings-btn';\r\n\r\n /** Toggler button*/\r\n toggler.innerHTML = '';\r\n\r\n return toggler;\r\n\r\n };\r\n\r\n /**\r\n * Redactor tools wrapper\r\n */\r\n\r\n draw.toolbox = function () {\r\n\r\n var wrapper = document.createElement('div');\r\n\r\n wrapper.className = 'ce-toolbar__tools';\r\n\r\n return wrapper;\r\n\r\n };\r\n\r\n /**\r\n * @protected\r\n *\r\n * Draws tool buttons for toolbox\r\n *\r\n * @param {String} type\r\n * @param {String} classname\r\n * @returns {Element}\r\n */\r\n draw.toolbarButton = function (type, classname) {\r\n\r\n var button = document.createElement('li'),\r\n toolIcon = document.createElement('i'),\r\n toolTitle = document.createElement('span');\r\n\r\n button.dataset.type = type;\r\n button.setAttribute('title', type);\r\n\r\n toolIcon.classList.add(classname);\r\n toolTitle.classList.add('ce_toolbar_tools--title');\r\n\r\n\r\n button.appendChild(toolIcon);\r\n button.appendChild(toolTitle);\r\n\r\n return button;\r\n\r\n };\r\n\r\n /**\r\n * @protected\r\n *\r\n * Draws tools for inline toolbar\r\n *\r\n * @param {String} type\r\n * @param {String} classname\r\n */\r\n draw.toolbarButtonInline = function (type, classname) {\r\n\r\n var button = document.createElement('BUTTON'),\r\n toolIcon = document.createElement('I');\r\n\r\n button.type = 'button';\r\n button.dataset.type = type;\r\n toolIcon.classList.add(classname);\r\n\r\n button.appendChild(toolIcon);\r\n\r\n return button;\r\n\r\n };\r\n\r\n /**\r\n * Redactor block\r\n */\r\n draw.block = function (tagName, content) {\r\n\r\n var node = document.createElement(tagName);\r\n\r\n node.innerHTML = content || '';\r\n\r\n return node;\r\n\r\n };\r\n\r\n /**\r\n * Creates Node with passed tagName and className\r\n * @param {string} tagName\r\n * @param {string} className\r\n * @param {object} properties - allow to assign properties\r\n */\r\n draw.node = function ( tagName, className, properties ) {\r\n\r\n var el = document.createElement( tagName );\r\n\r\n if ( className ) el.className = className;\r\n\r\n if ( properties ) {\r\n\r\n for (var name in properties) {\r\n\r\n el[name] = properties[name];\r\n\r\n }\r\n\r\n }\r\n\r\n return el;\r\n\r\n };\r\n\r\n /**\r\n * Unavailable plugin block\r\n */\r\n draw.unavailableBlock = function () {\r\n\r\n var wrapper = document.createElement('DIV');\r\n\r\n wrapper.classList.add('cdx-unavailable-block');\r\n\r\n return wrapper;\r\n\r\n };\r\n\r\n return draw;\r\n\r\n})({});\n\n\n// WEBPACK FOOTER //\n// ./modules/draw.js","/**\r\n * Codex Editor Caret Module\r\n *\r\n * @author Codex Team\r\n * @version 1.0\r\n */\r\n\r\nmodule.exports = (function (caret) {\r\n\r\n let editor = codex.editor;\r\n\r\n /**\r\n * @var {int} InputIndex - editable element in DOM\r\n */\r\n caret.inputIndex = null;\r\n\r\n /**\r\n * @var {int} offset - caret position in a text node.\r\n */\r\n caret.offset = null;\r\n\r\n /**\r\n * @var {int} focusedNodeIndex - we get index of child node from first-level block\r\n */\r\n caret.focusedNodeIndex = null;\r\n\r\n /**\r\n * Creates Document Range and sets caret to the element.\r\n * @protected\r\n * @uses caret.save — if you need to save caret position\r\n * @param {Element} el - Changed Node.\r\n */\r\n caret.set = function ( el, index, offset) {\r\n\r\n offset = offset || caret.offset || 0;\r\n index = index || caret.focusedNodeIndex || 0;\r\n\r\n var childs = el.childNodes,\r\n nodeToSet;\r\n\r\n if ( childs.length === 0 ) {\r\n\r\n nodeToSet = el;\r\n\r\n } else {\r\n\r\n nodeToSet = childs[index];\r\n\r\n }\r\n\r\n /** If Element is INPUT */\r\n if (el.contentEditable != 'true') {\r\n\r\n el.focus();\r\n return;\r\n\r\n }\r\n\r\n if (editor.core.isDomNode(nodeToSet)) {\r\n\r\n nodeToSet = editor.content.getDeepestTextNodeFromPosition(nodeToSet, nodeToSet.childNodes.length);\r\n\r\n }\r\n\r\n var range = document.createRange(),\r\n selection = window.getSelection();\r\n\r\n window.setTimeout(function () {\r\n\r\n range.setStart(nodeToSet, offset);\r\n range.setEnd(nodeToSet, offset);\r\n\r\n selection.removeAllRanges();\r\n selection.addRange(range);\r\n\r\n editor.caret.saveCurrentInputIndex();\r\n\r\n }, 20);\r\n\r\n };\r\n\r\n /**\r\n * @protected\r\n * Updates index of input and saves it in caret object\r\n */\r\n caret.saveCurrentInputIndex = function () {\r\n\r\n /** Index of Input that we paste sanitized content */\r\n var selection = window.getSelection(),\r\n inputs = editor.state.inputs,\r\n focusedNode = selection.anchorNode,\r\n focusedNodeHolder;\r\n\r\n if (!focusedNode) {\r\n\r\n return;\r\n\r\n }\r\n\r\n /** Looking for parent contentEditable block */\r\n while (focusedNode.contentEditable != 'true') {\r\n\r\n focusedNodeHolder = focusedNode.parentNode;\r\n focusedNode = focusedNodeHolder;\r\n\r\n }\r\n\r\n /** Input index in DOM level */\r\n var editableElementIndex = 0;\r\n\r\n while (focusedNode != inputs[editableElementIndex]) {\r\n\r\n editableElementIndex ++;\r\n\r\n }\r\n\r\n caret.inputIndex = editableElementIndex;\r\n\r\n };\r\n\r\n /**\r\n * Returns current input index (caret object)\r\n */\r\n caret.getCurrentInputIndex = function () {\r\n\r\n return caret.inputIndex;\r\n\r\n };\r\n\r\n /**\r\n * @param {int} index - index of first-level block after that we set caret into next input\r\n */\r\n caret.setToNextBlock = function (index) {\r\n\r\n var inputs = editor.state.inputs,\r\n nextInput = inputs[index + 1];\r\n\r\n if (!nextInput) {\r\n\r\n editor.core.log('We are reached the end');\r\n return;\r\n\r\n }\r\n\r\n /**\r\n * When new Block created or deleted content of input\r\n * We should add some text node to set caret\r\n */\r\n if (!nextInput.childNodes.length) {\r\n\r\n var emptyTextElement = document.createTextNode('');\r\n\r\n nextInput.appendChild(emptyTextElement);\r\n\r\n }\r\n\r\n editor.caret.inputIndex = index + 1;\r\n editor.caret.set(nextInput, 0, 0);\r\n editor.content.workingNodeChanged(nextInput);\r\n\r\n };\r\n\r\n /**\r\n * @param {int} index - index of target input.\r\n * Sets caret to input with this index\r\n */\r\n caret.setToBlock = function (index) {\r\n\r\n var inputs = editor.state.inputs,\r\n targetInput = inputs[index];\r\n\r\n if ( !targetInput ) {\r\n\r\n return;\r\n\r\n }\r\n\r\n /**\r\n * When new Block created or deleted content of input\r\n * We should add some text node to set caret\r\n */\r\n if (!targetInput.childNodes.length) {\r\n\r\n var emptyTextElement = document.createTextNode('');\r\n\r\n targetInput.appendChild(emptyTextElement);\r\n\r\n }\r\n\r\n editor.caret.inputIndex = index;\r\n editor.caret.set(targetInput, 0, 0);\r\n editor.content.workingNodeChanged(targetInput);\r\n\r\n };\r\n\r\n /**\r\n * @param {int} index - index of input\r\n */\r\n caret.setToPreviousBlock = function (index) {\r\n\r\n index = index || 0;\r\n\r\n var inputs = editor.state.inputs,\r\n previousInput = inputs[index - 1],\r\n lastChildNode,\r\n lengthOfLastChildNode,\r\n emptyTextElement;\r\n\r\n\r\n if (!previousInput) {\r\n\r\n editor.core.log('We are reached first node');\r\n return;\r\n\r\n }\r\n\r\n lastChildNode = editor.content.getDeepestTextNodeFromPosition(previousInput, previousInput.childNodes.length);\r\n lengthOfLastChildNode = lastChildNode.length;\r\n\r\n /**\r\n * When new Block created or deleted content of input\r\n * We should add some text node to set caret\r\n */\r\n if (!previousInput.childNodes.length) {\r\n\r\n emptyTextElement = document.createTextNode('');\r\n previousInput.appendChild(emptyTextElement);\r\n\r\n }\r\n editor.caret.inputIndex = index - 1;\r\n editor.caret.set(previousInput, previousInput.childNodes.length - 1, lengthOfLastChildNode);\r\n editor.content.workingNodeChanged(inputs[index - 1]);\r\n\r\n };\r\n\r\n caret.position = {\r\n\r\n atStart : function () {\r\n\r\n var selection = window.getSelection(),\r\n anchorOffset = selection.anchorOffset,\r\n anchorNode = selection.anchorNode,\r\n firstLevelBlock = editor.content.getFirstLevelBlock(anchorNode),\r\n pluginsRender = firstLevelBlock.childNodes[0];\r\n\r\n if (!editor.core.isDomNode(anchorNode)) {\r\n\r\n anchorNode = anchorNode.parentNode;\r\n\r\n }\r\n\r\n var isFirstNode = anchorNode === pluginsRender.childNodes[0],\r\n isOffsetZero = anchorOffset === 0;\r\n\r\n return isFirstNode && isOffsetZero;\r\n\r\n },\r\n\r\n atTheEnd : function () {\r\n\r\n var selection = window.getSelection(),\r\n anchorOffset = selection.anchorOffset,\r\n anchorNode = selection.anchorNode;\r\n\r\n /** Caret is at the end of input */\r\n return !anchorNode || !anchorNode.length || anchorOffset === anchorNode.length;\r\n\r\n }\r\n };\r\n\r\n\r\n /**\r\n * Inserts node at the caret location\r\n * @param node\r\n */\r\n caret.insertNode = function (node) {\r\n\r\n var selection, range;\r\n\r\n selection = window.getSelection();\r\n\r\n range = selection.getRangeAt(0);\r\n range.deleteContents();\r\n\r\n range.insertNode(node);\r\n\r\n range.setStartAfter(node);\r\n range.collapse(true);\r\n\r\n selection.removeAllRanges();\r\n selection.addRange(range);\r\n\r\n\r\n };\r\n\r\n return caret;\r\n\r\n})({});\n\n\n// WEBPACK FOOTER //\n// ./modules/caret.js","/**\r\n * Codex Editor Notification Module\r\n *\r\n * @author Codex Team\r\n * @version 1.0\r\n */\r\n\r\nmodule.exports = (function (notifications) {\r\n\r\n let editor = codex.editor;\r\n\r\n var queue = [];\r\n\r\n var addToQueue = function (settings) {\r\n\r\n queue.push(settings);\r\n\r\n var index = 0;\r\n\r\n while ( index < queue.length && queue.length > 5) {\r\n\r\n if (queue[index].type == 'confirm' || queue[index].type == 'prompt') {\r\n\r\n index++;\r\n continue;\r\n\r\n }\r\n\r\n queue[index].close();\r\n queue.splice(index, 1);\r\n\r\n }\r\n\r\n };\r\n\r\n notifications.createHolder = function () {\r\n\r\n var holder = editor.draw.node('DIV', 'cdx-notifications-block');\r\n\r\n editor.nodes.notifications = document.body.appendChild(holder);\r\n\r\n return holder;\r\n\r\n };\r\n\r\n\r\n /**\r\n * Error notificator. Shows block with message\r\n * @protected\r\n */\r\n notifications.errorThrown = function (errorMsg, event) {\r\n\r\n editor.notifications.notification({message: 'This action is not available currently', type: event.type});\r\n\r\n };\r\n\r\n /**\r\n *\r\n * Appends notification\r\n *\r\n * settings = {\r\n * type - notification type (reserved types: alert, confirm, prompt). Just add class 'cdx-notification-'+type\r\n * message - notification message\r\n * okMsg - confirm button text (default - 'Ok')\r\n * cancelBtn - cancel button text (default - 'Cancel'). Only for confirm and prompt types\r\n * confirm - function-handler for ok button click\r\n * cancel - function-handler for cancel button click. Only for confirm and prompt types\r\n * time - time (in seconds) after which notification will close (default - 10s)\r\n * }\r\n *\r\n * @param settings\r\n */\r\n notifications.notification = function (constructorSettings) {\r\n\r\n /** Private vars and methods */\r\n var notification = null,\r\n cancel = null,\r\n type = null,\r\n confirm = null,\r\n inputField = null;\r\n\r\n var confirmHandler = function () {\r\n\r\n close();\r\n\r\n if (typeof confirm !== 'function' ) {\r\n\r\n return;\r\n\r\n }\r\n\r\n if (type == 'prompt') {\r\n\r\n confirm(inputField.value);\r\n return;\r\n\r\n }\r\n\r\n confirm();\r\n\r\n };\r\n\r\n var cancelHandler = function () {\r\n\r\n close();\r\n\r\n if (typeof cancel !== 'function' ) {\r\n\r\n return;\r\n\r\n }\r\n\r\n cancel();\r\n\r\n };\r\n\r\n\r\n /** Public methods */\r\n function create(settings) {\r\n\r\n if (!(settings && settings.message)) {\r\n\r\n editor.core.log('Can\\'t create notification. Message is missed');\r\n return;\r\n\r\n }\r\n\r\n settings.type = settings.type || 'alert';\r\n settings.time = settings.time*1000 || 10000;\r\n\r\n var wrapper = editor.draw.node('DIV', 'cdx-notification'),\r\n message = editor.draw.node('DIV', 'cdx-notification__message'),\r\n input = editor.draw.node('INPUT', 'cdx-notification__input'),\r\n okBtn = editor.draw.node('SPAN', 'cdx-notification__ok-btn'),\r\n cancelBtn = editor.draw.node('SPAN', 'cdx-notification__cancel-btn');\r\n\r\n message.textContent = settings.message;\r\n okBtn.textContent = settings.okMsg || 'ОК';\r\n cancelBtn.textContent = settings.cancelMsg || 'Отмена';\r\n\r\n editor.listeners.add(okBtn, 'click', confirmHandler);\r\n editor.listeners.add(cancelBtn, 'click', cancelHandler);\r\n\r\n wrapper.appendChild(message);\r\n\r\n if (settings.type == 'prompt') {\r\n\r\n wrapper.appendChild(input);\r\n\r\n }\r\n\r\n wrapper.appendChild(okBtn);\r\n\r\n if (settings.type == 'prompt' || settings.type == 'confirm') {\r\n\r\n wrapper.appendChild(cancelBtn);\r\n\r\n }\r\n\r\n wrapper.classList.add('cdx-notification-' + settings.type);\r\n wrapper.dataset.type = settings.type;\r\n\r\n notification = wrapper;\r\n type = settings.type;\r\n confirm = settings.confirm;\r\n cancel = settings.cancel;\r\n inputField = input;\r\n\r\n if (settings.type != 'prompt' && settings.type != 'confirm') {\r\n\r\n window.setTimeout(close, settings.time);\r\n\r\n }\r\n\r\n };\r\n\r\n function send() {\r\n\r\n editor.nodes.notifications.appendChild(notification);\r\n inputField.focus();\r\n\r\n editor.nodes.notifications.classList.add('cdx-notification__notification-appending');\r\n\r\n window.setTimeout(function () {\r\n\r\n editor.nodes.notifications.classList.remove('cdx-notification__notification-appending');\r\n\r\n }, 100);\r\n\r\n addToQueue({type: type, close: close});\r\n\r\n };\r\n\r\n function close() {\r\n\r\n notification.remove();\r\n\r\n };\r\n\r\n\r\n if (constructorSettings) {\r\n\r\n create(constructorSettings);\r\n send();\r\n\r\n }\r\n\r\n return {\r\n create: create,\r\n send: send,\r\n close: close\r\n };\r\n\r\n };\r\n\r\n notifications.clear = function () {\r\n\r\n editor.nodes.notifications.innerHTML = '';\r\n queue = [];\r\n\r\n };\r\n\r\n return notifications;\r\n\r\n})({});\n\n\n// WEBPACK FOOTER //\n// ./modules/notifications.js","/**\r\n * Codex Editor Parser Module\r\n *\r\n * @author Codex Team\r\n * @version 1.1\r\n */\r\n\r\nmodule.exports = (function (parser) {\r\n\r\n let editor = codex.editor;\r\n\r\n /** inserting text */\r\n parser.insertPastedContent = function (blockType, tag) {\r\n\r\n editor.content.insertBlock({\r\n type : blockType.type,\r\n block : blockType.render({\r\n text : tag.innerHTML\r\n })\r\n });\r\n\r\n };\r\n\r\n /**\r\n * Check DOM node for display style: separated block or child-view\r\n */\r\n parser.isFirstLevelBlock = function (node) {\r\n\r\n return node.nodeType == editor.core.nodeTypes.TAG &&\r\n node.classList.contains(editor.ui.className.BLOCK_CLASSNAME);\r\n\r\n };\r\n\r\n return parser;\r\n\r\n})({});\r\n\n\n\n// WEBPACK FOOTER //\n// ./modules/parser.js","/**\r\n * Codex Sanitizer\r\n */\r\n\r\nmodule.exports = (function (sanitizer) {\r\n\r\n /** HTML Janitor library */\r\n let janitor = require('html-janitor');\r\n\r\n /** Codex Editor */\r\n let editor = codex.editor;\r\n\r\n sanitizer.prepare = function () {\r\n\r\n if (editor.settings.sanitizer && !editor.core.isEmpty(editor.settings.sanitizer)) {\r\n\r\n Config.CUSTOM = editor.settings.sanitizer;\r\n\r\n }\r\n\r\n };\r\n\r\n /**\r\n * Basic config\r\n */\r\n var Config = {\r\n\r\n /** User configuration */\r\n CUSTOM : null,\r\n\r\n BASIC : {\r\n\r\n tags: {\r\n p: {},\r\n a: {\r\n href: true,\r\n target: '_blank',\r\n rel: 'nofollow'\r\n },\r\n i: {},\r\n b: {},\r\n strong: {},\r\n em: {},\r\n span: {}\r\n }\r\n }\r\n };\r\n\r\n sanitizer.Config = Config;\r\n\r\n /**\r\n *\r\n * @param userCustomConfig\r\n * @returns {*}\r\n * @private\r\n *\r\n * @description If developer uses editor's API, then he can customize sane restrictions.\r\n * Or, sane config can be defined globally in editors initialization. That config will be used everywhere\r\n * At least, if there is no config overrides, that API uses BASIC Default configation\r\n */\r\n let init_ = function (userCustomConfig) {\r\n\r\n let configuration = userCustomConfig || Config.CUSTOM || Config.BASIC;\r\n\r\n return new janitor(configuration);\r\n\r\n };\r\n\r\n /**\r\n * Cleans string from unwanted tags\r\n * @protected\r\n * @param {String} dirtyString - taint string\r\n * @param {Object} customConfig - allowed tags\r\n */\r\n sanitizer.clean = function (dirtyString, customConfig) {\r\n\r\n let janitorInstance = init_(customConfig);\r\n\r\n return janitorInstance.clean(dirtyString);\r\n\r\n };\r\n\r\n return sanitizer;\r\n\r\n})({});\n\n\n// WEBPACK FOOTER //\n// ./modules/sanitizer.js","(function (root, factory) {\n if (typeof define === 'function' && define.amd) {\n define('html-janitor', factory);\n } else if (typeof exports === 'object') {\n module.exports = factory();\n } else {\n root.HTMLJanitor = factory();\n }\n}(this, function () {\n\n /**\n * @param {Object} config.tags Dictionary of allowed tags.\n * @param {boolean} config.keepNestedBlockElements Default false.\n */\n function HTMLJanitor(config) {\n\n var tagDefinitions = config['tags'];\n var tags = Object.keys(tagDefinitions);\n\n var validConfigValues = tags\n .map(function(k) { return typeof tagDefinitions[k]; })\n .every(function(type) { return type === 'object' || type === 'boolean' || type === 'function'; });\n\n if(!validConfigValues) {\n throw new Error(\"The configuration was invalid\");\n }\n\n this.config = config;\n }\n\n // TODO: not exhaustive?\n var blockElementNames = ['P', 'LI', 'TD', 'TH', 'DIV', 'H1', 'H2', 'H3', 'H4', 'H5', 'H6', 'PRE'];\n function isBlockElement(node) {\n return blockElementNames.indexOf(node.nodeName) !== -1;\n }\n\n var inlineElementNames = ['A', 'B', 'STRONG', 'I', 'EM', 'SUB', 'SUP', 'U', 'STRIKE'];\n function isInlineElement(node) {\n return inlineElementNames.indexOf(node.nodeName) !== -1;\n }\n\n HTMLJanitor.prototype.clean = function (html) {\n var sandbox = document.createElement('div');\n sandbox.innerHTML = html;\n\n this._sanitize(sandbox);\n\n return sandbox.innerHTML;\n };\n\n HTMLJanitor.prototype._sanitize = function (parentNode) {\n var treeWalker = createTreeWalker(parentNode);\n var node = treeWalker.firstChild();\n if (!node) { return; }\n\n do {\n // Ignore nodes that have already been sanitized\n if (node._sanitized) {\n continue;\n }\n\n if (node.nodeType === Node.TEXT_NODE) {\n // If this text node is just whitespace and the previous or next element\n // sibling is a block element, remove it\n // N.B.: This heuristic could change. Very specific to a bug with\n // `contenteditable` in Firefox: http://jsbin.com/EyuKase/1/edit?js,output\n // FIXME: make this an option?\n if (node.data.trim() === ''\n && ((node.previousElementSibling && isBlockElement(node.previousElementSibling))\n || (node.nextElementSibling && isBlockElement(node.nextElementSibling)))) {\n parentNode.removeChild(node);\n this._sanitize(parentNode);\n break;\n } else {\n continue;\n }\n }\n\n // Remove all comments\n if (node.nodeType === Node.COMMENT_NODE) {\n parentNode.removeChild(node);\n this._sanitize(parentNode);\n break;\n }\n\n var isInline = isInlineElement(node);\n var containsBlockElement;\n if (isInline) {\n containsBlockElement = Array.prototype.some.call(node.childNodes, isBlockElement);\n }\n\n // Block elements should not be nested (e.g.

  • ...); if\n // they are, we want to unwrap the inner block element.\n var isNotTopContainer = !! parentNode.parentNode;\n var isNestedBlockElement =\n isBlockElement(parentNode) &&\n isBlockElement(node) &&\n isNotTopContainer;\n\n var nodeName = node.nodeName.toLowerCase();\n\n var allowedAttrs = getAllowedAttrs(this.config, nodeName, node);\n\n var isInvalid = isInline && containsBlockElement;\n\n // Drop tag entirely according to the whitelist *and* if the markup\n // is invalid.\n if (isInvalid || shouldRejectNode(node, allowedAttrs)\n || (!this.config.keepNestedBlockElements && isNestedBlockElement)) {\n // Do not keep the inner text of SCRIPT/STYLE elements.\n if (! (node.nodeName === 'SCRIPT' || node.nodeName === 'STYLE')) {\n while (node.childNodes.length > 0) {\n parentNode.insertBefore(node.childNodes[0], node);\n }\n }\n parentNode.removeChild(node);\n\n this._sanitize(parentNode);\n break;\n }\n\n // Sanitize attributes\n for (var a = 0; a < node.attributes.length; a += 1) {\n var attr = node.attributes[a];\n\n if (shouldRejectAttr(attr, allowedAttrs, node)) {\n node.removeAttribute(attr.name);\n // Shift the array to continue looping.\n a = a - 1;\n }\n }\n\n // Sanitize children\n this._sanitize(node);\n\n // Mark node as sanitized so it's ignored in future runs\n node._sanitized = true;\n } while ((node = treeWalker.nextSibling()));\n };\n\n function createTreeWalker(node) {\n return document.createTreeWalker(node,\n NodeFilter.SHOW_TEXT | NodeFilter.SHOW_ELEMENT | NodeFilter.SHOW_COMMENT,\n null, false);\n }\n\n function getAllowedAttrs(config, nodeName, node){\n if (typeof config.tags[nodeName] === 'function') {\n return config.tags[nodeName](node);\n } else {\n return config.tags[nodeName];\n }\n }\n\n function shouldRejectNode(node, allowedAttrs){\n if (typeof allowedAttrs === 'undefined') {\n return true;\n } else if (typeof allowedAttrs === 'boolean') {\n return !allowedAttrs;\n }\n\n return false;\n }\n\n function shouldRejectAttr(attr, allowedAttrs, node){\n var attrName = attr.name.toLowerCase();\n\n if (allowedAttrs === true){\n return false;\n } else if (typeof allowedAttrs[attrName] === 'function'){\n return !allowedAttrs[attrName](attr.value, node);\n } else if (typeof allowedAttrs[attrName] === 'undefined'){\n return true;\n } else if (allowedAttrs[attrName] === false) {\n return true;\n } else if (typeof allowedAttrs[attrName] === 'string') {\n return (allowedAttrs[attrName] !== attr.value);\n }\n\n return false;\n }\n\n return HTMLJanitor;\n\n}));\n\n\n\n//////////////////\n// WEBPACK FOOTER\n// ./~/html-janitor/src/html-janitor.js\n// module id = 18\n// module chunks = 0","/**\r\n * Codex Editor Listeners module\r\n *\r\n * @author Codex Team\r\n * @version 1.0\r\n */\r\n\r\n/**\r\n * Module-decorator for event listeners assignment\r\n */\r\nmodule.exports = function (listeners) {\r\n\r\n var allListeners = [];\r\n\r\n /**\r\n * Search methods\r\n *\r\n * byElement, byType and byHandler returns array of suitable listeners\r\n * one and all takes element, eventType, and handler and returns first (all) suitable listener\r\n *\r\n */\r\n listeners.search = function () {\r\n\r\n var byElement = function (element, context) {\r\n\r\n var listenersOnElement = [];\r\n\r\n context = context || allListeners;\r\n\r\n for (var i = 0; i < context.length; i++) {\r\n\r\n var listener = context[i];\r\n\r\n if (listener.element === element) {\r\n\r\n listenersOnElement.push(listener);\r\n\r\n }\r\n\r\n }\r\n\r\n return listenersOnElement;\r\n\r\n };\r\n\r\n var byType = function (eventType, context) {\r\n\r\n var listenersWithType = [];\r\n\r\n context = context || allListeners;\r\n\r\n for (var i = 0; i < context.length; i++) {\r\n\r\n var listener = context[i];\r\n\r\n if (listener.type === eventType) {\r\n\r\n listenersWithType.push(listener);\r\n\r\n }\r\n\r\n }\r\n\r\n return listenersWithType;\r\n\r\n };\r\n\r\n var byHandler = function (handler, context) {\r\n\r\n var listenersWithHandler = [];\r\n\r\n context = context || allListeners;\r\n\r\n for (var i = 0; i < context.length; i++) {\r\n\r\n var listener = context[i];\r\n\r\n if (listener.handler === handler) {\r\n\r\n listenersWithHandler.push(listener);\r\n\r\n }\r\n\r\n }\r\n\r\n return listenersWithHandler;\r\n\r\n };\r\n\r\n var one = function (element, eventType, handler) {\r\n\r\n var result = allListeners;\r\n\r\n if (element)\r\n result = byElement(element, result);\r\n\r\n if (eventType)\r\n result = byType(eventType, result);\r\n\r\n if (handler)\r\n result = byHandler(handler, result);\r\n\r\n return result[0];\r\n\r\n };\r\n\r\n var all = function (element, eventType, handler) {\r\n\r\n var result = allListeners;\r\n\r\n if (element)\r\n result = byElement(element, result);\r\n\r\n if (eventType)\r\n result = byType(eventType, result);\r\n\r\n if (handler)\r\n result = byHandler(handler, result);\r\n\r\n return result;\r\n\r\n };\r\n\r\n return {\r\n byElement : byElement,\r\n byType : byType,\r\n byHandler : byHandler,\r\n one : one,\r\n all : all\r\n };\r\n\r\n }();\r\n\r\n listeners.add = function (element, eventType, handler, isCapture) {\r\n\r\n element.addEventListener(eventType, handler, isCapture);\r\n\r\n var data = {\r\n element: element,\r\n type: eventType,\r\n handler: handler\r\n };\r\n\r\n var alreadyAddedListener = listeners.search.one(element, eventType, handler);\r\n\r\n if (!alreadyAddedListener) {\r\n\r\n allListeners.push(data);\r\n\r\n }\r\n\r\n };\r\n\r\n listeners.remove = function (element, eventType, handler) {\r\n\r\n element.removeEventListener(eventType, handler);\r\n\r\n var existingListeners = listeners.search.all(element, eventType, handler);\r\n\r\n for (var i = 0; i < existingListeners.length; i++) {\r\n\r\n var index = allListeners.indexOf(existingListeners[i]);\r\n\r\n if (index > 0) {\r\n\r\n allListeners.splice(index, 1);\r\n\r\n }\r\n\r\n }\r\n\r\n };\r\n\r\n listeners.removeAll = function () {\r\n\r\n allListeners.map(function (current) {\r\n\r\n listeners.remove(current.element, current.type, current.handler);\r\n\r\n });\r\n\r\n };\r\n\r\n listeners.get = function (element, eventType, handler) {\r\n\r\n return listeners.search.all(element, eventType, handler);\r\n\r\n };\r\n\r\n return listeners;\r\n\r\n}({});\n\n\n// WEBPACK FOOTER //\n// ./modules/listeners.js","/**\r\n * Codex Editor Destroyer module\r\n *\r\n * @auhor Codex Team\r\n * @version 1.0\r\n */\r\n\r\nmodule.exports = function (destroyer) {\r\n\r\n let editor = codex.editor;\r\n\r\n destroyer.removeNodes = function () {\r\n\r\n editor.nodes.wrapper.remove();\r\n editor.nodes.notifications.remove();\r\n\r\n };\r\n\r\n destroyer.destroyPlugins = function () {\r\n\r\n for (var tool in editor.tools) {\r\n\r\n if (typeof editor.tools[tool].destroy === 'function') {\r\n\r\n editor.tools[tool].destroy();\r\n\r\n }\r\n\r\n }\r\n\r\n };\r\n\r\n destroyer.destroyScripts = function () {\r\n\r\n var scripts = document.getElementsByTagName('SCRIPT');\r\n\r\n for (var i = 0; i < scripts.length; i++) {\r\n\r\n if (scripts[i].id.indexOf(editor.scriptPrefix) + 1) {\r\n\r\n scripts[i].remove();\r\n i--;\r\n\r\n }\r\n\r\n }\r\n\r\n };\r\n\r\n\r\n /**\r\n * Delete editor data from webpage.\r\n * You should send settings argument with boolean flags:\r\n * @param settings.ui- remove redactor event listeners and DOM nodes\r\n * @param settings.scripts - remove redactor scripts from DOM\r\n * @param settings.plugins - remove plugin's objects\r\n * @param settings.core - remove editor core. You can remove core only if UI and scripts flags is true\r\n * }\r\n *\r\n */\r\n destroyer.destroy = function (settings) {\r\n\r\n if (!settings || typeof settings !== 'object') {\r\n\r\n return;\r\n\r\n }\r\n\r\n if (settings.ui) {\r\n\r\n destroyer.removeNodes();\r\n editor.listeners.removeAll();\r\n\r\n }\r\n\r\n if (settings.scripts) {\r\n\r\n destroyer.destroyScripts();\r\n\r\n }\r\n\r\n if (settings.plugins) {\r\n\r\n destroyer.destroyPlugins();\r\n\r\n }\r\n\r\n if (settings.ui && settings.scripts && settings.core) {\r\n\r\n delete codex.editor;\r\n\r\n }\r\n\r\n };\r\n\r\n return destroyer;\r\n\r\n}({});\n\n\n// WEBPACK FOOTER //\n// ./modules/destroyer.js","/**\r\n * Codex Editor Paste module\r\n *\r\n * @author Codex Team\r\n * @version 1.1.1\r\n */\r\n\r\nmodule.exports = function (paste) {\r\n\r\n let editor = codex.editor;\r\n\r\n var patterns = [];\r\n\r\n paste.prepare = function () {\r\n\r\n var tools = editor.tools;\r\n\r\n for (var tool in tools) {\r\n\r\n if (!tools[tool].renderOnPastePatterns || !Array.isArray(tools[tool].renderOnPastePatterns)) {\r\n\r\n continue;\r\n\r\n }\r\n\r\n tools[tool].renderOnPastePatterns.map(function (pattern) {\r\n\r\n\r\n patterns.push(pattern);\r\n\r\n });\r\n\r\n }\r\n\r\n return Promise.resolve();\r\n\r\n };\r\n\r\n /**\r\n * Saves data\r\n * @param event\r\n */\r\n paste.pasted = function (event) {\r\n\r\n var clipBoardData = event.clipboardData || window.clipboardData,\r\n content = clipBoardData.getData('Text');\r\n\r\n var result = analize(content);\r\n\r\n if (result) {\r\n\r\n event.preventDefault();\r\n event.stopImmediatePropagation();\r\n\r\n }\r\n\r\n return result;\r\n\r\n };\r\n\r\n /**\r\n * Analizes pated string and calls necessary method\r\n */\r\n\r\n var analize = function (string) {\r\n\r\n var result = false,\r\n content = editor.content.currentNode,\r\n plugin = content.dataset.tool;\r\n\r\n patterns.map( function (pattern) {\r\n\r\n var execArray = pattern.regex.exec(string),\r\n match = execArray && execArray[0];\r\n\r\n if ( match && match === string.trim()) {\r\n\r\n /** current block is not empty */\r\n if ( content.textContent.trim() && plugin == editor.settings.initialBlockPlugin ) {\r\n\r\n pasteToNewBlock_();\r\n\r\n }\r\n\r\n pattern.callback(string, pattern);\r\n result = true;\r\n\r\n }\r\n\r\n });\r\n\r\n return result;\r\n\r\n };\r\n\r\n var pasteToNewBlock_ = function () {\r\n\r\n /** Create new initial block */\r\n editor.content.insertBlock({\r\n\r\n type : editor.settings.initialBlockPlugin,\r\n block : editor.tools[editor.settings.initialBlockPlugin].render({\r\n text : ''\r\n })\r\n\r\n }, false);\r\n\r\n };\r\n\r\n /**\r\n * This method prevents default behaviour.\r\n *\r\n * @param {Object} event\r\n * @protected\r\n *\r\n * @description We get from clipboard pasted data, sanitize, make a fragment that contains of this sanitized nodes.\r\n * Firstly, we need to memorize the caret position. We can do that by getting the range of selection.\r\n * After all, we insert clear fragment into caret placed position. Then, we should move the caret to the last node\r\n */\r\n paste.blockPasteCallback = function (event) {\r\n\r\n\r\n if (!needsToHandlePasteEvent(event.target)) {\r\n\r\n return;\r\n\r\n }\r\n\r\n /** Prevent default behaviour */\r\n event.preventDefault();\r\n\r\n /** get html pasted data - dirty data */\r\n var htmlData = event.clipboardData.getData('text/html'),\r\n plainData = event.clipboardData.getData('text/plain');\r\n\r\n /** Temporary DIV that is used to work with text's paragraphs as DOM-elements*/\r\n var paragraphs = editor.draw.node('DIV', '', {}),\r\n cleanData,\r\n wrappedData;\r\n\r\n /** Create fragment, that we paste to range after proccesing */\r\n cleanData = editor.sanitizer.clean(htmlData);\r\n\r\n /**\r\n * We wrap pasted text with

    tags to split it logically\r\n * @type {string}\r\n */\r\n wrappedData = editor.content.wrapTextWithParagraphs(cleanData, plainData);\r\n paragraphs.innerHTML = wrappedData;\r\n\r\n /**\r\n * If there only one paragraph, lets user agent paste it\r\n */\r\n if (paragraphs.childNodes.length == 1) {\r\n\r\n var newNode,\r\n paragraph = paragraphs.firstChild;\r\n\r\n if (paragraph.childElementCount) {\r\n\r\n newNode = editor.draw.node('SPAN', '', {innerHTML: paragraph.innerHTML.trim()});\r\n\r\n } else {\r\n\r\n newNode = document.createTextNode(paragraph.textContent);\r\n\r\n }\r\n\r\n editor.caret.insertNode(newNode);\r\n return;\r\n\r\n }\r\n\r\n insertPastedParagraphs(paragraphs.childNodes);\r\n\r\n };\r\n\r\n /**\r\n * Checks if we should handle paste event on block\r\n * @param block\r\n *\r\n * @return {boolean}\r\n */\r\n var needsToHandlePasteEvent = function (block) {\r\n\r\n /** If area is input or textarea then allow default behaviour */\r\n if ( editor.core.isNativeInput(block) ) {\r\n\r\n return false;\r\n\r\n }\r\n\r\n var editableParent = editor.content.getEditableParent(block);\r\n\r\n /** Allow paste when event target placed in Editable element */\r\n if (!editableParent) {\r\n\r\n return false;\r\n\r\n }\r\n\r\n return true;\r\n\r\n };\r\n\r\n var insertPastedParagraphs = function (paragraphs) {\r\n\r\n var NEW_BLOCK_TYPE = editor.settings.initialBlockPlugin,\r\n currentBlockContent = editor.content.currentNode.firstChild.firstChild;\r\n\r\n paragraphs.forEach(function (paragraph, index) {\r\n\r\n /** Don't allow empty paragraphs */\r\n if (paragraph.innerHTML.trim() === '') {\r\n\r\n return;\r\n\r\n }\r\n\r\n /**\r\n * If there was no data in working node, replace it with first paragraph of pasted text\r\n */\r\n if (index == 0 && currentBlockContent.innerHTML.trim() === '') {\r\n\r\n editor.content.switchBlock(editor.content.currentNode, editor.tools[NEW_BLOCK_TYPE].render({\r\n text : paragraph.innerHTML\r\n }), NEW_BLOCK_TYPE);\r\n\r\n return;\r\n\r\n }\r\n\r\n editor.content.insertBlock({\r\n type : NEW_BLOCK_TYPE,\r\n block : editor.tools[NEW_BLOCK_TYPE].render({\r\n text : paragraph.innerHTML\r\n })\r\n });\r\n\r\n editor.caret.inputIndex++;\r\n\r\n });\r\n\r\n editor.caret.setToPreviousBlock(editor.caret.getCurrentInputIndex() + 1);\r\n\r\n\r\n };\r\n\r\n\r\n return paste;\r\n\r\n}({});\n\n\n// WEBPACK FOOTER //\n// ./modules/paste.js"],"sourceRoot":""} \ No newline at end of file diff --git a/modules/content.js b/modules/content.js index 4513a878..7171c975 100644 --- a/modules/content.js +++ b/modules/content.js @@ -647,7 +647,7 @@ module.exports = (function (content) { if (!htmlData) { - return '

    ' + plainData.split('\n\n').join('

    ') + '

    '; + return wrapPlainTextWithParagraphs(plainData); } @@ -713,6 +713,17 @@ module.exports = (function (content) { }; + /** + * Splits strings on new line and wraps paragraphs with

    tag + * @param plainText + * @returns {string} + */ + var wrapPlainTextWithParagraphs = function (plainText) { + + return '

    ' + plainText.split('\n\n').join('

    ') + '

    '; + + }; + /** * Finds closest Contenteditable parent from Element * @param {Element} node element looking from diff --git a/modules/paste.js b/modules/paste.js index d9082ce5..6a070005 100644 --- a/modules/paste.js +++ b/modules/paste.js @@ -126,6 +126,9 @@ module.exports = function (paste) { } + /** Prevent default behaviour */ + event.preventDefault(); + /** get html pasted data - dirty data */ var htmlData = event.clipboardData.getData('text/html'), plainData = event.clipboardData.getData('text/plain'); @@ -150,13 +153,24 @@ module.exports = function (paste) { */ if (paragraphs.childNodes.length == 1) { + var newNode, + paragraph = paragraphs.firstChild; + + if (paragraph.childElementCount) { + + newNode = editor.draw.node('SPAN', '', {innerHTML: paragraph.innerHTML.trim()}); + + } else { + + newNode = document.createTextNode(paragraph.textContent); + + } + + editor.caret.insertNode(newNode); return; } - /** Prevent default behaviour */ - event.preventDefault(); - insertPastedParagraphs(paragraphs.childNodes); }; @@ -196,6 +210,13 @@ module.exports = function (paste) { paragraphs.forEach(function (paragraph, index) { + /** Don't allow empty paragraphs */ + if (paragraph.innerHTML.trim() === '') { + + return; + + } + /** * If there was no data in working node, replace it with first paragraph of pasted text */ From de9572064d4c995c6a4a088047ec199c47d3c04a Mon Sep 17 00:00:00 2001 From: George Berezhnoy Date: Mon, 24 Apr 2017 03:16:42 +0300 Subject: [PATCH 05/12] Remove personality from example.html --- example.html | 23 ----------------------- 1 file changed, 23 deletions(-) diff --git a/example.html b/example.html index 911821ee..aa76ff86 100644 --- a/example.html +++ b/example.html @@ -222,21 +222,6 @@ maxSize: 50000, } }, - personality: { - type : 'personality', - displayInToolbox : true, - iconClassname : 'cdx-personality-icon', - prepare : cdxEditorPersonality.prepare, - render : cdxEditorPersonality.render, - save : cdxEditorPersonality.save, - validate : cdxEditorPersonality.validate, - destroy : cdxEditorPersonality.destroy, - enableLineBreaks : true, - showInlineToolbar: true, - config: { - uploadURL: '/uploadPhoto', - } - } }, data : { items: [ @@ -252,14 +237,6 @@ text : 'Пишите нам на team@ifmo.su' } }, - { - type : 'personality', - data : { - name : 'Красюк Светлана Ивановна', - cite : 'Заместитель директора по учебно-воспитательной работе (начальная школа)', - url : 'http://new.school332.ru/user/2' - } - }, { type : 'list', data : { From f8dda6c65d213ee3ac51561d96c8b52efb532f7a Mon Sep 17 00:00:00 2001 From: George Berezhnoy Date: Mon, 24 Apr 2017 03:17:01 +0300 Subject: [PATCH 06/12] Finally remove personality from example.html --- example.html | 3 --- 1 file changed, 3 deletions(-) diff --git a/example.html b/example.html index aa76ff86..896abed5 100644 --- a/example.html +++ b/example.html @@ -56,9 +56,6 @@ - - -