define([ 'jquery', '/common/diffMarked.js', '/bower_components/nthen/index.js', '/common/sframe-common.js', '/common/hyperscript.js', '/common/sframe-app-framework.js', '/common/sframe-common-codemirror.js', '/common/common-interface.js', '/common/common-util.js', '/common/common-hash.js', '/code/markers.js', '/common/visible.js', '/common/TypingTests.js', '/customize/messages.js', 'cm/lib/codemirror', 'css!cm/lib/codemirror.css', 'css!cm/addon/dialog/dialog.css', 'css!cm/addon/fold/foldgutter.css', 'cm/mode/gfm/gfm', 'cm/addon/mode/loadmode', 'cm/mode/meta', 'cm/addon/mode/overlay', 'cm/addon/mode/multiplex', 'cm/addon/mode/simple', 'cm/addon/edit/closebrackets', 'cm/addon/edit/matchbrackets', 'cm/addon/edit/trailingspace', 'cm/addon/selection/active-line', 'cm/addon/search/search', 'cm/addon/search/match-highlighter', 'cm/addon/search/searchcursor', 'cm/addon/dialog/dialog', 'cm/addon/fold/foldcode', 'cm/addon/fold/foldgutter', 'cm/addon/fold/brace-fold', 'cm/addon/fold/xml-fold', 'cm/addon/fold/markdown-fold', 'cm/addon/fold/comment-fold', 'cm/addon/display/placeholder', 'css!/customize/src/print.css', 'less!/code/app-code.less' ], function ( $, DiffMd, nThen, SFCommon, h, Framework, SFCodeMirror, UI, Util, Hash, Markers, Visible, TypingTest, Messages, CMeditor) { window.CodeMirror = CMeditor; var MEDIA_TAG_MODES = Object.freeze([ 'markdown', 'gfm', 'html', 'htmlembedded', 'htmlmixed', 'index.html', 'php', 'velocity', 'xml', ]); var mkThemeButton = function (framework) { var $theme = $(h('button.cp-toolbar-appmenu', [ h('i.cptools.cptools-palette'), h('span.cp-button-name', Messages.toolbar_theme) ])); var $content = $(h('div.cp-toolbar-drawer-content', { tabindex: 1 })).hide(); // set up all the necessary events UI.createDrawer($theme, $content); framework._.toolbar.$theme = $content; framework._.toolbar.$bottomL.append($theme); }; var mkCbaButton = function (framework, markers) { var $showAuthorColorsButton = framework._.sfCommon.createButton('', true, { text: Messages.cba_hide, name: 'authormarks', icon: 'fa-paint-brush', }).hide(); framework._.toolbar.$theme.append($showAuthorColorsButton); markers.setButton($showAuthorColorsButton); }; var mkPrintButton = function (framework, $content, $print) { var $printButton = framework._.sfCommon.createButton('print', true); $printButton.click(function () { $print.html($content.html()); window.focus(); window.print(); framework.feedback('PRINT_CODE'); }); framework._.toolbar.$drawer.append($printButton); }; var mkMarkdownTb = function (editor, framework) { var $codeMirrorContainer = $('#cp-app-code-container'); var markdownTb = framework._.sfCommon.createMarkdownToolbar(editor); $codeMirrorContainer.prepend(markdownTb.toolbar); framework._.toolbar.$bottomL.append(markdownTb.button); var modeChange = function (mode) { if (['markdown', 'gfm'].indexOf(mode) !== -1) { return void markdownTb.setState(true); } markdownTb.setState(false); }; return { modeChange: modeChange }; }; var mkHelpMenu = function (framework) { var $codeMirrorContainer = $('#cp-app-code-container'); var helpMenu = framework._.sfCommon.createHelpMenu(['text', 'code']); $codeMirrorContainer.prepend(helpMenu.menu); framework._.toolbar.$drawer.append(helpMenu.button); }; var previews = {}; previews['gfm'] = function (val, $div, common) { DiffMd.apply(DiffMd.render(val), $div, common); }; previews['markdown'] = previews['gfm']; previews['htmlmixed'] = function (val, $div, common) { DiffMd.apply(val, $div, common); }; var mkPreviewPane = function (editor, CodeMirror, framework, isPresentMode) { var $previewContainer = $('#cp-app-code-preview'); var $preview = $('#cp-app-code-preview-content'); var $editorContainer = $('#cp-app-code-editor'); var $codeMirrorContainer = $('#cp-app-code-container'); var $codeMirror = $('.CodeMirror'); $('<img>', { src: '/customize/CryptPad_logo_grey.svg', alt: '', class: 'cp-app-code-preview-empty' }).appendTo($previewContainer); var $previewButton = framework._.sfCommon.createButton('preview', true); var forceDrawPreview = function () { var f = previews[CodeMirror.highlightMode]; if (!f) { return; } try { if (editor.getValue() === '') { $previewContainer.addClass('cp-app-code-preview-isempty'); return; } $previewContainer.removeClass('cp-app-code-preview-isempty'); f(editor.getValue(), $preview, framework._.sfCommon); } catch (e) { console.error(e); } }; var drawPreview = Util.throttle(function () { if (!previews[CodeMirror.highlightMode]) { return; } if (!$previewButton.is('.cp-toolbar-button-active')) { return; } forceDrawPreview(); }, 400); var previewTo; $previewButton.click(function () { clearTimeout(previewTo); $codeMirror.addClass('transition'); previewTo = setTimeout(function () { $codeMirror.removeClass('transition'); }, 500); if (!previews[CodeMirror.highlightMode]) { $previewContainer.show(); } $previewContainer.toggle(); if ($previewContainer.is(':visible')) { forceDrawPreview(); $codeMirrorContainer.removeClass('cp-app-code-fullpage'); $previewButton.addClass('cp-toolbar-button-active'); framework._.sfCommon.setPadAttribute('previewMode', true, function (e) { if (e) { return console.log(e); } }); } else { $codeMirrorContainer.addClass('cp-app-code-fullpage'); $previewButton.removeClass('cp-toolbar-button-active'); framework._.sfCommon.setPadAttribute('previewMode', false, function (e) { if (e) { return console.log(e); } }); } }); framework._.toolbar.$bottomM.append($previewButton); $preview.click(function (e) { if (!e.target) { return; } var $t = $(e.target); if ($t.is('a') || $t.parents('a').length) { e.preventDefault(); var $a = $t.is('a') ? $t : $t.parents('a').first(); var href = $a.attr('href'); if (/^\/[^\/]/.test(href)) { var privateData = framework._.cpNfInner.metadataMgr.getPrivateData(); href = privateData.origin + href; } else if (/^#/.test(href)) { var target = document.getElementById('cp-md-0-'+href.slice(1)); if (target) { target.scrollIntoView(); } return; } framework._.sfCommon.openUnsafeURL(href); } }); var modeChange = function (mode) { if (previews[mode]) { $previewButton.show(); framework._.sfCommon.getPadAttribute('previewMode', function (e, data) { if (e) { return void console.error(e); } if (data !== false) { $previewContainer.show(); $previewButton.addClass('cp-toolbar-button-active'); $codeMirrorContainer.removeClass('cp-app-code-fullpage'); if (isPresentMode) { $editorContainer.addClass('cp-app-code-present'); } } }); return; } $editorContainer.removeClass('cp-app-code-present'); $previewButton.hide(); $previewContainer.hide(); $previewButton.removeClass('active'); $codeMirrorContainer.addClass('cp-app-code-fullpage'); }; var isVisible = function () { return $previewContainer.is(':visible'); }; framework.onReady(function () { // add the splitter var splitter = $('<div>', { 'class': 'cp-splitter' }).appendTo($previewContainer); $preview.on('scroll', function() { splitter.css('top', $preview.scrollTop() + 'px'); }); var $target = $codeMirrorContainer; splitter.on('mousedown', function (e) { e.preventDefault(); var x = e.pageX; var w = $target.width(); var handler = function (evt) { if (evt.type === 'mouseup') { $(window).off('mouseup mousemove', handler); return; } $target.css('width', (w - x + evt.pageX) + 'px'); editor.refresh(); }; $(window).off('mouseup mousemove', handler); $(window).on('mouseup mousemove', handler); }); var previewInt; var clear = function () { clearInterval(previewInt); }; // keep trying to draw until you're confident it has been drawn previewInt = setInterval(function () { // give up if it's not a valid preview mode if (!previews[CodeMirror.highlightMode]) { return void clear(); } // give up if content has been drawn if ($preview.text()) { return void clear(); } // only draw if there is actually content to display if (editor && !editor.getValue().trim()) { return void clear(); } forceDrawPreview(); }, 1000); }); framework._.sfCommon.getPadAttribute('previewMode', function (e, data) { if (e) { return void console.error(e); } if (data === false && $previewButton) { $previewButton.click(); } }); Visible.onChange(function (visible) { if (visible) { drawPreview(); } }); DiffMd.onPluginLoaded(drawPreview); return { forceDraw: forceDrawPreview, draw: drawPreview, modeChange: modeChange, isVisible: isVisible }; }; var mkColorByAuthor = function (framework, markers) { var common = framework._.sfCommon; var $cbaButton = framework._.sfCommon.createButton(null, true, { icon: 'fa-paint-brush', text: Messages.cba_title, name: 'cba' }, function () { var div = h('div'); var $div = $(div); var content = h('div', [ h('h4', Messages.cba_properties), h('p', Messages.cba_hint), div ]); var setButton = function (state) { var button = h('button.btn'); var $button = $(button); $div.html('').append($button); if (state) { // Add "enable" button $button.addClass('btn-secondary').text(Messages.cba_enable); UI.confirmButton(button, { classes: 'btn-primary' }, function () { $button.remove(); markers.setState(true); common.setAttribute(['code', 'enableColors'], true); setButton(false); }); return; } // Add "disable" button $button.addClass('btn-danger-alt').text(Messages.cba_disable); UI.confirmButton(button, { classes: 'btn-danger' }, function () { $button.remove(); markers.setState(false); common.setAttribute(['code', 'enableColors'], false); setButton(true); }); }; setButton(!markers.getState()); UI.alert(content); }); framework._.toolbar.$theme.append($cbaButton); }; var mkFilePicker = function (framework, editor, evModeChange) { evModeChange.reg(function (mode) { if (MEDIA_TAG_MODES.indexOf(mode) !== -1) { // Embedding is endabled framework.setMediaTagEmbedder(function (mt) { editor.focus(); editor.replaceSelection($(mt)[0].outerHTML); }); } else { // Embedding is disabled framework.setMediaTagEmbedder(); } }); }; ///////////////////////////////////////////////////////////////////////////////////////////////////////// ///////////////////////////////////////////////////////////////////////////////////////////////////////// ///////////////////////////////////////////////////////////////////////////////////////////////////////// ///////////////////////////////////////////////////////////////////////////////////////////////////////// var andThen2 = function (editor, CodeMirror, framework, isPresentMode) { var common = framework._.sfCommon; var privateData = common.getMetadataMgr().getPrivateData(); var previewPane = mkPreviewPane(editor, CodeMirror, framework, isPresentMode); var markdownTb = mkMarkdownTb(editor, framework); mkThemeButton(framework); var markers = Markers.create({ common: common, framework: framework, CodeMirror: CodeMirror, devMode: privateData.devMode, editor: editor }); mkCbaButton(framework, markers); var $print = $('#cp-app-code-print'); var $content = $('#cp-app-code-preview-content'); mkPrintButton(framework, $content, $print); if (!privateData.isEmbed) { mkHelpMenu(framework); } var evModeChange = Util.mkEvent(); evModeChange.reg(previewPane.modeChange); evModeChange.reg(markdownTb.modeChange); CodeMirror.mkIndentSettings(framework._.cpNfInner.metadataMgr); CodeMirror.init(framework.localChange, framework._.title, framework._.toolbar); mkFilePicker(framework, editor, evModeChange); if (!framework.isReadOnly()) { CodeMirror.configureTheme(common, function () { CodeMirror.configureLanguage(common, null, evModeChange.fire); }); } else { CodeMirror.configureTheme(common); } framework.onContentUpdate(function (newContent) { var highlightMode = newContent.highlightMode; if (highlightMode && highlightMode !== CodeMirror.highlightMode) { CodeMirror.setMode(highlightMode, evModeChange.fire); } // Fix the markers offsets markers.checkMarks(newContent); // Apply the text content CodeMirror.contentUpdate(newContent); previewPane.draw(); // Apply the markers markers.setMarks(); framework.localChange(); }); framework.setContentGetter(function () { CodeMirror.removeCursors(); var content = CodeMirror.getContent(); content.highlightMode = CodeMirror.highlightMode; previewPane.draw(); markers.updateAuthorMarks(); content.authormarks = markers.getAuthorMarks(); return content; }); var cursorTo; var updateCursor = function () { if (cursorTo) { clearTimeout(cursorTo); } if (editor._noCursorUpdate) { return; } cursorTo = setTimeout(function () { framework.updateCursor(); }, 500); // 500ms to make sure it is sent after chainpad sync }; framework.onCursorUpdate(CodeMirror.setRemoteCursor); framework.setCursorGetter(CodeMirror.getCursor); editor.on('cursorActivity', updateCursor); framework.onEditableChange(function () { editor.setOption('readOnly', framework.isLocked() || framework.isReadOnly()); }); framework.setTitleRecommender(CodeMirror.getHeadingText); framework.onReady(function (newPad) { editor.focus(); if (newPad && !CodeMirror.highlightMode) { CodeMirror.setMode('gfm', evModeChange.fire); //console.log("%s => %s", CodeMirror.highlightMode, CodeMirror.$language.val()); } markers.ready(); common.getPadMetadata(null, function (md) { if (md && md.error) { return; } if (!Array.isArray(md.owners)) { return void markers.setState(false); } if (!common.isOwned(md.owners)) { return; } // We're the owner: add the button and enable the colors if needed mkColorByAuthor(framework, markers); if (newPad && Util.find(privateData, ['settings', 'code', 'enableColors'])) { markers.setState(true); } }); var fmConfig = { dropArea: $('.CodeMirror'), body: $('body'), onUploaded: function (ev, data) { var parsed = Hash.parsePadUrl(data.url); var secret = Hash.getSecrets('file', parsed.hash, data.password); var fileHost = privateData.fileHost || privateData.origin; var src = fileHost + Hash.getBlobPathFromHex(secret.channel); var key = Hash.encodeBase64(secret.keys.cryptKey); var mt = '<media-tag src="' + src + '" data-crypto-key="cryptpad:' + key + '"></media-tag>'; editor.replaceSelection(mt); } }; common.createFileManager(fmConfig); }); framework.onDefaultContentNeeded(function () { editor.setValue(''); }); framework.setFileExporter(CodeMirror.getContentExtension, CodeMirror.fileExporter); framework.setFileImporter({}, function () { /* setFileImporter currently takes a function with the following signature: (content, file) => {} I used 'apply' with 'arguments' to avoid breaking things if this API ever changes. */ var ret = CodeMirror.fileImporter.apply(null, Array.prototype.slice.call(arguments)); previewPane.modeChange(ret.mode); return ret; }); framework.setNormalizer(function (c) { return { content: c.content, highlightMode: c.highlightMode, authormarks: c.authormarks }; }); editor.on('change', function( cm, change ) { markers.localChange(change, framework.localChange); }); framework.start(); window.easyTest = function () { var test = TypingTest.testCode(editor); return test; }; }; var getThumbnailContainer = function () { var $preview = $('#cp-app-code-preview-content'); if ($preview.length && $preview.is(':visible')) { return $preview[0]; } }; var main = function () { var CodeMirror; var editor; var framework; nThen(function (waitFor) { Framework.create({ toolbarContainer: '#cme_toolbox', contentContainer: '#cp-app-code-editor', thumbnail: { getContainer: getThumbnailContainer, filter: function (el, before) { if (before) { //$(el).parents().css('overflow', 'visible'); $(el).css('max-height', Math.max(600, $(el).width()) + 'px'); return; } $(el).parents().css('overflow', ''); $(el).css('max-height', ''); editor.refresh(); } } }, waitFor(function (fw) { framework = fw; })); nThen(function (waitFor) { $(waitFor()); }).nThen(function () { CodeMirror = SFCodeMirror.create(null, CMeditor); $('#cp-app-code-container').addClass('cp-app-code-fullpage'); editor = CodeMirror.editor; }).nThen(waitFor()); }).nThen(function (/*waitFor*/) { framework._.sfCommon.isPresentUrl(function (err, val) { andThen2(editor, CodeMirror, framework, val); }); }); }; main(); });