From d2e755086a75734202d79385a077bdbbe9420ce4 Mon Sep 17 00:00:00 2001 From: George Berezhnoy Date: Mon, 13 Feb 2017 20:54:18 +0300 Subject: [PATCH] Destroy module (#157) * listeners module added * Destroy method added * Destroy method for plugins added * Delete plugins properties from window obj * Revert "Delete plugins properties from window obj" This reverts commit 6c91f81229f65e396d0e6d0cecadd8cd31bd4953. * Twitter and instagram api's destroy * Scripts destoy added * Fix * Replace regex with String.indexOf * Settings for destroyer --- .eslintrc | 3 +- codex-editor.js | 4 +- codex-editor.js.map | 2 +- codex.js | 3 + example.html | 10 ++ modules/callbacks.js | 3 +- modules/caret.js | 3 +- modules/content.js | 3 +- modules/core.js | 10 +- modules/destroyer.js | 98 +++++++++++++++++ modules/listeners.js | 192 +++++++++++++++++++++++++++++++++ modules/notifications.js | 4 +- modules/parser.js | 3 +- modules/paste.js | 1 - modules/renderer.js | 4 +- modules/sanitizer.js | 4 +- modules/saver.js | 4 +- modules/toolbar/inline.js | 6 +- modules/toolbar/settings.js | 20 ++-- modules/toolbar/toolbar.js | 4 +- modules/toolbar/toolbox.js | 4 +- modules/transport.js | 5 +- modules/ui.js | 26 ++--- package.json | 2 +- plugins/code/code.js | 16 ++- plugins/embed/embed.js | 49 +++++---- plugins/header/header.js | 20 ++-- plugins/image/image.js | 52 +++++---- plugins/instagram/instagram.js | 27 +++-- plugins/link/link.js | 24 +++-- plugins/list/list.js | 21 ++-- plugins/paragraph/paragraph.js | 20 ++-- plugins/quote/quote.js | 30 +++--- plugins/twitter/twitter.js | 25 +++-- whatwg-fetch.js.map | 2 +- 35 files changed, 535 insertions(+), 169 deletions(-) create mode 100644 modules/destroyer.js create mode 100644 modules/listeners.js diff --git a/.eslintrc b/.eslintrc index d53f9594..c06c48f7 100644 --- a/.eslintrc +++ b/.eslintrc @@ -70,7 +70,8 @@ "MutationObserver": true, "FormData": true, "XMLHttpRequest": true, - "ActiveXObject": true + "ActiveXObject": true, + "RegExp": true } } \ No newline at end of file diff --git a/codex-editor.js b/codex-editor.js index b70d493c..4a21be2e 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.4.10";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.anchors=n(19),e.paste=n(20)};return e.settings={tools:["paragraph","header","picture","list","quote","code","twitter","instagram","smile"],textareaId:"codex-editor",uploadImagesUrl:"/editor/transport/",initialBlockPlugin:"paragraph"},e.nodes={textarea: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.make).then(e.ui.addTools).then(e.ui.bindEvents).then(e.tools.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},o=codex.editor;e.exports=function(e){return e.prepare=function(e){return new Promise(function(t,r){e&&(o.settings.tools=e.tools||o.settings.tools),e.data&&(o.state.blocks=e.data),e.initialBlockPlugin&&(o.settings.initialBlockPlugin=e.initialBlockPlugin),e.uploadImagesUrl&&(o.settings.uploadImagesUrl=e.uploadImagesUrl),o.nodes.textarea=document.getElementById(e.textareaId||o.settings.textareaId),void 0===n(o.nodes.textarea)||null===o.nodes.textarea?r(Error("Textarea wasn't found by ID: #"+e.textareaId)):t()})},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=window.XMLHttpRequest?new XMLHttpRequest:new ActiveXObject("Microsoft.XMLHTTP"),o=function(){},r="";if(e.async=!0,e.type=e.type||"GET",e.data=e.data||"",e["content-type"]=e["content-type"]||"application/json; charset=utf-8",o=e.success||o,"GET"==e.type&&e.data)e.url=/\?/.test(e.url)?e.url+"&"+e.data:e.url+"?"+e.data;else for(t in e.data)r+=t+"="+encodeURIComponent(e.data[t])+"&";e.withCredentials&&(n.withCredentials=!0),e.beforeSend&&"function"==typeof e.beforeSend&&e.beforeSend.call(),n.open(e.type,e.url,e.async),n.setRequestHeader("X-Requested-With","XMLHttpRequest"),n.setRequestHeader("Content-type","application/x-www-form-urlencoded"),n.onreadystatechange=function(){4==n.readyState&&200==n.status&&o(n.responseText)},n.send(r)}},e.importScript=function(e,t){return new Promise(function(n,o){var r="cdx-script-",i=void 0;t?document.getElementById(r+t)&&n(e):o("Instance name is missed"),i=document.createElement("SCRIPT"),i.async=!0,i.defer=!0,i.id=r+t,i.onload=function(){n(e)},i.onerror=function(){o(e)},i.src=e,document.head.appendChild(i)})},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";var n=codex.editor;e.exports=function(e){return e.className={BLOCK_CLASSNAME:"ce-block",BLOCK_CONTENT:"ce-block__content",BLOCK_STRETCHED:"ce-block--stretched",BLOCK_HIGHLIGHTED:"ce-block--focused",BLOCK_IN_FEED_MODE:"ce-block--feed-mode",BLOCK_WITH_ANCHOR:"ce-block--anchor",SETTINGS_ITEM:"ce-settings__item"},e.make=function(){var e,t,o,r,i,a,s,c,l,d;e=n.draw.wrapper(),n.core.insertAfter(n.nodes.textarea,e),n.notifications.createHolder(),t=n.draw.toolbar(),o=n.draw.toolbarContent(),d=n.draw.plusButton(),s=n.draw.settingsButton(),c=n.toolbar.settings.makeRemoveBlockButton(),a=n.draw.blockSettings(),i=n.draw.blockButtons(),l=n.draw.toolbox(),r=n.draw.redactor();var u=n.draw.defaultSettings(),p=n.draw.pluginsSettings();a.appendChild(p),a.appendChild(u),i.appendChild(s),i.appendChild(c),i.appendChild(a),o.appendChild(d),o.appendChild(l),t.appendChild(i),t.appendChild(o),e.appendChild(t),e.appendChild(r),n.nodes.wrapper=e,n.nodes.toolbar=t,n.nodes.plusButton=d,n.nodes.toolbox=l,n.nodes.blockSettings=a,n.nodes.pluginSettings=p,n.nodes.defaultSettings=u,n.nodes.showSettingsButton=s,n.nodes.showTrashButton=c,n.nodes.redactor=r,n.ui.makeInlineToolbar(),n.toolbar.settings.addDefaultSettings()},e.makeInlineToolbar=function(){var e=n.draw.inlineToolbar();n.nodes.inlineToolbar.wrapper=e,n.nodes.inlineToolbar.buttons=n.draw.inlineToolbarButtons(),n.nodes.inlineToolbar.actions=n.draw.inlineToolbarActions(),n.nodes.inlineToolbar.wrapper.appendChild(n.nodes.inlineToolbar.buttons),n.nodes.inlineToolbar.wrapper.appendChild(n.nodes.inlineToolbar.actions),n.nodes.wrapper.appendChild(n.nodes.inlineToolbar.wrapper)},e.addTools=function(){var e,t,o;for(t in n.settings.tools)e=n.settings.tools[t],n.tools[t]=e,e.iconClassname?"function"==typeof e.render?e.displayInToolbox&&(o=n.draw.toolbarButton(t,e.iconClassname),n.nodes.toolbox.appendChild(o),n.nodes.toolbarButtons[t]=o):n.core.log("render method missed. Tool %o skipped","warn",t):n.core.log("Toolbar icon classname missed. Tool %o skipped","warn",t);n.ui.addInlineToolbarTools()},e.addInlineToolbarTools=function(){var e,t,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)t=o[r],e=n.draw.toolbarButtonInline(r,t.icon),n.nodes.inlineToolbar.buttons.appendChild(e),n.ui.setInlineToolbarButtonBehaviour(e,t.command)},e.bindEvents=function(){n.core.log("ui.bindEvents fired","info"),document.addEventListener("keydown",n.callback.globalKeydown,!1),n.nodes.redactor.addEventListener("keydown",n.callback.redactorKeyDown,!1),document.addEventListener("keyup",n.callback.globalKeyup,!1),n.nodes.redactor.addEventListener("click",n.callback.redactorClicked,!1),n.nodes.plusButton.addEventListener("click",n.callback.plusButtonClicked,!1),n.nodes.showSettingsButton.addEventListener("click",n.callback.showSettingsButtonClicked,!1);for(var e in n.nodes.toolbarButtons)n.nodes.toolbarButtons[e].addEventListener("click",n.callback.toolbarButtonClicked,!1)},e.addBlockHandlers=function(e){e&&(e.addEventListener("keydown",n.callback.blockKeydown,!1),e.addEventListener("paste",n.callback.blockPasteCallback,!1),e.addEventListener("mouseup",n.toolbar.inline.show,!1))},e.saveInputs=function(){var e=n.nodes.redactor;n.state.inputs=e.querySelectorAll("[contenteditable], input")},e.addInitialBlock=function(){var e,t=n.settings.initialBlockPlugin;return n.tools[t]?(e=n.tools[t].render(),e.setAttribute("data-placeholder","Расскажите свою историю..."),n.content.insertBlock({type:t,block:e}),void n.content.workingNodeChanged(e)):void n.core.log("Plugin %o was not implemented and can't be used as initial block","warn",t)},e.setInlineToolbarButtonBehaviour=function(e,t){e.addEventListener("mousedown",function(e){n.toolbar.inline.toolClicked(e,t)},!1)},e}({})},function(e,t){"use strict";var n=codex.editor;e.exports=function(e){return e.input=null,e.arguments=null,e.prepare=function(){var e=document.createElement("INPUT");e.type="file",e.addEventListener("change",n.transport.fileSelected),n.transport.input=e},e.clearInput=function(){this.input=null,this.prepare()},e.fileSelected=function(){var e=this,t=e.files,o=new FormData;o.append("files",t[0],t[0].name),n.transport.ajax({data:o,beforeSend:n.transport.arguments.beforeSend,success:n.transport.arguments.success,error:n.transport.arguments.error})},e.selectAndUpload=function(e){this.arguments=e,this.input.click()},e.ajax=function(e){var t=new XMLHttpRequest,o="function"==typeof e.beforeSend?e.beforeSend:function(){},r="function"==typeof e.success?e.success:function(){},i="function"==typeof e.error?e.error:function(){};o(),t.open("POST",n.settings.uploadImagesUrl,!0),t.setRequestHeader("X-Requested-With","XMLHttpRequest"),t.onload=function(){200===t.status?r(t.responseText):(n.core.log("request error: %o",t),i())},t.send(e.data),this.clearInput()},e}({})},function(e,t){"use strict";var n=codex.editor;e.exports=function(e){return e.makeBlocksFromData=function(){return n.core.isEmpty(n.state.blocks)||!n.state.blocks.items.length?void n.ui.addInitialBlock():void Promise.resolve().then(function(){return n.state.blocks}).then(n.renderer.appendBlocks).catch(function(e){n.core.log("Error while parsing JSON: %o","error",e)})},e.appendBlocks=function(e){for(var t=e.items,o=Promise.resolve(),r=0;rНе выводить в ленте'}:{innerHTML:'Вывести в ленте'},e=n.draw.node("DIV",n.ui.className.SETTINGS_ITEM,t),e.addEventListener("click",n.toolbar.settings.updateFeedMode,!1),e},e.updateFeedMode=function(){var e=n.content.currentNode;e.classList.toggle(n.ui.className.BLOCK_IN_FEED_MODE),n.toolbar.settings.close()},e.isFeedModeActivated=function(){var e=n.content.currentNode;return!!e&&e.classList.contains(n.ui.className.BLOCK_IN_FEED_MODE)},e.makeAnchorInput=function(){var e=n.draw.node("div","ce-settings__anchor-wrapper ce-settings__item",{}),t=n.draw.node("i","ce-settings__anchor-hash",{}),o=n.draw.node("input","ce-settings__anchor-input",{placeholder:"Якорь"});return o.addEventListener("keydown",n.anchors.keyDownOnAnchorInput),o.addEventListener("keyup",n.anchors.keyUpOnAnchorInput),o.addEventListener("input",n.anchors.anchorChanged),o.addEventListener("blur",n.anchors.anchorChanged),e.appendChild(t),e.appendChild(o),n.anchors.input=o,e},e.makeRemoveBlockButton=function(){var e=n.draw.node("SPAN","ce-toolbar__remove-btn",{}),t=n.draw.node("SPAN","ce-toolbar__remove-setting",{innerHTML:''}),o=n.draw.node("DIV","ce-toolbar__remove-confirmation",{}),r=n.draw.node("DIV","ce-toolbar__remove-confirm",{textContent:"Удалить блок"}),i=n.draw.node("DIV","ce-toolbar__remove-cancel",{textContent:"Отмена"});return t.addEventListener("click",n.toolbar.settings.removeButtonClicked,!1),r.addEventListener("click",n.toolbar.settings.confirmRemovingRequest,!1),i.addEventListener("click",n.toolbar.settings.cancelRemovingRequest,!1),o.appendChild(r),o.appendChild(i),e.appendChild(t),e.appendChild(o),n.toolbar.settings.setting=t,n.toolbar.settings.actions=o,e},e.removeButtonClicked=function(){var e=n.toolbar.settings.actions;e.classList.contains("opened")?n.toolbar.settings.hideRemoveActions():n.toolbar.settings.showRemoveActions(),n.toolbar.toolbox.close(),n.toolbar.settings.close()},e.cancelRemovingRequest=function(){n.toolbar.settings.actions.classList.remove("opened")},e.confirmRemovingRequest=function(){var e,t=n.content.currentNode;t.remove(),e=n.nodes.redactor.childNodes.length,0===e&&(n.content.currentNode=null,n.ui.addInitialBlock()),n.ui.saveInputs(),n.toolbar.close()},e.showRemoveActions=function(){n.toolbar.settings.actions.classList.add("opened")},e.hideRemoveActions=function(){n.toolbar.settings.actions.classList.remove("opened")},e}({})},function(e,t){"use strict";var n=codex.editor;e.exports=function(e){e.buttonsOpened=null,e.actionsOpened=null,e.wrappersOffset=null,e.storedSelection=null,e.show=function(){var t,o=n.content.currentNode,r=o.dataset.tool;if(t=n.tools[r],t.showInlineToolbar){var i=e.getSelectionText(),a=n.nodes.inlineToolbar.wrapper;i.length>0&&(n.toolbar.inline.move(),a.classList.add("opened"),n.toolbar.inline.showButtons())}},e.close=function(){var e=n.nodes.inlineToolbar.wrapper;e.classList.remove("opened")},e.move=function(){this.wrappersOffset||(this.wrappersOffset=this.getWrappersOffset());var e,t,o=this.getSelectionCoords(),r=0,i=n.nodes.inlineToolbar.wrapper;0===i.offsetHeight&&(r=40),e=o.x-this.wrappersOffset.left,t=o.y+window.scrollY-this.wrappersOffset.top-r-i.offsetHeight,i.style.transform="translate3D("+Math.floor(e)+"px, "+Math.floor(t)+"px, 0)",n.toolbar.inline.closeButtons(),n.toolbar.inline.closeAction()},e.toolClicked=function(e,t){switch(t){case"createLink":n.toolbar.inline.createLinkAction(e,t);break;default:n.toolbar.inline.defaultToolAction(t)}n.nodes.inlineToolbar.buttons.childNodes.forEach(n.toolbar.inline.hightlight)},e.getWrappersOffset=function(){var e=n.nodes.wrapper,t=this.getOffset(e);return this.wrappersOffset=t,t},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=n.nodes.inlineToolbar.buttons;e.classList.add("opened"),n.toolbar.inline.buttonsOpened=!0,n.nodes.inlineToolbar.buttons.childNodes.forEach(n.toolbar.inline.hightlight)},e.closeButtons=function(){var e=n.nodes.inlineToolbar.buttons;e.classList.remove("opened"),n.toolbar.inline.buttonsOpened=!1},e.showActions=function(){var e=n.nodes.inlineToolbar.actions;e.classList.add("opened"),n.toolbar.inline.actionsOpened=!0},e.closeAction=function(){var e=n.nodes.inlineToolbar.actions;e.innerHTML="",e.classList.remove("opened"),n.toolbar.inline.actionsOpened=!1};var t=function(e){if(e.keyCode==n.core.keys.ENTER){var t=n.content.currentNode,o=n.toolbar.inline.storedSelection;n.toolbar.inline.restoreSelection(t,o),n.toolbar.inline.setAnchor(this.value),e.preventDefault(),e.stopImmediatePropagation(),n.toolbar.inline.clearRange()}};return e.createLinkAction=function(e){var o=this.isLinkActive(),r=n.content.currentNode,i=n.toolbar.inline.saveSelection(r);if(n.toolbar.inline.storedSelection=i,o)n.toolbar.inline.restoreSelection(r,i),n.toolbar.inline.defaultToolAction("unlink");else{var a=n.draw.inputForLink();n.nodes.inlineToolbar.actions.appendChild(a),n.toolbar.inline.closeButtons(),n.toolbar.inline.showActions(),a.focus(),e.preventDefault(),a.addEventListener("keydown",t,!1)}},e.isLinkActive=function(){var e=!1;return n.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),n.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 t=e.dataset.type;document.queryCommandState(t)?n.toolbar.inline.setButtonHighlighted(e):n.toolbar.inline.removeButtonsHighLight(e);var o=window.getSelection(),r=o.anchorNode.parentNode;"A"==r.tagName&&"link"==t&&n.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";var n=codex.editor;e.exports=function(e){return e.opened=!1,e.open=function(){n.toolbar.settings.opened&&n.toolbar.settings.close(),n.nodes.toolbox.classList.add("opened"),n.nodes.plusButton.classList.add("clicked"),n.toolbar.toolbox.opened=!0},e.close=function(){n.nodes.toolbox.classList.remove("opened"),n.nodes.plusButton.classList.remove("clicked"),n.toolbar.toolbox.opened=!1},e.leaf=function(){var e=n.toolbar.current,t=Object.keys(n.tools),o=n.nodes.toolbarButtons,r=0,i=void 0,a=void 0,s=void 0;if(e)for(r=t.indexOf(e)+1,a=t[r];!n.tools[a].displayInToolbox;)r++,a=t[r],r==t.length&&(r=0,a=t[r]);else for(s in n.tools){if(n.tools[s].displayInToolbox)break;r++}i=t[r];for(var c in o)o[c].classList.remove("selected");o[i].classList.add("selected"),n.toolbar.current=i},e.toolClicked=function(e){var t,o,r,i=["image","link","list","instagram","twitter","embed"],a=n.tools[n.toolbar.current],s=n.content.currentNode,c=n.caret.inputIndex;t=a.render(),r={block:t,type:a.type,stretched:!1},s&&i.indexOf(s.dataset.tool)===-1&&""===s.textContent.trim()?n.content.switchBlock(s,t,a.type):(n.content.insertBlock(r),c++),o=a.appendCallback,o&&"function"==typeof o&&o.call(e),window.setTimeout(function(){n.caret.setToBlock(c)},10),n.content.workingNodeChanged(),n.toolbar.move()},e}({})},function(e,t){"use strict";var n=codex.editor;e.exports=function(e){return e.globalKeydown=function(e){switch(e.keyCode){case n.core.keys.ENTER:n.callback.enterKeyPressed(e)}},e.redactorKeyDown=function(e){switch(e.keyCode){case n.core.keys.TAB:n.callback.tabKeyPressed(e);break;case n.core.keys.ENTER:n.callback.enterKeyPressedOnRedactorZone(e);break;case n.core.keys.ESC:n.callback.escapeKeyPressed(e);break;default:n.callback.defaultKeyPressed(e)}},e.globalKeyup=function(e){switch(e.keyCode){case n.core.keys.UP:case n.core.keys.LEFT:case n.core.keys.RIGHT:case n.core.keys.DOWN:n.callback.arrowKeyPressed(e)}},e.tabKeyPressed=function(e){n.toolbar.opened||n.toolbar.open(),n.toolbar.opened&&!n.toolbar.toolbox.opened?n.toolbar.toolbox.open():n.toolbar.toolbox.leaf(),e.preventDefault()},e.enterKeyPressed=function(){n.content.editorAreaHightlighted&&(n.caret.inputIndex=-1,n.callback.enterPressedOnBlock())},e.enterKeyPressedOnRedactorZone=function(e){"true"==e.target.contentEditable&&n.caret.saveCurrentInputIndex();var t=n.caret.getCurrentInputIndex()||0,o=n.content.currentNode,r=o.dataset.tool,i=n.toolbar.opened&&n.toolbar.current&&e.target==n.state.inputs[t],a=n.tools[r].enableLineBreaks,s=n.settings.initialBlockPlugin;if(i)return e.preventDefault(),n.toolbar.toolbox.toolClicked(e),n.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=n.caret.position.atTheEnd(),u=!1;if(e.shiftKey&&!a)return n.callback.enterPressedOnBlock(n.content.currentBlock,e),void e.preventDefault();if(u=l&&"true"!=l.parentNode.contentEditable,l.nodeType!=n.core.nodeTypes.TEXT||u||d){var p=n.content.isLastNode(l);p&&d&&(e.preventDefault(),e.stopPropagation(),e.stopImmediatePropagation(),n.core.log("ENTER clicked in last textNode. Create new BLOCK"),n.content.insertBlock({type:s,block:n.tools[s].render()},!0),n.toolbar.move(),n.toolbar.open(),n.toolbar.showPlusButton())}else e.preventDefault(),n.core.log("Splitting Text node..."),n.content.splitBlock(t),n.state.inputs[t+1].textContent.trim()||n.toolbar.showPlusButton();n.ui.saveInputs()},e.escapeKeyPressed=function(e){n.toolbar.close(),n.toolbar.toolbox.close(),e.preventDefault()},e.arrowKeyPressed=function(){n.content.workingNodeChanged(),n.toolbar.close(),n.toolbar.move()},e.defaultKeyPressed=function(){n.toolbar.close(),n.toolbar.inline.actionsOpened||(n.toolbar.inline.close(),n.content.clearMark())},e.redactorClicked=function(t){e.detectWhenClickedOnFirstLevelBlockArea(),n.content.workingNodeChanged(t.target),n.ui.saveInputs();var o,r=n.toolbar.inline.getSelectionText();if(0===r.length&&n.toolbar.inline.close(),"true"==t.target.contentEditable&&n.caret.saveCurrentInputIndex(),null===n.content.currentNode){var i=n.state.inputs.length>0?n.state.inputs.length-1:0;if(n.state.inputs.length&&(o=n.content.getFirstLevelBlock(n.state.inputs[i])),n.state.inputs.length&&""===n.state.inputs[i].textContent&&o.dataset.tool==n.settings.initialBlockPlugin)n.caret.setToBlock(i);else{var a=n.settings.initialBlockPlugin;n.content.insertBlock({type:a,block:n.tools[a].render()}),1===n.state.inputs.length?n.caret.setToBlock(i):n.caret.setToNextBlock(i)}n.toolbar.move(),n.toolbar.open()}else n.toolbar.move(),n.toolbar.open(),n.toolbar.settings.close(),n.toolbar.toolbox.close();var s=!n.content.currentNode.textContent.trim(),c=n.content.currentNode.dataset.tool,l=c==n.settings.initialBlockPlugin; -n.toolbar.hidePlusButton(),n.content.markBlock(),l&&s&&n.toolbar.showPlusButton()},e.detectWhenClickedOnFirstLevelBlockArea=function(){var e=window.getSelection(),t=e.anchorNode,o=!1;if(0===e.rangeCount)n.content.editorAreaHightlighted=!0;else{for(n.core.isDomNode(t)||(t=t.parentNode),"true"==t.contentEditable&&(o=!0);"true"!=t.contentEditable&&(t=t.parentNode,"true"==t.contentEditable&&(o=!0),t!=document.body););n.content.editorAreaHightlighted=!o}},e.toolbarButtonClicked=function(e){var t=this;n.toolbar.current=t.dataset.type,n.toolbar.toolbox.toolClicked(e),n.toolbar.close()},e.plusButtonClicked=function(){n.nodes.toolbox.classList.contains("opened")?n.toolbar.toolbox.close():n.toolbar.toolbox.open()},e.blockKeydown=function(e){var t=this;switch(e.keyCode){case n.core.keys.DOWN:case n.core.keys.RIGHT:n.callback.blockRightOrDownArrowPressed();break;case n.core.keys.BACKSPACE:n.callback.backspacePressed(t,e);break;case n.core.keys.UP:case n.core.keys.LEFT:n.callback.blockLeftOrUpArrowPressed()}},e.blockRightOrDownArrowPressed=function(){var e,t=window.getSelection(),o=n.state.inputs,r=t.anchorNode;if(!r)return!1;for(;"true"!=r.contentEditable;)e=r.parentNode,r=e;for(var i=0;r!=o[i];)i++;if(!r.textContent)return void n.caret.setToNextBlock(i);var a,s,c=!1,l=!1;return a=r.childNodes[r.childNodes.length-1],s=n.core.isDomNode(a)?n.content.getDeepestTextNodeFromPosition(a,a.childNodes.length):a,c=t.anchorNode==s,l=s.length==t.anchorOffset,c&&l?void n.caret.setToNextBlock(i):(n.core.log("arrow [down|right] : caret does not reached the end"),!1)},e.blockLeftOrUpArrowPressed=function(){var e,t=window.getSelection(),o=n.state.inputs,r=t.anchorNode;if(!r)return!1;if(0!==t.anchorOffset)return!1;for(;"true"!=r.contentEditable;)e=r.parentNode,r=e;for(var i=0;r!=o[i];)i++;var a,s,c=!1,l=!1;return r.textContent?(a=r.childNodes[0],s=n.core.isDomNode(a)?n.content.getDeepestTextNodeFromPosition(a,0):a,c=t.anchorNode==s,l=0===t.anchorOffset,void(c&&l&&n.caret.setToPreviousBlock(i))):void n.caret.setToPreviousBlock(i)},e.enterPressedOnBlock=function(){var e=n.settings.initialBlockPlugin;n.content.insertBlock({type:e,block:n.tools[e].render()},!0),n.toolbar.move(),n.toolbar.open()},e.backspacePressed=function(e,t){var o,r,i,a=n.caret.getCurrentInputIndex();if(e.textContent.trim()){if(o=n.content.getRange(),r=o.endOffset-o.startOffset,!n.caret.position.atStart()||r||!n.state.inputs[a-1])return;n.content.mergeBlocks(a)}r||e.remove(),i=n.nodes.redactor.childNodes.length,0===i?(n.content.currentNode=null,n.ui.addInitialBlock(),n.ui.saveInputs(),window.setTimeout(function(){n.caret.setToPreviousBlock(1)},10)):0!==n.caret.inputIndex?n.caret.setToPreviousBlock(n.caret.inputIndex):n.caret.setToNextBlock(n.caret.inputIndex),n.toolbar.move(),n.toolbar.opened||n.toolbar.open(),n.ui.saveInputs(),t.preventDefault()},e._blockPasteCallback=function(){var e=n.caret.getCurrentInputIndex(),t=new MutationObserver(n.callback.handleMutationsOnPaste),o={attributes:!0,childList:!1,characterData:!1,subtree:!0};t.observe(n.state.inputs[e],o)},e.blockPasteCallback=function(e){e.preventDefault();var t=n.content.getEditableParent(e.target),o=n.content.getFirstLevelBlock(e.target);if(t){var r,i,a=e.clipboardData.getData("text/html")||e.clipboardData.getData("text/plain"),s=n.draw.node("DIV","",{}),c=new n.sanitizer.init(n.sanitizer.Config.BASIC);i=document.createDocumentFragment(),r=c.clean(a),s.innerHTML=r;for(var l,d;l=s.firstChild;)d=i.appendChild(l);if(!n.tools[o.dataset.tool].allowRenderOnPaste||!n.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.handleMutationsOnPaste=function(e){var t=this;e.forEach(function(e){n.content.paste.call(t,e)})},e.showSettingsButtonClicked=function(){var e=n.content.currentNode.dataset.tool;n.toolbar.settings.toggle(e),n.toolbar.toolbox.close(),n.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";var n=codex.editor;e.exports=function(e){return e.inputIndex=null,e.offset=null,e.focusedNodeIndex=null,e.set=function(t,o,r){r=r||e.offset||0,o=o||e.focusedNodeIndex||0;var i,a=t.childNodes;if(i=0===a.length?t:a[o],"INPUT"==t.tagName)return void t.focus();n.core.isDomNode(i)&&(i=n.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),n.caret.saveCurrentInputIndex()},20)},e.saveCurrentInputIndex=function(){var t,o=window.getSelection(),r=n.state.inputs,i=o.anchorNode;if(i){for(;"true"!=i.contentEditable;)t=i.parentNode,i=t;for(var a=0;i!=r[a];)a++;e.inputIndex=a}},e.getCurrentInputIndex=function(){return e.inputIndex},e.setToNextBlock=function(e){var t=n.state.inputs,o=t[e+1];if(!o)return void n.core.log("We are reached the end");if(!o.childNodes.length){var r=document.createTextNode("");o.appendChild(r)}n.caret.inputIndex=e+1,n.caret.set(o,0,0),n.content.workingNodeChanged(o)},e.setToBlock=function(e){var t=n.state.inputs,o=t[e];if(o){if(!o.childNodes.length){var r=document.createTextNode("");o.appendChild(r)}n.caret.inputIndex=e,n.caret.set(o,0,0),n.content.workingNodeChanged(o)}},e.setToPreviousBlock=function(e){e=e||0;var t,o,r,i=n.state.inputs,a=i[e-1];return a?(t=n.content.getDeepestTextNodeFromPosition(a,a.childNodes.length),o=t.length,a.childNodes.length||(r=document.createTextNode(""),a.appendChild(r)),n.caret.inputIndex=e-1,n.caret.set(a,a.childNodes.length-1,o),void n.content.workingNodeChanged(i[e-1])):void n.core.log("We are reached first node")},e.position={atStart:function(){var e=window.getSelection(),t=e.anchorOffset,o=e.anchorNode,r=n.content.getFirstLevelBlock(o),i=r.childNodes[0];n.core.isDomNode(o)||(o=o.parentNode);var a=o===i.childNodes[0],s=0===t;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||"Отмена",f.addEventListener("click",u),g.addEventListener("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";var n=codex.editor;e.exports=function(e){return e.insertPastedContent=function(e,t){n.content.insertBlock({type:e.type,block:e.render({text:t.innerHTML})})},e.isFirstLevelBlock=function(e){return e.nodeType==n.core.nodeTypes.TAG&&e.classList.contains(n.ui.className.BLOCK_CLASSNAME)},e}({})},function(e,t,n){"use strict";var o=n(18);e.exports=function(e){var t={BASIC:{tags:{p:{},a:{href:!0,target:"_blank",rel:"nofollow"},i:{},b:{},strong:{},em:{},span:{}}}};return e.Config=t,e.init=o,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;v=t.core.keys.LEFT&&e.keyCode<=t.core.keys.DOWN&&e.stopPropagation()},e.rusToTranslit=function(e){for(var t=["А","Б","В","Г","Д","Е","Ё","Ж","З","И","Й","К","Л","М","Н","О","П","Р","С","Т","У","Ф","Х","Ц","Ч","Ш","Щ","Ь","Ы","Ь","Э","Ю","Я"],n=["A","B","V","G","D","E","E","Zh","Z","I","Y","K","L","M","N","O","P","R","S","T","U","F","H","C","Ch","Sh","Sch","","Y","","E","Yu","Ya"],o=0;oНе выводить в ленте'}:{innerHTML:'Вывести в ленте'},e=t.draw.node("DIV",t.ui.className.SETTINGS_ITEM,n),t.listeners.add(e,"click",t.toolbar.settings.updateFeedMode,!1),e},e.updateFeedMode=function(){var e=t.content.currentNode;e.classList.toggle(t.ui.className.BLOCK_IN_FEED_MODE),t.toolbar.settings.close()},e.isFeedModeActivated=function(){var e=t.content.currentNode;return!!e&&e.classList.contains(t.ui.className.BLOCK_IN_FEED_MODE)},e.makeAnchorInput=function(){var e=t.draw.node("div","ce-settings__anchor-wrapper ce-settings__item",{}),n=t.draw.node("i","ce-settings__anchor-hash",{}),o=t.draw.node("input","ce-settings__anchor-input",{placeholder:"Якорь"});return t.listeners.add(o,"keydown",t.anchors.keyDownOnAnchorInput),t.listeners.add(o,"keyup",t.anchors.keyUpOnAnchorInput),t.listeners.add(o,"input",t.anchors.anchorChanged),t.listeners.add(o,"blur",t.anchors.anchorChanged),e.appendChild(n),e.appendChild(o),t.anchors.input=o,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},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,a=n[r];!t.tools[a].displayInToolbox;)r++,a=n[r],r==n.length&&(r=0,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;return e.globalKeydown=function(e){switch(e.keyCode){case t.core.keys.ENTER:t.callback.enterKeyPressed(e)}},e.redactorKeyDown=function(e){switch(e.keyCode){case t.core.keys.TAB:t.callback.tabKeyPressed(e);break;case t.core.keys.ENTER:t.callback.enterKeyPressedOnRedactorZone(e);break;case t.core.keys.ESC:t.callback.escapeKeyPressed(e);break;default:t.callback.defaultKeyPressed(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:t.callback.arrowKeyPressed(e)}},e.tabKeyPressed=function(e){t.toolbar.opened||t.toolbar.open(),t.toolbar.opened&&!t.toolbar.toolbox.opened?t.toolbar.toolbox.open():t.toolbar.toolbox.leaf(),e.preventDefault()},e.enterKeyPressed=function(){t.content.editorAreaHightlighted&&(t.caret.inputIndex=-1,t.callback.enterPressedOnBlock())},e.enterKeyPressedOnRedactorZone=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()},e.escapeKeyPressed=function(e){t.toolbar.close(),t.toolbar.toolbox.close(),e.preventDefault()},e.arrowKeyPressed=function(){t.content.workingNodeChanged(),t.toolbar.close(),t.toolbar.move()},e.defaultKeyPressed=function(){t.toolbar.close(),t.toolbar.inline.actionsOpened||(t.toolbar.inline.close(),t.content.clearMark())},e.redactorClicked=function(n){e.detectWhenClickedOnFirstLevelBlockArea(),t.content.workingNodeChanged(n.target),t.ui.saveInputs();var o,r=t.toolbar.inline.getSelectionText();if(0===r.length&&t.toolbar.inline.close(),"true"==n.target.contentEditable&&t.caret.saveCurrentInputIndex(),null===t.content.currentNode){var i=t.state.inputs.length>0?t.state.inputs.length-1:0;if(t.state.inputs.length&&(o=t.content.getFirstLevelBlock(t.state.inputs[i])),t.state.inputs.length&&""===t.state.inputs[i].textContent&&o.dataset.tool==t.settings.initialBlockPlugin)t.caret.setToBlock(i);else{var a=t.settings.initialBlockPlugin;t.content.insertBlock({type:a,block:t.tools[a].render()}),1===t.state.inputs.length?t.caret.setToBlock(i):t.caret.setToNextBlock(i)}t.toolbar.move(),t.toolbar.open()}else t.toolbar.move(),t.toolbar.open(),t.toolbar.settings.close(), +t.toolbar.toolbox.close();var s=!t.content.currentNode.textContent.trim(),c=t.content.currentNode.dataset.tool,l=c==t.settings.initialBlockPlugin;t.toolbar.hidePlusButton(),t.content.markBlock(),l&&s&&t.toolbar.showPlusButton()},e.detectWhenClickedOnFirstLevelBlockArea=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=this;switch(e.keyCode){case t.core.keys.DOWN:case t.core.keys.RIGHT:t.callback.blockRightOrDownArrowPressed();break;case t.core.keys.BACKSPACE:t.callback.backspacePressed(n,e);break;case t.core.keys.UP:case t.core.keys.LEFT:t.callback.blockLeftOrUpArrowPressed()}},e.blockRightOrDownArrowPressed=function(){var e,n=window.getSelection(),o=t.state.inputs,r=n.anchorNode;if(!r)return!1;for(;"true"!=r.contentEditable;)e=r.parentNode,r=e;for(var i=0;r!=o[i];)i++;if(!r.textContent)return void t.caret.setToNextBlock(i);var a,s,c=!1,l=!1;return a=r.childNodes[r.childNodes.length-1],s=t.core.isDomNode(a)?t.content.getDeepestTextNodeFromPosition(a,a.childNodes.length):a,c=n.anchorNode==s,l=s.length==n.anchorOffset,c&&l?void t.caret.setToNextBlock(i):(t.core.log("arrow [down|right] : caret does not reached the end"),!1)},e.blockLeftOrUpArrowPressed=function(){var e,n=window.getSelection(),o=t.state.inputs,r=n.anchorNode;if(!r)return!1;if(0!==n.anchorOffset)return!1;for(;"true"!=r.contentEditable;)e=r.parentNode,r=e;for(var i=0;r!=o[i];)i++;var a,s,c=!1,l=!1;return r.textContent?(a=r.childNodes[0],s=t.core.isDomNode(a)?t.content.getDeepestTextNodeFromPosition(a,0):a,c=n.anchorNode==s,l=0===n.anchorOffset,void(c&&l&&t.caret.setToPreviousBlock(i))):void t.caret.setToPreviousBlock(i)},e.enterPressedOnBlock=function(){var e=t.settings.initialBlockPlugin;t.content.insertBlock({type:e,block:t.tools[e].render()},!0),t.toolbar.move(),t.toolbar.open()},e.backspacePressed=function(e,n){var o,r,i,a=t.caret.getCurrentInputIndex();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(){var e=t.caret.getCurrentInputIndex(),n=new MutationObserver(t.callback.handleMutationsOnPaste),o={attributes:!0,childList:!1,characterData:!1,subtree:!0};n.observe(t.state.inputs[e],o)},e.blockPasteCallback=function(e){e.preventDefault();var n=t.content.getEditableParent(e.target),o=t.content.getFirstLevelBlock(e.target);if(n){var r,i,a=e.clipboardData.getData("text/html")||e.clipboardData.getData("text/plain"),s=t.draw.node("DIV","",{}),c=new t.sanitizer.init(t.sanitizer.Config.BASIC);i=document.createDocumentFragment(),r=c.clean(a),s.innerHTML=r;for(var l,d;l=s.firstChild;)d=i.appendChild(l);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.handleMutationsOnPaste=function(e){var n=this;e.forEach(function(e){t.content.paste.call(n,e)})},e.showSettingsButtonClicked=function(){var e=t.content.currentNode.dataset.tool;t.toolbar.settings.toggle(e),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],"INPUT"==n.tagName)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}({})},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={BASIC:{tags:{p:{},a:{href:!0,target:"_blank",rel:"nofollow"},i:{},b:{},strong:{},em:{},span:{}}}};return e.Config=o,e.init=t,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;v=t.core.keys.LEFT&&e.keyCode<=t.core.keys.DOWN&&e.stopPropagation()},e.rusToTranslit=function(e){for(var t=["А","Б","В","Г","Д","Е","Ё","Ж","З","И","Й","К","Л","М","Н","О","П","Р","С","Т","У","Ф","Х","Ц","Ч","Ш","Щ","Ь","Ы","Ь","Э","Ю","Я"],n=["A","B","V","G","D","E","E","Zh","Z","I","Y","K","L","M","N","O","P","R","S","T","U","F","H","C","Ch","Sh","Sch","","Y","","E","Yu","Ya"],o=0;o0&&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 anchor = tool.anchor,\n\t cover = tool.cover;\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 cover: cover,\n\t anchor: anchor\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\tvar editor = codex.editor;\n\t\n\tmodule.exports = function (saver) {\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 anchor = block.dataset.anchor;\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 coverFlag = false;\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\n\t savedData = codex.editor.state.blocks.items[position].data;\n\t coverFlag = codex.editor.state.blocks.items[position].cover;\n\t anchor = codex.editor.state.blocks.items[position].anchor;\n\t } else {\n\t\n\t savedData = editor.tools[pluginName].save(pluginsContent);\n\t coverFlag = block.classList.contains(editor.ui.className.BLOCK_IN_FEED_MODE);\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 anchor: anchor,\n\t data: savedData\n\t };\n\t\n\t /** Marks Blocks that will be in main page */\n\t output.cover = coverFlag;\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 * @author Codex Team\r\n\t * @version 1.3.11\r\n\t */\n\tvar editor = codex.editor;\n\t\n\tmodule.exports = function (content) {\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 * 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 * @deprecated\r\n\t */\n\t content.getNodeFocused = function () {\n\t\n\t var selection = window.getSelection(),\n\t focused;\n\t\n\t if (selection.anchorNode === null) {\n\t\n\t return null;\n\t }\n\t\n\t if (selection.anchorNode.nodeType == editor.core.nodeTypes.TAG) {\n\t\n\t focused = selection.anchorNode;\n\t } else {\n\t\n\t focused = selection.focusNode.parentElement;\n\t }\n\t\n\t if (!editor.parser.isFirstLevelBlock(focused)) {\n\t\n\t /** Iterate with parent nodes to find first-level*/\n\t var parent = focused.parentNode;\n\t\n\t while (parent && !editor.parser.isFirstLevelBlock(parent)) {\n\t\n\t parent = parent.parentNode;\n\t }\n\t\n\t focused = parent;\n\t }\n\t\n\t if (focused != editor.nodes.redactor) {\n\t\n\t return focused;\n\t }\n\t\n\t return null;\n\t };\n\t\n\t /**\r\n\t * Appends background to the block\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 */\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 * @private\r\n\t *\r\n\t * Finds first-level block\r\n\t * @param {Element} node - selected or clicked in redactors area node\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 * 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 this.currentNode = this.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 /**\r\n\t * Check is this block was in feed\r\n\t * If true, than set switched block also covered\r\n\t */\n\t if (targetBlock.classList.contains(editor.ui.className.BLOCK_IN_FEED_MODE)) {\n\t\n\t newBlock.classList.add(editor.ui.className.BLOCK_IN_FEED_MODE);\n\t }\n\t\n\t if (targetBlock.classList.contains(editor.ui.className.BLOCK_WITH_ANCHOR)) {\n\t\n\t newBlock.classList.add(editor.ui.className.BLOCK_WITH_ANCHOR);\n\t }\n\t\n\t /**\r\n\t * Saving anchor\r\n\t */\n\t newBlock.dataset.anchor = targetBlock.dataset.anchor;\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 * @private\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 cover = blockData.cover,\n\t anchor = blockData.anchor,\n\t isStretched = blockData.stretched;\n\t\n\t var newBlock = editor.content.composeNewBlock(newBlockContent, blockType, isStretched, anchor);\n\t\n\t if (cover === true) {\n\t\n\t newBlock.classList.add(editor.ui.className.BLOCK_IN_FEED_MODE);\n\t }\n\t\n\t if (anchor) {\n\t\n\t newBlock.classList.add(editor.ui.className.BLOCK_WITH_ANCHOR);\n\t }\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 var newBlockComposed = editor.content.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 * @private\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 */\n\t content.composeNewBlock = function (block, tool, isStretched, anchor) {\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 newBlock.dataset.anchor = anchor || '';\n\t return newBlock;\n\t };\n\t\n\t /**\r\n\t * Returns Range object of current selection\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 * @private\r\n\t * @param {Int} inputIndex - target input index\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 */\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 * @private\r\n\t *\r\n\t * Callback for HTML Mutations\r\n\t * @param {Array} mutation - Mutation Record\r\n\t */\n\t content.paste = function (mutation) {\n\t\n\t var workingNode = editor.content.currentNode,\n\t tool = workingNode.dataset.tool;\n\t\n\t if (editor.tools[tool].allowedToPaste) {\n\t\n\t editor.content.sanitize.call(this, mutation.target);\n\t } else {\n\t\n\t editor.content.pasteTextContent(mutation.addedNodes);\n\t }\n\t };\n\t\n\t /**\r\n\t * @private\r\n\t *\r\n\t * gets only text/plain content of node\r\n\t * @param {Element} target - HTML node\r\n\t */\n\t content.pasteTextContent = function (nodes) {\n\t\n\t var node = nodes[0],\n\t textNode;\n\t\n\t if (!node) {\n\t\n\t return;\n\t }\n\t\n\t if (node.nodeType == editor.core.nodeTypes.TEXT) {\n\t\n\t textNode = document.createTextNode(node);\n\t } else {\n\t\n\t textNode = document.createTextNode(node.textContent);\n\t }\n\t\n\t if (editor.core.isDomNode(node)) {\n\t\n\t node.parentNode.replaceChild(textNode, node);\n\t }\n\t };\n\t\n\t /**\r\n\t * @private\r\n\t *\r\n\t * Sanitizes HTML content\r\n\t * @param {Element} target - inserted element\r\n\t * @uses Sanitize library html-janitor\r\n\t */\n\t content.sanitize = function (target) {\n\t\n\t if (!target) {\n\t\n\t return;\n\t }\n\t\n\t var node = target[0];\n\t\n\t if (!node) {\n\t\n\t return;\n\t }\n\t\n\t /**\r\n\t * Disconnect Observer\r\n\t * hierarchy of function calls inherits context of observer\r\n\t */\n\t this.disconnect();\n\t\n\t /**\r\n\t * Don't sanitize text node\r\n\t */\n\t if (node.nodeType == editor.core.nodeTypes.TEXT) {\n\t\n\t return;\n\t }\n\t\n\t /**\r\n\t * Clear dirty content\r\n\t */\n\t var cleaner = editor.sanitizer.init(editor.satinizer.Config.BASIC),\n\t clean = cleaner.clean(target.outerHTML);\n\t\n\t var div = editor.draw.node('DIV', [], { innerHTML: clean });\n\t\n\t node.replaceWith(div.childNodes[0]);\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] htmlString - html content as string\r\n\t * @return {string} - html content as string\r\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 /**\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 = 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 /**\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\tvar editor = codex.editor;\n\t\n\tmodule.exports = function (toolbar) {\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 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\tvar editor = codex.editor;\n\t\n\tmodule.exports = function (settings) {\n\t\n\t settings.opened = false;\n\t\n\t settings.setting = null;\n\t settings.actions = null;\n\t\n\t settings.cover = 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 = `Плагин «${toolType}» не имеет настроек`;\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 editor.toolbar.settings.addDefaultSettings();\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 editor.anchors.settingsOpened(editor.content.currentNode);\n\t } else {\n\t\n\t this.close();\n\t }\n\t };\n\t\n\t /**\r\n\t * This function adds default core settings\r\n\t */\n\t settings.addDefaultSettings = function () {\n\t\n\t /** list of default settings */\n\t var feedModeToggler, anchorInput;\n\t\n\t /** Clear block and append initialized settings */\n\t editor.nodes.defaultSettings.innerHTML = '';\n\t\n\t /** Init all default setting buttons */\n\t feedModeToggler = editor.toolbar.settings.makeFeedModeToggler();\n\t anchorInput = editor.toolbar.settings.makeAnchorInput();\n\t\n\t /**\r\n\t * Fill defaultSettings\r\n\t */\n\t\n\t /**\r\n\t * Input for anchor for block\r\n\t */\n\t editor.nodes.defaultSettings.appendChild(anchorInput);\n\t\n\t /**\r\n\t * Button that enables/disables Feed-mode\r\n\t * Feed-mode means that block will be showed in articles-feed like cover\r\n\t */\n\t editor.nodes.defaultSettings.appendChild(feedModeToggler);\n\t };\n\t\n\t /**\r\n\t * Cover setting.\r\n\t * This tune highlights block, so that it may be used for showing target block on main page\r\n\t * Draw different setting when block is marked for main page\r\n\t * If TRUE, then we show button that removes this selection\r\n\t * Also defined setting \"Click\" events will be listened and have separate callbacks\r\n\t *\r\n\t * @return {Element} node/button that we place in default settings block\r\n\t */\n\t settings.makeFeedModeToggler = function () {\n\t\n\t var isFeedModeActivated = editor.toolbar.settings.isFeedModeActivated(),\n\t setting,\n\t data;\n\t\n\t if (!isFeedModeActivated) {\n\t\n\t data = {\n\t innerHTML: 'Вывести в ленте'\n\t };\n\t } else {\n\t\n\t data = {\n\t innerHTML: 'Не выводить в ленте'\n\t };\n\t }\n\t\n\t setting = editor.draw.node('DIV', editor.ui.className.SETTINGS_ITEM, data);\n\t setting.addEventListener('click', editor.toolbar.settings.updateFeedMode, false);\n\t\n\t return setting;\n\t };\n\t\n\t /**\r\n\t * Updates Feed-mode\r\n\t */\n\t settings.updateFeedMode = function () {\n\t\n\t var currentNode = editor.content.currentNode;\n\t\n\t currentNode.classList.toggle(editor.ui.className.BLOCK_IN_FEED_MODE);\n\t\n\t editor.toolbar.settings.close();\n\t };\n\t\n\t settings.isFeedModeActivated = function () {\n\t\n\t var currentBlock = editor.content.currentNode;\n\t\n\t if (currentBlock) {\n\t\n\t return currentBlock.classList.contains(editor.ui.className.BLOCK_IN_FEED_MODE);\n\t } else {\n\t\n\t return false;\n\t }\n\t };\n\t\n\t settings.makeAnchorInput = function () {\n\t\n\t var anchorWrapper = editor.draw.node('div', 'ce-settings__anchor-wrapper ce-settings__item', {}),\n\t hash = editor.draw.node('i', 'ce-settings__anchor-hash', {}),\n\t anchor = editor.draw.node('input', 'ce-settings__anchor-input', { placeholder: 'Якорь' });\n\t\n\t anchor.addEventListener('keydown', editor.anchors.keyDownOnAnchorInput);\n\t anchor.addEventListener('keyup', editor.anchors.keyUpOnAnchorInput);\n\t anchor.addEventListener('input', editor.anchors.anchorChanged);\n\t anchor.addEventListener('blur', editor.anchors.anchorChanged);\n\t\n\t anchorWrapper.appendChild(hash);\n\t anchorWrapper.appendChild(anchor);\n\t\n\t editor.anchors.input = anchor;\n\t\n\t return anchorWrapper;\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 settingButton.addEventListener('click', editor.toolbar.settings.removeButtonClicked, false);\n\t\n\t confirmAction.addEventListener('click', editor.toolbar.settings.confirmRemovingRequest, false);\n\t\n\t cancelAction.addEventListener('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\tvar editor = codex.editor;\n\t\n\tmodule.exports = function (inline) {\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 action.addEventListener('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\tvar editor = codex.editor;\n\t\n\tmodule.exports = function (toolbox) {\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\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;\n\t visibleTool = tools[nextToolIndex];\n\t\n\t while (!editor.tools[visibleTool].displayInToolbox) {\n\t\n\t nextToolIndex++;\n\t visibleTool = tools[nextToolIndex];\n\t\n\t if (nextToolIndex == tools.length) {\n\t\n\t nextToolIndex = 0;\n\t visibleTool = tools[nextToolIndex];\n\t }\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 * Codex Editor callbacks module\r\n\t *\r\n\t * @author Codex Team\r\n\t * @version 1.3.7\r\n\t */\n\tvar editor = codex.editor;\n\t\n\tmodule.exports = function (callbacks) {\n\t\n\t callbacks.globalKeydown = function (event) {\n\t\n\t switch (event.keyCode) {\n\t case editor.core.keys.ENTER:\n\t editor.callback.enterKeyPressed(event);break;\n\t }\n\t };\n\t\n\t callbacks.redactorKeyDown = function (event) {\n\t\n\t switch (event.keyCode) {\n\t case editor.core.keys.TAB:\n\t editor.callback.tabKeyPressed(event);break;\n\t case editor.core.keys.ENTER:\n\t editor.callback.enterKeyPressedOnRedactorZone(event);break;\n\t case editor.core.keys.ESC:\n\t editor.callback.escapeKeyPressed(event);break;\n\t default:\n\t editor.callback.defaultKeyPressed(event);break;\n\t }\n\t };\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 editor.callback.arrowKeyPressed(event);break;\n\t }\n\t };\n\t\n\t callbacks.tabKeyPressed = function (event) {\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 event.preventDefault();\n\t };\n\t\n\t /**\r\n\t * @param {Event} event\r\n\t */\n\t callbacks.enterKeyPressed = function () {\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 editor.callback.enterPressedOnBlock();\n\t }\n\t };\n\t\n\t /**\r\n\t * ENTER key handler\r\n\t * Makes new paragraph block\r\n\t */\n\t callbacks.enterKeyPressedOnRedactorZone = function (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 callbacks.escapeKeyPressed = function (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 */\n\t callbacks.arrowKeyPressed = function () {\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 * @param {Event} event\r\n\t */\n\t callbacks.defaultKeyPressed = function () {\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 callbacks.redactorClicked = function (event) {\n\t\n\t callbacks.detectWhenClickedOnFirstLevelBlockArea();\n\t\n\t editor.content.workingNodeChanged(event.target);\n\t\n\t editor.ui.saveInputs();\n\t\n\t var selectedText = editor.toolbar.inline.getSelectionText(),\n\t firstLevelBlock;\n\t\n\t /**\r\n\t * If selection range took off, then we hide inline toolbar\r\n\t */\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 /**\r\n\t * @todo Refactor\r\n\t */\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\n\t /**\r\n\t * Move toolbar to the right position and open\r\n\t */\n\t editor.toolbar.move();\n\t editor.toolbar.open();\n\t } else {\n\t\n\t /**\r\n\t * Move toolbar to the new position and open\r\n\t */\n\t editor.toolbar.move();\n\t editor.toolbar.open();\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 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 /** Mark current block */\n\t editor.content.markBlock();\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 * 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 callbacks.detectWhenClickedOnFirstLevelBlockArea = function () {\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 ? false : true;\n\t }\n\t };\n\t\n\t /**\r\n\t * Toolbar button click handler\r\n\t * @param this - cursor to the button\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 /** Show or Hide toolbox when plus button is clicked */\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 */\n\t callbacks.blockKeydown = function (event) {\n\t\n\t var block = this; // event.target 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 editor.callback.blockRightOrDownArrowPressed();\n\t break;\n\t\n\t case editor.core.keys.BACKSPACE:\n\t editor.callback.backspacePressed(block, event);\n\t break;\n\t\n\t case editor.core.keys.UP:\n\t case editor.core.keys.LEFT:\n\t editor.callback.blockLeftOrUpArrowPressed();\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 */\n\t callbacks.blockRightOrDownArrowPressed = function () {\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 */\n\t callbacks.blockLeftOrUpArrowPressed = function () {\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 * Callback for enter key pressing in first-level block area\r\n\t * @param {Event} event\r\n\t */\n\t callbacks.enterPressedOnBlock = function () {\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 callbacks.backspacePressed = function (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 (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 is used to observe pasted dirty data.\r\n\t *\r\n\t * Mutation handlers send to separate observers each mutation (added, changed and so on), which will be\r\n\t * passed from handler that sanitizes and replaces data.\r\n\t *\r\n\t * Probably won't be used\r\n\t *\r\n\t * @deprecated\r\n\t *\r\n\t * @param event\r\n\t * @private\r\n\t */\n\t callbacks._blockPasteCallback = function () {\n\t\n\t var currentInputIndex = editor.caret.getCurrentInputIndex();\n\t\n\t /**\r\n\t * create an observer instance\r\n\t */\n\t var observer = new MutationObserver(editor.callback.handleMutationsOnPaste);\n\t\n\t /**\r\n\t * configuration of the observer:\r\n\t */\n\t var config = {\n\t attributes: true,\n\t childList: false,\n\t characterData: false,\n\t subtree: true\n\t };\n\t\n\t // pass in the target node, as well as the observer options\n\t observer.observe(editor.state.inputs[currentInputIndex], config);\n\t };\n\t\n\t /**\r\n\t * This method prevents default behaviour.\r\n\t *\r\n\t * 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 *\r\n\t * @param event\r\n\t */\n\t callbacks.blockPasteCallback = function (event) {\n\t\n\t /** Prevent default behaviour */\n\t event.preventDefault();\n\t\n\t var editableParent = editor.content.getEditableParent(event.target),\n\t firstLevelBlock = editor.content.getFirstLevelBlock(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 data = event.clipboardData.getData('text/html') || 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 cleaner = new editor.sanitizer.init(editor.sanitizer.Config.BASIC),\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 cleanData = cleaner.clean(data);\n\t\n\t div.innerHTML = cleanData;\n\t\n\t var node, lastNode;\n\t\n\t /**\r\n\t * and fill in fragment\r\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[firstLevelBlock.dataset.tool].allowRenderOnPaste) {\n\t\n\t if (editor.paste.pasted(event)) return;\n\t }\n\t\n\t /**\r\n\t * work with selection and range\r\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 /**\r\n\t * Sends all mutations to paste handler\r\n\t */\n\t callbacks.handleMutationsOnPaste = function (mutations) {\n\t\n\t var self = this;\n\t\n\t /**\r\n\t * Calling function with context of this function.\r\n\t * Also, we should sanitize pasted or changed data one time and ignore\r\n\t * changings which makes sanitize method.\r\n\t * For that, we need to send Context, MutationObserver.__proto__ that contains\r\n\t * observer disconnect method.\r\n\t */\n\t mutations.forEach(function (mutation) {\n\t\n\t editor.content.paste.call(self, mutation);\n\t });\n\t };\n\t\n\t /**\r\n\t * Clicks on block settings button\r\n\t */\n\t callbacks.showSettingsButtonClicked = function () {\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\tvar editor = codex.editor;\n\t\n\tmodule.exports = function (caret) {\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.tagName == 'INPUT') {\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 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 okBtn.addEventListener('click', confirmHandler);\n\t cancelBtn.addEventListener('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\tvar editor = codex.editor;\n\t\n\tmodule.exports = function (parser) {\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\tvar janitor = __webpack_require__(18);\n\t\n\tmodule.exports = function (sanitizer) {\n\t\n\t /**\r\n\t * Basic config\r\n\t */\n\t var Config = {\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 sanitizer.init = janitor;\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 Anchors 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 (anchors) {\n\t\n\t var editor = codex.editor;\n\t\n\t anchors.input = null;\n\t anchors.currentNode = null;\n\t\n\t anchors.settingsOpened = function (currentBlock) {\n\t\n\t anchors.currentNode = currentBlock;\n\t anchors.input.value = anchors.currentNode.dataset.anchor || '';\n\t };\n\t\n\t anchors.anchorChanged = function (e) {\n\t\n\t var newAnchor = e.target.value = anchors.rusToTranslit(e.target.value);\n\t\n\t anchors.currentNode.dataset.anchor = newAnchor;\n\t\n\t if (newAnchor.trim() !== '') {\n\t\n\t anchors.currentNode.classList.add(editor.ui.className.BLOCK_WITH_ANCHOR);\n\t } else {\n\t\n\t anchors.currentNode.classList.remove(editor.ui.className.BLOCK_WITH_ANCHOR);\n\t }\n\t };\n\t\n\t anchors.keyDownOnAnchorInput = function (e) {\n\t\n\t if (e.keyCode == editor.core.keys.ENTER) {\n\t\n\t e.preventDefault();\n\t e.stopPropagation();\n\t\n\t e.target.blur();\n\t editor.toolbar.settings.close();\n\t }\n\t };\n\t\n\t anchors.keyUpOnAnchorInput = function (e) {\n\t\n\t if (e.keyCode >= editor.core.keys.LEFT && e.keyCode <= editor.core.keys.DOWN) {\n\t\n\t e.stopPropagation();\n\t }\n\t };\n\t\n\t anchors.rusToTranslit = function (string) {\n\t\n\t var ru = ['А', 'Б', 'В', 'Г', 'Д', 'Е', 'Ё', 'Ж', 'З', 'И', 'Й', 'К', 'Л', 'М', 'Н', 'О', 'П', 'Р', 'С', 'Т', 'У', 'Ф', 'Х', 'Ц', 'Ч', 'Ш', 'Щ', 'Ь', 'Ы', 'Ь', 'Э', 'Ю', 'Я'],\n\t en = ['A', 'B', 'V', 'G', 'D', 'E', 'E', 'Zh', 'Z', 'I', 'Y', 'K', 'L', 'M', 'N', 'O', 'P', 'R', 'S', 'T', 'U', 'F', 'H', 'C', 'Ch', 'Sh', 'Sch', '', 'Y', '', 'E', 'Yu', 'Ya'];\n\t\n\t for (var i = 0; i < ru.length; i++) {\n\t\n\t string = string.split(ru[i]).join(en[i]);\n\t string = string.split(ru[i].toLowerCase()).join(en[i].toLowerCase());\n\t }\n\t\n\t string = string.replace(/[^0-9a-zA-Z_]+/g, '-');\n\t\n\t return string;\n\t };\n\t\n\t return anchors;\n\t}({});\n\n/***/ },\n/* 20 */\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.0\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 if (pattern.regex.test(string)) {\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 f80ecde20480b792f648","/**\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\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.anchors = require('./modules/anchors');\r\n editor.paste = require('./modules/paste');\r\n\r\n };\r\n\r\n /**\r\n * @public\r\n *\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 textareaId: 'codex-editor',\r\n uploadImagesUrl: '/editor/transport/',\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 textarea : 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 {} userSettings are :\r\n * - tools [],\r\n * - textareaId String\r\n * ...\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.make)\r\n .then(editor.ui.addTools)\r\n .then(editor.ui.bindEvents)\r\n .then(editor.tools.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})({});\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.2\r\n */\r\n\r\nlet editor = codex.editor;\r\n\r\nmodule.exports = (function (core) {\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.uploadImagesUrl) {\r\n\r\n editor.settings.uploadImagesUrl = userSettings.uploadImagesUrl;\r\n\r\n }\r\n\r\n editor.nodes.textarea = document.getElementById(userSettings.textareaId || editor.settings.textareaId);\r\n\r\n if (typeof editor.nodes.textarea === undefined || editor.nodes.textarea === null) {\r\n\r\n reject(Error(\"Textarea wasn't found by ID: #\" + userSettings.textareaId));\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 */\r\n core.ajax = function (data) {\r\n\r\n if (!data || !data.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 successFunction = function () {},\r\n params = '',\r\n obj;\r\n\r\n data.async = true;\r\n data.type = data.type || 'GET';\r\n data.data = data.data || '';\r\n data['content-type'] = data['content-type'] || 'application/json; charset=utf-8';\r\n successFunction = data.success || successFunction ;\r\n\r\n if (data.type == 'GET' && data.data) {\r\n\r\n data.url = /\\?/.test(data.url) ? data.url + '&' + data.data : data.url + '?' + data.data;\r\n\r\n } else {\r\n\r\n for(obj in data.data) {\r\n\r\n params += (obj + '=' + encodeURIComponent(data.data[obj]) + '&');\r\n\r\n }\r\n\r\n }\r\n\r\n if (data.withCredentials) {\r\n\r\n XMLHTTP.withCredentials = true;\r\n\r\n }\r\n\r\n if (data.beforeSend && typeof data.beforeSend == 'function') {\r\n\r\n data.beforeSend.call();\r\n\r\n }\r\n\r\n XMLHTTP.open( data.type, data.url, data.async );\r\n XMLHTTP.setRequestHeader('X-Requested-With', 'XMLHttpRequest');\r\n XMLHTTP.setRequestHeader('Content-type', 'application/x-www-form-urlencoded');\r\n\r\n XMLHTTP.onreadystatechange = function () {\r\n\r\n if (XMLHTTP.readyState == 4 && XMLHTTP.status == 200) {\r\n\r\n successFunction(XMLHTTP.responseText);\r\n\r\n }\r\n\r\n };\r\n\r\n XMLHTTP.send(params);\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 const instancePrefix = 'cdx-script-';\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(instancePrefix + 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 = instancePrefix + 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 return core;\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.1\r\n */\r\n\r\nlet editor = codex.editor;\r\n\r\nmodule.exports = (function (ui) {\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} - highlights covered blocks\r\n */\r\n BLOCK_IN_FEED_MODE : 'ce-block--feed-mode',\r\n\r\n /**\r\n * @const {String} - Block with anchor\r\n */\r\n BLOCK_WITH_ANCHOR : 'ce-block--anchor',\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.make = function () {\r\n\r\n var wrapper,\r\n toolbar,\r\n toolbarContent,\r\n redactor,\r\n blockButtons,\r\n blockSettings,\r\n showSettingsButton,\r\n showTrashButton,\r\n toolbox,\r\n plusButton;\r\n\r\n /** Make editor wrapper */\r\n wrapper = editor.draw.wrapper();\r\n\r\n /** Append editor wrapper after initial textarea */\r\n editor.core.insertAfter(editor.nodes.textarea, wrapper);\r\n\r\n /** Append block with notifications to the document */\r\n editor.notifications.createHolder();\r\n\r\n /** Make toolbar and content-editable redactor */\r\n toolbar = editor.draw.toolbar();\r\n toolbarContent = editor.draw.toolbarContent();\r\n plusButton = editor.draw.plusButton();\r\n showSettingsButton = editor.draw.settingsButton();\r\n showTrashButton = editor.toolbar.settings.makeRemoveBlockButton();\r\n blockSettings = editor.draw.blockSettings();\r\n blockButtons = editor.draw.blockButtons();\r\n toolbox = editor.draw.toolbox();\r\n redactor = editor.draw.redactor();\r\n\r\n /** settings */\r\n var defaultSettings = editor.draw.defaultSettings(),\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 /** 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 /** Append plus button */\r\n toolbarContent.appendChild(plusButton);\r\n\r\n /** Appending toolbar tools */\r\n toolbarContent.appendChild(toolbox);\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 wrapper.appendChild(toolbar);\r\n\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.toolbar = toolbar;\r\n editor.nodes.plusButton = plusButton;\r\n editor.nodes.toolbox = toolbox;\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 editor.nodes.redactor = redactor;\r\n\r\n /** Make container for inline toolbar */\r\n editor.ui.makeInlineToolbar();\r\n\r\n /** fill in default settings */\r\n editor.toolbar.settings.addDefaultSettings();\r\n\r\n };\r\n\r\n ui.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 /**\r\n * @private\r\n * Append tools passed in editor.tools\r\n */\r\n ui.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) {\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 * Add inline toolbar tools\r\n */\r\n editor.ui.addInlineToolbarTools();\r\n\r\n\r\n };\r\n\r\n ui.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 ui.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 document.addEventListener('keydown', editor.callback.globalKeydown, false );\r\n\r\n /** All keydowns on Redactor zone */\r\n editor.nodes.redactor.addEventListener('keydown', editor.callback.redactorKeyDown, false);\r\n\r\n /** All keydowns on Document */\r\n document.addEventListener('keyup', editor.callback.globalKeyup, false );\r\n\r\n /**\r\n * Mouse click to radactor\r\n */\r\n editor.nodes.redactor.addEventListener('click', editor.callback.redactorClicked, false );\r\n\r\n /**\r\n * Clicks to the Plus button\r\n */\r\n editor.nodes.plusButton.addEventListener('click', editor.callback.plusButtonClicked, false);\r\n\r\n /**\r\n * Clicks to SETTINGS button in toolbar\r\n */\r\n editor.nodes.showSettingsButton.addEventListener('click', editor.callback.showSettingsButtonClicked, false );\r\n\r\n /**\r\n * @deprecated ( but now in use for syncronization );\r\n * Any redactor changes: keyboard input, mouse cut/paste, drag-n-drop text\r\n */\r\n // editor.nodes.redactor.addEventListener('input', editor.callback.redactorInputEvent, 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.nodes.toolbarButtons[button].addEventListener('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 block.addEventListener('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 block.addEventListener('paste', editor.callback.blockPasteCallback, false);\r\n\r\n block.addEventListener('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 /** Save all inputs in global variable state */\r\n editor.state.inputs = redactor.querySelectorAll('[contenteditable], input');\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 button.addEventListener('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})({});\n\n\n// WEBPACK FOOTER //\n// ./modules/ui.js","/**\r\n *\r\n * Codex.Editor Transport Module\r\n *\r\n * @author Codex Team\r\n * @version 1.0\r\n */\r\nlet editor = codex.editor;\r\n\r\nmodule.exports = (function (transport) {\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 transport.prepare = function () {\r\n\r\n var input = document.createElement('INPUT');\r\n\r\n input.type = 'file';\r\n input.addEventListener('change', editor.transport.fileSelected);\r\n\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 this.input = null;\r\n\r\n /** Prepare new one */\r\n this.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 files = input.files,\r\n formdData = new FormData();\r\n\r\n formdData.append('files', files[0], files[0].name);\r\n\r\n editor.transport.ajax({\r\n data : formdData,\r\n beforeSend : editor.transport.arguments.beforeSend,\r\n success : editor.transport.arguments.success,\r\n error : editor.transport.arguments.error\r\n });\r\n\r\n };\r\n\r\n /**\r\n * Use plugin callbacks\r\n * @protected\r\n */\r\n transport.selectAndUpload = function (args) {\r\n\r\n this.arguments = args;\r\n this.input.click();\r\n\r\n };\r\n\r\n /**\r\n * Ajax requests module\r\n * @todo use core.ajax\r\n */\r\n transport.ajax = function (params) {\r\n\r\n var xhr = new XMLHttpRequest(),\r\n beforeSend = typeof params.beforeSend == 'function' ? params.beforeSend : function () {},\r\n success = typeof params.success == 'function' ? params.success : function () {},\r\n error = typeof params.error == 'function' ? params.error : function () {};\r\n\r\n beforeSend();\r\n\r\n xhr.open('POST', editor.settings.uploadImagesUrl, true);\r\n\r\n xhr.setRequestHeader('X-Requested-With', 'XMLHttpRequest');\r\n\r\n xhr.onload = function () {\r\n\r\n if (xhr.status === 200) {\r\n\r\n success(xhr.responseText);\r\n\r\n } else {\r\n\r\n editor.core.log('request error: %o', xhr);\r\n error();\r\n\r\n }\r\n\r\n };\r\n\r\n xhr.send(params.data);\r\n this.clearInput();\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\nlet editor = codex.editor;\r\n\r\nmodule.exports = (function (renderer) {\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 anchor = tool.anchor,\r\n cover = tool.cover;\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 cover : cover,\r\n anchor : anchor\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\nlet editor = codex.editor;\r\n\r\nmodule.exports = (function (saver) {\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 anchor = block.dataset.anchor;\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 coverFlag = false;\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\r\n savedData = codex.editor.state.blocks.items[position].data;\r\n coverFlag = codex.editor.state.blocks.items[position].cover;\r\n anchor = codex.editor.state.blocks.items[position].anchor;\r\n\r\n } else {\r\n\r\n savedData = editor.tools[pluginName].save(pluginsContent);\r\n coverFlag = block.classList.contains(editor.ui.className.BLOCK_IN_FEED_MODE);\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 anchor : anchor,\r\n data : savedData\r\n };\r\n\r\n /** Marks Blocks that will be in main page */\r\n output.cover = coverFlag;\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 * @author Codex Team\r\n * @version 1.3.11\r\n */\r\nlet editor = codex.editor;\r\n\r\nmodule.exports = (function (content) {\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 * 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 * @deprecated\r\n */\r\n content.getNodeFocused = function () {\r\n\r\n var selection = window.getSelection(),\r\n focused;\r\n\r\n if (selection.anchorNode === null) {\r\n\r\n return null;\r\n\r\n }\r\n\r\n if ( selection.anchorNode.nodeType == editor.core.nodeTypes.TAG ) {\r\n\r\n focused = selection.anchorNode;\r\n\r\n } else {\r\n\r\n focused = selection.focusNode.parentElement;\r\n\r\n }\r\n\r\n if ( !editor.parser.isFirstLevelBlock(focused) ) {\r\n\r\n /** Iterate with parent nodes to find first-level*/\r\n var parent = focused.parentNode;\r\n\r\n while (parent && !editor.parser.isFirstLevelBlock(parent)) {\r\n\r\n parent = parent.parentNode;\r\n\r\n }\r\n\r\n focused = parent;\r\n\r\n }\r\n\r\n if (focused != editor.nodes.redactor) {\r\n\r\n return focused;\r\n\r\n }\r\n\r\n return null;\r\n\r\n };\r\n\r\n /**\r\n * Appends background to the block\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 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 * @private\r\n *\r\n * Finds first-level block\r\n * @param {Element} node - selected or clicked in redactors area node\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 * 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 this.currentNode = this.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 /**\r\n * Check is this block was in feed\r\n * If true, than set switched block also covered\r\n */\r\n if (targetBlock.classList.contains(editor.ui.className.BLOCK_IN_FEED_MODE)) {\r\n\r\n newBlock.classList.add(editor.ui.className.BLOCK_IN_FEED_MODE);\r\n\r\n }\r\n\r\n if (targetBlock.classList.contains(editor.ui.className.BLOCK_WITH_ANCHOR)) {\r\n\r\n newBlock.classList.add(editor.ui.className.BLOCK_WITH_ANCHOR);\r\n\r\n }\r\n\r\n /**\r\n * Saving anchor\r\n */\r\n newBlock.dataset.anchor = targetBlock.dataset.anchor;\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 * @private\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 cover = blockData.cover,\r\n anchor = blockData.anchor,\r\n isStretched = blockData.stretched;\r\n\r\n var newBlock = editor.content.composeNewBlock(newBlockContent, blockType, isStretched, anchor);\r\n\r\n if (cover === true) {\r\n\r\n newBlock.classList.add(editor.ui.className.BLOCK_IN_FEED_MODE);\r\n\r\n }\r\n\r\n if (anchor) {\r\n\r\n newBlock.classList.add(editor.ui.className.BLOCK_WITH_ANCHOR);\r\n\r\n }\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 var newBlockComposed = editor.content.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 * @private\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 */\r\n content.composeNewBlock = function (block, tool, isStretched, anchor) {\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 newBlock.dataset.anchor = anchor || '';\r\n return newBlock;\r\n\r\n };\r\n\r\n /**\r\n * Returns Range object of current selection\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 * @private\r\n * @param {Int} inputIndex - target input index\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 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 * @private\r\n *\r\n * Callback for HTML Mutations\r\n * @param {Array} mutation - Mutation Record\r\n */\r\n content.paste = function (mutation) {\r\n\r\n var workingNode = editor.content.currentNode,\r\n tool = workingNode.dataset.tool;\r\n\r\n if (editor.tools[tool].allowedToPaste) {\r\n\r\n editor.content.sanitize.call(this, mutation.target);\r\n\r\n } else {\r\n\r\n editor.content.pasteTextContent(mutation.addedNodes);\r\n\r\n }\r\n\r\n };\r\n\r\n /**\r\n * @private\r\n *\r\n * gets only text/plain content of node\r\n * @param {Element} target - HTML node\r\n */\r\n content.pasteTextContent = function (nodes) {\r\n\r\n var node = nodes[0],\r\n textNode;\r\n\r\n if (!node) {\r\n\r\n return;\r\n\r\n }\r\n\r\n if (node.nodeType == editor.core.nodeTypes.TEXT) {\r\n\r\n textNode = document.createTextNode(node);\r\n\r\n } else {\r\n\r\n textNode = document.createTextNode(node.textContent);\r\n\r\n }\r\n\r\n if (editor.core.isDomNode(node)) {\r\n\r\n node.parentNode.replaceChild(textNode, node);\r\n\r\n }\r\n\r\n };\r\n\r\n /**\r\n * @private\r\n *\r\n * Sanitizes HTML content\r\n * @param {Element} target - inserted element\r\n * @uses Sanitize library html-janitor\r\n */\r\n content.sanitize = function (target) {\r\n\r\n if (!target) {\r\n\r\n return;\r\n\r\n }\r\n\r\n var node = target[0];\r\n\r\n if (!node) {\r\n\r\n return;\r\n\r\n }\r\n\r\n /**\r\n * Disconnect Observer\r\n * hierarchy of function calls inherits context of observer\r\n */\r\n this.disconnect();\r\n\r\n /**\r\n * Don't sanitize text node\r\n */\r\n if (node.nodeType == editor.core.nodeTypes.TEXT) {\r\n\r\n return;\r\n\r\n }\r\n\r\n /**\r\n * Clear dirty content\r\n */\r\n var cleaner = editor.sanitizer.init(editor.satinizer.Config.BASIC),\r\n clean = cleaner.clean(target.outerHTML);\r\n\r\n var div = editor.draw.node('DIV', [], { innerHTML: clean });\r\n\r\n node.replaceWith(div.childNodes[0]);\r\n\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] htmlString - html content as string\r\n * @return {string} - html content as string\r\n */\r\n content.wrapTextWithParagraphs = function (htmlString) {\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 = htmlString;\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\nlet editor = codex.editor;\r\n\r\nmodule.exports = (function (toolbar) {\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 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\nlet editor = codex.editor;\r\n\r\nmodule.exports = (function (settings) {\r\n\r\n settings.opened = false;\r\n\r\n settings.setting = null;\r\n settings.actions = null;\r\n\r\n settings.cover = 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 editor.toolbar.settings.addDefaultSettings();\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 editor.anchors.settingsOpened(editor.content.currentNode);\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 * This function adds default core settings\r\n */\r\n settings.addDefaultSettings = function () {\r\n\r\n /** list of default settings */\r\n var feedModeToggler,\r\n anchorInput;\r\n\r\n /** Clear block and append initialized settings */\r\n editor.nodes.defaultSettings.innerHTML = '';\r\n\r\n\r\n /** Init all default setting buttons */\r\n feedModeToggler = editor.toolbar.settings.makeFeedModeToggler();\r\n anchorInput = editor.toolbar.settings.makeAnchorInput();\r\n\r\n /**\r\n * Fill defaultSettings\r\n */\r\n\r\n /**\r\n * Input for anchor for block\r\n */\r\n editor.nodes.defaultSettings.appendChild(anchorInput);\r\n\r\n /**\r\n * Button that enables/disables Feed-mode\r\n * Feed-mode means that block will be showed in articles-feed like cover\r\n */\r\n editor.nodes.defaultSettings.appendChild(feedModeToggler);\r\n\r\n };\r\n\r\n /**\r\n * Cover setting.\r\n * This tune highlights block, so that it may be used for showing target block on main page\r\n * Draw different setting when block is marked for main page\r\n * If TRUE, then we show button that removes this selection\r\n * Also defined setting \"Click\" events will be listened and have separate callbacks\r\n *\r\n * @return {Element} node/button that we place in default settings block\r\n */\r\n settings.makeFeedModeToggler = function () {\r\n\r\n var isFeedModeActivated = editor.toolbar.settings.isFeedModeActivated(),\r\n setting,\r\n data;\r\n\r\n if (!isFeedModeActivated) {\r\n\r\n data = {\r\n innerHTML : 'Вывести в ленте'\r\n };\r\n\r\n } else {\r\n\r\n data = {\r\n innerHTML : 'Не выводить в ленте'\r\n };\r\n\r\n }\r\n\r\n setting = editor.draw.node('DIV', editor.ui.className.SETTINGS_ITEM, data);\r\n setting.addEventListener('click', editor.toolbar.settings.updateFeedMode, false);\r\n\r\n return setting;\r\n\r\n };\r\n\r\n /**\r\n * Updates Feed-mode\r\n */\r\n settings.updateFeedMode = function () {\r\n\r\n var currentNode = editor.content.currentNode;\r\n\r\n currentNode.classList.toggle(editor.ui.className.BLOCK_IN_FEED_MODE);\r\n\r\n editor.toolbar.settings.close();\r\n\r\n };\r\n\r\n settings.isFeedModeActivated = function () {\r\n\r\n var currentBlock = editor.content.currentNode;\r\n\r\n if (currentBlock) {\r\n\r\n return currentBlock.classList.contains(editor.ui.className.BLOCK_IN_FEED_MODE);\r\n\r\n } else {\r\n\r\n return false;\r\n\r\n }\r\n\r\n };\r\n\r\n settings.makeAnchorInput = function () {\r\n\r\n var anchorWrapper = editor.draw.node('div', 'ce-settings__anchor-wrapper ce-settings__item', {}),\r\n hash = editor.draw.node('i', 'ce-settings__anchor-hash', {}),\r\n anchor = editor.draw.node('input', 'ce-settings__anchor-input', { placeholder: 'Якорь' });\r\n\r\n anchor.addEventListener('keydown', editor.anchors.keyDownOnAnchorInput );\r\n anchor.addEventListener('keyup', editor.anchors.keyUpOnAnchorInput );\r\n anchor.addEventListener('input', editor.anchors.anchorChanged );\r\n anchor.addEventListener('blur', editor.anchors.anchorChanged );\r\n\r\n anchorWrapper.appendChild(hash);\r\n anchorWrapper.appendChild(anchor);\r\n\r\n editor.anchors.input = anchor;\r\n\r\n return anchorWrapper;\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 settingButton.addEventListener('click', editor.toolbar.settings.removeButtonClicked, false);\r\n\r\n confirmAction.addEventListener('click', editor.toolbar.settings.confirmRemovingRequest, false);\r\n\r\n cancelAction.addEventListener('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\nlet editor = codex.editor;\r\n\r\nmodule.exports = (function (inline) {\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 action.addEventListener('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\nlet editor = codex.editor;\r\n\r\nmodule.exports = (function (toolbox) {\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 };\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;\r\n visibleTool = tools[nextToolIndex];\r\n\r\n while (!editor.tools[visibleTool].displayInToolbox) {\r\n\r\n nextToolIndex++;\r\n visibleTool = tools[nextToolIndex];\r\n\r\n if ( nextToolIndex == tools.length ) {\r\n\r\n nextToolIndex = 0;\r\n visibleTool = tools[nextToolIndex];\r\n\r\n }\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 * Codex Editor callbacks module\r\n *\r\n * @author Codex Team\r\n * @version 1.3.7\r\n */\r\nlet editor = codex.editor;\r\n\r\nmodule.exports = (function (callbacks) {\r\n\r\n callbacks.globalKeydown = function (event) {\r\n\r\n switch (event.keyCode) {\r\n case editor.core.keys.ENTER : editor.callback.enterKeyPressed(event); break;\r\n }\r\n\r\n };\r\n\r\n callbacks.redactorKeyDown = function (event) {\r\n\r\n switch (event.keyCode) {\r\n case editor.core.keys.TAB : editor.callback.tabKeyPressed(event); break;\r\n case editor.core.keys.ENTER : editor.callback.enterKeyPressedOnRedactorZone(event); break;\r\n case editor.core.keys.ESC : editor.callback.escapeKeyPressed(event); break;\r\n default : editor.callback.defaultKeyPressed(event); break;\r\n }\r\n\r\n };\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 : editor.callback.arrowKeyPressed(event); break;\r\n }\r\n\r\n };\r\n\r\n callbacks.tabKeyPressed = function (event) {\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 event.preventDefault();\r\n\r\n };\r\n\r\n /**\r\n * @param {Event} event\r\n */\r\n callbacks.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 editor.callback.enterPressedOnBlock();\r\n\r\n }\r\n\r\n };\r\n\r\n /**\r\n * ENTER key handler\r\n * Makes new paragraph block\r\n */\r\n callbacks.enterKeyPressedOnRedactorZone = 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 callbacks.escapeKeyPressed = 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 */\r\n callbacks.arrowKeyPressed = function () {\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 * @param {Event} event\r\n */\r\n callbacks.defaultKeyPressed = 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 callbacks.redactorClicked = function (event) {\r\n\r\n callbacks.detectWhenClickedOnFirstLevelBlockArea();\r\n\r\n editor.content.workingNodeChanged(event.target);\r\n\r\n editor.ui.saveInputs();\r\n\r\n var selectedText = editor.toolbar.inline.getSelectionText(),\r\n firstLevelBlock;\r\n\r\n /**\r\n * If selection range took off, then we hide inline toolbar\r\n */\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 /**\r\n * @todo Refactor\r\n */\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 /**\r\n * Move toolbar to the right position and open\r\n */\r\n editor.toolbar.move();\r\n editor.toolbar.open();\r\n\r\n } else {\r\n\r\n /**\r\n * Move toolbar to the new position and open\r\n */\r\n editor.toolbar.move();\r\n editor.toolbar.open();\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 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 /** Mark current block */\r\n editor.content.markBlock();\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 * 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 callbacks.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 ? false : true;\r\n\r\n }\r\n\r\n };\r\n\r\n /**\r\n * Toolbar button click handler\r\n * @param this - cursor to the button\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 /** Show or Hide toolbox when plus button is clicked */\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 callbacks.blockKeydown = function (event) {\r\n\r\n let block = this; // event.target 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 editor.callback.blockRightOrDownArrowPressed();\r\n break;\r\n\r\n case editor.core.keys.BACKSPACE:\r\n editor.callback.backspacePressed(block, event);\r\n break;\r\n\r\n case editor.core.keys.UP:\r\n case editor.core.keys.LEFT:\r\n editor.callback.blockLeftOrUpArrowPressed();\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 callbacks.blockRightOrDownArrowPressed = function () {\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 callbacks.blockLeftOrUpArrowPressed = function () {\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 * Callback for enter key pressing in first-level block area\r\n * @param {Event} event\r\n */\r\n callbacks.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 callbacks.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 (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 is used to observe pasted dirty data.\r\n *\r\n * Mutation handlers send to separate observers each mutation (added, changed and so on), which will be\r\n * passed from handler that sanitizes and replaces data.\r\n *\r\n * Probably won't be used\r\n *\r\n * @deprecated\r\n *\r\n * @param event\r\n * @private\r\n */\r\n callbacks._blockPasteCallback = function () {\r\n\r\n var currentInputIndex = editor.caret.getCurrentInputIndex();\r\n\r\n /**\r\n * create an observer instance\r\n */\r\n var observer = new MutationObserver(editor.callback.handleMutationsOnPaste);\r\n\r\n /**\r\n * configuration of the observer:\r\n */\r\n var config = {\r\n attributes: true,\r\n childList: false,\r\n characterData: false,\r\n subtree : true\r\n };\r\n\r\n // pass in the target node, as well as the observer options\r\n observer.observe(editor.state.inputs[currentInputIndex], config);\r\n\r\n };\r\n\r\n /**\r\n * This method prevents default behaviour.\r\n *\r\n * 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 * @param event\r\n */\r\n callbacks.blockPasteCallback = function (event) {\r\n\r\n /** Prevent default behaviour */\r\n event.preventDefault();\r\n\r\n var editableParent = editor.content.getEditableParent(event.target),\r\n firstLevelBlock = editor.content.getFirstLevelBlock(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 data = event.clipboardData.getData('text/html') || event.clipboardData.getData('text/plain');\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 cleaner = new editor.sanitizer.init(editor.sanitizer.Config.BASIC),\r\n cleanData,\r\n fragment;\r\n\r\n /** Create fragment, that we paste to range after proccesing */\r\n fragment = document.createDocumentFragment();\r\n\r\n cleanData = cleaner.clean(data);\r\n\r\n div.innerHTML = cleanData;\r\n\r\n var node, lastNode;\r\n\r\n /**\r\n * and fill in fragment\r\n */\r\n while (( node = div.firstChild) ) {\r\n\r\n lastNode = fragment.appendChild(node);\r\n\r\n }\r\n\r\n\r\n if (editor.tools[firstLevelBlock.dataset.tool].allowRenderOnPaste) {\r\n\r\n if (editor.paste.pasted(event)) return;\r\n\r\n }\r\n\r\n /**\r\n * work with selection and range\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(fragment);\r\n\r\n /** Preserve the selection */\r\n if (lastNode) {\r\n\r\n range = range.cloneRange();\r\n range.setStartAfter(lastNode);\r\n range.collapse(true);\r\n selection.removeAllRanges();\r\n selection.addRange(range);\r\n\r\n }\r\n\r\n };\r\n\r\n /**\r\n * Sends all mutations to paste handler\r\n */\r\n callbacks.handleMutationsOnPaste = function (mutations) {\r\n\r\n var self = this;\r\n\r\n /**\r\n * Calling function with context of this function.\r\n * Also, we should sanitize pasted or changed data one time and ignore\r\n * changings which makes sanitize method.\r\n * For that, we need to send Context, MutationObserver.__proto__ that contains\r\n * observer disconnect method.\r\n */\r\n mutations.forEach(function (mutation) {\r\n\r\n editor.content.paste.call(self, mutation);\r\n\r\n });\r\n\r\n };\r\n\r\n /**\r\n * Clicks on block settings button\r\n */\r\n callbacks.showSettingsButtonClicked = function () {\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\nlet editor = codex.editor;\r\n\r\nmodule.exports = (function (caret) {\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.tagName == 'INPUT') {\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 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 okBtn.addEventListener('click', confirmHandler);\r\n cancelBtn.addEventListener('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\nlet editor = codex.editor;\r\n\r\nmodule.exports = (function (parser) {\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\nvar janitor = require('html-janitor');\r\n\r\nmodule.exports = (function (sanitizer) {\r\n\r\n /**\r\n * Basic config\r\n */\r\n var Config = {\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 sanitizer.init = janitor;\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 Anchors module\r\n *\r\n * @author Codex Team\r\n * @version 1.0\r\n */\r\n\r\nmodule.exports = function (anchors) {\r\n\r\n let editor = codex.editor;\r\n\r\n anchors.input = null;\r\n anchors.currentNode = null;\r\n\r\n anchors.settingsOpened = function (currentBlock) {\r\n\r\n anchors.currentNode = currentBlock;\r\n anchors.input.value = anchors.currentNode.dataset.anchor || '';\r\n\r\n };\r\n\r\n anchors.anchorChanged = function (e) {\r\n\r\n var newAnchor = e.target.value = anchors.rusToTranslit(e.target.value);\r\n\r\n anchors.currentNode.dataset.anchor = newAnchor;\r\n\r\n if (newAnchor.trim() !== '') {\r\n\r\n anchors.currentNode.classList.add(editor.ui.className.BLOCK_WITH_ANCHOR);\r\n\r\n } else {\r\n\r\n anchors.currentNode.classList.remove(editor.ui.className.BLOCK_WITH_ANCHOR);\r\n\r\n }\r\n\r\n };\r\n\r\n anchors.keyDownOnAnchorInput = function (e) {\r\n\r\n if (e.keyCode == editor.core.keys.ENTER) {\r\n\r\n e.preventDefault();\r\n e.stopPropagation();\r\n\r\n e.target.blur();\r\n editor.toolbar.settings.close();\r\n\r\n }\r\n\r\n };\r\n\r\n anchors.keyUpOnAnchorInput = function (e) {\r\n\r\n if (e.keyCode >= editor.core.keys.LEFT && e.keyCode <= editor.core.keys.DOWN) {\r\n\r\n e.stopPropagation();\r\n\r\n }\r\n\r\n };\r\n\r\n anchors.rusToTranslit = function (string) {\r\n\r\n var ru = [\r\n 'А', 'Б', 'В', 'Г', 'Д', 'Е', 'Ё', 'Ж', 'З', 'И', 'Й',\r\n 'К', 'Л', 'М', 'Н', 'О', 'П', 'Р', 'С', 'Т', 'У', 'Ф',\r\n 'Х', 'Ц', 'Ч', 'Ш', 'Щ', 'Ь', 'Ы', 'Ь', 'Э', 'Ю', 'Я'\r\n ],\r\n en = [\r\n 'A', 'B', 'V', 'G', 'D', 'E', 'E', 'Zh', 'Z', 'I', 'Y',\r\n 'K', 'L', 'M', 'N', 'O', 'P', 'R', 'S', 'T', 'U', 'F',\r\n 'H', 'C', 'Ch', 'Sh', 'Sch', '', 'Y', '', 'E', 'Yu', 'Ya'\r\n ];\r\n\r\n for (var i = 0; i < ru.length; i++) {\r\n\r\n string = string.split(ru[i]).join(en[i]);\r\n string = string.split(ru[i].toLowerCase()).join(en[i].toLowerCase());\r\n\r\n }\r\n\r\n string = string.replace(/[^0-9a-zA-Z_]+/g, '-');\r\n\r\n return string;\r\n\r\n };\r\n\r\n return anchors;\r\n\r\n}({});\n\n\n// WEBPACK FOOTER //\n// ./modules/anchors.js","/**\r\n * Codex Editor Paste module\r\n *\r\n * @author Codex Team\r\n * @version 1.0\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 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 if (pattern.regex.test(string)) {\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 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 7633cb891787da44e4d8","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/anchors.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","anchors","listeners","destroyer","paste","settings","textareaId","uploadImagesUrl","initialBlockPlugin","nodes","textarea","wrapper","inlineToolbar","buttons","actions","toolbox","plusButton","showSettingsButton","showTrashButton","blockSettings","pluginSettings","defaultSettings","toolbarButtons","redactor","state","jsonOutput","blocks","inputs","start","userSettings","prepare","then","make","addTools","bindEvents","makeBlocksFromData","saveInputs","catch","error","log","_typeof","Symbol","iterator","obj","constructor","prototype","Promise","resolve","reject","data","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","XMLHTTP","XMLHttpRequest","ActiveXObject","successFunction","params","async","success","test","encodeURIComponent","withCredentials","beforeSend","open","setRequestHeader","onreadystatechange","readyState","status","responseText","send","importScript","scriptPath","instanceName","script","createElement","defer","onload","onerror","src","head","appendChild","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","BLOCK_IN_FEED_MODE","BLOCK_WITH_ANCHOR","SETTINGS_ITEM","toolbarContent","blockButtons","createHolder","settingsButton","makeRemoveBlockButton","pluginsSettings","makeInlineToolbar","addDefaultSettings","container","inlineToolbarButtons","inlineToolbarActions","tool","toolName","toolButton","iconClassname","render","displayInToolbox","toolbarButton","addInlineToolbarTools","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","addInitialBlock","initialBlock","initialBlockType","setAttribute","insertBlock","workingNodeChanged","event","toolClicked","input","arguments","fileSelected","clearInput","files","formdData","FormData","append","selectAndUpload","args","click","xhr","items","appendBlocks","nodeSequence","index","appendNodeAtIndex","getNodeAsync","createBlockFromData","blockData","blocksList","position","toolData","anchor","cover","unavailableBlock","innerHTML","dataset","inputPosition","stretched","isStretched","saveBlocks","html","childNodes","makeQueue","queue","getBlockData","makeFormDataFromBlocks","save","savedData","output","blockContent","pluginsContent","coverFlag","classList","contains","validate","result","currentNode","editorAreaHightlighted","sync","getNodeFocused","focused","selection","getSelection","anchorNode","focusNode","parentElement","isFirstLevelBlock","parent","markBlock","clearMark","remove","getFirstLevelBlock","node","body","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","getRangeAt","splitBlock","inputIndex","textBeforeCaret","textNodeBeforeCaret","textAfterCaret","textNodeAfterCaret","anchorNodeText","caretOffset","anchorOffset","currentBlock","substring","previousChilds","nextChilds","reachedCurrent","child","i","previousChildsLength","nextChildsLength","newNode","NEW_BLOCK_TYPE","mergeBlocks","targetInputIndex","targetInput","currentInputContent","mutation","workingNode","allowedToPaste","sanitize","pasteTextContent","addedNodes","textNode","disconnect","cleaner","satinizer","Config","BASIC","clean","outerHTML","div","replaceWith","isLastNode","allChecked","allSiblingsEmpty_","sibling","wrapTextWithParagraphs","htmlString","paragraph","blockTyped","newWrapper","firstLevelBlocks","indexOf","tagName","cloneNode","getEditableParent","contentEditable","defaultToolbarHeight","defaultOffset","opened","current","close","toggle","hidePlusButton","newYCoordinate","offsetTop","style","transform","Math","floor","hideRemoveActions","setting","toolType","makeSettings","settingsBlock","settingsOpened","feedModeToggler","anchorInput","makeFeedModeToggler","makeAnchorInput","isFeedModeActivated","updateFeedMode","anchorWrapper","hash","placeholder","keyDownOnAnchorInput","keyUpOnAnchorInput","anchorChanged","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","setToBlock","callbacks","enterKeyPressed","tabKeyPressed","enterKeyPressedOnRedactorZone","escapeKeyPressed","defaultKeyPressed","arrowKeyPressed","enterPressedOnBlock","saveCurrentInputIndex","isEnterPressedOnToolbar","enableLineBreaks","stopPropagation","shiftKey","currentSelection","currentSelectedNode","caretAtTheEndOfText","atTheEnd","isTextNodeHasParentBetweenContenteditable","islastNode","detectWhenClickedOnFirstLevelBlockArea","firstLevelBlock","indexOfLastInput","inputIsEmpty","currentNodeType","isInitialType","flag","blockRightOrDownArrowPressed","backspacePressed","blockLeftOrUpArrowPressed","focusedNodeHolder","focusedNode","editableElementIndex","lastChild","deepestTextnode","caretInLastChild","firstChild","caretInFirstChild","caretAtTheBeginning","setToPreviousBlock","selectionLength","endOffset","atStart","_blockPasteCallback","observer","MutationObserver","handleMutationsOnPaste","attributes","childList","characterData","subtree","observe","editableParent","cleanData","fragment","clipboardData","getData","createDocumentFragment","lastNode","allowRenderOnPaste","pasted","deleteContents","insertNode","setStartAfter","mutations","self","currentToolType","ceBlock","bar","toggler","classname","toolIcon","toolTitle","properties","focusedNodeIndex","nodeToSet","childs","nextInput","emptyTextElement","lastChildNode","lengthOfLastChildNode","previousInput","pluginsRender","isFirstNode","isOffsetZero","addToQueue","splice","holder","errorThrown","errorMsg","notification","message","constructorSettings","create","time","okBtn","cancelBtn","okMsg","cancelMsg","confirmHandler","cancelHandler","confirm","cancel","inputField","clear","insertPastedContent","janitor","tags","a","href","rel","b","strong","em","span","__WEBPACK_AMD_DEFINE_FACTORY__","__WEBPACK_AMD_DEFINE_RESULT__","root","factory","HTMLJanitor","tagDefinitions","validConfigValues","map","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","Array","some","isNotTopContainer","isNestedBlockElement","isInvalid","keepNestedBlockElements","removeAttribute","previousElementSibling","nextElementSibling","newAnchor","rusToTranslit","blur","string","ru","en","split","join","replace","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","clipBoardData","analize","regex","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.5.0\");\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.anchors = __webpack_require__(19);\n\t editor.listeners = __webpack_require__(20);\n\t editor.destroyer = __webpack_require__(21);\n\t editor.paste = __webpack_require__(22);\n\t };\n\t\n\t /**\r\n\t * @public\r\n\t *\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 textareaId: 'codex-editor',\n\t uploadImagesUrl: '/editor/transport/',\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 textarea: 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 {} userSettings are :\r\n\t * - tools [],\r\n\t * - textareaId String\r\n\t * ...\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.make).then(editor.ui.addTools).then(editor.ui.bindEvents).then(editor.tools.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.2\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.uploadImagesUrl) {\n\t\n\t editor.settings.uploadImagesUrl = userSettings.uploadImagesUrl;\n\t }\n\t\n\t editor.nodes.textarea = document.getElementById(userSettings.textareaId || editor.settings.textareaId);\n\t\n\t if (_typeof(editor.nodes.textarea) === undefined || editor.nodes.textarea === null) {\n\t\n\t reject(Error(\"Textarea wasn't found by ID: #\" + userSettings.textareaId));\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 */\n\t core.ajax = function (data) {\n\t\n\t if (!data || !data.url) {\n\t\n\t return;\n\t }\n\t\n\t var XMLHTTP = window.XMLHttpRequest ? new XMLHttpRequest() : new ActiveXObject('Microsoft.XMLHTTP'),\n\t successFunction = function successFunction() {},\n\t params = '',\n\t obj;\n\t\n\t data.async = true;\n\t data.type = data.type || 'GET';\n\t data.data = data.data || '';\n\t data['content-type'] = data['content-type'] || 'application/json; charset=utf-8';\n\t successFunction = data.success || successFunction;\n\t\n\t if (data.type == 'GET' && data.data) {\n\t\n\t data.url = /\\?/.test(data.url) ? data.url + '&' + data.data : data.url + '?' + data.data;\n\t } else {\n\t\n\t for (obj in data.data) {\n\t\n\t params += obj + '=' + encodeURIComponent(data.data[obj]) + '&';\n\t }\n\t }\n\t\n\t if (data.withCredentials) {\n\t\n\t XMLHTTP.withCredentials = true;\n\t }\n\t\n\t if (data.beforeSend && typeof data.beforeSend == 'function') {\n\t\n\t data.beforeSend.call();\n\t }\n\t\n\t XMLHTTP.open(data.type, data.url, data.async);\n\t XMLHTTP.setRequestHeader('X-Requested-With', 'XMLHttpRequest');\n\t XMLHTTP.setRequestHeader('Content-type', 'application/x-www-form-urlencoded');\n\t\n\t XMLHTTP.onreadystatechange = function () {\n\t\n\t if (XMLHTTP.readyState == 4 && XMLHTTP.status == 200) {\n\t\n\t successFunction(XMLHTTP.responseText);\n\t }\n\t };\n\t\n\t XMLHTTP.send(params);\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 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.1\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} - highlights covered blocks\r\n\t */\n\t BLOCK_IN_FEED_MODE: 'ce-block--feed-mode',\n\t\n\t /**\r\n\t * @const {String} - Block with anchor\r\n\t */\n\t BLOCK_WITH_ANCHOR: 'ce-block--anchor',\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.make = function () {\n\t\n\t var wrapper, toolbar, toolbarContent, redactor, blockButtons, blockSettings, showSettingsButton, showTrashButton, toolbox, plusButton;\n\t\n\t /** Make editor wrapper */\n\t wrapper = editor.draw.wrapper();\n\t\n\t /** Append editor wrapper after initial textarea */\n\t editor.core.insertAfter(editor.nodes.textarea, wrapper);\n\t\n\t /** Append block with notifications to the document */\n\t editor.notifications.createHolder();\n\t\n\t /** Make toolbar and content-editable redactor */\n\t toolbar = editor.draw.toolbar();\n\t toolbarContent = editor.draw.toolbarContent();\n\t plusButton = editor.draw.plusButton();\n\t showSettingsButton = editor.draw.settingsButton();\n\t showTrashButton = editor.toolbar.settings.makeRemoveBlockButton();\n\t blockSettings = editor.draw.blockSettings();\n\t blockButtons = editor.draw.blockButtons();\n\t toolbox = editor.draw.toolbox();\n\t redactor = editor.draw.redactor();\n\t\n\t /** settings */\n\t var defaultSettings = editor.draw.defaultSettings(),\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 /** 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 /** Append plus button */\n\t toolbarContent.appendChild(plusButton);\n\t\n\t /** Appending toolbar tools */\n\t toolbarContent.appendChild(toolbox);\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 wrapper.appendChild(toolbar);\n\t\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.toolbar = toolbar;\n\t editor.nodes.plusButton = plusButton;\n\t editor.nodes.toolbox = toolbox;\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 editor.nodes.redactor = redactor;\n\t\n\t /** Make container for inline toolbar */\n\t editor.ui.makeInlineToolbar();\n\t\n\t /** fill in default settings */\n\t editor.toolbar.settings.addDefaultSettings();\n\t };\n\t\n\t ui.makeInlineToolbar = function () {\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 /**\r\n\t * @private\r\n\t * Append tools passed in editor.tools\r\n\t */\n\t ui.addTools = function () {\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) {\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 /**\r\n\t * Add inline toolbar tools\r\n\t */\n\t editor.ui.addInlineToolbarTools();\n\t };\n\t\n\t ui.addInlineToolbarTools = function () {\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 ui.bindEvents = function () {\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 /**\r\n\t * @deprecated ( but now in use for syncronization );\r\n\t * Any redactor changes: keyboard input, mouse cut/paste, drag-n-drop text\r\n\t */\n\t // editor.nodes.redactor.addEventListener('input', editor.callback.redactorInputEvent, 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 /** Save all inputs in global variable state */\n\t editor.state.inputs = redactor.querySelectorAll('[contenteditable], input');\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 * @author Codex Team\r\n\t * @version 1.0\r\n\t */\n\t\n\tmodule.exports = function (transport) {\n\t\n\t var editor = codex.editor;\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 transport.prepare = function () {\n\t\n\t var input = document.createElement('INPUT');\n\t\n\t input.type = 'file';\n\t editor.listeners.add(input, 'change', editor.transport.fileSelected);\n\t\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 this.input = null;\n\t\n\t /** Prepare new one */\n\t this.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 files = input.files,\n\t formdData = new FormData();\n\t\n\t formdData.append('files', files[0], files[0].name);\n\t\n\t editor.transport.ajax({\n\t data: formdData,\n\t beforeSend: editor.transport.arguments.beforeSend,\n\t success: editor.transport.arguments.success,\n\t error: editor.transport.arguments.error\n\t });\n\t };\n\t\n\t /**\r\n\t * Use plugin callbacks\r\n\t * @protected\r\n\t */\n\t transport.selectAndUpload = function (args) {\n\t\n\t this.arguments = args;\n\t this.input.click();\n\t };\n\t\n\t /**\r\n\t * Ajax requests module\r\n\t * @todo use core.ajax\r\n\t */\n\t transport.ajax = function (params) {\n\t\n\t var xhr = new XMLHttpRequest(),\n\t beforeSend = typeof params.beforeSend == 'function' ? params.beforeSend : function () {},\n\t success = typeof params.success == 'function' ? params.success : function () {},\n\t error = typeof params.error == 'function' ? params.error : function () {};\n\t\n\t beforeSend();\n\t\n\t xhr.open('POST', editor.settings.uploadImagesUrl, true);\n\t\n\t xhr.setRequestHeader('X-Requested-With', 'XMLHttpRequest');\n\t\n\t xhr.onload = function () {\n\t\n\t if (xhr.status === 200) {\n\t\n\t success(xhr.responseText);\n\t } else {\n\t\n\t editor.core.log('request error: %o', xhr);\n\t error();\n\t }\n\t };\n\t\n\t xhr.send(params.data);\n\t this.clearInput();\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 anchor = tool.anchor,\n\t cover = tool.cover;\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 cover: cover,\n\t anchor: anchor\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 anchor = block.dataset.anchor;\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 coverFlag = false;\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\n\t savedData = codex.editor.state.blocks.items[position].data;\n\t coverFlag = codex.editor.state.blocks.items[position].cover;\n\t anchor = codex.editor.state.blocks.items[position].anchor;\n\t } else {\n\t\n\t savedData = editor.tools[pluginName].save(pluginsContent);\n\t coverFlag = block.classList.contains(editor.ui.className.BLOCK_IN_FEED_MODE);\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 anchor: anchor,\n\t data: savedData\n\t };\n\t\n\t /** Marks Blocks that will be in main page */\n\t output.cover = coverFlag;\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 * @author Codex Team\r\n\t * @version 1.3.11\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 * 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 * @deprecated\r\n\t */\n\t content.getNodeFocused = function () {\n\t\n\t var selection = window.getSelection(),\n\t focused;\n\t\n\t if (selection.anchorNode === null) {\n\t\n\t return null;\n\t }\n\t\n\t if (selection.anchorNode.nodeType == editor.core.nodeTypes.TAG) {\n\t\n\t focused = selection.anchorNode;\n\t } else {\n\t\n\t focused = selection.focusNode.parentElement;\n\t }\n\t\n\t if (!editor.parser.isFirstLevelBlock(focused)) {\n\t\n\t /** Iterate with parent nodes to find first-level*/\n\t var parent = focused.parentNode;\n\t\n\t while (parent && !editor.parser.isFirstLevelBlock(parent)) {\n\t\n\t parent = parent.parentNode;\n\t }\n\t\n\t focused = parent;\n\t }\n\t\n\t if (focused != editor.nodes.redactor) {\n\t\n\t return focused;\n\t }\n\t\n\t return null;\n\t };\n\t\n\t /**\r\n\t * Appends background to the block\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 */\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 * @private\r\n\t *\r\n\t * Finds first-level block\r\n\t * @param {Element} node - selected or clicked in redactors area node\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 * 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 this.currentNode = this.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 /**\r\n\t * Check is this block was in feed\r\n\t * If true, than set switched block also covered\r\n\t */\n\t if (targetBlock.classList.contains(editor.ui.className.BLOCK_IN_FEED_MODE)) {\n\t\n\t newBlock.classList.add(editor.ui.className.BLOCK_IN_FEED_MODE);\n\t }\n\t\n\t if (targetBlock.classList.contains(editor.ui.className.BLOCK_WITH_ANCHOR)) {\n\t\n\t newBlock.classList.add(editor.ui.className.BLOCK_WITH_ANCHOR);\n\t }\n\t\n\t /**\r\n\t * Saving anchor\r\n\t */\n\t newBlock.dataset.anchor = targetBlock.dataset.anchor;\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 * @private\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 cover = blockData.cover,\n\t anchor = blockData.anchor,\n\t isStretched = blockData.stretched;\n\t\n\t var newBlock = editor.content.composeNewBlock(newBlockContent, blockType, isStretched, anchor);\n\t\n\t if (cover === true) {\n\t\n\t newBlock.classList.add(editor.ui.className.BLOCK_IN_FEED_MODE);\n\t }\n\t\n\t if (anchor) {\n\t\n\t newBlock.classList.add(editor.ui.className.BLOCK_WITH_ANCHOR);\n\t }\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 var newBlockComposed = editor.content.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 * @private\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 */\n\t content.composeNewBlock = function (block, tool, isStretched, anchor) {\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 newBlock.dataset.anchor = anchor || '';\n\t return newBlock;\n\t };\n\t\n\t /**\r\n\t * Returns Range object of current selection\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 * @private\r\n\t * @param {Int} inputIndex - target input index\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 */\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 * @private\r\n\t *\r\n\t * Callback for HTML Mutations\r\n\t * @param {Array} mutation - Mutation Record\r\n\t */\n\t content.paste = function (mutation) {\n\t\n\t var workingNode = editor.content.currentNode,\n\t tool = workingNode.dataset.tool;\n\t\n\t if (editor.tools[tool].allowedToPaste) {\n\t\n\t editor.content.sanitize.call(this, mutation.target);\n\t } else {\n\t\n\t editor.content.pasteTextContent(mutation.addedNodes);\n\t }\n\t };\n\t\n\t /**\r\n\t * @private\r\n\t *\r\n\t * gets only text/plain content of node\r\n\t * @param {Element} target - HTML node\r\n\t */\n\t content.pasteTextContent = function (nodes) {\n\t\n\t var node = nodes[0],\n\t textNode;\n\t\n\t if (!node) {\n\t\n\t return;\n\t }\n\t\n\t if (node.nodeType == editor.core.nodeTypes.TEXT) {\n\t\n\t textNode = document.createTextNode(node);\n\t } else {\n\t\n\t textNode = document.createTextNode(node.textContent);\n\t }\n\t\n\t if (editor.core.isDomNode(node)) {\n\t\n\t node.parentNode.replaceChild(textNode, node);\n\t }\n\t };\n\t\n\t /**\r\n\t * @private\r\n\t *\r\n\t * Sanitizes HTML content\r\n\t * @param {Element} target - inserted element\r\n\t * @uses Sanitize library html-janitor\r\n\t */\n\t content.sanitize = function (target) {\n\t\n\t if (!target) {\n\t\n\t return;\n\t }\n\t\n\t var node = target[0];\n\t\n\t if (!node) {\n\t\n\t return;\n\t }\n\t\n\t /**\r\n\t * Disconnect Observer\r\n\t * hierarchy of function calls inherits context of observer\r\n\t */\n\t this.disconnect();\n\t\n\t /**\r\n\t * Don't sanitize text node\r\n\t */\n\t if (node.nodeType == editor.core.nodeTypes.TEXT) {\n\t\n\t return;\n\t }\n\t\n\t /**\r\n\t * Clear dirty content\r\n\t */\n\t var cleaner = editor.sanitizer.init(editor.satinizer.Config.BASIC),\n\t clean = cleaner.clean(target.outerHTML);\n\t\n\t var div = editor.draw.node('DIV', [], { innerHTML: clean });\n\t\n\t node.replaceWith(div.childNodes[0]);\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] htmlString - html content as string\r\n\t * @return {string} - html content as string\r\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 /**\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 = 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 /**\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 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 settings.cover = 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 = `Плагин «${toolType}» не имеет настроек`;\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 editor.toolbar.settings.addDefaultSettings();\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 editor.anchors.settingsOpened(editor.content.currentNode);\n\t } else {\n\t\n\t this.close();\n\t }\n\t };\n\t\n\t /**\r\n\t * This function adds default core settings\r\n\t */\n\t settings.addDefaultSettings = function () {\n\t\n\t /** list of default settings */\n\t var feedModeToggler, anchorInput;\n\t\n\t /** Clear block and append initialized settings */\n\t editor.nodes.defaultSettings.innerHTML = '';\n\t\n\t /** Init all default setting buttons */\n\t feedModeToggler = editor.toolbar.settings.makeFeedModeToggler();\n\t anchorInput = editor.toolbar.settings.makeAnchorInput();\n\t\n\t /**\r\n\t * Fill defaultSettings\r\n\t */\n\t\n\t /**\r\n\t * Input for anchor for block\r\n\t */\n\t editor.nodes.defaultSettings.appendChild(anchorInput);\n\t\n\t /**\r\n\t * Button that enables/disables Feed-mode\r\n\t * Feed-mode means that block will be showed in articles-feed like cover\r\n\t */\n\t editor.nodes.defaultSettings.appendChild(feedModeToggler);\n\t };\n\t\n\t /**\r\n\t * Cover setting.\r\n\t * This tune highlights block, so that it may be used for showing target block on main page\r\n\t * Draw different setting when block is marked for main page\r\n\t * If TRUE, then we show button that removes this selection\r\n\t * Also defined setting \"Click\" events will be listened and have separate callbacks\r\n\t *\r\n\t * @return {Element} node/button that we place in default settings block\r\n\t */\n\t settings.makeFeedModeToggler = function () {\n\t\n\t var isFeedModeActivated = editor.toolbar.settings.isFeedModeActivated(),\n\t setting,\n\t data;\n\t\n\t if (!isFeedModeActivated) {\n\t\n\t data = {\n\t innerHTML: 'Вывести в ленте'\n\t };\n\t } else {\n\t\n\t data = {\n\t innerHTML: 'Не выводить в ленте'\n\t };\n\t }\n\t\n\t setting = editor.draw.node('DIV', editor.ui.className.SETTINGS_ITEM, data);\n\t editor.listeners.add(setting, 'click', editor.toolbar.settings.updateFeedMode, false);\n\t\n\t return setting;\n\t };\n\t\n\t /**\r\n\t * Updates Feed-mode\r\n\t */\n\t settings.updateFeedMode = function () {\n\t\n\t var currentNode = editor.content.currentNode;\n\t\n\t currentNode.classList.toggle(editor.ui.className.BLOCK_IN_FEED_MODE);\n\t\n\t editor.toolbar.settings.close();\n\t };\n\t\n\t settings.isFeedModeActivated = function () {\n\t\n\t var currentBlock = editor.content.currentNode;\n\t\n\t if (currentBlock) {\n\t\n\t return currentBlock.classList.contains(editor.ui.className.BLOCK_IN_FEED_MODE);\n\t } else {\n\t\n\t return false;\n\t }\n\t };\n\t\n\t settings.makeAnchorInput = function () {\n\t\n\t var anchorWrapper = editor.draw.node('div', 'ce-settings__anchor-wrapper ce-settings__item', {}),\n\t hash = editor.draw.node('i', 'ce-settings__anchor-hash', {}),\n\t anchor = editor.draw.node('input', 'ce-settings__anchor-input', { placeholder: 'Якорь' });\n\t\n\t editor.listeners.add(anchor, 'keydown', editor.anchors.keyDownOnAnchorInput);\n\t editor.listeners.add(anchor, 'keyup', editor.anchors.keyUpOnAnchorInput);\n\t editor.listeners.add(anchor, 'input', editor.anchors.anchorChanged);\n\t editor.listeners.add(anchor, 'blur', editor.anchors.anchorChanged);\n\t\n\t anchorWrapper.appendChild(hash);\n\t anchorWrapper.appendChild(anchor);\n\t\n\t editor.anchors.input = anchor;\n\t\n\t return anchorWrapper;\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\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;\n\t visibleTool = tools[nextToolIndex];\n\t\n\t while (!editor.tools[visibleTool].displayInToolbox) {\n\t\n\t nextToolIndex++;\n\t visibleTool = tools[nextToolIndex];\n\t\n\t if (nextToolIndex == tools.length) {\n\t\n\t nextToolIndex = 0;\n\t visibleTool = tools[nextToolIndex];\n\t }\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 * Codex Editor callbacks module\r\n\t *\r\n\t * @author Codex Team\r\n\t * @version 1.3.7\r\n\t */\n\t\n\tmodule.exports = function (callbacks) {\n\t\n\t var editor = codex.editor;\n\t\n\t callbacks.globalKeydown = function (event) {\n\t\n\t switch (event.keyCode) {\n\t case editor.core.keys.ENTER:\n\t editor.callback.enterKeyPressed(event);break;\n\t }\n\t };\n\t\n\t callbacks.redactorKeyDown = function (event) {\n\t\n\t switch (event.keyCode) {\n\t case editor.core.keys.TAB:\n\t editor.callback.tabKeyPressed(event);break;\n\t case editor.core.keys.ENTER:\n\t editor.callback.enterKeyPressedOnRedactorZone(event);break;\n\t case editor.core.keys.ESC:\n\t editor.callback.escapeKeyPressed(event);break;\n\t default:\n\t editor.callback.defaultKeyPressed(event);break;\n\t }\n\t };\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 editor.callback.arrowKeyPressed(event);break;\n\t }\n\t };\n\t\n\t callbacks.tabKeyPressed = function (event) {\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 event.preventDefault();\n\t };\n\t\n\t /**\r\n\t * @param {Event} event\r\n\t */\n\t callbacks.enterKeyPressed = function () {\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 editor.callback.enterPressedOnBlock();\n\t }\n\t };\n\t\n\t /**\r\n\t * ENTER key handler\r\n\t * Makes new paragraph block\r\n\t */\n\t callbacks.enterKeyPressedOnRedactorZone = function (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 callbacks.escapeKeyPressed = function (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 */\n\t callbacks.arrowKeyPressed = function () {\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 * @param {Event} event\r\n\t */\n\t callbacks.defaultKeyPressed = function () {\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 callbacks.redactorClicked = function (event) {\n\t\n\t callbacks.detectWhenClickedOnFirstLevelBlockArea();\n\t\n\t editor.content.workingNodeChanged(event.target);\n\t\n\t editor.ui.saveInputs();\n\t\n\t var selectedText = editor.toolbar.inline.getSelectionText(),\n\t firstLevelBlock;\n\t\n\t /**\r\n\t * If selection range took off, then we hide inline toolbar\r\n\t */\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 /**\r\n\t * @todo Refactor\r\n\t */\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\n\t /**\r\n\t * Move toolbar to the right position and open\r\n\t */\n\t editor.toolbar.move();\n\t editor.toolbar.open();\n\t } else {\n\t\n\t /**\r\n\t * Move toolbar to the new position and open\r\n\t */\n\t editor.toolbar.move();\n\t editor.toolbar.open();\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 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 /** Mark current block */\n\t editor.content.markBlock();\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 * 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 callbacks.detectWhenClickedOnFirstLevelBlockArea = function () {\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 ? false : true;\n\t }\n\t };\n\t\n\t /**\r\n\t * Toolbar button click handler\r\n\t * @param this - cursor to the button\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 /** Show or Hide toolbox when plus button is clicked */\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 */\n\t callbacks.blockKeydown = function (event) {\n\t\n\t var block = this; // event.target 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 editor.callback.blockRightOrDownArrowPressed();\n\t break;\n\t\n\t case editor.core.keys.BACKSPACE:\n\t editor.callback.backspacePressed(block, event);\n\t break;\n\t\n\t case editor.core.keys.UP:\n\t case editor.core.keys.LEFT:\n\t editor.callback.blockLeftOrUpArrowPressed();\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 */\n\t callbacks.blockRightOrDownArrowPressed = function () {\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 */\n\t callbacks.blockLeftOrUpArrowPressed = function () {\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 * Callback for enter key pressing in first-level block area\r\n\t * @param {Event} event\r\n\t */\n\t callbacks.enterPressedOnBlock = function () {\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 callbacks.backspacePressed = function (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 (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 is used to observe pasted dirty data.\r\n\t *\r\n\t * Mutation handlers send to separate observers each mutation (added, changed and so on), which will be\r\n\t * passed from handler that sanitizes and replaces data.\r\n\t *\r\n\t * Probably won't be used\r\n\t *\r\n\t * @deprecated\r\n\t *\r\n\t * @param event\r\n\t * @private\r\n\t */\n\t callbacks._blockPasteCallback = function () {\n\t\n\t var currentInputIndex = editor.caret.getCurrentInputIndex();\n\t\n\t /**\r\n\t * create an observer instance\r\n\t */\n\t var observer = new MutationObserver(editor.callback.handleMutationsOnPaste);\n\t\n\t /**\r\n\t * configuration of the observer:\r\n\t */\n\t var config = {\n\t attributes: true,\n\t childList: false,\n\t characterData: false,\n\t subtree: true\n\t };\n\t\n\t // pass in the target node, as well as the observer options\n\t observer.observe(editor.state.inputs[currentInputIndex], config);\n\t };\n\t\n\t /**\r\n\t * This method prevents default behaviour.\r\n\t *\r\n\t * 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 *\r\n\t * @param event\r\n\t */\n\t callbacks.blockPasteCallback = function (event) {\n\t\n\t /** Prevent default behaviour */\n\t event.preventDefault();\n\t\n\t var editableParent = editor.content.getEditableParent(event.target),\n\t firstLevelBlock = editor.content.getFirstLevelBlock(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 data = event.clipboardData.getData('text/html') || 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 cleaner = new editor.sanitizer.init(editor.sanitizer.Config.BASIC),\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 cleanData = cleaner.clean(data);\n\t\n\t div.innerHTML = cleanData;\n\t\n\t var node, lastNode;\n\t\n\t /**\r\n\t * and fill in fragment\r\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[firstLevelBlock.dataset.tool].allowRenderOnPaste) {\n\t\n\t if (editor.paste.pasted(event)) return;\n\t }\n\t\n\t /**\r\n\t * work with selection and range\r\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 /**\r\n\t * Sends all mutations to paste handler\r\n\t */\n\t callbacks.handleMutationsOnPaste = function (mutations) {\n\t\n\t var self = this;\n\t\n\t /**\r\n\t * Calling function with context of this function.\r\n\t * Also, we should sanitize pasted or changed data one time and ignore\r\n\t * changings which makes sanitize method.\r\n\t * For that, we need to send Context, MutationObserver.__proto__ that contains\r\n\t * observer disconnect method.\r\n\t */\n\t mutations.forEach(function (mutation) {\n\t\n\t editor.content.paste.call(self, mutation);\n\t });\n\t };\n\t\n\t /**\r\n\t * Clicks on block settings button\r\n\t */\n\t callbacks.showSettingsButtonClicked = function () {\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.tagName == 'INPUT') {\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 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 var janitor = __webpack_require__(18);\n\t\n\t /**\r\n\t * Basic config\r\n\t */\n\t var Config = {\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 sanitizer.init = janitor;\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 Anchors 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 (anchors) {\n\t\n\t var editor = codex.editor;\n\t\n\t anchors.input = null;\n\t anchors.currentNode = null;\n\t\n\t anchors.settingsOpened = function (currentBlock) {\n\t\n\t anchors.currentNode = currentBlock;\n\t anchors.input.value = anchors.currentNode.dataset.anchor || '';\n\t };\n\t\n\t anchors.anchorChanged = function (e) {\n\t\n\t var newAnchor = e.target.value = anchors.rusToTranslit(e.target.value);\n\t\n\t anchors.currentNode.dataset.anchor = newAnchor;\n\t\n\t if (newAnchor.trim() !== '') {\n\t\n\t anchors.currentNode.classList.add(editor.ui.className.BLOCK_WITH_ANCHOR);\n\t } else {\n\t\n\t anchors.currentNode.classList.remove(editor.ui.className.BLOCK_WITH_ANCHOR);\n\t }\n\t };\n\t\n\t anchors.keyDownOnAnchorInput = function (e) {\n\t\n\t if (e.keyCode == editor.core.keys.ENTER) {\n\t\n\t e.preventDefault();\n\t e.stopPropagation();\n\t\n\t e.target.blur();\n\t editor.toolbar.settings.close();\n\t }\n\t };\n\t\n\t anchors.keyUpOnAnchorInput = function (e) {\n\t\n\t if (e.keyCode >= editor.core.keys.LEFT && e.keyCode <= editor.core.keys.DOWN) {\n\t\n\t e.stopPropagation();\n\t }\n\t };\n\t\n\t anchors.rusToTranslit = function (string) {\n\t\n\t var ru = ['А', 'Б', 'В', 'Г', 'Д', 'Е', 'Ё', 'Ж', 'З', 'И', 'Й', 'К', 'Л', 'М', 'Н', 'О', 'П', 'Р', 'С', 'Т', 'У', 'Ф', 'Х', 'Ц', 'Ч', 'Ш', 'Щ', 'Ь', 'Ы', 'Ь', 'Э', 'Ю', 'Я'],\n\t en = ['A', 'B', 'V', 'G', 'D', 'E', 'E', 'Zh', 'Z', 'I', 'Y', 'K', 'L', 'M', 'N', 'O', 'P', 'R', 'S', 'T', 'U', 'F', 'H', 'C', 'Ch', 'Sh', 'Sch', '', 'Y', '', 'E', 'Yu', 'Ya'];\n\t\n\t for (var i = 0; i < ru.length; i++) {\n\t\n\t string = string.split(ru[i]).join(en[i]);\n\t string = string.split(ru[i].toLowerCase()).join(en[i].toLowerCase());\n\t }\n\t\n\t string = string.replace(/[^0-9a-zA-Z_]+/g, '-');\n\t\n\t return string;\n\t };\n\t\n\t return anchors;\n\t}({});\n\n/***/ },\n/* 20 */\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/* 21 */\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/* 22 */\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.0\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 if (pattern.regex.test(string)) {\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 7633cb891787da44e4d8","/**\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.anchors = require('./modules/anchors');\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 *\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 textareaId: 'codex-editor',\r\n uploadImagesUrl: '/editor/transport/',\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 textarea : 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 {} userSettings are :\r\n * - tools [],\r\n * - textareaId String\r\n * ...\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.make)\r\n .then(editor.ui.addTools)\r\n .then(editor.ui.bindEvents)\r\n .then(editor.tools.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})({});\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.2\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.uploadImagesUrl) {\r\n\r\n editor.settings.uploadImagesUrl = userSettings.uploadImagesUrl;\r\n\r\n }\r\n\r\n editor.nodes.textarea = document.getElementById(userSettings.textareaId || editor.settings.textareaId);\r\n\r\n if (typeof editor.nodes.textarea === undefined || editor.nodes.textarea === null) {\r\n\r\n reject(Error(\"Textarea wasn't found by ID: #\" + userSettings.textareaId));\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 */\r\n core.ajax = function (data) {\r\n\r\n if (!data || !data.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 successFunction = function () {},\r\n params = '',\r\n obj;\r\n\r\n data.async = true;\r\n data.type = data.type || 'GET';\r\n data.data = data.data || '';\r\n data['content-type'] = data['content-type'] || 'application/json; charset=utf-8';\r\n successFunction = data.success || successFunction ;\r\n\r\n if (data.type == 'GET' && data.data) {\r\n\r\n data.url = /\\?/.test(data.url) ? data.url + '&' + data.data : data.url + '?' + data.data;\r\n\r\n } else {\r\n\r\n for(obj in data.data) {\r\n\r\n params += (obj + '=' + encodeURIComponent(data.data[obj]) + '&');\r\n\r\n }\r\n\r\n }\r\n\r\n if (data.withCredentials) {\r\n\r\n XMLHTTP.withCredentials = true;\r\n\r\n }\r\n\r\n if (data.beforeSend && typeof data.beforeSend == 'function') {\r\n\r\n data.beforeSend.call();\r\n\r\n }\r\n\r\n XMLHTTP.open( data.type, data.url, data.async );\r\n XMLHTTP.setRequestHeader('X-Requested-With', 'XMLHttpRequest');\r\n XMLHTTP.setRequestHeader('Content-type', 'application/x-www-form-urlencoded');\r\n\r\n XMLHTTP.onreadystatechange = function () {\r\n\r\n if (XMLHTTP.readyState == 4 && XMLHTTP.status == 200) {\r\n\r\n successFunction(XMLHTTP.responseText);\r\n\r\n }\r\n\r\n };\r\n\r\n XMLHTTP.send(params);\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 return core;\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.1\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} - highlights covered blocks\r\n */\r\n BLOCK_IN_FEED_MODE : 'ce-block--feed-mode',\r\n\r\n /**\r\n * @const {String} - Block with anchor\r\n */\r\n BLOCK_WITH_ANCHOR : 'ce-block--anchor',\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.make = function () {\r\n\r\n var wrapper,\r\n toolbar,\r\n toolbarContent,\r\n redactor,\r\n blockButtons,\r\n blockSettings,\r\n showSettingsButton,\r\n showTrashButton,\r\n toolbox,\r\n plusButton;\r\n\r\n /** Make editor wrapper */\r\n wrapper = editor.draw.wrapper();\r\n\r\n /** Append editor wrapper after initial textarea */\r\n editor.core.insertAfter(editor.nodes.textarea, wrapper);\r\n\r\n /** Append block with notifications to the document */\r\n editor.notifications.createHolder();\r\n\r\n /** Make toolbar and content-editable redactor */\r\n toolbar = editor.draw.toolbar();\r\n toolbarContent = editor.draw.toolbarContent();\r\n plusButton = editor.draw.plusButton();\r\n showSettingsButton = editor.draw.settingsButton();\r\n showTrashButton = editor.toolbar.settings.makeRemoveBlockButton();\r\n blockSettings = editor.draw.blockSettings();\r\n blockButtons = editor.draw.blockButtons();\r\n toolbox = editor.draw.toolbox();\r\n redactor = editor.draw.redactor();\r\n\r\n /** settings */\r\n var defaultSettings = editor.draw.defaultSettings(),\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 /** 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 /** Append plus button */\r\n toolbarContent.appendChild(plusButton);\r\n\r\n /** Appending toolbar tools */\r\n toolbarContent.appendChild(toolbox);\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 wrapper.appendChild(toolbar);\r\n\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.toolbar = toolbar;\r\n editor.nodes.plusButton = plusButton;\r\n editor.nodes.toolbox = toolbox;\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 editor.nodes.redactor = redactor;\r\n\r\n /** Make container for inline toolbar */\r\n editor.ui.makeInlineToolbar();\r\n\r\n /** fill in default settings */\r\n editor.toolbar.settings.addDefaultSettings();\r\n\r\n };\r\n\r\n ui.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 /**\r\n * @private\r\n * Append tools passed in editor.tools\r\n */\r\n ui.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) {\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 * Add inline toolbar tools\r\n */\r\n editor.ui.addInlineToolbarTools();\r\n\r\n\r\n };\r\n\r\n ui.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 ui.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 /**\r\n * @deprecated ( but now in use for syncronization );\r\n * Any redactor changes: keyboard input, mouse cut/paste, drag-n-drop text\r\n */\r\n // editor.nodes.redactor.addEventListener('input', editor.callback.redactorInputEvent, 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 /** Save all inputs in global variable state */\r\n editor.state.inputs = redactor.querySelectorAll('[contenteditable], input');\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})({});\n\n\n// WEBPACK FOOTER //\n// ./modules/ui.js","/**\r\n *\r\n * Codex.Editor Transport Module\r\n *\r\n * @author Codex Team\r\n * @version 1.0\r\n */\r\n\r\nmodule.exports = (function (transport) {\r\n\r\n let editor = codex.editor;\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 transport.prepare = function () {\r\n\r\n var input = document.createElement('INPUT');\r\n\r\n input.type = 'file';\r\n editor.listeners.add(input, 'change', editor.transport.fileSelected);\r\n\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 this.input = null;\r\n\r\n /** Prepare new one */\r\n this.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 files = input.files,\r\n formdData = new FormData();\r\n\r\n formdData.append('files', files[0], files[0].name);\r\n\r\n editor.transport.ajax({\r\n data : formdData,\r\n beforeSend : editor.transport.arguments.beforeSend,\r\n success : editor.transport.arguments.success,\r\n error : editor.transport.arguments.error\r\n });\r\n\r\n };\r\n\r\n /**\r\n * Use plugin callbacks\r\n * @protected\r\n */\r\n transport.selectAndUpload = function (args) {\r\n\r\n this.arguments = args;\r\n this.input.click();\r\n\r\n };\r\n\r\n /**\r\n * Ajax requests module\r\n * @todo use core.ajax\r\n */\r\n transport.ajax = function (params) {\r\n\r\n var xhr = new XMLHttpRequest(),\r\n beforeSend = typeof params.beforeSend == 'function' ? params.beforeSend : function () {},\r\n success = typeof params.success == 'function' ? params.success : function () {},\r\n error = typeof params.error == 'function' ? params.error : function () {};\r\n\r\n beforeSend();\r\n\r\n xhr.open('POST', editor.settings.uploadImagesUrl, true);\r\n\r\n xhr.setRequestHeader('X-Requested-With', 'XMLHttpRequest');\r\n\r\n xhr.onload = function () {\r\n\r\n if (xhr.status === 200) {\r\n\r\n success(xhr.responseText);\r\n\r\n } else {\r\n\r\n editor.core.log('request error: %o', xhr);\r\n error();\r\n\r\n }\r\n\r\n };\r\n\r\n xhr.send(params.data);\r\n this.clearInput();\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 anchor = tool.anchor,\r\n cover = tool.cover;\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 cover : cover,\r\n anchor : anchor\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 anchor = block.dataset.anchor;\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 coverFlag = false;\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\r\n savedData = codex.editor.state.blocks.items[position].data;\r\n coverFlag = codex.editor.state.blocks.items[position].cover;\r\n anchor = codex.editor.state.blocks.items[position].anchor;\r\n\r\n } else {\r\n\r\n savedData = editor.tools[pluginName].save(pluginsContent);\r\n coverFlag = block.classList.contains(editor.ui.className.BLOCK_IN_FEED_MODE);\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 anchor : anchor,\r\n data : savedData\r\n };\r\n\r\n /** Marks Blocks that will be in main page */\r\n output.cover = coverFlag;\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 * @author Codex Team\r\n * @version 1.3.11\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 * 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 * @deprecated\r\n */\r\n content.getNodeFocused = function () {\r\n\r\n var selection = window.getSelection(),\r\n focused;\r\n\r\n if (selection.anchorNode === null) {\r\n\r\n return null;\r\n\r\n }\r\n\r\n if ( selection.anchorNode.nodeType == editor.core.nodeTypes.TAG ) {\r\n\r\n focused = selection.anchorNode;\r\n\r\n } else {\r\n\r\n focused = selection.focusNode.parentElement;\r\n\r\n }\r\n\r\n if ( !editor.parser.isFirstLevelBlock(focused) ) {\r\n\r\n /** Iterate with parent nodes to find first-level*/\r\n var parent = focused.parentNode;\r\n\r\n while (parent && !editor.parser.isFirstLevelBlock(parent)) {\r\n\r\n parent = parent.parentNode;\r\n\r\n }\r\n\r\n focused = parent;\r\n\r\n }\r\n\r\n if (focused != editor.nodes.redactor) {\r\n\r\n return focused;\r\n\r\n }\r\n\r\n return null;\r\n\r\n };\r\n\r\n /**\r\n * Appends background to the block\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 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 * @private\r\n *\r\n * Finds first-level block\r\n * @param {Element} node - selected or clicked in redactors area node\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 * 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 this.currentNode = this.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 /**\r\n * Check is this block was in feed\r\n * If true, than set switched block also covered\r\n */\r\n if (targetBlock.classList.contains(editor.ui.className.BLOCK_IN_FEED_MODE)) {\r\n\r\n newBlock.classList.add(editor.ui.className.BLOCK_IN_FEED_MODE);\r\n\r\n }\r\n\r\n if (targetBlock.classList.contains(editor.ui.className.BLOCK_WITH_ANCHOR)) {\r\n\r\n newBlock.classList.add(editor.ui.className.BLOCK_WITH_ANCHOR);\r\n\r\n }\r\n\r\n /**\r\n * Saving anchor\r\n */\r\n newBlock.dataset.anchor = targetBlock.dataset.anchor;\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 * @private\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 cover = blockData.cover,\r\n anchor = blockData.anchor,\r\n isStretched = blockData.stretched;\r\n\r\n var newBlock = editor.content.composeNewBlock(newBlockContent, blockType, isStretched, anchor);\r\n\r\n if (cover === true) {\r\n\r\n newBlock.classList.add(editor.ui.className.BLOCK_IN_FEED_MODE);\r\n\r\n }\r\n\r\n if (anchor) {\r\n\r\n newBlock.classList.add(editor.ui.className.BLOCK_WITH_ANCHOR);\r\n\r\n }\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 var newBlockComposed = editor.content.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 * @private\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 */\r\n content.composeNewBlock = function (block, tool, isStretched, anchor) {\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 newBlock.dataset.anchor = anchor || '';\r\n return newBlock;\r\n\r\n };\r\n\r\n /**\r\n * Returns Range object of current selection\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 * @private\r\n * @param {Int} inputIndex - target input index\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 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 * @private\r\n *\r\n * Callback for HTML Mutations\r\n * @param {Array} mutation - Mutation Record\r\n */\r\n content.paste = function (mutation) {\r\n\r\n var workingNode = editor.content.currentNode,\r\n tool = workingNode.dataset.tool;\r\n\r\n if (editor.tools[tool].allowedToPaste) {\r\n\r\n editor.content.sanitize.call(this, mutation.target);\r\n\r\n } else {\r\n\r\n editor.content.pasteTextContent(mutation.addedNodes);\r\n\r\n }\r\n\r\n };\r\n\r\n /**\r\n * @private\r\n *\r\n * gets only text/plain content of node\r\n * @param {Element} target - HTML node\r\n */\r\n content.pasteTextContent = function (nodes) {\r\n\r\n var node = nodes[0],\r\n textNode;\r\n\r\n if (!node) {\r\n\r\n return;\r\n\r\n }\r\n\r\n if (node.nodeType == editor.core.nodeTypes.TEXT) {\r\n\r\n textNode = document.createTextNode(node);\r\n\r\n } else {\r\n\r\n textNode = document.createTextNode(node.textContent);\r\n\r\n }\r\n\r\n if (editor.core.isDomNode(node)) {\r\n\r\n node.parentNode.replaceChild(textNode, node);\r\n\r\n }\r\n\r\n };\r\n\r\n /**\r\n * @private\r\n *\r\n * Sanitizes HTML content\r\n * @param {Element} target - inserted element\r\n * @uses Sanitize library html-janitor\r\n */\r\n content.sanitize = function (target) {\r\n\r\n if (!target) {\r\n\r\n return;\r\n\r\n }\r\n\r\n var node = target[0];\r\n\r\n if (!node) {\r\n\r\n return;\r\n\r\n }\r\n\r\n /**\r\n * Disconnect Observer\r\n * hierarchy of function calls inherits context of observer\r\n */\r\n this.disconnect();\r\n\r\n /**\r\n * Don't sanitize text node\r\n */\r\n if (node.nodeType == editor.core.nodeTypes.TEXT) {\r\n\r\n return;\r\n\r\n }\r\n\r\n /**\r\n * Clear dirty content\r\n */\r\n var cleaner = editor.sanitizer.init(editor.satinizer.Config.BASIC),\r\n clean = cleaner.clean(target.outerHTML);\r\n\r\n var div = editor.draw.node('DIV', [], { innerHTML: clean });\r\n\r\n node.replaceWith(div.childNodes[0]);\r\n\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] htmlString - html content as string\r\n * @return {string} - html content as string\r\n */\r\n content.wrapTextWithParagraphs = function (htmlString) {\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 = htmlString;\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 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 settings.cover = 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 editor.toolbar.settings.addDefaultSettings();\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 editor.anchors.settingsOpened(editor.content.currentNode);\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 * This function adds default core settings\r\n */\r\n settings.addDefaultSettings = function () {\r\n\r\n /** list of default settings */\r\n var feedModeToggler,\r\n anchorInput;\r\n\r\n /** Clear block and append initialized settings */\r\n editor.nodes.defaultSettings.innerHTML = '';\r\n\r\n\r\n /** Init all default setting buttons */\r\n feedModeToggler = editor.toolbar.settings.makeFeedModeToggler();\r\n anchorInput = editor.toolbar.settings.makeAnchorInput();\r\n\r\n /**\r\n * Fill defaultSettings\r\n */\r\n\r\n /**\r\n * Input for anchor for block\r\n */\r\n editor.nodes.defaultSettings.appendChild(anchorInput);\r\n\r\n /**\r\n * Button that enables/disables Feed-mode\r\n * Feed-mode means that block will be showed in articles-feed like cover\r\n */\r\n editor.nodes.defaultSettings.appendChild(feedModeToggler);\r\n\r\n };\r\n\r\n /**\r\n * Cover setting.\r\n * This tune highlights block, so that it may be used for showing target block on main page\r\n * Draw different setting when block is marked for main page\r\n * If TRUE, then we show button that removes this selection\r\n * Also defined setting \"Click\" events will be listened and have separate callbacks\r\n *\r\n * @return {Element} node/button that we place in default settings block\r\n */\r\n settings.makeFeedModeToggler = function () {\r\n\r\n var isFeedModeActivated = editor.toolbar.settings.isFeedModeActivated(),\r\n setting,\r\n data;\r\n\r\n if (!isFeedModeActivated) {\r\n\r\n data = {\r\n innerHTML : 'Вывести в ленте'\r\n };\r\n\r\n } else {\r\n\r\n data = {\r\n innerHTML : 'Не выводить в ленте'\r\n };\r\n\r\n }\r\n\r\n setting = editor.draw.node('DIV', editor.ui.className.SETTINGS_ITEM, data);\r\n editor.listeners.add(setting, 'click', editor.toolbar.settings.updateFeedMode, false);\r\n\r\n return setting;\r\n\r\n };\r\n\r\n /**\r\n * Updates Feed-mode\r\n */\r\n settings.updateFeedMode = function () {\r\n\r\n var currentNode = editor.content.currentNode;\r\n\r\n currentNode.classList.toggle(editor.ui.className.BLOCK_IN_FEED_MODE);\r\n\r\n editor.toolbar.settings.close();\r\n\r\n };\r\n\r\n settings.isFeedModeActivated = function () {\r\n\r\n var currentBlock = editor.content.currentNode;\r\n\r\n if (currentBlock) {\r\n\r\n return currentBlock.classList.contains(editor.ui.className.BLOCK_IN_FEED_MODE);\r\n\r\n } else {\r\n\r\n return false;\r\n\r\n }\r\n\r\n };\r\n\r\n settings.makeAnchorInput = function () {\r\n\r\n var anchorWrapper = editor.draw.node('div', 'ce-settings__anchor-wrapper ce-settings__item', {}),\r\n hash = editor.draw.node('i', 'ce-settings__anchor-hash', {}),\r\n anchor = editor.draw.node('input', 'ce-settings__anchor-input', { placeholder: 'Якорь' });\r\n\r\n editor.listeners.add(anchor, 'keydown', editor.anchors.keyDownOnAnchorInput );\r\n editor.listeners.add(anchor, 'keyup', editor.anchors.keyUpOnAnchorInput );\r\n editor.listeners.add(anchor, 'input', editor.anchors.anchorChanged );\r\n editor.listeners.add(anchor, 'blur', editor.anchors.anchorChanged );\r\n\r\n anchorWrapper.appendChild(hash);\r\n anchorWrapper.appendChild(anchor);\r\n\r\n editor.anchors.input = anchor;\r\n\r\n return anchorWrapper;\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 };\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;\r\n visibleTool = tools[nextToolIndex];\r\n\r\n while (!editor.tools[visibleTool].displayInToolbox) {\r\n\r\n nextToolIndex++;\r\n visibleTool = tools[nextToolIndex];\r\n\r\n if ( nextToolIndex == tools.length ) {\r\n\r\n nextToolIndex = 0;\r\n visibleTool = tools[nextToolIndex];\r\n\r\n }\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 * Codex Editor callbacks module\r\n *\r\n * @author Codex Team\r\n * @version 1.3.7\r\n */\r\n\r\nmodule.exports = (function (callbacks) {\r\n\r\n let editor = codex.editor;\r\n\r\n callbacks.globalKeydown = function (event) {\r\n\r\n switch (event.keyCode) {\r\n case editor.core.keys.ENTER : editor.callback.enterKeyPressed(event); break;\r\n }\r\n\r\n };\r\n\r\n callbacks.redactorKeyDown = function (event) {\r\n\r\n switch (event.keyCode) {\r\n case editor.core.keys.TAB : editor.callback.tabKeyPressed(event); break;\r\n case editor.core.keys.ENTER : editor.callback.enterKeyPressedOnRedactorZone(event); break;\r\n case editor.core.keys.ESC : editor.callback.escapeKeyPressed(event); break;\r\n default : editor.callback.defaultKeyPressed(event); break;\r\n }\r\n\r\n };\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 : editor.callback.arrowKeyPressed(event); break;\r\n }\r\n\r\n };\r\n\r\n callbacks.tabKeyPressed = function (event) {\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 event.preventDefault();\r\n\r\n };\r\n\r\n /**\r\n * @param {Event} event\r\n */\r\n callbacks.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 editor.callback.enterPressedOnBlock();\r\n\r\n }\r\n\r\n };\r\n\r\n /**\r\n * ENTER key handler\r\n * Makes new paragraph block\r\n */\r\n callbacks.enterKeyPressedOnRedactorZone = 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 callbacks.escapeKeyPressed = 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 */\r\n callbacks.arrowKeyPressed = function () {\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 * @param {Event} event\r\n */\r\n callbacks.defaultKeyPressed = 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 callbacks.redactorClicked = function (event) {\r\n\r\n callbacks.detectWhenClickedOnFirstLevelBlockArea();\r\n\r\n editor.content.workingNodeChanged(event.target);\r\n\r\n editor.ui.saveInputs();\r\n\r\n var selectedText = editor.toolbar.inline.getSelectionText(),\r\n firstLevelBlock;\r\n\r\n /**\r\n * If selection range took off, then we hide inline toolbar\r\n */\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 /**\r\n * @todo Refactor\r\n */\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 /**\r\n * Move toolbar to the right position and open\r\n */\r\n editor.toolbar.move();\r\n editor.toolbar.open();\r\n\r\n } else {\r\n\r\n /**\r\n * Move toolbar to the new position and open\r\n */\r\n editor.toolbar.move();\r\n editor.toolbar.open();\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 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 /** Mark current block */\r\n editor.content.markBlock();\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 * 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 callbacks.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 ? false : true;\r\n\r\n }\r\n\r\n };\r\n\r\n /**\r\n * Toolbar button click handler\r\n * @param this - cursor to the button\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 /** Show or Hide toolbox when plus button is clicked */\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 callbacks.blockKeydown = function (event) {\r\n\r\n let block = this; // event.target 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 editor.callback.blockRightOrDownArrowPressed();\r\n break;\r\n\r\n case editor.core.keys.BACKSPACE:\r\n editor.callback.backspacePressed(block, event);\r\n break;\r\n\r\n case editor.core.keys.UP:\r\n case editor.core.keys.LEFT:\r\n editor.callback.blockLeftOrUpArrowPressed();\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 callbacks.blockRightOrDownArrowPressed = function () {\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 callbacks.blockLeftOrUpArrowPressed = function () {\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 * Callback for enter key pressing in first-level block area\r\n * @param {Event} event\r\n */\r\n callbacks.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 callbacks.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 (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 is used to observe pasted dirty data.\r\n *\r\n * Mutation handlers send to separate observers each mutation (added, changed and so on), which will be\r\n * passed from handler that sanitizes and replaces data.\r\n *\r\n * Probably won't be used\r\n *\r\n * @deprecated\r\n *\r\n * @param event\r\n * @private\r\n */\r\n callbacks._blockPasteCallback = function () {\r\n\r\n var currentInputIndex = editor.caret.getCurrentInputIndex();\r\n\r\n /**\r\n * create an observer instance\r\n */\r\n var observer = new MutationObserver(editor.callback.handleMutationsOnPaste);\r\n\r\n /**\r\n * configuration of the observer:\r\n */\r\n var config = {\r\n attributes: true,\r\n childList: false,\r\n characterData: false,\r\n subtree : true\r\n };\r\n\r\n // pass in the target node, as well as the observer options\r\n observer.observe(editor.state.inputs[currentInputIndex], config);\r\n\r\n };\r\n\r\n /**\r\n * This method prevents default behaviour.\r\n *\r\n * 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 * @param event\r\n */\r\n callbacks.blockPasteCallback = function (event) {\r\n\r\n /** Prevent default behaviour */\r\n event.preventDefault();\r\n\r\n var editableParent = editor.content.getEditableParent(event.target),\r\n firstLevelBlock = editor.content.getFirstLevelBlock(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 data = event.clipboardData.getData('text/html') || event.clipboardData.getData('text/plain');\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 cleaner = new editor.sanitizer.init(editor.sanitizer.Config.BASIC),\r\n cleanData,\r\n fragment;\r\n\r\n /** Create fragment, that we paste to range after proccesing */\r\n fragment = document.createDocumentFragment();\r\n\r\n cleanData = cleaner.clean(data);\r\n\r\n div.innerHTML = cleanData;\r\n\r\n var node, lastNode;\r\n\r\n /**\r\n * and fill in fragment\r\n */\r\n while (( node = div.firstChild) ) {\r\n\r\n lastNode = fragment.appendChild(node);\r\n\r\n }\r\n\r\n\r\n if (editor.tools[firstLevelBlock.dataset.tool].allowRenderOnPaste) {\r\n\r\n if (editor.paste.pasted(event)) return;\r\n\r\n }\r\n\r\n /**\r\n * work with selection and range\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(fragment);\r\n\r\n /** Preserve the selection */\r\n if (lastNode) {\r\n\r\n range = range.cloneRange();\r\n range.setStartAfter(lastNode);\r\n range.collapse(true);\r\n selection.removeAllRanges();\r\n selection.addRange(range);\r\n\r\n }\r\n\r\n };\r\n\r\n /**\r\n * Sends all mutations to paste handler\r\n */\r\n callbacks.handleMutationsOnPaste = function (mutations) {\r\n\r\n var self = this;\r\n\r\n /**\r\n * Calling function with context of this function.\r\n * Also, we should sanitize pasted or changed data one time and ignore\r\n * changings which makes sanitize method.\r\n * For that, we need to send Context, MutationObserver.__proto__ that contains\r\n * observer disconnect method.\r\n */\r\n mutations.forEach(function (mutation) {\r\n\r\n editor.content.paste.call(self, mutation);\r\n\r\n });\r\n\r\n };\r\n\r\n /**\r\n * Clicks on block settings button\r\n */\r\n callbacks.showSettingsButtonClicked = function () {\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.tagName == 'INPUT') {\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 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 var janitor = require('html-janitor');\r\n\r\n /**\r\n * Basic config\r\n */\r\n var Config = {\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 sanitizer.init = janitor;\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 Anchors module\r\n *\r\n * @author Codex Team\r\n * @version 1.0\r\n */\r\n\r\nmodule.exports = function (anchors) {\r\n\r\n let editor = codex.editor;\r\n\r\n anchors.input = null;\r\n anchors.currentNode = null;\r\n\r\n anchors.settingsOpened = function (currentBlock) {\r\n\r\n anchors.currentNode = currentBlock;\r\n anchors.input.value = anchors.currentNode.dataset.anchor || '';\r\n\r\n };\r\n\r\n anchors.anchorChanged = function (e) {\r\n\r\n var newAnchor = e.target.value = anchors.rusToTranslit(e.target.value);\r\n\r\n anchors.currentNode.dataset.anchor = newAnchor;\r\n\r\n if (newAnchor.trim() !== '') {\r\n\r\n anchors.currentNode.classList.add(editor.ui.className.BLOCK_WITH_ANCHOR);\r\n\r\n } else {\r\n\r\n anchors.currentNode.classList.remove(editor.ui.className.BLOCK_WITH_ANCHOR);\r\n\r\n }\r\n\r\n };\r\n\r\n anchors.keyDownOnAnchorInput = function (e) {\r\n\r\n if (e.keyCode == editor.core.keys.ENTER) {\r\n\r\n e.preventDefault();\r\n e.stopPropagation();\r\n\r\n e.target.blur();\r\n editor.toolbar.settings.close();\r\n\r\n }\r\n\r\n };\r\n\r\n anchors.keyUpOnAnchorInput = function (e) {\r\n\r\n if (e.keyCode >= editor.core.keys.LEFT && e.keyCode <= editor.core.keys.DOWN) {\r\n\r\n e.stopPropagation();\r\n\r\n }\r\n\r\n };\r\n\r\n anchors.rusToTranslit = function (string) {\r\n\r\n var ru = [\r\n 'А', 'Б', 'В', 'Г', 'Д', 'Е', 'Ё', 'Ж', 'З', 'И', 'Й',\r\n 'К', 'Л', 'М', 'Н', 'О', 'П', 'Р', 'С', 'Т', 'У', 'Ф',\r\n 'Х', 'Ц', 'Ч', 'Ш', 'Щ', 'Ь', 'Ы', 'Ь', 'Э', 'Ю', 'Я'\r\n ],\r\n en = [\r\n 'A', 'B', 'V', 'G', 'D', 'E', 'E', 'Zh', 'Z', 'I', 'Y',\r\n 'K', 'L', 'M', 'N', 'O', 'P', 'R', 'S', 'T', 'U', 'F',\r\n 'H', 'C', 'Ch', 'Sh', 'Sch', '', 'Y', '', 'E', 'Yu', 'Ya'\r\n ];\r\n\r\n for (var i = 0; i < ru.length; i++) {\r\n\r\n string = string.split(ru[i]).join(en[i]);\r\n string = string.split(ru[i].toLowerCase()).join(en[i].toLowerCase());\r\n\r\n }\r\n\r\n string = string.replace(/[^0-9a-zA-Z_]+/g, '-');\r\n\r\n return string;\r\n\r\n };\r\n\r\n return anchors;\r\n\r\n}({});\n\n\n// WEBPACK FOOTER //\n// ./modules/anchors.js","/**\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.0\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 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 if (pattern.regex.test(string)) {\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/codex.js b/codex.js index 3d09da19..deef008a 100644 --- a/codex.js +++ b/codex.js @@ -10,6 +10,7 @@ module.exports = (function (editor) { 'use strict'; editor.version = VERSION; + editor.scriptPrefix = 'cdx-script-'; var init = function () { @@ -28,6 +29,8 @@ module.exports = (function (editor) { editor.parser = require('./modules/parser'); editor.sanitizer = require('./modules/sanitizer'); editor.anchors = require('./modules/anchors'); + editor.listeners = require('./modules/listeners'); + editor.destroyer = require('./modules/destroyer'); editor.paste = require('./modules/paste'); }; diff --git a/example.html b/example.html index 14eb8108..7cef20da 100644 --- a/example.html +++ b/example.html @@ -62,6 +62,7 @@ render: paragraph.render, validate: paragraph.validate, save: paragraph.save, + destroy: paragraph.destroy, allowedToPaste: true, showInlineToolbar: true, allowRenderOnPaste: true @@ -74,6 +75,7 @@ render: header.render, validate: header.validate, save: header.save, + destroy: header.destroy, displayInToolbox: true }, code: { @@ -85,6 +87,7 @@ render: code.render, validate: code.validate, save: code.save, + destroy: code.destroy, displayInToolbox: true, enableLineBreaks: true }, @@ -97,6 +100,7 @@ render: link.render, validate: link.validate, save: link.save, + destroy: link.destroy, displayInToolbox: true, enableLineBreaks: true }, @@ -109,6 +113,7 @@ render: list.render, validate: list.validate, save: list.save, + destroy: list.destroy, displayInToolbox: true, showInlineToolbar: true, enableLineBreaks: true, @@ -122,6 +127,7 @@ render: quote.render, validate: quote.validate, save: quote.save, + destroy: quote.destroy, displayInToolbox: true, enableLineBreaks: true, showInlineToolbar: true, @@ -138,6 +144,7 @@ makeSettings: image.makeSettings, render: image.render, save: image.save, + destroy: image.destroy, isStretched: true, showInlineToolbar: true, displayInToolbox: true, @@ -153,6 +160,7 @@ render: instagram.reneder, validate: instagram.validate, save: instagram.save, + destroy: instagram.destroy, renderOnPastePatterns: instagram.pastePatterns, }, tweet: { @@ -162,6 +170,7 @@ render: twitter.render, validate: twitter.validate, save: twitter.save, + destroy: twitter.destroy, showInlineToolbar : true, renderOnPastePatterns: twitter.pastePatterns, config : { @@ -173,6 +182,7 @@ make: embed.make, render: embed.render, save: embed.save, + destroy: embed.destroy, validate: embed.validate, renderOnPastePatterns: embed.pastePatterns, } diff --git a/modules/callbacks.js b/modules/callbacks.js index 4bc92ee9..251bbad7 100644 --- a/modules/callbacks.js +++ b/modules/callbacks.js @@ -4,10 +4,11 @@ * @author Codex Team * @version 1.3.7 */ -let editor = codex.editor; module.exports = (function (callbacks) { + let editor = codex.editor; + callbacks.globalKeydown = function (event) { switch (event.keyCode) { diff --git a/modules/caret.js b/modules/caret.js index dc2596e2..156cf75c 100644 --- a/modules/caret.js +++ b/modules/caret.js @@ -4,10 +4,11 @@ * @author Codex Team * @version 1.0 */ -let editor = codex.editor; module.exports = (function (caret) { + let editor = codex.editor; + /** * @var {int} InputIndex - editable element in DOM */ diff --git a/modules/content.js b/modules/content.js index 2a18ffd5..afa38406 100644 --- a/modules/content.js +++ b/modules/content.js @@ -5,10 +5,11 @@ * @author Codex Team * @version 1.3.11 */ -let editor = codex.editor; module.exports = (function (content) { + let editor = codex.editor; + /** * Links to current active block * @type {null | Element} diff --git a/modules/core.js b/modules/core.js index 7a51700a..eb628222 100644 --- a/modules/core.js +++ b/modules/core.js @@ -5,10 +5,10 @@ * @version 1.1.2 */ -let editor = codex.editor; - module.exports = (function (core) { + let editor = codex.editor; + /** * @public * @@ -215,8 +215,6 @@ module.exports = (function (core) { return new Promise(function (resolve, reject) { - const instancePrefix = 'cdx-script-'; - let script; /** Script is already loaded */ @@ -224,7 +222,7 @@ module.exports = (function (core) { reject('Instance name is missed'); - } else if ( document.getElementById(instancePrefix + instanceName) ) { + } else if ( document.getElementById(editor.scriptPrefix + instanceName) ) { resolve(scriptPath); @@ -233,7 +231,7 @@ module.exports = (function (core) { script = document.createElement('SCRIPT'); script.async = true; script.defer = true; - script.id = instancePrefix + instanceName; + script.id = editor.scriptPrefix + instanceName; script.onload = function () { diff --git a/modules/destroyer.js b/modules/destroyer.js new file mode 100644 index 00000000..989f9478 --- /dev/null +++ b/modules/destroyer.js @@ -0,0 +1,98 @@ +/** + * Codex Editor Destroyer module + * + * @auhor Codex Team + * @version 1.0 + */ + +module.exports = function (destroyer) { + + let editor = codex.editor; + + destroyer.removeNodes = function () { + + editor.nodes.wrapper.remove(); + editor.nodes.notifications.remove(); + + }; + + destroyer.destroyPlugins = function () { + + for (var tool in editor.tools) { + + if (typeof editor.tools[tool].destroy === 'function') { + + editor.tools[tool].destroy(); + + } + + } + + }; + + destroyer.destroyScripts = function () { + + var scripts = document.getElementsByTagName('SCRIPT'); + + for (var i = 0; i < scripts.length; i++) { + + if (scripts[i].id.indexOf(editor.scriptPrefix) + 1) { + + scripts[i].remove(); + i--; + + } + + } + + }; + + + /** + * Delete editor data from webpage. + * You should send settings argument with boolean flags: + * @param settings.ui- remove redactor event listeners and DOM nodes + * @param settings.scripts - remove redactor scripts from DOM + * @param settings.plugins - remove plugin's objects + * @param settings.core - remove editor core. You can remove core only if UI and scripts flags is true + * } + * + */ + destroyer.destroy = function (settings) { + + if (!settings || typeof settings !== 'object') { + + return; + + } + + if (settings.ui) { + + destroyer.removeNodes(); + editor.listeners.removeAll(); + + } + + if (settings.scripts) { + + destroyer.destroyScripts(); + + } + + if (settings.plugins) { + + destroyer.destroyPlugins(); + + } + + if (settings.ui && settings.scripts && settings.core) { + + delete codex.editor; + + } + + }; + + return destroyer; + +}({}); \ No newline at end of file diff --git a/modules/listeners.js b/modules/listeners.js new file mode 100644 index 00000000..6b353ae2 --- /dev/null +++ b/modules/listeners.js @@ -0,0 +1,192 @@ +/** + * Codex Editor Listeners module + * + * @author Codex Team + * @version 1.0 + */ + +/** + * Module-decorator for event listeners assignment + */ +module.exports = function (listeners) { + + var allListeners = []; + + /** + * Search methods + * + * byElement, byType and byHandler returns array of suitable listeners + * one and all takes element, eventType, and handler and returns first (all) suitable listener + * + */ + listeners.search = function () { + + var byElement = function (element, context) { + + var listenersOnElement = []; + + context = context || allListeners; + + for (var i = 0; i < context.length; i++) { + + var listener = context[i]; + + if (listener.element === element) { + + listenersOnElement.push(listener); + + } + + } + + return listenersOnElement; + + }; + + var byType = function (eventType, context) { + + var listenersWithType = []; + + context = context || allListeners; + + for (var i = 0; i < context.length; i++) { + + var listener = context[i]; + + if (listener.type === eventType) { + + listenersWithType.push(listener); + + } + + } + + return listenersWithType; + + }; + + var byHandler = function (handler, context) { + + var listenersWithHandler = []; + + context = context || allListeners; + + for (var i = 0; i < context.length; i++) { + + var listener = context[i]; + + if (listener.handler === handler) { + + listenersWithHandler.push(listener); + + } + + } + + return listenersWithHandler; + + }; + + var one = function (element, eventType, handler) { + + var result = allListeners; + + if (element) + result = byElement(element, result); + + if (eventType) + result = byType(eventType, result); + + if (handler) + result = byHandler(handler, result); + + return result[0]; + + }; + + var all = function (element, eventType, handler) { + + var result = allListeners; + + if (element) + result = byElement(element, result); + + if (eventType) + result = byType(eventType, result); + + if (handler) + result = byHandler(handler, result); + + return result; + + }; + + return { + byElement : byElement, + byType : byType, + byHandler : byHandler, + one : one, + all : all + }; + + }(); + + listeners.add = function (element, eventType, handler, isCapture) { + + element.addEventListener(eventType, handler, isCapture); + + var data = { + element: element, + type: eventType, + handler: handler + }; + + var alreadyAddedListener = listeners.search.one(element, eventType, handler); + + if (!alreadyAddedListener) { + + allListeners.push(data); + + } + + }; + + listeners.remove = function (element, eventType, handler) { + + element.removeEventListener(eventType, handler); + + var existingListeners = listeners.search.all(element, eventType, handler); + + for (var i = 0; i < existingListeners.length; i++) { + + var index = allListeners.indexOf(existingListeners[i]); + + if (index > 0) { + + allListeners.splice(index, 1); + + } + + } + + }; + + listeners.removeAll = function () { + + allListeners.map(function (current) { + + listeners.remove(current.element, current.type, current.handler); + + }); + + }; + + listeners.get = function (element, eventType, handler) { + + return listeners.search.all(element, eventType, handler); + + }; + + return listeners; + +}({}); \ No newline at end of file diff --git a/modules/notifications.js b/modules/notifications.js index 850ca8af..dff23505 100644 --- a/modules/notifications.js +++ b/modules/notifications.js @@ -138,8 +138,8 @@ module.exports = (function (notifications) { okBtn.textContent = settings.okMsg || 'ОК'; cancelBtn.textContent = settings.cancelMsg || 'Отмена'; - okBtn.addEventListener('click', confirmHandler); - cancelBtn.addEventListener('click', cancelHandler); + editor.listeners.add(okBtn, 'click', confirmHandler); + editor.listeners.add(cancelBtn, 'click', cancelHandler); wrapper.appendChild(message); diff --git a/modules/parser.js b/modules/parser.js index d35daeb6..70debc9c 100644 --- a/modules/parser.js +++ b/modules/parser.js @@ -4,10 +4,11 @@ * @author Codex Team * @version 1.1 */ -let editor = codex.editor; module.exports = (function (parser) { + let editor = codex.editor; + /** inserting text */ parser.insertPastedContent = function (blockType, tag) { diff --git a/modules/paste.js b/modules/paste.js index 774a0a74..c1dade58 100644 --- a/modules/paste.js +++ b/modules/paste.js @@ -103,7 +103,6 @@ module.exports = function (paste) { }; - return paste; }({}); \ No newline at end of file diff --git a/modules/renderer.js b/modules/renderer.js index c47b23ea..1913b21a 100644 --- a/modules/renderer.js +++ b/modules/renderer.js @@ -5,10 +5,10 @@ * @version 1.0 */ -let editor = codex.editor; - module.exports = (function (renderer) { + let editor = codex.editor; + /** * Asyncronously parses input JSON to redactor blocks */ diff --git a/modules/sanitizer.js b/modules/sanitizer.js index 3fe2065d..8c4cbc4e 100644 --- a/modules/sanitizer.js +++ b/modules/sanitizer.js @@ -2,10 +2,10 @@ * Codex Sanitizer */ -var janitor = require('html-janitor'); - module.exports = (function (sanitizer) { + var janitor = require('html-janitor'); + /** * Basic config */ diff --git a/modules/saver.js b/modules/saver.js index 6363d6b7..e5128d8f 100644 --- a/modules/saver.js +++ b/modules/saver.js @@ -5,10 +5,10 @@ * @version 1.0.2 */ -let editor = codex.editor; - module.exports = (function (saver) { + let editor = codex.editor; + /** * Saves blocks * @private diff --git a/modules/toolbar/inline.js b/modules/toolbar/inline.js index 81b7beed..0ba41a62 100644 --- a/modules/toolbar/inline.js +++ b/modules/toolbar/inline.js @@ -8,10 +8,10 @@ * @version 1.0 */ -let editor = codex.editor; - module.exports = (function (inline) { + let editor = codex.editor; + inline.buttonsOpened = null; inline.actionsOpened = null; inline.wrappersOffset = null; @@ -363,7 +363,7 @@ module.exports = (function (inline) { event.preventDefault(); /** Callback to link action */ - action.addEventListener('keydown', inlineToolbarAnchorInputKeydown_, false); + editor.listeners.add(action, 'keydown', inlineToolbarAnchorInputKeydown_, false); } diff --git a/modules/toolbar/settings.js b/modules/toolbar/settings.js index c4898548..541d64da 100644 --- a/modules/toolbar/settings.js +++ b/modules/toolbar/settings.js @@ -4,10 +4,10 @@ * @version 1.0.4 */ -let editor = codex.editor; - module.exports = (function (settings) { + let editor = codex.editor; + settings.opened = false; settings.setting = null; @@ -141,7 +141,7 @@ module.exports = (function (settings) { } setting = editor.draw.node('DIV', editor.ui.className.SETTINGS_ITEM, data); - setting.addEventListener('click', editor.toolbar.settings.updateFeedMode, false); + editor.listeners.add(setting, 'click', editor.toolbar.settings.updateFeedMode, false); return setting; @@ -182,10 +182,10 @@ module.exports = (function (settings) { hash = editor.draw.node('i', 'ce-settings__anchor-hash', {}), anchor = editor.draw.node('input', 'ce-settings__anchor-input', { placeholder: 'Якорь' }); - anchor.addEventListener('keydown', editor.anchors.keyDownOnAnchorInput ); - anchor.addEventListener('keyup', editor.anchors.keyUpOnAnchorInput ); - anchor.addEventListener('input', editor.anchors.anchorChanged ); - anchor.addEventListener('blur', editor.anchors.anchorChanged ); + editor.listeners.add(anchor, 'keydown', editor.anchors.keyDownOnAnchorInput ); + editor.listeners.add(anchor, 'keyup', editor.anchors.keyUpOnAnchorInput ); + editor.listeners.add(anchor, 'input', editor.anchors.anchorChanged ); + editor.listeners.add(anchor, 'blur', editor.anchors.anchorChanged ); anchorWrapper.appendChild(hash); anchorWrapper.appendChild(anchor); @@ -207,11 +207,11 @@ module.exports = (function (settings) { confirmAction = editor.draw.node('DIV', 'ce-toolbar__remove-confirm', { textContent : 'Удалить блок' }), cancelAction = editor.draw.node('DIV', 'ce-toolbar__remove-cancel', { textContent : 'Отмена' }); - settingButton.addEventListener('click', editor.toolbar.settings.removeButtonClicked, false); + editor.listeners.add(settingButton, 'click', editor.toolbar.settings.removeButtonClicked, false); - confirmAction.addEventListener('click', editor.toolbar.settings.confirmRemovingRequest, false); + editor.listeners.add(confirmAction, 'click', editor.toolbar.settings.confirmRemovingRequest, false); - cancelAction.addEventListener('click', editor.toolbar.settings.cancelRemovingRequest, false); + editor.listeners.add(cancelAction, 'click', editor.toolbar.settings.cancelRemovingRequest, false); actionWrapper.appendChild(confirmAction); actionWrapper.appendChild(cancelAction); diff --git a/modules/toolbar/toolbar.js b/modules/toolbar/toolbar.js index 8cc9167b..95c54a33 100644 --- a/modules/toolbar/toolbar.js +++ b/modules/toolbar/toolbar.js @@ -10,10 +10,10 @@ * @version 1.0 */ -let editor = codex.editor; - module.exports = (function (toolbar) { + let editor = codex.editor; + toolbar.settings = require('./settings'); toolbar.inline = require('./inline'); toolbar.toolbox = require('./toolbox'); diff --git a/modules/toolbar/toolbox.js b/modules/toolbar/toolbox.js index a8451f92..183876a9 100644 --- a/modules/toolbar/toolbox.js +++ b/modules/toolbar/toolbox.js @@ -7,10 +7,10 @@ * @version 1.0 */ -let editor = codex.editor; - module.exports = (function (toolbox) { + let editor = codex.editor; + toolbox.opened = false; /** Shows toolbox */ diff --git a/modules/transport.js b/modules/transport.js index 9c310b5d..9bd1540d 100644 --- a/modules/transport.js +++ b/modules/transport.js @@ -5,10 +5,11 @@ * @author Codex Team * @version 1.0 */ -let editor = codex.editor; module.exports = (function (transport) { + let editor = codex.editor; + transport.input = null; /** @@ -21,7 +22,7 @@ module.exports = (function (transport) { var input = document.createElement('INPUT'); input.type = 'file'; - input.addEventListener('change', editor.transport.fileSelected); + editor.listeners.add(input, 'change', editor.transport.fileSelected); editor.transport.input = input; diff --git a/modules/ui.js b/modules/ui.js index a1c099d7..f3520588 100644 --- a/modules/ui.js +++ b/modules/ui.js @@ -5,10 +5,10 @@ * @version 1.1 */ -let editor = codex.editor; - module.exports = (function (ui) { + let editor = codex.editor; + /** * Basic editor classnames */ @@ -274,28 +274,28 @@ module.exports = (function (ui) { // }, false ); /** All keydowns on Document */ - document.addEventListener('keydown', editor.callback.globalKeydown, false ); + editor.listeners.add(document, 'keydown', editor.callback.globalKeydown, false); /** All keydowns on Redactor zone */ - editor.nodes.redactor.addEventListener('keydown', editor.callback.redactorKeyDown, false); + editor.listeners.add(editor.nodes.redactor, 'keydown', editor.callback.redactorKeyDown, false); /** All keydowns on Document */ - document.addEventListener('keyup', editor.callback.globalKeyup, false ); + editor.listeners.add(document, 'keyup', editor.callback.globalKeyup, false ); /** * Mouse click to radactor */ - editor.nodes.redactor.addEventListener('click', editor.callback.redactorClicked, false ); + editor.listeners.add(editor.nodes.redactor, 'click', editor.callback.redactorClicked, false ); /** * Clicks to the Plus button */ - editor.nodes.plusButton.addEventListener('click', editor.callback.plusButtonClicked, false); + editor.listeners.add(editor.nodes.plusButton, 'click', editor.callback.plusButtonClicked, false); /** * Clicks to SETTINGS button in toolbar */ - editor.nodes.showSettingsButton.addEventListener('click', editor.callback.showSettingsButtonClicked, false ); + editor.listeners.add(editor.nodes.showSettingsButton, 'click', editor.callback.showSettingsButtonClicked, false ); /** * @deprecated ( but now in use for syncronization ); @@ -306,7 +306,7 @@ module.exports = (function (ui) { /** Bind click listeners on toolbar buttons */ for (var button in editor.nodes.toolbarButtons) { - editor.nodes.toolbarButtons[button].addEventListener('click', editor.callback.toolbarButtonClicked, false); + editor.listeners.add(editor.nodes.toolbarButtons[button], 'click', editor.callback.toolbarButtonClicked, false); } @@ -319,7 +319,7 @@ module.exports = (function (ui) { /** * Block keydowns */ - block.addEventListener('keydown', editor.callback.blockKeydown, false); + editor.listeners.add(block, 'keydown', editor.callback.blockKeydown, false); /** * Pasting content from another source @@ -340,9 +340,9 @@ module.exports = (function (ui) { * @example editor.callback.blockPasteViaSanitize(event), the second method. * */ - block.addEventListener('paste', editor.callback.blockPasteCallback, false); + editor.listeners.add(block, 'paste', editor.callback.blockPasteCallback, false); - block.addEventListener('mouseup', editor.toolbar.inline.show, false); + editor.listeners.add(block, 'mouseup', editor.toolbar.inline.show, false); }; @@ -386,7 +386,7 @@ module.exports = (function (ui) { ui.setInlineToolbarButtonBehaviour = function (button, type) { - button.addEventListener('mousedown', function (event) { + editor.listeners.add(button, 'mousedown', function (event) { editor.toolbar.inline.toolClicked(event, type); diff --git a/package.json b/package.json index 32110513..892e3a52 100644 --- a/package.json +++ b/package.json @@ -1,6 +1,6 @@ { "name": "codex.editor", - "version": "1.4.10", + "version": "1.5.0", "description": "Codex Editor. Native JS, based on API and Open Source", "main": "index.js", "scripts": { diff --git a/plugins/code/code.js b/plugins/code/code.js index d9e71897..63d0267d 100644 --- a/plugins/code/code.js +++ b/plugins/code/code.js @@ -3,7 +3,7 @@ * Creates code tag and adds content to this tag */ -var code = (function(code) { +var code = (function(code_plugin) { var baseClass = "ce-code"; @@ -28,7 +28,7 @@ var code = (function(code) { /** * Method to render HTML block from JSON */ - code.render = function (data) { + code_plugin.render = function (data) { return make_(data); }; @@ -36,7 +36,7 @@ var code = (function(code) { /** * Method to extract JSON data from HTML block */ - code.save = function (blockContent){ + code_plugin.save = function (blockContent) { var data = { text : blockContent.innerHTML @@ -45,7 +45,7 @@ var code = (function(code) { }; - code.validate = function(data) { + code_plugin.validate = function (data) { if (data.text.trim() == '') return; @@ -53,6 +53,12 @@ var code = (function(code) { return true; }; - return code; + code_plugin.destroy = function () { + + code = null; + + }; + + return code_plugin; })({}); diff --git a/plugins/embed/embed.js b/plugins/embed/embed.js index 0a4aa8fd..1ddaa345 100644 --- a/plugins/embed/embed.js +++ b/plugins/embed/embed.js @@ -2,7 +2,7 @@ * Embed plugin by gohabereg * @version 1.0.0 */ -var embed = function(embed){ +var embed = function(embed_plugin){ var methods = { @@ -122,7 +122,7 @@ var embed = function(embed){ }; - embed.make = function(data, isInternal) { + embed_plugin.make = function(data, isInternal) { if (!data.remote_id) return; @@ -152,7 +152,7 @@ var embed = function(embed){ * Saving JSON output. * Upload data via ajax */ - embed.save = function(blockContent) { + embed_plugin.save = function(blockContent) { if (!blockContent) return; @@ -175,11 +175,11 @@ var embed = function(embed){ /** * Render data */ - embed.render = function(data) { - return embed.make(data); + embed_plugin.render = function(data) { + return embed_plugin.make(data); }; - embed.urlPastedCallback = function(url, pattern) { + embed_plugin.urlPastedCallback = function(url, pattern) { var execArray = pattern.regex.exec(url), id = methods.getRemoteId(pattern.type, execArray); @@ -190,10 +190,10 @@ var embed = function(embed){ thumbnailUrl: url }; - embed.make(data, true); + embed_plugin.make(data, true); }; - embed.validate = function(savedData) { + embed_plugin.validate = function(savedData) { var source = savedData.source, execArray = services[source].regex.exec(savedData.thumbnailUrl), @@ -203,69 +203,74 @@ var embed = function(embed){ }; - embed.pastePatterns = [ + embed_plugin.pastePatterns = [ { type: 'vk', regex: /https?:\/\/vk\.com\/.*(?:video)([-0-9]+_[0-9]+)/, ///https?.+vk?.com\/feed\?w=wall\d+_\d+/, - callback: embed.urlPastedCallback + callback: embed_plugin.urlPastedCallback }, { type: 'youtube', regex: /(?:https?:\/{2})?(?:w{3}\.)?youtu(?:be)?\.(?:com|be)(?:\/watch\?v=|\/)([^\s&]+)/, - callback: embed.urlPastedCallback + callback: embed_plugin.urlPastedCallback }, { type: 'vimeo', regex: /(?:http[s]?:\/\/)?(?:www.)?vimeo\.co(?:.+\/([^\/]\d+)(?:#t=[\d]+)?s?$)/, - callback: embed.urlPastedCallback + callback: embed_plugin.urlPastedCallback }, { type: 'coub', regex: /https?:\/\/coub\.com\/view\/([^\/\?\&]+)/, - callback: embed.urlPastedCallback + callback: embed_plugin.urlPastedCallback }, { type: 'vine', regex: /https?:\/\/vine\.co\/v\/([^\/\?\&]+)/, - callback: embed.urlPastedCallback + callback: embed_plugin.urlPastedCallback }, { type: 'imgur', regex: /https?:\/\/(?:i\.)?imgur\.com.*\/([a-zA-Z0-9]+)(?:\.gifv)?/, - callback: embed.urlPastedCallback + callback: embed_plugin.urlPastedCallback }, { type: 'gfycat', regex: /https?:\/\/gfycat\.com(?:\/detail)?\/([a-zA-Z]+)/, - callback: embed.urlPastedCallback + callback: embed_plugin.urlPastedCallback }, { type: 'twitch-channel', regex: /https?:\/\/www.twitch.tv\/([^\/\?\&]*)/, - callback: embed.urlPastedCallback + callback: embed_plugin.urlPastedCallback }, { type: 'twitch-video', regex: /https?:\/\/www.twitch.tv\/(?:[^\/\?\&]*\/v|videos)\/([0-9]*)/, - callback: embed.urlPastedCallback + callback: embed_plugin.urlPastedCallback }, { type: 'yandex-music-album', regex: /https?:\/\/music.yandex.ru\/album\/([0-9]*)/, - callback: embed.urlPastedCallback + callback: embed_plugin.urlPastedCallback }, { type: 'yandex-music-track', regex: /https?:\/\/music.yandex.ru\/album\/([0-9]*)\/track\/([0-9]*)/, - callback: embed.urlPastedCallback + callback: embed_plugin.urlPastedCallback }, { type: 'yandex-music-playlist', regex: /https?:\/\/music.yandex.ru\/users\/([^\/\?\&]*)\/playlists\/([0-9]*)/, - callback: embed.urlPastedCallback + callback: embed_plugin.urlPastedCallback } ]; + embed_plugin.destroy = function () { - return embed; + embed = null; + + }; + + return embed_plugin; }({}); \ No newline at end of file diff --git a/plugins/header/header.js b/plugins/header/header.js index 63ba8e7e..2544fc6d 100644 --- a/plugins/header/header.js +++ b/plugins/header/header.js @@ -3,7 +3,7 @@ * H e a d e r */ -var header = (function(header) { +var header = (function(header_plugin) { /** * @private @@ -92,14 +92,14 @@ var header = (function(header) { }; - header.prepareDataForSave = function(data) { + header_plugin.prepareDataForSave = function(data) { }; /** * Method to render HTML block from JSON */ - header.render = function (data) { + header_plugin.render = function (data) { return make_(data); @@ -108,7 +108,7 @@ var header = (function(header) { /** * Method to extract JSON data from HTML block */ - header.save = function (blockContent) { + header_plugin.save = function (blockContent) { var data = { "heading-styles": blockContent.dataset.headerData, @@ -126,7 +126,7 @@ var header = (function(header) { * - - - - - - - - - - - - - * @return {Element} element contains all settings */ - header.makeSettings = function () { + header_plugin.makeSettings = function () { var holder = codex.editor.draw.node('DIV', ['cdx-plugin-settings--horisontal'], {} ), types = { @@ -148,7 +148,7 @@ var header = (function(header) { return holder; }; - header.validate = function(data) { + header_plugin.validate = function(data) { if (data.text.trim() === '' || data['heading-styles'].trim() === ''){ return false; @@ -157,7 +157,13 @@ var header = (function(header) { return true; }; - return header; + header_plugin.destroy = function () { + + header = null; + + } + + return header_plugin; })({}); diff --git a/plugins/image/image.js b/plugins/image/image.js index 938a89f9..3bd9a63f 100644 --- a/plugins/image/image.js +++ b/plugins/image/image.js @@ -4,7 +4,7 @@ * * @version 1.2.0 */ -var image = (function(image) { +var image = (function(image_plugin) { /** * @private @@ -428,9 +428,9 @@ var image = (function(image) { */ uploadImageFromUrl : function(path) { - var ajaxUrl = image.config.uploadUrl, + var ajaxUrl = image_plugin.config.uploadUrl, file, - image_plugin, + image, current = codex.editor.content.currentNode, beforeSend, success_callback; @@ -440,7 +440,7 @@ var image = (function(image) { var imageInfo = JSON.parse(data); - var newImage = image_plugin.getElementsByTagName('IMG')[0]; + var newImage = image.getElementsByTagName('IMG')[0]; newImage.dataset.stretched = false; newImage.dataset.src = imageInfo.file.url; @@ -449,7 +449,7 @@ var image = (function(image) { newImage.dataset.height = imageInfo.file.height; newImage.dataset.additionalData = imageInfo.file.additionalData; - image_plugin.classList.remove(elementClasses_.imagePreview); + image.classList.remove(elementClasses_.imagePreview); }; @@ -473,19 +473,19 @@ var image = (function(image) { cover: null }; - image_plugin = codex.editor.tools.image_extended.render(data); + image = codex.editor.tools.image_extended.render(data); - image_plugin.classList.add(elementClasses_.imagePreview); + image.classList.add(elementClasses_.imagePreview); - var img = image_plugin.querySelector('img'); + var img = image.querySelector('img'); - codex.editor.content.switchBlock(codex.editor.content.currentNode, image_plugin, 'image_extended'); + codex.editor.content.switchBlock(codex.editor.content.currentNode, image, 'image_extended'); }; /** Preparing data for XMLHTTP */ var data = { - url: image.config.uploadUrl, + url: image_plugin.config.uploadUrl, type: "POST", data : { url: path @@ -504,12 +504,12 @@ var image = (function(image) { * Image path * @type {null} */ - image.path = null; + image_plugin.path = null; /** * Plugin configuration */ - image.config = null; + image_plugin.config = null; /** * @@ -567,9 +567,9 @@ var image = (function(image) { * @public * @param config */ - image.prepare = function(config) { + image_plugin.prepare = function(config) { - image.config = config; + image_plugin.config = config; return Promise.resolve(); }; @@ -579,7 +579,7 @@ var image = (function(image) { * * this tool works when tool is clicked in toolbox */ - image.appendCallback = function(event) { + image_plugin.appendCallback = function(event) { /** Upload image and call success callback*/ uploadButtonClicked_(event); @@ -592,7 +592,7 @@ var image = (function(image) { * @param data * @return {*} */ - image.render = function( data ) { + image_plugin.render = function( data ) { return make_(data); }; @@ -603,7 +603,7 @@ var image = (function(image) { * @param block * @return {{background: boolean, border: boolean, isstretch: boolean, file: {url: (*|string|Object), bigUrl: (null|*), width: *, height: *, additionalData: null}, caption: (string|*|string), cover: null}} */ - image.save = function ( block ) { + image_plugin.save = function ( block ) { var content = block, image = ui_.getImage(content), @@ -633,7 +633,7 @@ var image = (function(image) { * Settings panel content * @return {Element} element contains all settings */ - image.makeSettings = function () { + image_plugin.makeSettings = function () { var currentNode = codex.editor.content.currentNode, wrapper = currentNode.querySelector('.' + elementClasses_.imageWrapper), @@ -688,20 +688,26 @@ var image = (function(image) { /** * Share as API */ - image.uploadImageFromUri = uploadingCallbacks_.ByPaste.uploadImageFromUrl; + image_plugin.uploadImageFromUri = uploadingCallbacks_.ByPaste.uploadImageFromUrl; - image.pastePatterns = [ + image_plugin.pastePatterns = [ { type: 'image', regex: /(?:([^:\/?#]+):)?(?:\/\/([^\/?#]*))?([^?#]*\.(?:jpe?g|gif|png))(?:\?([^#]*))?(?:#(.*))?/i, - callback: image.uploadImageFromUri + callback: image_plugin.uploadImageFromUri }, { type: 'uploadCare', regex: /^https:\/\/(uploadcare\.cmtt\.ru|ucarecdn\.com|static[0-9]+\.siliconrus\.cmtt\.ru|static[0-9]+\.cmtt\.ru)/i, - callback: image.uploadImageFromUri + callback: image_plugin.uploadImageFromUri } ]; - return image; + image_plugin.destroy = function () { + + image = null; + + }; + + return image_plugin; })({}); \ No newline at end of file diff --git a/plugins/instagram/instagram.js b/plugins/instagram/instagram.js index 8148a0c6..f6d2c0b9 100644 --- a/plugins/instagram/instagram.js +++ b/plugins/instagram/instagram.js @@ -2,7 +2,7 @@ * Instagram plugin * @version 1.0.0 */ -var instagram = (function(instagram) { +var instagram = (function(instagram_plugin) { var methods = { @@ -44,7 +44,7 @@ var instagram = (function(instagram) { * Prepare before usage * Load important scripts to render embed */ - instagram.prepare = function() { + instagram_plugin.prepare = function() { return new Promise(function(resolve, reject){ @@ -86,7 +86,7 @@ var instagram = (function(instagram) { return block; }; - instagram.validate = function(data) { + instagram_plugin.validate = function(data) { return true; }; @@ -94,7 +94,7 @@ var instagram = (function(instagram) { * Saving JSON output. * Upload data via ajax */ - instagram.save = function(blockContent) { + instagram_plugin.save = function(blockContent) { var data; @@ -110,7 +110,7 @@ var instagram = (function(instagram) { }; - instagram.validate = function(data) { + instagram_plugin.validate = function(data) { var checkUrl = new RegExp("http?.+instagram.com\/p?."); @@ -123,7 +123,7 @@ var instagram = (function(instagram) { /** * Render data */ - instagram.render = function(data) { + instagram_plugin.render = function(data) { return make_(data); }; @@ -132,7 +132,7 @@ var instagram = (function(instagram) { * Using instagram Embed Widgete to render * @param url */ - instagram.urlPastedCallback = function(url) { + instagram_plugin.urlPastedCallback = function(url) { var data = { instagram_url: url }; @@ -141,15 +141,22 @@ var instagram = (function(instagram) { }; - instagram.pastePatterns = [ + instagram_plugin.pastePatterns = [ { type: 'instagram', regex: /http?.+instagram.com\/p\/([a-zA-Z0-9]*)/, - callback: instagram.urlPastedCallback + callback: instagram_plugin.urlPastedCallback } ]; - return instagram; + instagram_plugin.destroy = function () { + + instagram = null; + delete window.instgrm + + }; + + return instagram_plugin; })({}); diff --git a/plugins/link/link.js b/plugins/link/link.js index 2b4c6436..2763015f 100644 --- a/plugins/link/link.js +++ b/plugins/link/link.js @@ -6,7 +6,7 @@ * Link tool plugin */ -var link = (function(link) { +var link = (function(link_plugin) { var settings = { defaultText : 'Вставьте ссылку ...', @@ -178,7 +178,7 @@ var link = (function(link) { /* Show loader gif **/ block.classList.add(settings.elementClasses.loader); - return fetch( link.config.fetchUrl + '?url=' + encodeURI(url) ); + return fetch( link_plugin.config.fetchUrl + '?url=' + encodeURI(url) ); }) .then(function (response) { @@ -242,9 +242,9 @@ var link = (function(link) { } }; - link.prepare = function (config) { + link_plugin.prepare = function (config) { - link.config = config; + link_plugin.config = config; return Promise.resolve(); @@ -255,7 +255,7 @@ var link = (function(link) { * @param {object} JSON with block data * @return {Element} element to append */ - link.makeNewBlock = function (data) { + link_plugin.makeNewBlock = function (data) { var wrapper = ui.mainBlock(), tag = ui.input(); @@ -278,7 +278,7 @@ var link = (function(link) { /** * Method to render HTML block from JSON */ - link.render = function (json) { + link_plugin.render = function (json) { if ( json ) { @@ -312,7 +312,7 @@ var link = (function(link) { }; - link.validate = function (data) { + link_plugin.validate = function (data) { if (data.url.trim() == '' || data.title.trim() == '' || data.description.trim() == '') return; @@ -323,7 +323,7 @@ var link = (function(link) { /** * Method to extract JSON data from HTML block */ - link.save = function (blockContent){ + link_plugin.save = function (blockContent){ var linkElement = settings.elementClasses.link; @@ -339,6 +339,12 @@ var link = (function(link) { }; - return link; + link_plugin.destroy = function () { + + link = null; + + }; + + return link_plugin; })({}); diff --git a/plugins/list/list.js b/plugins/list/list.js index a933b918..6a060ced 100644 --- a/plugins/list/list.js +++ b/plugins/list/list.js @@ -2,7 +2,7 @@ * Code Plugin\ * Creates code tag and adds content to this tag */ -var list = (function(list) { +var list = (function(list_plugin) { /** * CSS class names @@ -76,7 +76,6 @@ var list = (function(list) { codex.editor.content.switchBlock(currentBlock, newEditable, 'list'); }, - keyDown: function (e) { var controlKeyPressed = e.ctrlKey || e.metaKey, @@ -117,7 +116,7 @@ var list = (function(list) { } - range.selectNode(currentSelectedNode); + range.selectNodeContents(currentSelectedNode); selection.removeAllRanges(); selection.addRange(range); @@ -128,7 +127,7 @@ var list = (function(list) { /** * Method to render HTML block from JSON */ - list.render = function (data) { + list_plugin.render = function (data) { var type = data && data.type == 'ordered' ? 'OL' : 'UL', tag = ui.make(type), @@ -158,7 +157,7 @@ var list = (function(list) { }; - list.validate = function(data) { + list_plugin.validate = function(data) { var items = data.items.every(function(item){ return item.trim() !== ''; @@ -176,7 +175,7 @@ var list = (function(list) { /** * Method to extract JSON data from HTML block */ - list.save = function (blockContent){ + list_plugin.save = function (blockContent){ var data = { type : null, @@ -192,7 +191,7 @@ var list = (function(list) { }; - list.makeSettings = function () { + list_plugin.makeSettings = function () { var holder = document.createElement('DIV'); @@ -219,6 +218,12 @@ var list = (function(list) { }; - return list; + list_plugin.destroy = function () { + + list = null; + + }; + + return list_plugin; })({}); diff --git a/plugins/paragraph/paragraph.js b/plugins/paragraph/paragraph.js index c9019a52..2df4ff7d 100644 --- a/plugins/paragraph/paragraph.js +++ b/plugins/paragraph/paragraph.js @@ -3,12 +3,12 @@ * Creates DIV tag and adds content to this tag */ -var paragraph = (function(paragraph) { +var paragraph = (function(paragraph_plugin) { /** * @private * - * Make initial header block + * Make initial paragraph block * @param {object} JSON with block data * @return {Element} element to append */ @@ -44,7 +44,7 @@ var paragraph = (function(paragraph) { * Plugins should have prepare method * @param config */ - paragraph.prepare = function(config) { + paragraph_plugin.prepare = function(config) { }; @@ -53,7 +53,7 @@ var paragraph = (function(paragraph) { * * Method to render HTML block from JSON */ - paragraph.render = function (data) { + paragraph_plugin.render = function (data) { return make_(data); @@ -65,7 +65,7 @@ var paragraph = (function(paragraph) { * Check output data for validity. * Should be defined by developer */ - paragraph.validate = function(output) { + paragraph_plugin.validate = function(output) { if (output.text === '') return; @@ -78,7 +78,7 @@ var paragraph = (function(paragraph) { * * Method to extract JSON data from HTML block */ - paragraph.save = function (blockContent){ + paragraph_plugin.save = function (blockContent){ var wrappedText = codex.editor.content.wrapTextWithParagraphs(blockContent.innerHTML); @@ -92,6 +92,12 @@ var paragraph = (function(paragraph) { }; - return paragraph; + paragraph_plugin.destroy = function () { + + paragraph = null; + + }; + + return paragraph_plugin; })({}); diff --git a/plugins/quote/quote.js b/plugins/quote/quote.js index 5728ec88..8ec53265 100644 --- a/plugins/quote/quote.js +++ b/plugins/quote/quote.js @@ -3,7 +3,7 @@ * Quote plugin */ -var quote = (function(quote) { +var quote = (function(quote_plugin) { /** * @private @@ -401,7 +401,7 @@ var quote = (function(quote) { if (data && data.size) { - data.style = quote.config.defaultStyle; + data.style = quote_plugin.config.defaultStyle; /** * Supported types @@ -457,11 +457,11 @@ var quote = (function(quote) { * * @param data */ - quote.render = function(data) { + quote_plugin.render = function(data) { return make_(data); }; - quote.validate = function(output) { + quote_plugin.validate = function(output) { if (typeof output.text != "string") { return; @@ -470,7 +470,7 @@ var quote = (function(quote) { return output; }; - quote.save = function(blockContent) { + quote_plugin.save = function(blockContent) { /** * Extracts JSON quote data from HTML block @@ -495,7 +495,7 @@ var quote = (function(quote) { * * Draws settings */ - quote.makeSettings = function(data) { + quote_plugin.makeSettings = function(data) { var holder = document.createElement('DIV'), types = { @@ -517,7 +517,7 @@ var quote = (function(quote) { selectTypeButton.dataset.style = type; - if ( type == quote.config.defaultStyle ){ + if ( type == quote_plugin.config.defaultStyle ){ selectTypeButton.classList.add(quoteTools.styles.settings.selectedType); } @@ -539,27 +539,33 @@ var quote = (function(quote) { * Default path to redactors images * @type {null} */ - quote.path = null; + quote_plugin.path = null; /** * @public * * @type {null} */ - quote.config = null; + quote_plugin.config = null; /** * @public * * @param config */ - quote.prepare = function(config) { + quote_plugin.prepare = function(config) { - quote.config = config; + quote_plugin.config = config; return Promise.resolve(); }; - return quote; + quote_plugin.destroy = function () { + + quote = null; + + }; + + return quote_plugin; })({}); \ No newline at end of file diff --git a/plugins/twitter/twitter.js b/plugins/twitter/twitter.js index aefd54a3..762fe6e9 100644 --- a/plugins/twitter/twitter.js +++ b/plugins/twitter/twitter.js @@ -3,7 +3,7 @@ * @version 1.0.0 */ -var twitter = (function(twitter) { +var twitter = (function(twitter_plugin) { /** * User's configuration object @@ -158,7 +158,7 @@ var twitter = (function(twitter) { * Prepare twitter scripts * @param {object} config */ - twitter.prepare = function(config) { + twitter_plugin.prepare = function(config) { /** * Save configs @@ -199,11 +199,11 @@ var twitter = (function(twitter) { return tweet; }; - twitter.validate = function(data) { + twitter_plugin.validate = function(data) { return true; }; - twitter.save = function(blockContent) { + twitter_plugin.save = function(blockContent) { var data, caption = blockContent.querySelector('.ce-twitter__caption'); @@ -228,11 +228,11 @@ var twitter = (function(twitter) { return data; }; - twitter.render = function(data) { + twitter_plugin.render = function(data) { return make_(data); }; - twitter.urlPastedCallback = function(url) { + twitter_plugin.urlPastedCallback = function(url) { var tweetId, arr, @@ -256,15 +256,22 @@ var twitter = (function(twitter) { make_(data); }; - twitter.pastePatterns = [ + twitter_plugin.pastePatterns = [ { type: 'twitter', regex: /http?.+twitter.com?.+\//, - callback: twitter.urlPastedCallback + callback: twitter_plugin.urlPastedCallback } ]; - return twitter; + twitter_plugin.destroy = function () { + + twitter = null; + delete window.twttr; + + }; + + return twitter_plugin; })({}); diff --git a/whatwg-fetch.js.map b/whatwg-fetch.js.map index 93bdbc35..ce2be656 100644 --- a/whatwg-fetch.js.map +++ b/whatwg-fetch.js.map @@ -1 +1 @@ -{"version":3,"sources":["webpack:///whatwg-fetch.js","webpack:///webpack/bootstrap f80ecde20480b792f648?0a00","webpack:///./~/whatwg-fetch/fetch.js"],"names":["codex","modules","__webpack_require__","moduleId","installedModules","exports","module","id","loaded","call","m","c","p","self","normalizeName","name","String","test","TypeError","toLowerCase","normalizeValue","value","iteratorFor","items","iterator","next","shift","done","undefined","support","iterable","Symbol","Headers","headers","this","map","forEach","append","Object","getOwnPropertyNames","consumed","body","bodyUsed","Promise","reject","fileReaderReady","reader","resolve","onload","result","onerror","error","readBlobAsArrayBuffer","blob","FileReader","promise","readAsArrayBuffer","readBlobAsText","readAsText","readArrayBufferAsText","buf","view","Uint8Array","chars","Array","length","i","fromCharCode","join","bufferClone","slice","byteLength","set","buffer","Body","_initBody","_bodyInit","_bodyText","Blob","prototype","isPrototypeOf","_bodyBlob","formData","FormData","_bodyFormData","searchParams","URLSearchParams","toString","arrayBuffer","isDataView","_bodyArrayBuffer","ArrayBuffer","isArrayBufferView","Error","get","type","rejected","then","text","decode","json","JSON","parse","normalizeMethod","method","upcased","toUpperCase","methods","indexOf","Request","input","options","url","credentials","mode","referrer","form","trim","split","bytes","replace","decodeURIComponent","parseHeaders","rawHeaders","line","parts","key","Response","bodyInit","status","ok","statusText","fetch","e","viewClasses","obj","DataView","isView","oldValue","has","hasOwnProperty","callback","thisArg","keys","push","values","entries","clone","response","redirectStatuses","redirect","RangeError","location","init","request","xhr","XMLHttpRequest","getAllResponseHeaders","responseURL","responseText","ontimeout","open","withCredentials","responseType","setRequestHeader","send","polyfill"],"mappings":"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","file":"whatwg-fetch.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) {\n\n\t(function(self) {\n\t 'use strict';\n\t\n\t if (self.fetch) {\n\t return\n\t }\n\t\n\t var support = {\n\t searchParams: 'URLSearchParams' in self,\n\t iterable: 'Symbol' in self && 'iterator' in Symbol,\n\t blob: 'FileReader' in self && 'Blob' in self && (function() {\n\t try {\n\t new Blob()\n\t return true\n\t } catch(e) {\n\t return false\n\t }\n\t })(),\n\t formData: 'FormData' in self,\n\t arrayBuffer: 'ArrayBuffer' in self\n\t }\n\t\n\t if (support.arrayBuffer) {\n\t var viewClasses = [\n\t '[object Int8Array]',\n\t '[object Uint8Array]',\n\t '[object Uint8ClampedArray]',\n\t '[object Int16Array]',\n\t '[object Uint16Array]',\n\t '[object Int32Array]',\n\t '[object Uint32Array]',\n\t '[object Float32Array]',\n\t '[object Float64Array]'\n\t ]\n\t\n\t var isDataView = function(obj) {\n\t return obj && DataView.prototype.isPrototypeOf(obj)\n\t }\n\t\n\t var isArrayBufferView = ArrayBuffer.isView || function(obj) {\n\t return obj && viewClasses.indexOf(Object.prototype.toString.call(obj)) > -1\n\t }\n\t }\n\t\n\t function normalizeName(name) {\n\t if (typeof name !== 'string') {\n\t name = String(name)\n\t }\n\t if (/[^a-z0-9\\-#$%&'*+.\\^_`|~]/i.test(name)) {\n\t throw new TypeError('Invalid character in header field name')\n\t }\n\t return name.toLowerCase()\n\t }\n\t\n\t function normalizeValue(value) {\n\t if (typeof value !== 'string') {\n\t value = String(value)\n\t }\n\t return value\n\t }\n\t\n\t // Build a destructive iterator for the value list\n\t function iteratorFor(items) {\n\t var iterator = {\n\t next: function() {\n\t var value = items.shift()\n\t return {done: value === undefined, value: value}\n\t }\n\t }\n\t\n\t if (support.iterable) {\n\t iterator[Symbol.iterator] = function() {\n\t return iterator\n\t }\n\t }\n\t\n\t return iterator\n\t }\n\t\n\t function Headers(headers) {\n\t this.map = {}\n\t\n\t if (headers instanceof Headers) {\n\t headers.forEach(function(value, name) {\n\t this.append(name, value)\n\t }, this)\n\t\n\t } else if (headers) {\n\t Object.getOwnPropertyNames(headers).forEach(function(name) {\n\t this.append(name, headers[name])\n\t }, this)\n\t }\n\t }\n\t\n\t Headers.prototype.append = function(name, value) {\n\t name = normalizeName(name)\n\t value = normalizeValue(value)\n\t var oldValue = this.map[name]\n\t this.map[name] = oldValue ? oldValue+','+value : value\n\t }\n\t\n\t Headers.prototype['delete'] = function(name) {\n\t delete this.map[normalizeName(name)]\n\t }\n\t\n\t Headers.prototype.get = function(name) {\n\t name = normalizeName(name)\n\t return this.has(name) ? this.map[name] : null\n\t }\n\t\n\t Headers.prototype.has = function(name) {\n\t return this.map.hasOwnProperty(normalizeName(name))\n\t }\n\t\n\t Headers.prototype.set = function(name, value) {\n\t this.map[normalizeName(name)] = normalizeValue(value)\n\t }\n\t\n\t Headers.prototype.forEach = function(callback, thisArg) {\n\t for (var name in this.map) {\n\t if (this.map.hasOwnProperty(name)) {\n\t callback.call(thisArg, this.map[name], name, this)\n\t }\n\t }\n\t }\n\t\n\t Headers.prototype.keys = function() {\n\t var items = []\n\t this.forEach(function(value, name) { items.push(name) })\n\t return iteratorFor(items)\n\t }\n\t\n\t Headers.prototype.values = function() {\n\t var items = []\n\t this.forEach(function(value) { items.push(value) })\n\t return iteratorFor(items)\n\t }\n\t\n\t Headers.prototype.entries = function() {\n\t var items = []\n\t this.forEach(function(value, name) { items.push([name, value]) })\n\t return iteratorFor(items)\n\t }\n\t\n\t if (support.iterable) {\n\t Headers.prototype[Symbol.iterator] = Headers.prototype.entries\n\t }\n\t\n\t function consumed(body) {\n\t if (body.bodyUsed) {\n\t return Promise.reject(new TypeError('Already read'))\n\t }\n\t body.bodyUsed = true\n\t }\n\t\n\t function fileReaderReady(reader) {\n\t return new Promise(function(resolve, reject) {\n\t reader.onload = function() {\n\t resolve(reader.result)\n\t }\n\t reader.onerror = function() {\n\t reject(reader.error)\n\t }\n\t })\n\t }\n\t\n\t function readBlobAsArrayBuffer(blob) {\n\t var reader = new FileReader()\n\t var promise = fileReaderReady(reader)\n\t reader.readAsArrayBuffer(blob)\n\t return promise\n\t }\n\t\n\t function readBlobAsText(blob) {\n\t var reader = new FileReader()\n\t var promise = fileReaderReady(reader)\n\t reader.readAsText(blob)\n\t return promise\n\t }\n\t\n\t function readArrayBufferAsText(buf) {\n\t var view = new Uint8Array(buf)\n\t var chars = new Array(view.length)\n\t\n\t for (var i = 0; i < view.length; i++) {\n\t chars[i] = String.fromCharCode(view[i])\n\t }\n\t return chars.join('')\n\t }\n\t\n\t function bufferClone(buf) {\n\t if (buf.slice) {\n\t return buf.slice(0)\n\t } else {\n\t var view = new Uint8Array(buf.byteLength)\n\t view.set(new Uint8Array(buf))\n\t return view.buffer\n\t }\n\t }\n\t\n\t function Body() {\n\t this.bodyUsed = false\n\t\n\t this._initBody = function(body) {\n\t this._bodyInit = body\n\t if (!body) {\n\t this._bodyText = ''\n\t } else if (typeof body === 'string') {\n\t this._bodyText = body\n\t } else if (support.blob && Blob.prototype.isPrototypeOf(body)) {\n\t this._bodyBlob = body\n\t } else if (support.formData && FormData.prototype.isPrototypeOf(body)) {\n\t this._bodyFormData = body\n\t } else if (support.searchParams && URLSearchParams.prototype.isPrototypeOf(body)) {\n\t this._bodyText = body.toString()\n\t } else if (support.arrayBuffer && support.blob && isDataView(body)) {\n\t this._bodyArrayBuffer = bufferClone(body.buffer)\n\t // IE 10-11 can't handle a DataView body.\n\t this._bodyInit = new Blob([this._bodyArrayBuffer])\n\t } else if (support.arrayBuffer && (ArrayBuffer.prototype.isPrototypeOf(body) || isArrayBufferView(body))) {\n\t this._bodyArrayBuffer = bufferClone(body)\n\t } else {\n\t throw new Error('unsupported BodyInit type')\n\t }\n\t\n\t if (!this.headers.get('content-type')) {\n\t if (typeof body === 'string') {\n\t this.headers.set('content-type', 'text/plain;charset=UTF-8')\n\t } else if (this._bodyBlob && this._bodyBlob.type) {\n\t this.headers.set('content-type', this._bodyBlob.type)\n\t } else if (support.searchParams && URLSearchParams.prototype.isPrototypeOf(body)) {\n\t this.headers.set('content-type', 'application/x-www-form-urlencoded;charset=UTF-8')\n\t }\n\t }\n\t }\n\t\n\t if (support.blob) {\n\t this.blob = function() {\n\t var rejected = consumed(this)\n\t if (rejected) {\n\t return rejected\n\t }\n\t\n\t if (this._bodyBlob) {\n\t return Promise.resolve(this._bodyBlob)\n\t } else if (this._bodyArrayBuffer) {\n\t return Promise.resolve(new Blob([this._bodyArrayBuffer]))\n\t } else if (this._bodyFormData) {\n\t throw new Error('could not read FormData body as blob')\n\t } else {\n\t return Promise.resolve(new Blob([this._bodyText]))\n\t }\n\t }\n\t\n\t this.arrayBuffer = function() {\n\t if (this._bodyArrayBuffer) {\n\t return consumed(this) || Promise.resolve(this._bodyArrayBuffer)\n\t } else {\n\t return this.blob().then(readBlobAsArrayBuffer)\n\t }\n\t }\n\t }\n\t\n\t this.text = function() {\n\t var rejected = consumed(this)\n\t if (rejected) {\n\t return rejected\n\t }\n\t\n\t if (this._bodyBlob) {\n\t return readBlobAsText(this._bodyBlob)\n\t } else if (this._bodyArrayBuffer) {\n\t return Promise.resolve(readArrayBufferAsText(this._bodyArrayBuffer))\n\t } else if (this._bodyFormData) {\n\t throw new Error('could not read FormData body as text')\n\t } else {\n\t return Promise.resolve(this._bodyText)\n\t }\n\t }\n\t\n\t if (support.formData) {\n\t this.formData = function() {\n\t return this.text().then(decode)\n\t }\n\t }\n\t\n\t this.json = function() {\n\t return this.text().then(JSON.parse)\n\t }\n\t\n\t return this\n\t }\n\t\n\t // HTTP methods whose capitalization should be normalized\n\t var methods = ['DELETE', 'GET', 'HEAD', 'OPTIONS', 'POST', 'PUT']\n\t\n\t function normalizeMethod(method) {\n\t var upcased = method.toUpperCase()\n\t return (methods.indexOf(upcased) > -1) ? upcased : method\n\t }\n\t\n\t function Request(input, options) {\n\t options = options || {}\n\t var body = options.body\n\t\n\t if (typeof input === 'string') {\n\t this.url = input\n\t } else {\n\t if (input.bodyUsed) {\n\t throw new TypeError('Already read')\n\t }\n\t this.url = input.url\n\t this.credentials = input.credentials\n\t if (!options.headers) {\n\t this.headers = new Headers(input.headers)\n\t }\n\t this.method = input.method\n\t this.mode = input.mode\n\t if (!body && input._bodyInit != null) {\n\t body = input._bodyInit\n\t input.bodyUsed = true\n\t }\n\t }\n\t\n\t this.credentials = options.credentials || this.credentials || 'omit'\n\t if (options.headers || !this.headers) {\n\t this.headers = new Headers(options.headers)\n\t }\n\t this.method = normalizeMethod(options.method || this.method || 'GET')\n\t this.mode = options.mode || this.mode || null\n\t this.referrer = null\n\t\n\t if ((this.method === 'GET' || this.method === 'HEAD') && body) {\n\t throw new TypeError('Body not allowed for GET or HEAD requests')\n\t }\n\t this._initBody(body)\n\t }\n\t\n\t Request.prototype.clone = function() {\n\t return new Request(this, { body: this._bodyInit })\n\t }\n\t\n\t function decode(body) {\n\t var form = new FormData()\n\t body.trim().split('&').forEach(function(bytes) {\n\t if (bytes) {\n\t var split = bytes.split('=')\n\t var name = split.shift().replace(/\\+/g, ' ')\n\t var value = split.join('=').replace(/\\+/g, ' ')\n\t form.append(decodeURIComponent(name), decodeURIComponent(value))\n\t }\n\t })\n\t return form\n\t }\n\t\n\t function parseHeaders(rawHeaders) {\n\t var headers = new Headers()\n\t rawHeaders.split('\\r\\n').forEach(function(line) {\n\t var parts = line.split(':')\n\t var key = parts.shift().trim()\n\t if (key) {\n\t var value = parts.join(':').trim()\n\t headers.append(key, value)\n\t }\n\t })\n\t return headers\n\t }\n\t\n\t Body.call(Request.prototype)\n\t\n\t function Response(bodyInit, options) {\n\t if (!options) {\n\t options = {}\n\t }\n\t\n\t this.type = 'default'\n\t this.status = 'status' in options ? options.status : 200\n\t this.ok = this.status >= 200 && this.status < 300\n\t this.statusText = 'statusText' in options ? options.statusText : 'OK'\n\t this.headers = new Headers(options.headers)\n\t this.url = options.url || ''\n\t this._initBody(bodyInit)\n\t }\n\t\n\t Body.call(Response.prototype)\n\t\n\t Response.prototype.clone = function() {\n\t return new Response(this._bodyInit, {\n\t status: this.status,\n\t statusText: this.statusText,\n\t headers: new Headers(this.headers),\n\t url: this.url\n\t })\n\t }\n\t\n\t Response.error = function() {\n\t var response = new Response(null, {status: 0, statusText: ''})\n\t response.type = 'error'\n\t return response\n\t }\n\t\n\t var redirectStatuses = [301, 302, 303, 307, 308]\n\t\n\t Response.redirect = function(url, status) {\n\t if (redirectStatuses.indexOf(status) === -1) {\n\t throw new RangeError('Invalid status code')\n\t }\n\t\n\t return new Response(null, {status: status, headers: {location: url}})\n\t }\n\t\n\t self.Headers = Headers\n\t self.Request = Request\n\t self.Response = Response\n\t\n\t self.fetch = function(input, init) {\n\t return new Promise(function(resolve, reject) {\n\t var request = new Request(input, init)\n\t var xhr = new XMLHttpRequest()\n\t\n\t xhr.onload = function() {\n\t var options = {\n\t status: xhr.status,\n\t statusText: xhr.statusText,\n\t headers: parseHeaders(xhr.getAllResponseHeaders() || '')\n\t }\n\t options.url = 'responseURL' in xhr ? xhr.responseURL : options.headers.get('X-Request-URL')\n\t var body = 'response' in xhr ? xhr.response : xhr.responseText\n\t resolve(new Response(body, options))\n\t }\n\t\n\t xhr.onerror = function() {\n\t reject(new TypeError('Network request failed'))\n\t }\n\t\n\t xhr.ontimeout = function() {\n\t reject(new TypeError('Network request failed'))\n\t }\n\t\n\t xhr.open(request.method, request.url, true)\n\t\n\t if (request.credentials === 'include') {\n\t xhr.withCredentials = true\n\t }\n\t\n\t if ('responseType' in xhr && support.blob) {\n\t xhr.responseType = 'blob'\n\t }\n\t\n\t request.headers.forEach(function(value, name) {\n\t xhr.setRequestHeader(name, value)\n\t })\n\t\n\t xhr.send(typeof request._bodyInit === 'undefined' ? null : request._bodyInit)\n\t })\n\t }\n\t self.fetch.polyfill = true\n\t})(typeof self !== 'undefined' ? self : this);\n\n\n/***/ }\n/******/ ]);\n\n\n// WEBPACK FOOTER //\n// whatwg-fetch.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 f80ecde20480b792f648","(function(self) {\n 'use strict';\n\n if (self.fetch) {\n return\n }\n\n var support = {\n searchParams: 'URLSearchParams' in self,\n iterable: 'Symbol' in self && 'iterator' in Symbol,\n blob: 'FileReader' in self && 'Blob' in self && (function() {\n try {\n new Blob()\n return true\n } catch(e) {\n return false\n }\n })(),\n formData: 'FormData' in self,\n arrayBuffer: 'ArrayBuffer' in self\n }\n\n if (support.arrayBuffer) {\n var viewClasses = [\n '[object Int8Array]',\n '[object Uint8Array]',\n '[object Uint8ClampedArray]',\n '[object Int16Array]',\n '[object Uint16Array]',\n '[object Int32Array]',\n '[object Uint32Array]',\n '[object Float32Array]',\n '[object Float64Array]'\n ]\n\n var isDataView = function(obj) {\n return obj && DataView.prototype.isPrototypeOf(obj)\n }\n\n var isArrayBufferView = ArrayBuffer.isView || function(obj) {\n return obj && viewClasses.indexOf(Object.prototype.toString.call(obj)) > -1\n }\n }\n\n function normalizeName(name) {\n if (typeof name !== 'string') {\n name = String(name)\n }\n if (/[^a-z0-9\\-#$%&'*+.\\^_`|~]/i.test(name)) {\n throw new TypeError('Invalid character in header field name')\n }\n return name.toLowerCase()\n }\n\n function normalizeValue(value) {\n if (typeof value !== 'string') {\n value = String(value)\n }\n return value\n }\n\n // Build a destructive iterator for the value list\n function iteratorFor(items) {\n var iterator = {\n next: function() {\n var value = items.shift()\n return {done: value === undefined, value: value}\n }\n }\n\n if (support.iterable) {\n iterator[Symbol.iterator] = function() {\n return iterator\n }\n }\n\n return iterator\n }\n\n function Headers(headers) {\n this.map = {}\n\n if (headers instanceof Headers) {\n headers.forEach(function(value, name) {\n this.append(name, value)\n }, this)\n\n } else if (headers) {\n Object.getOwnPropertyNames(headers).forEach(function(name) {\n this.append(name, headers[name])\n }, this)\n }\n }\n\n Headers.prototype.append = function(name, value) {\n name = normalizeName(name)\n value = normalizeValue(value)\n var oldValue = this.map[name]\n this.map[name] = oldValue ? oldValue+','+value : value\n }\n\n Headers.prototype['delete'] = function(name) {\n delete this.map[normalizeName(name)]\n }\n\n Headers.prototype.get = function(name) {\n name = normalizeName(name)\n return this.has(name) ? this.map[name] : null\n }\n\n Headers.prototype.has = function(name) {\n return this.map.hasOwnProperty(normalizeName(name))\n }\n\n Headers.prototype.set = function(name, value) {\n this.map[normalizeName(name)] = normalizeValue(value)\n }\n\n Headers.prototype.forEach = function(callback, thisArg) {\n for (var name in this.map) {\n if (this.map.hasOwnProperty(name)) {\n callback.call(thisArg, this.map[name], name, this)\n }\n }\n }\n\n Headers.prototype.keys = function() {\n var items = []\n this.forEach(function(value, name) { items.push(name) })\n return iteratorFor(items)\n }\n\n Headers.prototype.values = function() {\n var items = []\n this.forEach(function(value) { items.push(value) })\n return iteratorFor(items)\n }\n\n Headers.prototype.entries = function() {\n var items = []\n this.forEach(function(value, name) { items.push([name, value]) })\n return iteratorFor(items)\n }\n\n if (support.iterable) {\n Headers.prototype[Symbol.iterator] = Headers.prototype.entries\n }\n\n function consumed(body) {\n if (body.bodyUsed) {\n return Promise.reject(new TypeError('Already read'))\n }\n body.bodyUsed = true\n }\n\n function fileReaderReady(reader) {\n return new Promise(function(resolve, reject) {\n reader.onload = function() {\n resolve(reader.result)\n }\n reader.onerror = function() {\n reject(reader.error)\n }\n })\n }\n\n function readBlobAsArrayBuffer(blob) {\n var reader = new FileReader()\n var promise = fileReaderReady(reader)\n reader.readAsArrayBuffer(blob)\n return promise\n }\n\n function readBlobAsText(blob) {\n var reader = new FileReader()\n var promise = fileReaderReady(reader)\n reader.readAsText(blob)\n return promise\n }\n\n function readArrayBufferAsText(buf) {\n var view = new Uint8Array(buf)\n var chars = new Array(view.length)\n\n for (var i = 0; i < view.length; i++) {\n chars[i] = String.fromCharCode(view[i])\n }\n return chars.join('')\n }\n\n function bufferClone(buf) {\n if (buf.slice) {\n return buf.slice(0)\n } else {\n var view = new Uint8Array(buf.byteLength)\n view.set(new Uint8Array(buf))\n return view.buffer\n }\n }\n\n function Body() {\n this.bodyUsed = false\n\n this._initBody = function(body) {\n this._bodyInit = body\n if (!body) {\n this._bodyText = ''\n } else if (typeof body === 'string') {\n this._bodyText = body\n } else if (support.blob && Blob.prototype.isPrototypeOf(body)) {\n this._bodyBlob = body\n } else if (support.formData && FormData.prototype.isPrototypeOf(body)) {\n this._bodyFormData = body\n } else if (support.searchParams && URLSearchParams.prototype.isPrototypeOf(body)) {\n this._bodyText = body.toString()\n } else if (support.arrayBuffer && support.blob && isDataView(body)) {\n this._bodyArrayBuffer = bufferClone(body.buffer)\n // IE 10-11 can't handle a DataView body.\n this._bodyInit = new Blob([this._bodyArrayBuffer])\n } else if (support.arrayBuffer && (ArrayBuffer.prototype.isPrototypeOf(body) || isArrayBufferView(body))) {\n this._bodyArrayBuffer = bufferClone(body)\n } else {\n throw new Error('unsupported BodyInit type')\n }\n\n if (!this.headers.get('content-type')) {\n if (typeof body === 'string') {\n this.headers.set('content-type', 'text/plain;charset=UTF-8')\n } else if (this._bodyBlob && this._bodyBlob.type) {\n this.headers.set('content-type', this._bodyBlob.type)\n } else if (support.searchParams && URLSearchParams.prototype.isPrototypeOf(body)) {\n this.headers.set('content-type', 'application/x-www-form-urlencoded;charset=UTF-8')\n }\n }\n }\n\n if (support.blob) {\n this.blob = function() {\n var rejected = consumed(this)\n if (rejected) {\n return rejected\n }\n\n if (this._bodyBlob) {\n return Promise.resolve(this._bodyBlob)\n } else if (this._bodyArrayBuffer) {\n return Promise.resolve(new Blob([this._bodyArrayBuffer]))\n } else if (this._bodyFormData) {\n throw new Error('could not read FormData body as blob')\n } else {\n return Promise.resolve(new Blob([this._bodyText]))\n }\n }\n\n this.arrayBuffer = function() {\n if (this._bodyArrayBuffer) {\n return consumed(this) || Promise.resolve(this._bodyArrayBuffer)\n } else {\n return this.blob().then(readBlobAsArrayBuffer)\n }\n }\n }\n\n this.text = function() {\n var rejected = consumed(this)\n if (rejected) {\n return rejected\n }\n\n if (this._bodyBlob) {\n return readBlobAsText(this._bodyBlob)\n } else if (this._bodyArrayBuffer) {\n return Promise.resolve(readArrayBufferAsText(this._bodyArrayBuffer))\n } else if (this._bodyFormData) {\n throw new Error('could not read FormData body as text')\n } else {\n return Promise.resolve(this._bodyText)\n }\n }\n\n if (support.formData) {\n this.formData = function() {\n return this.text().then(decode)\n }\n }\n\n this.json = function() {\n return this.text().then(JSON.parse)\n }\n\n return this\n }\n\n // HTTP methods whose capitalization should be normalized\n var methods = ['DELETE', 'GET', 'HEAD', 'OPTIONS', 'POST', 'PUT']\n\n function normalizeMethod(method) {\n var upcased = method.toUpperCase()\n return (methods.indexOf(upcased) > -1) ? upcased : method\n }\n\n function Request(input, options) {\n options = options || {}\n var body = options.body\n\n if (typeof input === 'string') {\n this.url = input\n } else {\n if (input.bodyUsed) {\n throw new TypeError('Already read')\n }\n this.url = input.url\n this.credentials = input.credentials\n if (!options.headers) {\n this.headers = new Headers(input.headers)\n }\n this.method = input.method\n this.mode = input.mode\n if (!body && input._bodyInit != null) {\n body = input._bodyInit\n input.bodyUsed = true\n }\n }\n\n this.credentials = options.credentials || this.credentials || 'omit'\n if (options.headers || !this.headers) {\n this.headers = new Headers(options.headers)\n }\n this.method = normalizeMethod(options.method || this.method || 'GET')\n this.mode = options.mode || this.mode || null\n this.referrer = null\n\n if ((this.method === 'GET' || this.method === 'HEAD') && body) {\n throw new TypeError('Body not allowed for GET or HEAD requests')\n }\n this._initBody(body)\n }\n\n Request.prototype.clone = function() {\n return new Request(this, { body: this._bodyInit })\n }\n\n function decode(body) {\n var form = new FormData()\n body.trim().split('&').forEach(function(bytes) {\n if (bytes) {\n var split = bytes.split('=')\n var name = split.shift().replace(/\\+/g, ' ')\n var value = split.join('=').replace(/\\+/g, ' ')\n form.append(decodeURIComponent(name), decodeURIComponent(value))\n }\n })\n return form\n }\n\n function parseHeaders(rawHeaders) {\n var headers = new Headers()\n rawHeaders.split('\\r\\n').forEach(function(line) {\n var parts = line.split(':')\n var key = parts.shift().trim()\n if (key) {\n var value = parts.join(':').trim()\n headers.append(key, value)\n }\n })\n return headers\n }\n\n Body.call(Request.prototype)\n\n function Response(bodyInit, options) {\n if (!options) {\n options = {}\n }\n\n this.type = 'default'\n this.status = 'status' in options ? options.status : 200\n this.ok = this.status >= 200 && this.status < 300\n this.statusText = 'statusText' in options ? options.statusText : 'OK'\n this.headers = new Headers(options.headers)\n this.url = options.url || ''\n this._initBody(bodyInit)\n }\n\n Body.call(Response.prototype)\n\n Response.prototype.clone = function() {\n return new Response(this._bodyInit, {\n status: this.status,\n statusText: this.statusText,\n headers: new Headers(this.headers),\n url: this.url\n })\n }\n\n Response.error = function() {\n var response = new Response(null, {status: 0, statusText: ''})\n response.type = 'error'\n return response\n }\n\n var redirectStatuses = [301, 302, 303, 307, 308]\n\n Response.redirect = function(url, status) {\n if (redirectStatuses.indexOf(status) === -1) {\n throw new RangeError('Invalid status code')\n }\n\n return new Response(null, {status: status, headers: {location: url}})\n }\n\n self.Headers = Headers\n self.Request = Request\n self.Response = Response\n\n self.fetch = function(input, init) {\n return new Promise(function(resolve, reject) {\n var request = new Request(input, init)\n var xhr = new XMLHttpRequest()\n\n xhr.onload = function() {\n var options = {\n status: xhr.status,\n statusText: xhr.statusText,\n headers: parseHeaders(xhr.getAllResponseHeaders() || '')\n }\n options.url = 'responseURL' in xhr ? xhr.responseURL : options.headers.get('X-Request-URL')\n var body = 'response' in xhr ? xhr.response : xhr.responseText\n resolve(new Response(body, options))\n }\n\n xhr.onerror = function() {\n reject(new TypeError('Network request failed'))\n }\n\n xhr.ontimeout = function() {\n reject(new TypeError('Network request failed'))\n }\n\n xhr.open(request.method, request.url, true)\n\n if (request.credentials === 'include') {\n xhr.withCredentials = true\n }\n\n if ('responseType' in xhr && support.blob) {\n xhr.responseType = 'blob'\n }\n\n request.headers.forEach(function(value, name) {\n xhr.setRequestHeader(name, value)\n })\n\n xhr.send(typeof request._bodyInit === 'undefined' ? null : request._bodyInit)\n })\n }\n self.fetch.polyfill = true\n})(typeof self !== 'undefined' ? self : this);\n\n\n\n//////////////////\n// WEBPACK FOOTER\n// ./~/whatwg-fetch/fetch.js\n// module id = 0\n// module chunks = 1"],"sourceRoot":""} \ No newline at end of file +{"version":3,"sources":["webpack:///whatwg-fetch.js","webpack:///webpack/bootstrap 7633cb891787da44e4d8?e80a","webpack:///./~/whatwg-fetch/fetch.js"],"names":["codex","modules","__webpack_require__","moduleId","installedModules","exports","module","id","loaded","call","m","c","p","self","normalizeName","name","String","test","TypeError","toLowerCase","normalizeValue","value","iteratorFor","items","iterator","next","shift","done","undefined","support","iterable","Symbol","Headers","headers","this","map","forEach","append","Object","getOwnPropertyNames","consumed","body","bodyUsed","Promise","reject","fileReaderReady","reader","resolve","onload","result","onerror","error","readBlobAsArrayBuffer","blob","FileReader","promise","readAsArrayBuffer","readBlobAsText","readAsText","readArrayBufferAsText","buf","view","Uint8Array","chars","Array","length","i","fromCharCode","join","bufferClone","slice","byteLength","set","buffer","Body","_initBody","_bodyInit","_bodyText","Blob","prototype","isPrototypeOf","_bodyBlob","formData","FormData","_bodyFormData","searchParams","URLSearchParams","toString","arrayBuffer","isDataView","_bodyArrayBuffer","ArrayBuffer","isArrayBufferView","Error","get","type","rejected","then","text","decode","json","JSON","parse","normalizeMethod","method","upcased","toUpperCase","methods","indexOf","Request","input","options","url","credentials","mode","referrer","form","trim","split","bytes","replace","decodeURIComponent","parseHeaders","rawHeaders","line","parts","key","Response","bodyInit","status","ok","statusText","fetch","e","viewClasses","obj","DataView","isView","oldValue","has","hasOwnProperty","callback","thisArg","keys","push","values","entries","clone","response","redirectStatuses","redirect","RangeError","location","init","request","xhr","XMLHttpRequest","getAllResponseHeaders","responseURL","responseText","ontimeout","open","withCredentials","responseType","setRequestHeader","send","polyfill"],"mappings":"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","file":"whatwg-fetch.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) {\n\n\t(function(self) {\n\t 'use strict';\n\t\n\t if (self.fetch) {\n\t return\n\t }\n\t\n\t var support = {\n\t searchParams: 'URLSearchParams' in self,\n\t iterable: 'Symbol' in self && 'iterator' in Symbol,\n\t blob: 'FileReader' in self && 'Blob' in self && (function() {\n\t try {\n\t new Blob()\n\t return true\n\t } catch(e) {\n\t return false\n\t }\n\t })(),\n\t formData: 'FormData' in self,\n\t arrayBuffer: 'ArrayBuffer' in self\n\t }\n\t\n\t if (support.arrayBuffer) {\n\t var viewClasses = [\n\t '[object Int8Array]',\n\t '[object Uint8Array]',\n\t '[object Uint8ClampedArray]',\n\t '[object Int16Array]',\n\t '[object Uint16Array]',\n\t '[object Int32Array]',\n\t '[object Uint32Array]',\n\t '[object Float32Array]',\n\t '[object Float64Array]'\n\t ]\n\t\n\t var isDataView = function(obj) {\n\t return obj && DataView.prototype.isPrototypeOf(obj)\n\t }\n\t\n\t var isArrayBufferView = ArrayBuffer.isView || function(obj) {\n\t return obj && viewClasses.indexOf(Object.prototype.toString.call(obj)) > -1\n\t }\n\t }\n\t\n\t function normalizeName(name) {\n\t if (typeof name !== 'string') {\n\t name = String(name)\n\t }\n\t if (/[^a-z0-9\\-#$%&'*+.\\^_`|~]/i.test(name)) {\n\t throw new TypeError('Invalid character in header field name')\n\t }\n\t return name.toLowerCase()\n\t }\n\t\n\t function normalizeValue(value) {\n\t if (typeof value !== 'string') {\n\t value = String(value)\n\t }\n\t return value\n\t }\n\t\n\t // Build a destructive iterator for the value list\n\t function iteratorFor(items) {\n\t var iterator = {\n\t next: function() {\n\t var value = items.shift()\n\t return {done: value === undefined, value: value}\n\t }\n\t }\n\t\n\t if (support.iterable) {\n\t iterator[Symbol.iterator] = function() {\n\t return iterator\n\t }\n\t }\n\t\n\t return iterator\n\t }\n\t\n\t function Headers(headers) {\n\t this.map = {}\n\t\n\t if (headers instanceof Headers) {\n\t headers.forEach(function(value, name) {\n\t this.append(name, value)\n\t }, this)\n\t\n\t } else if (headers) {\n\t Object.getOwnPropertyNames(headers).forEach(function(name) {\n\t this.append(name, headers[name])\n\t }, this)\n\t }\n\t }\n\t\n\t Headers.prototype.append = function(name, value) {\n\t name = normalizeName(name)\n\t value = normalizeValue(value)\n\t var oldValue = this.map[name]\n\t this.map[name] = oldValue ? oldValue+','+value : value\n\t }\n\t\n\t Headers.prototype['delete'] = function(name) {\n\t delete this.map[normalizeName(name)]\n\t }\n\t\n\t Headers.prototype.get = function(name) {\n\t name = normalizeName(name)\n\t return this.has(name) ? this.map[name] : null\n\t }\n\t\n\t Headers.prototype.has = function(name) {\n\t return this.map.hasOwnProperty(normalizeName(name))\n\t }\n\t\n\t Headers.prototype.set = function(name, value) {\n\t this.map[normalizeName(name)] = normalizeValue(value)\n\t }\n\t\n\t Headers.prototype.forEach = function(callback, thisArg) {\n\t for (var name in this.map) {\n\t if (this.map.hasOwnProperty(name)) {\n\t callback.call(thisArg, this.map[name], name, this)\n\t }\n\t }\n\t }\n\t\n\t Headers.prototype.keys = function() {\n\t var items = []\n\t this.forEach(function(value, name) { items.push(name) })\n\t return iteratorFor(items)\n\t }\n\t\n\t Headers.prototype.values = function() {\n\t var items = []\n\t this.forEach(function(value) { items.push(value) })\n\t return iteratorFor(items)\n\t }\n\t\n\t Headers.prototype.entries = function() {\n\t var items = []\n\t this.forEach(function(value, name) { items.push([name, value]) })\n\t return iteratorFor(items)\n\t }\n\t\n\t if (support.iterable) {\n\t Headers.prototype[Symbol.iterator] = Headers.prototype.entries\n\t }\n\t\n\t function consumed(body) {\n\t if (body.bodyUsed) {\n\t return Promise.reject(new TypeError('Already read'))\n\t }\n\t body.bodyUsed = true\n\t }\n\t\n\t function fileReaderReady(reader) {\n\t return new Promise(function(resolve, reject) {\n\t reader.onload = function() {\n\t resolve(reader.result)\n\t }\n\t reader.onerror = function() {\n\t reject(reader.error)\n\t }\n\t })\n\t }\n\t\n\t function readBlobAsArrayBuffer(blob) {\n\t var reader = new FileReader()\n\t var promise = fileReaderReady(reader)\n\t reader.readAsArrayBuffer(blob)\n\t return promise\n\t }\n\t\n\t function readBlobAsText(blob) {\n\t var reader = new FileReader()\n\t var promise = fileReaderReady(reader)\n\t reader.readAsText(blob)\n\t return promise\n\t }\n\t\n\t function readArrayBufferAsText(buf) {\n\t var view = new Uint8Array(buf)\n\t var chars = new Array(view.length)\n\t\n\t for (var i = 0; i < view.length; i++) {\n\t chars[i] = String.fromCharCode(view[i])\n\t }\n\t return chars.join('')\n\t }\n\t\n\t function bufferClone(buf) {\n\t if (buf.slice) {\n\t return buf.slice(0)\n\t } else {\n\t var view = new Uint8Array(buf.byteLength)\n\t view.set(new Uint8Array(buf))\n\t return view.buffer\n\t }\n\t }\n\t\n\t function Body() {\n\t this.bodyUsed = false\n\t\n\t this._initBody = function(body) {\n\t this._bodyInit = body\n\t if (!body) {\n\t this._bodyText = ''\n\t } else if (typeof body === 'string') {\n\t this._bodyText = body\n\t } else if (support.blob && Blob.prototype.isPrototypeOf(body)) {\n\t this._bodyBlob = body\n\t } else if (support.formData && FormData.prototype.isPrototypeOf(body)) {\n\t this._bodyFormData = body\n\t } else if (support.searchParams && URLSearchParams.prototype.isPrototypeOf(body)) {\n\t this._bodyText = body.toString()\n\t } else if (support.arrayBuffer && support.blob && isDataView(body)) {\n\t this._bodyArrayBuffer = bufferClone(body.buffer)\n\t // IE 10-11 can't handle a DataView body.\n\t this._bodyInit = new Blob([this._bodyArrayBuffer])\n\t } else if (support.arrayBuffer && (ArrayBuffer.prototype.isPrototypeOf(body) || isArrayBufferView(body))) {\n\t this._bodyArrayBuffer = bufferClone(body)\n\t } else {\n\t throw new Error('unsupported BodyInit type')\n\t }\n\t\n\t if (!this.headers.get('content-type')) {\n\t if (typeof body === 'string') {\n\t this.headers.set('content-type', 'text/plain;charset=UTF-8')\n\t } else if (this._bodyBlob && this._bodyBlob.type) {\n\t this.headers.set('content-type', this._bodyBlob.type)\n\t } else if (support.searchParams && URLSearchParams.prototype.isPrototypeOf(body)) {\n\t this.headers.set('content-type', 'application/x-www-form-urlencoded;charset=UTF-8')\n\t }\n\t }\n\t }\n\t\n\t if (support.blob) {\n\t this.blob = function() {\n\t var rejected = consumed(this)\n\t if (rejected) {\n\t return rejected\n\t }\n\t\n\t if (this._bodyBlob) {\n\t return Promise.resolve(this._bodyBlob)\n\t } else if (this._bodyArrayBuffer) {\n\t return Promise.resolve(new Blob([this._bodyArrayBuffer]))\n\t } else if (this._bodyFormData) {\n\t throw new Error('could not read FormData body as blob')\n\t } else {\n\t return Promise.resolve(new Blob([this._bodyText]))\n\t }\n\t }\n\t\n\t this.arrayBuffer = function() {\n\t if (this._bodyArrayBuffer) {\n\t return consumed(this) || Promise.resolve(this._bodyArrayBuffer)\n\t } else {\n\t return this.blob().then(readBlobAsArrayBuffer)\n\t }\n\t }\n\t }\n\t\n\t this.text = function() {\n\t var rejected = consumed(this)\n\t if (rejected) {\n\t return rejected\n\t }\n\t\n\t if (this._bodyBlob) {\n\t return readBlobAsText(this._bodyBlob)\n\t } else if (this._bodyArrayBuffer) {\n\t return Promise.resolve(readArrayBufferAsText(this._bodyArrayBuffer))\n\t } else if (this._bodyFormData) {\n\t throw new Error('could not read FormData body as text')\n\t } else {\n\t return Promise.resolve(this._bodyText)\n\t }\n\t }\n\t\n\t if (support.formData) {\n\t this.formData = function() {\n\t return this.text().then(decode)\n\t }\n\t }\n\t\n\t this.json = function() {\n\t return this.text().then(JSON.parse)\n\t }\n\t\n\t return this\n\t }\n\t\n\t // HTTP methods whose capitalization should be normalized\n\t var methods = ['DELETE', 'GET', 'HEAD', 'OPTIONS', 'POST', 'PUT']\n\t\n\t function normalizeMethod(method) {\n\t var upcased = method.toUpperCase()\n\t return (methods.indexOf(upcased) > -1) ? upcased : method\n\t }\n\t\n\t function Request(input, options) {\n\t options = options || {}\n\t var body = options.body\n\t\n\t if (typeof input === 'string') {\n\t this.url = input\n\t } else {\n\t if (input.bodyUsed) {\n\t throw new TypeError('Already read')\n\t }\n\t this.url = input.url\n\t this.credentials = input.credentials\n\t if (!options.headers) {\n\t this.headers = new Headers(input.headers)\n\t }\n\t this.method = input.method\n\t this.mode = input.mode\n\t if (!body && input._bodyInit != null) {\n\t body = input._bodyInit\n\t input.bodyUsed = true\n\t }\n\t }\n\t\n\t this.credentials = options.credentials || this.credentials || 'omit'\n\t if (options.headers || !this.headers) {\n\t this.headers = new Headers(options.headers)\n\t }\n\t this.method = normalizeMethod(options.method || this.method || 'GET')\n\t this.mode = options.mode || this.mode || null\n\t this.referrer = null\n\t\n\t if ((this.method === 'GET' || this.method === 'HEAD') && body) {\n\t throw new TypeError('Body not allowed for GET or HEAD requests')\n\t }\n\t this._initBody(body)\n\t }\n\t\n\t Request.prototype.clone = function() {\n\t return new Request(this, { body: this._bodyInit })\n\t }\n\t\n\t function decode(body) {\n\t var form = new FormData()\n\t body.trim().split('&').forEach(function(bytes) {\n\t if (bytes) {\n\t var split = bytes.split('=')\n\t var name = split.shift().replace(/\\+/g, ' ')\n\t var value = split.join('=').replace(/\\+/g, ' ')\n\t form.append(decodeURIComponent(name), decodeURIComponent(value))\n\t }\n\t })\n\t return form\n\t }\n\t\n\t function parseHeaders(rawHeaders) {\n\t var headers = new Headers()\n\t rawHeaders.split('\\r\\n').forEach(function(line) {\n\t var parts = line.split(':')\n\t var key = parts.shift().trim()\n\t if (key) {\n\t var value = parts.join(':').trim()\n\t headers.append(key, value)\n\t }\n\t })\n\t return headers\n\t }\n\t\n\t Body.call(Request.prototype)\n\t\n\t function Response(bodyInit, options) {\n\t if (!options) {\n\t options = {}\n\t }\n\t\n\t this.type = 'default'\n\t this.status = 'status' in options ? options.status : 200\n\t this.ok = this.status >= 200 && this.status < 300\n\t this.statusText = 'statusText' in options ? options.statusText : 'OK'\n\t this.headers = new Headers(options.headers)\n\t this.url = options.url || ''\n\t this._initBody(bodyInit)\n\t }\n\t\n\t Body.call(Response.prototype)\n\t\n\t Response.prototype.clone = function() {\n\t return new Response(this._bodyInit, {\n\t status: this.status,\n\t statusText: this.statusText,\n\t headers: new Headers(this.headers),\n\t url: this.url\n\t })\n\t }\n\t\n\t Response.error = function() {\n\t var response = new Response(null, {status: 0, statusText: ''})\n\t response.type = 'error'\n\t return response\n\t }\n\t\n\t var redirectStatuses = [301, 302, 303, 307, 308]\n\t\n\t Response.redirect = function(url, status) {\n\t if (redirectStatuses.indexOf(status) === -1) {\n\t throw new RangeError('Invalid status code')\n\t }\n\t\n\t return new Response(null, {status: status, headers: {location: url}})\n\t }\n\t\n\t self.Headers = Headers\n\t self.Request = Request\n\t self.Response = Response\n\t\n\t self.fetch = function(input, init) {\n\t return new Promise(function(resolve, reject) {\n\t var request = new Request(input, init)\n\t var xhr = new XMLHttpRequest()\n\t\n\t xhr.onload = function() {\n\t var options = {\n\t status: xhr.status,\n\t statusText: xhr.statusText,\n\t headers: parseHeaders(xhr.getAllResponseHeaders() || '')\n\t }\n\t options.url = 'responseURL' in xhr ? xhr.responseURL : options.headers.get('X-Request-URL')\n\t var body = 'response' in xhr ? xhr.response : xhr.responseText\n\t resolve(new Response(body, options))\n\t }\n\t\n\t xhr.onerror = function() {\n\t reject(new TypeError('Network request failed'))\n\t }\n\t\n\t xhr.ontimeout = function() {\n\t reject(new TypeError('Network request failed'))\n\t }\n\t\n\t xhr.open(request.method, request.url, true)\n\t\n\t if (request.credentials === 'include') {\n\t xhr.withCredentials = true\n\t }\n\t\n\t if ('responseType' in xhr && support.blob) {\n\t xhr.responseType = 'blob'\n\t }\n\t\n\t request.headers.forEach(function(value, name) {\n\t xhr.setRequestHeader(name, value)\n\t })\n\t\n\t xhr.send(typeof request._bodyInit === 'undefined' ? null : request._bodyInit)\n\t })\n\t }\n\t self.fetch.polyfill = true\n\t})(typeof self !== 'undefined' ? self : this);\n\n\n/***/ }\n/******/ ]);\n\n\n// WEBPACK FOOTER //\n// whatwg-fetch.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 7633cb891787da44e4d8","(function(self) {\n 'use strict';\n\n if (self.fetch) {\n return\n }\n\n var support = {\n searchParams: 'URLSearchParams' in self,\n iterable: 'Symbol' in self && 'iterator' in Symbol,\n blob: 'FileReader' in self && 'Blob' in self && (function() {\n try {\n new Blob()\n return true\n } catch(e) {\n return false\n }\n })(),\n formData: 'FormData' in self,\n arrayBuffer: 'ArrayBuffer' in self\n }\n\n if (support.arrayBuffer) {\n var viewClasses = [\n '[object Int8Array]',\n '[object Uint8Array]',\n '[object Uint8ClampedArray]',\n '[object Int16Array]',\n '[object Uint16Array]',\n '[object Int32Array]',\n '[object Uint32Array]',\n '[object Float32Array]',\n '[object Float64Array]'\n ]\n\n var isDataView = function(obj) {\n return obj && DataView.prototype.isPrototypeOf(obj)\n }\n\n var isArrayBufferView = ArrayBuffer.isView || function(obj) {\n return obj && viewClasses.indexOf(Object.prototype.toString.call(obj)) > -1\n }\n }\n\n function normalizeName(name) {\n if (typeof name !== 'string') {\n name = String(name)\n }\n if (/[^a-z0-9\\-#$%&'*+.\\^_`|~]/i.test(name)) {\n throw new TypeError('Invalid character in header field name')\n }\n return name.toLowerCase()\n }\n\n function normalizeValue(value) {\n if (typeof value !== 'string') {\n value = String(value)\n }\n return value\n }\n\n // Build a destructive iterator for the value list\n function iteratorFor(items) {\n var iterator = {\n next: function() {\n var value = items.shift()\n return {done: value === undefined, value: value}\n }\n }\n\n if (support.iterable) {\n iterator[Symbol.iterator] = function() {\n return iterator\n }\n }\n\n return iterator\n }\n\n function Headers(headers) {\n this.map = {}\n\n if (headers instanceof Headers) {\n headers.forEach(function(value, name) {\n this.append(name, value)\n }, this)\n\n } else if (headers) {\n Object.getOwnPropertyNames(headers).forEach(function(name) {\n this.append(name, headers[name])\n }, this)\n }\n }\n\n Headers.prototype.append = function(name, value) {\n name = normalizeName(name)\n value = normalizeValue(value)\n var oldValue = this.map[name]\n this.map[name] = oldValue ? oldValue+','+value : value\n }\n\n Headers.prototype['delete'] = function(name) {\n delete this.map[normalizeName(name)]\n }\n\n Headers.prototype.get = function(name) {\n name = normalizeName(name)\n return this.has(name) ? this.map[name] : null\n }\n\n Headers.prototype.has = function(name) {\n return this.map.hasOwnProperty(normalizeName(name))\n }\n\n Headers.prototype.set = function(name, value) {\n this.map[normalizeName(name)] = normalizeValue(value)\n }\n\n Headers.prototype.forEach = function(callback, thisArg) {\n for (var name in this.map) {\n if (this.map.hasOwnProperty(name)) {\n callback.call(thisArg, this.map[name], name, this)\n }\n }\n }\n\n Headers.prototype.keys = function() {\n var items = []\n this.forEach(function(value, name) { items.push(name) })\n return iteratorFor(items)\n }\n\n Headers.prototype.values = function() {\n var items = []\n this.forEach(function(value) { items.push(value) })\n return iteratorFor(items)\n }\n\n Headers.prototype.entries = function() {\n var items = []\n this.forEach(function(value, name) { items.push([name, value]) })\n return iteratorFor(items)\n }\n\n if (support.iterable) {\n Headers.prototype[Symbol.iterator] = Headers.prototype.entries\n }\n\n function consumed(body) {\n if (body.bodyUsed) {\n return Promise.reject(new TypeError('Already read'))\n }\n body.bodyUsed = true\n }\n\n function fileReaderReady(reader) {\n return new Promise(function(resolve, reject) {\n reader.onload = function() {\n resolve(reader.result)\n }\n reader.onerror = function() {\n reject(reader.error)\n }\n })\n }\n\n function readBlobAsArrayBuffer(blob) {\n var reader = new FileReader()\n var promise = fileReaderReady(reader)\n reader.readAsArrayBuffer(blob)\n return promise\n }\n\n function readBlobAsText(blob) {\n var reader = new FileReader()\n var promise = fileReaderReady(reader)\n reader.readAsText(blob)\n return promise\n }\n\n function readArrayBufferAsText(buf) {\n var view = new Uint8Array(buf)\n var chars = new Array(view.length)\n\n for (var i = 0; i < view.length; i++) {\n chars[i] = String.fromCharCode(view[i])\n }\n return chars.join('')\n }\n\n function bufferClone(buf) {\n if (buf.slice) {\n return buf.slice(0)\n } else {\n var view = new Uint8Array(buf.byteLength)\n view.set(new Uint8Array(buf))\n return view.buffer\n }\n }\n\n function Body() {\n this.bodyUsed = false\n\n this._initBody = function(body) {\n this._bodyInit = body\n if (!body) {\n this._bodyText = ''\n } else if (typeof body === 'string') {\n this._bodyText = body\n } else if (support.blob && Blob.prototype.isPrototypeOf(body)) {\n this._bodyBlob = body\n } else if (support.formData && FormData.prototype.isPrototypeOf(body)) {\n this._bodyFormData = body\n } else if (support.searchParams && URLSearchParams.prototype.isPrototypeOf(body)) {\n this._bodyText = body.toString()\n } else if (support.arrayBuffer && support.blob && isDataView(body)) {\n this._bodyArrayBuffer = bufferClone(body.buffer)\n // IE 10-11 can't handle a DataView body.\n this._bodyInit = new Blob([this._bodyArrayBuffer])\n } else if (support.arrayBuffer && (ArrayBuffer.prototype.isPrototypeOf(body) || isArrayBufferView(body))) {\n this._bodyArrayBuffer = bufferClone(body)\n } else {\n throw new Error('unsupported BodyInit type')\n }\n\n if (!this.headers.get('content-type')) {\n if (typeof body === 'string') {\n this.headers.set('content-type', 'text/plain;charset=UTF-8')\n } else if (this._bodyBlob && this._bodyBlob.type) {\n this.headers.set('content-type', this._bodyBlob.type)\n } else if (support.searchParams && URLSearchParams.prototype.isPrototypeOf(body)) {\n this.headers.set('content-type', 'application/x-www-form-urlencoded;charset=UTF-8')\n }\n }\n }\n\n if (support.blob) {\n this.blob = function() {\n var rejected = consumed(this)\n if (rejected) {\n return rejected\n }\n\n if (this._bodyBlob) {\n return Promise.resolve(this._bodyBlob)\n } else if (this._bodyArrayBuffer) {\n return Promise.resolve(new Blob([this._bodyArrayBuffer]))\n } else if (this._bodyFormData) {\n throw new Error('could not read FormData body as blob')\n } else {\n return Promise.resolve(new Blob([this._bodyText]))\n }\n }\n\n this.arrayBuffer = function() {\n if (this._bodyArrayBuffer) {\n return consumed(this) || Promise.resolve(this._bodyArrayBuffer)\n } else {\n return this.blob().then(readBlobAsArrayBuffer)\n }\n }\n }\n\n this.text = function() {\n var rejected = consumed(this)\n if (rejected) {\n return rejected\n }\n\n if (this._bodyBlob) {\n return readBlobAsText(this._bodyBlob)\n } else if (this._bodyArrayBuffer) {\n return Promise.resolve(readArrayBufferAsText(this._bodyArrayBuffer))\n } else if (this._bodyFormData) {\n throw new Error('could not read FormData body as text')\n } else {\n return Promise.resolve(this._bodyText)\n }\n }\n\n if (support.formData) {\n this.formData = function() {\n return this.text().then(decode)\n }\n }\n\n this.json = function() {\n return this.text().then(JSON.parse)\n }\n\n return this\n }\n\n // HTTP methods whose capitalization should be normalized\n var methods = ['DELETE', 'GET', 'HEAD', 'OPTIONS', 'POST', 'PUT']\n\n function normalizeMethod(method) {\n var upcased = method.toUpperCase()\n return (methods.indexOf(upcased) > -1) ? upcased : method\n }\n\n function Request(input, options) {\n options = options || {}\n var body = options.body\n\n if (typeof input === 'string') {\n this.url = input\n } else {\n if (input.bodyUsed) {\n throw new TypeError('Already read')\n }\n this.url = input.url\n this.credentials = input.credentials\n if (!options.headers) {\n this.headers = new Headers(input.headers)\n }\n this.method = input.method\n this.mode = input.mode\n if (!body && input._bodyInit != null) {\n body = input._bodyInit\n input.bodyUsed = true\n }\n }\n\n this.credentials = options.credentials || this.credentials || 'omit'\n if (options.headers || !this.headers) {\n this.headers = new Headers(options.headers)\n }\n this.method = normalizeMethod(options.method || this.method || 'GET')\n this.mode = options.mode || this.mode || null\n this.referrer = null\n\n if ((this.method === 'GET' || this.method === 'HEAD') && body) {\n throw new TypeError('Body not allowed for GET or HEAD requests')\n }\n this._initBody(body)\n }\n\n Request.prototype.clone = function() {\n return new Request(this, { body: this._bodyInit })\n }\n\n function decode(body) {\n var form = new FormData()\n body.trim().split('&').forEach(function(bytes) {\n if (bytes) {\n var split = bytes.split('=')\n var name = split.shift().replace(/\\+/g, ' ')\n var value = split.join('=').replace(/\\+/g, ' ')\n form.append(decodeURIComponent(name), decodeURIComponent(value))\n }\n })\n return form\n }\n\n function parseHeaders(rawHeaders) {\n var headers = new Headers()\n rawHeaders.split('\\r\\n').forEach(function(line) {\n var parts = line.split(':')\n var key = parts.shift().trim()\n if (key) {\n var value = parts.join(':').trim()\n headers.append(key, value)\n }\n })\n return headers\n }\n\n Body.call(Request.prototype)\n\n function Response(bodyInit, options) {\n if (!options) {\n options = {}\n }\n\n this.type = 'default'\n this.status = 'status' in options ? options.status : 200\n this.ok = this.status >= 200 && this.status < 300\n this.statusText = 'statusText' in options ? options.statusText : 'OK'\n this.headers = new Headers(options.headers)\n this.url = options.url || ''\n this._initBody(bodyInit)\n }\n\n Body.call(Response.prototype)\n\n Response.prototype.clone = function() {\n return new Response(this._bodyInit, {\n status: this.status,\n statusText: this.statusText,\n headers: new Headers(this.headers),\n url: this.url\n })\n }\n\n Response.error = function() {\n var response = new Response(null, {status: 0, statusText: ''})\n response.type = 'error'\n return response\n }\n\n var redirectStatuses = [301, 302, 303, 307, 308]\n\n Response.redirect = function(url, status) {\n if (redirectStatuses.indexOf(status) === -1) {\n throw new RangeError('Invalid status code')\n }\n\n return new Response(null, {status: status, headers: {location: url}})\n }\n\n self.Headers = Headers\n self.Request = Request\n self.Response = Response\n\n self.fetch = function(input, init) {\n return new Promise(function(resolve, reject) {\n var request = new Request(input, init)\n var xhr = new XMLHttpRequest()\n\n xhr.onload = function() {\n var options = {\n status: xhr.status,\n statusText: xhr.statusText,\n headers: parseHeaders(xhr.getAllResponseHeaders() || '')\n }\n options.url = 'responseURL' in xhr ? xhr.responseURL : options.headers.get('X-Request-URL')\n var body = 'response' in xhr ? xhr.response : xhr.responseText\n resolve(new Response(body, options))\n }\n\n xhr.onerror = function() {\n reject(new TypeError('Network request failed'))\n }\n\n xhr.ontimeout = function() {\n reject(new TypeError('Network request failed'))\n }\n\n xhr.open(request.method, request.url, true)\n\n if (request.credentials === 'include') {\n xhr.withCredentials = true\n }\n\n if ('responseType' in xhr && support.blob) {\n xhr.responseType = 'blob'\n }\n\n request.headers.forEach(function(value, name) {\n xhr.setRequestHeader(name, value)\n })\n\n xhr.send(typeof request._bodyInit === 'undefined' ? null : request._bodyInit)\n })\n }\n self.fetch.polyfill = true\n})(typeof self !== 'undefined' ? self : this);\n\n\n\n//////////////////\n// WEBPACK FOOTER\n// ./~/whatwg-fetch/fetch.js\n// module id = 0\n// module chunks = 1"],"sourceRoot":""} \ No newline at end of file