require(['/api/config'], function (ApiConfig) { // see ckeditor_base.js getUrl() window.CKEDITOR_GETURL = function (resource) { if (resource.indexOf( '/' ) === 0) { resource = window.CKEDITOR.basePath.replace(/\/bower_components\/.*/, '') + resource; } else if (resource.indexOf(':/') === -1) { resource = window.CKEDITOR.basePath + resource; } if (resource[resource.length - 1] !== '/' && resource.indexOf('ver=') === -1) { var args = ApiConfig.requireConf.urlArgs; if (resource.indexOf('/bower_components/') !== -1) { args = 'ver=' + window.CKEDITOR.timestamp; } resource += (resource.indexOf('?') >= 0 ? '&' : '?') + args; } return resource; }; require(['/bower_components/ckeditor/ckeditor.js']); }); define([ 'jquery', '/bower_components/hyperjson/hyperjson.js', '/common/sframe-app-framework.js', '/common/cursor.js', '/common/TypingTests.js', '/customize/messages.js', '/pad/links.js', '/bower_components/nthen/index.js', '/common/media-tag.js', '/api/config', '/common/cryptpad-common.js', '/bower_components/diff-dom/diffDOM.js', 'css!/bower_components/bootstrap/dist/css/bootstrap.min.css', 'less!/bower_components/components-font-awesome/css/font-awesome.min.css', 'less!/customize/src/less2/main.less', ], function ( $, Hyperjson, Framework, Cursor, TypingTest, Messages, Links, nThen, MediaTag, ApiConfig, Cryptpad) { var DiffDom = window.diffDOM; var slice = function (coll) { return Array.prototype.slice.call(coll); }; var removeListeners = function (root) { slice(root.attributes).map(function (attr) { if (/^on/.test(attr.name)) { root.attributes.removeNamedItem(attr.name); } }); slice(root.children).forEach(removeListeners); }; var hjsonToDom = function (H) { var dom = Hyperjson.toDOM(H); removeListeners(dom); return dom; }; var module = window.REALTIME_MODULE = window.APP = { Hyperjson: Hyperjson, logFights: true, fights: [], Cursor: Cursor, }; var isNotMagicLine = function (el) { return !(el && typeof(el.getAttribute) === 'function' && el.getAttribute('class') && el.getAttribute('class').split(' ').indexOf('non-realtime') !== -1); }; var hjsonFilters = function (hj) { /* catch `type="_moz"` before it goes over the wire */ var brFilter = function (hj) { if (hj[1].type === '_moz') { hj[1].type = undefined; } return hj; }; var mediatagContentFilter = function (hj) { if (hj[0] === 'MEDIA-TAG') { hj[2] = []; } return hj; }; brFilter(hj); mediatagContentFilter(hj); return hj; }; var domFromHTML = function (html) { return new DOMParser().parseFromString(html, 'text/html'); }; var forbiddenTags = [ 'SCRIPT', //'IFRAME', 'OBJECT', 'APPLET', //'VIDEO', //'AUDIO' ]; var getHTML = function (inner) { return ('\n' + '\n' + inner.innerHTML); }; var CKEDITOR_CHECK_INTERVAL = 100; var ckEditorAvailable = function (cb) { var intr; var check = function () { if (window.CKEDITOR) { clearTimeout(intr); cb(window.CKEDITOR); } }; intr = setInterval(function () { console.log("Ckeditor was not defined. Trying again in %sms", CKEDITOR_CHECK_INTERVAL); check(); }, CKEDITOR_CHECK_INTERVAL); check(); }; var mkDiffOptions = function (cursor, readOnly) { return { preDiffApply: function (info) { /* Don't accept attributes that begin with 'on' these are probably listeners, and we don't want to send scripts over the wire. */ if (['addAttribute', 'modifyAttribute'].indexOf(info.diff.action) !== -1) { if (info.diff.name === 'href') { // console.log(info.diff); //var href = info.diff.newValue; // TODO normalize HTML entities if (/javascript *: */.test(info.diff.newValue)) { // TODO remove javascript: links } } if (/^on/.test(info.diff.name)) { console.log("Rejecting forbidden element attribute with name (%s)", info.diff.name); return true; } } /* Also reject any elements which would insert any one of our forbidden tag types: script, iframe, object, applet, video, or audio */ if (['addElement', 'replaceElement'].indexOf(info.diff.action) !== -1) { if (info.diff.element && forbiddenTags.indexOf(info.diff.element.nodeName) !== -1) { console.log("Rejecting forbidden tag of type (%s)", info.diff.element.nodeName); return true; } else if (info.diff.newValue && forbiddenTags.indexOf(info.diff.newValue.nodeType) !== -1) { console.log("Rejecting forbidden tag of type (%s)", info.diff.newValue.nodeName); return true; } } if (info.node && info.node.tagName === 'BODY') { if (info.diff.action === 'removeAttribute' && ['class', 'spellcheck'].indexOf(info.diff.name) !== -1) { return true; } } /* DiffDOM will filter out magicline plugin elements in practice this will make it impossible to use it while someone else is typing, which could be annoying. we should check when such an element is going to be removed, and prevent that from happening. */ if (info.node && info.node.tagName === 'SPAN' && info.node.getAttribute('contentEditable') === "false") { // it seems to be a magicline plugin element... if (info.diff.action === 'removeElement') { // and you're about to remove it... // this probably isn't what you want /* I have never seen this in the console, but the magic line is still getting removed on remote edits. This suggests that it's getting removed by something other than diffDom. */ console.log("preventing removal of the magic line!"); // return true to prevent diff application return true; } } // Do not change the contenteditable value in view mode if (readOnly && info.node && info.node.tagName === 'BODY' && info.diff.action === 'modifyAttribute' && info.diff.name === 'contenteditable') { return true; } // no use trying to recover the cursor if it doesn't exist if (!cursor.exists()) { return; } /* frame is either 0, 1, 2, or 3, depending on which cursor frames were affected: none, first, last, or both */ var frame = info.frame = cursor.inNode(info.node); if (!frame) { return; } if (typeof info.diff.oldValue === 'string' && typeof info.diff.newValue === 'string') { var pushes = cursor.pushDelta(info.diff.oldValue, info.diff.newValue); if (frame & 1) { // push cursor start if necessary if (pushes.commonStart < cursor.Range.start.offset) { cursor.Range.start.offset += pushes.delta; } } if (frame & 2) { // push cursor end if necessary if (pushes.commonStart < cursor.Range.end.offset) { cursor.Range.end.offset += pushes.delta; } } } }, postDiffApply: function (info) { if (info.frame) { if (info.node) { if (info.frame & 1) { cursor.fixStart(info.node); } if (info.frame & 2) { cursor.fixEnd(info.node); } } else { console.error("info.node did not exist"); } var sel = cursor.makeSelection(); var range = cursor.makeRange(); cursor.fixSelection(sel, range); } } }; }; //////////////////////////////////////////////////////////////////////////////////////////////////////////// //////////////////////////////////////////////////////////////////////////////////////////////////////////// //////////////////////////////////////////////////////////////////////////////////////////////////////////// var addToolbarHideBtn = function (framework, $bar) { // Expand / collapse the toolbar var $collapse = framework._.sfCommon.createButton(null, true); $collapse.removeClass('fa-question'); var updateIcon = function (isVisible) { $collapse.removeClass('fa-caret-down').removeClass('fa-caret-up'); if (!isVisible) { framework.feedback('HIDETOOLBAR_PAD'); $collapse.addClass('fa-caret-down'); } else { framework.feedback('SHOWTOOLBAR_PAD'); $collapse.addClass('fa-caret-up'); } }; updateIcon(); $collapse.click(function () { $(window).trigger('resize'); $('.cke_toolbox_main').toggle(); $(window).trigger('cryptpad-ck-toolbar'); var isVisible = $bar.find('.cke_toolbox_main').is(':visible'); framework._.sfCommon.setAttribute(['pad', 'showToolbar'], isVisible); updateIcon(isVisible); }); framework._.sfCommon.getAttribute(['pad', 'showToolbar'], function (err, data) { if (typeof(data) === "undefined" || data) { $('.cke_toolbox_main').show(); updateIcon(true); return; } $('.cke_toolbox_main').hide(); updateIcon(false); }); framework._.toolbar.$rightside.append($collapse); }; var andThen2 = function (editor, Ckeditor, framework) { var $bar = $('#cke_1_toolbox'); var $html = $bar.closest('html'); var $faLink = $html.find('head link[href*="/bower_components/components-font-awesome/css/font-awesome.min.css"]'); if ($faLink.length) { $html.find('iframe').contents().find('head').append($faLink.clone()); } var ml = Ckeditor.instances.editor1.plugins.magicline.backdoor.that.line.$; [ml, ml.parentElement].forEach(function (el) { el.setAttribute('class', 'non-realtime'); }); var ifrWindow = $html.find('iframe')[0].contentWindow; var documentBody = ifrWindow.document.body; var inner = window.inner = documentBody; var cursor = module.cursor = Cursor(inner); var openLink = function (e) { var el = e.currentTarget; if (!el || el.nodeName !== 'A') { return; } var href = el.getAttribute('href'); var bounceHref = window.location.origin + '/bounce/#' + encodeURIComponent(href); if (href) { ifrWindow.open(bounceHref, '_blank'); } }; if (!framework.isReadOnly()) { console.log('\n\n\n\n\nREGISTER\n\n\n\n\n'); framework.onEditableChange(function () { console.log("Editable change"); var locked = framework.isLocked(); $(inner).css({ 'background-color': ((locked) ? '#aaa' : '') }); inner.setAttribute('contenteditable', !locked); }); var fileDialogCfg = { onSelect: function (data) { if (data.type === 'file') { var mt = ''; editor.insertElement(window.CKEDITOR.dom.element.createFromHtml(mt)); return; } } }; framework._.sfCommon.initFilePicker(fileDialogCfg); window.APP.$mediaTagButton = $('