cryptpad/www/code/inner.js

958 lines
38 KiB
JavaScript
Raw Normal View History

define([
'jquery',
2017-09-05 09:35:15 +00:00
'/common/diffMarked.js',
'/bower_components/nthen/index.js',
'/common/sframe-common.js',
'/common/sframe-app-framework.js',
'/common/sframe-common-codemirror.js',
'/common/common-util.js',
2017-11-13 15:32:40 +00:00
'/common/common-hash.js',
'/common/modes.js',
'/common/visible.js',
2018-11-22 13:45:12 +00:00
'/common/TypingTests.js',
'/customize/messages.js',
'cm/lib/codemirror',
2020-04-09 14:06:04 +00:00
'/bower_components/chainpad/chainpad.dist.js',
2017-07-03 08:55:25 +00:00
'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',
2017-09-05 09:35:15 +00:00
2018-07-14 13:15:23 +00:00
'less!/code/app-code.less'
2017-09-05 09:35:15 +00:00
], function (
$,
DiffMd,
nThen,
SFCommon,
Framework,
SFCodeMirror,
Util,
2017-11-13 15:32:40 +00:00
Hash,
Modes,
Visible,
2018-11-22 13:45:12 +00:00
TypingTest,
Messages,
2020-04-09 14:06:04 +00:00
CMeditor,
ChainPad)
2017-09-05 09:35:15 +00:00
{
window.CodeMirror = CMeditor;
2017-09-05 09:35:15 +00:00
var MEDIA_TAG_MODES = Object.freeze([
'markdown',
'gfm',
'html',
'htmlembedded',
'htmlmixed',
'index.html',
'php',
'velocity',
'xml',
]);
2017-09-07 16:56:58 +00:00
2018-12-06 15:43:48 +00:00
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);
};
2018-02-27 16:38:29 +00:00
var mkMarkdownTb = function (editor, framework) {
var $codeMirrorContainer = $('#cp-app-code-container');
var markdownTb = framework._.sfCommon.createMarkdownToolbar(editor);
$codeMirrorContainer.prepend(markdownTb.toolbar);
framework._.toolbar.$rightside.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');
2020-01-09 16:30:15 +00:00
$codeMirrorContainer.prepend(framework._.sfCommon.getBurnAfterReadingWarning());
var helpMenu = framework._.sfCommon.createHelpMenu(['text', 'code']);
2018-02-27 16:38:29 +00:00
$codeMirrorContainer.prepend(helpMenu.menu);
framework._.toolbar.$drawer.append(helpMenu.button);
2018-02-27 16:38:29 +00:00
};
var mkPreviewPane = function (editor, CodeMirror, framework, isPresentMode) {
2017-09-05 09:35:15 +00:00
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');
2018-02-27 16:38:29 +00:00
$('<img>', {
src: '/customize/main-favicon.png',
alt: '',
class: 'cp-app-code-preview-empty'
}).appendTo($previewContainer);
var $previewButton = framework._.sfCommon.createButton('preview', true);
2017-09-05 09:35:15 +00:00
var forceDrawPreview = function () {
try {
2018-02-27 16:38:29 +00:00
if (editor.getValue() === '') {
$previewContainer.addClass('cp-app-code-preview-isempty');
return;
}
$previewContainer.removeClass('cp-app-code-preview-isempty');
DiffMd.apply(DiffMd.render(editor.getValue()), $preview, framework._.sfCommon);
2017-09-05 09:35:15 +00:00
} catch (e) { console.error(e); }
};
var drawPreview = Util.throttle(function () {
if (['markdown', 'gfm'].indexOf(CodeMirror.highlightMode) === -1) { return; }
if (!$previewButton.is('.cp-toolbar-button-active')) { return; }
2017-09-05 09:35:15 +00:00
forceDrawPreview();
}, 400);
2017-09-05 09:35:15 +00:00
var previewTo;
$previewButton.click(function () {
clearTimeout(previewTo);
2017-09-05 09:35:15 +00:00
$codeMirror.addClass('transition');
previewTo = setTimeout(function () {
2017-09-05 09:35:15 +00:00
$codeMirror.removeClass('transition');
}, 500);
if (['markdown', 'gfm'].indexOf(CodeMirror.highlightMode) === -1) {
$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); }
});
}
});
2018-02-27 16:38:29 +00:00
framework._.toolbar.$rightside.append($previewButton);
2017-09-05 09:35:15 +00:00
$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)) {
2020-02-11 17:20:45 +00:00
var privateData = framework._.cpNfInner.metadataMgr.getPrivateData();
href = privateData.origin + href;
} else if (/^#/.test(href)) {
2020-02-11 17:24:12 +00:00
var target = document.getElementById('cp-md-0-'+href.slice(1));
if (target) { target.scrollIntoView(); }
return;
}
framework._.sfCommon.openUnsafeURL(href);
}
});
var modeChange = function (mode) {
if (['markdown', 'gfm'].indexOf(mode) !== -1) {
$previewButton.show();
framework._.sfCommon.getPadAttribute('previewMode', function (e, data) {
2017-09-05 09:35:15 +00:00
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');
}
2017-09-05 09:35:15 +00:00
}
});
return;
2017-09-05 09:35:15 +00:00
}
$editorContainer.removeClass('cp-app-code-present');
$previewButton.hide();
2017-09-05 09:35:15 +00:00
$previewContainer.hide();
$previewButton.removeClass('active');
$codeMirrorContainer.addClass('cp-app-code-fullpage');
2017-09-05 09:35:15 +00:00
};
2017-10-24 16:49:58 +00:00
var isVisible = function () {
return $previewContainer.is(':visible');
};
framework.onReady(function () {
// add the splitter
var splitter = $('<div>', {
'class': 'cp-splitter'
}).appendTo($previewContainer);
2017-09-05 09:35:15 +00:00
$preview.on('scroll', function() {
splitter.css('top', $preview.scrollTop() + 'px');
2017-09-05 09:35:15 +00:00
});
var $target = $codeMirrorContainer;
2017-09-05 09:35:15 +00:00
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 (['markdown', 'gfm'].indexOf(CodeMirror.highlightMode) === -1) { 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);
});
2017-09-05 09:35:15 +00:00
framework._.sfCommon.getPadAttribute('previewMode', function (e, data) {
if (e) { return void console.error(e); }
if (data === false && $previewButton) {
$previewButton.click();
}
});
2017-09-05 09:35:15 +00:00
Visible.onChange(function (visible) {
if (visible) {
drawPreview();
}
});
return {
forceDraw: forceDrawPreview,
draw: drawPreview,
2017-10-24 16:49:58 +00:00
modeChange: modeChange,
isVisible: isVisible
};
};
2017-09-05 09:35:15 +00:00
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.replaceSelection($(mt)[0].outerHTML);
});
} else {
// Embedding is disabled
framework.setMediaTagEmbedder();
}
});
};
/////////////////////////////////////////////////////////////////////////////////////////////////////////
/////////////////////////////////////////////////////////////////////////////////////////////////////////
/////////////////////////////////////////////////////////////////////////////////////////////////////////
/////////////////////////////////////////////////////////////////////////////////////////////////////////
2017-09-05 09:35:15 +00:00
2020-04-09 14:06:04 +00:00
var authorUid = function (existing) {
if (!Array.isArray(existing)) { existing = []; }
var n;
var i = 0;
while (!n || existing.indexOf(n) !== -1 && i++ < 1000) {
n = Math.floor(Math.random() * 1000000);
}
// If we can't find a valid number in 1000 iterations, use 0...
if (existing.indexOf(n) !== -1) { n = 0; }
return n;
};
var andThen2 = function (editor, CodeMirror, framework, isPresentMode) {
2017-09-05 09:35:15 +00:00
var common = framework._.sfCommon;
2019-08-27 13:31:22 +00:00
var privateData = common.getMetadataMgr().getPrivateData();
2017-09-05 09:35:15 +00:00
var previewPane = mkPreviewPane(editor, CodeMirror, framework, isPresentMode);
2018-02-27 16:38:29 +00:00
var markdownTb = mkMarkdownTb(editor, framework);
2018-12-06 15:43:48 +00:00
2020-04-09 14:06:04 +00:00
var $removeAuthorColorsButton = framework._.sfCommon.createButton('removeauthorcolors', true, {icon: 'fa-paint-brush', title: 'Autorenfarben entfernen'}); // XXX
2020-04-07 14:52:38 +00:00
framework._.toolbar.$rightside.append($removeAuthorColorsButton);
$removeAuthorColorsButton.click(function() {
var selfrom = editor.getCursor("from");
var selto = editor.getCursor("to");
if (!editor.somethingSelected() || selfrom === selto) {
editor.getAllMarks().forEach(function (marker) {
marker.clear();
});
2020-04-09 14:06:04 +00:00
authormarks.authors = {};
authormarks.marks = [];
2020-04-07 14:52:38 +00:00
} else {
editor.findMarks(selfrom, selto).forEach(function (marker) {
marker.clear();
});
2020-04-07 14:52:38 +00:00
}
framework.localChange();
});
2020-04-09 14:06:04 +00:00
var authormarks = {
marks: [],
authors: {}
};
2020-04-07 14:52:38 +00:00
var authormarksLocal = [];
2020-04-09 14:06:04 +00:00
var myAuthorId = 0;
var MARK_OPACITY = 90;
var addMark = function (from, to, uid) {
var author = authormarks.authors[uid] || {};
2020-04-10 12:41:51 +00:00
uid = Number(uid);
return editor.markText(from, to, {
2020-04-09 14:06:04 +00:00
inclusiveLeft: uid === myAuthorId,
inclusiveRight: uid === myAuthorId,
css: "background-color: " + author.color + MARK_OPACITY,
attributes: {
'data-type': 'authormark',
'data-uid': uid
}
});
};
var sortMarks = function (a, b) {
if (!Array.isArray(b)) { return -1; }
if (!Array.isArray(a)) { return 1; }
// Check line
if (a[1] < b[1]) { return -1; }
if (a[1] > b[1]) { return 1; }
// Same line: check start offset
if (a[2] < b[2]) { return -1; }
if (a[2] > b[2]) { return 1; }
return 0;
};
2020-04-10 12:41:51 +00:00
var parseMark = function (array) {
if (!Array.isArray(array)) { return {}; }
var multiline = typeof(array[4]) !== "undefined";
var singleChar = typeof(array[3]) === "undefined";
return {
startLine: array[1],
startCh: array[2],
endLine: multiline ? array[3] : array[1],
2020-04-14 15:11:47 +00:00
endCh: singleChar ? (array[2]+1) : (multiline ? array[4] : array[3])
2020-04-10 12:41:51 +00:00
};
};
2020-04-07 14:08:28 +00:00
2018-12-06 15:43:48 +00:00
var $print = $('#cp-app-code-print');
var $content = $('#cp-app-code-preview-content');
mkPrintButton(framework, $content, $print);
2018-02-27 16:38:29 +00:00
mkHelpMenu(framework);
var evModeChange = Util.mkEvent();
evModeChange.reg(previewPane.modeChange);
2018-02-27 16:38:29 +00:00
evModeChange.reg(markdownTb.modeChange);
2017-09-05 09:35:15 +00:00
CodeMirror.mkIndentSettings(framework._.cpNfInner.metadataMgr);
CodeMirror.init(framework.localChange, framework._.title, framework._.toolbar);
mkFilePicker(framework, editor, evModeChange);
2017-09-05 09:35:15 +00:00
if (!framework.isReadOnly()) {
CodeMirror.configureTheme(common, function () {
CodeMirror.configureLanguage(common, null, evModeChange.fire);
});
} else {
CodeMirror.configureTheme(common);
}
2017-09-05 09:35:15 +00:00
2020-04-10 12:41:51 +00:00
var oldMarks = authormarks;
// Remove marks added by OT and fix the incorrect ones
// first: data about the change with the lowest offset
// last: data about the change with the latest offset
// in the comments, "I" am "first"
2020-04-14 15:11:47 +00:00
var fixMarks = function (first, last, content, toKeepEnd) {
console.log(first, last);
var toKeep = [];
// Get their start position compared to the authDoc
var lastAuthOffset = last.offset + last.total;
var lastAuthPos = SFCodeMirror.posToCursor(lastAuthOffset, last.doc);
// Get their start position compared to the localDoc
var lastLocalOffset = last.offset + first.total;
var lastLocalPos = SFCodeMirror.posToCursor(lastLocalOffset, first.doc);
console.log(lastAuthPos, lastAuthOffset);
console.log(lastLocalPos, lastLocalOffset);
2020-04-10 12:41:51 +00:00
// Keep their changes in the marks (after their offset)
last.marks.some(function (array, i) {
var p = parseMark(array);
// End of the mark before offset? ignore
2020-04-14 15:11:47 +00:00
if (p.endLine < lastAuthPos.line) { return; }
2020-04-10 12:41:51 +00:00
// Take everything from the first mark ending after the pos
2020-04-14 15:11:47 +00:00
if (p.endLine > lastAuthPos.line || p.endCh >= lastAuthPos.ch) {
toKeep = last.marks.slice(i);
last.marks.splice(i);
2020-04-10 12:41:51 +00:00
return true;
}
});
// Keep my marks (based on currentDoc) before their changes
2020-04-14 15:11:47 +00:00
var toJoin = {};
2020-04-10 12:41:51 +00:00
first.marks.some(function (array, i) {
var p = parseMark(array);
// End of the mark before offset? ignore
2020-04-14 15:11:47 +00:00
if (p.endLine < lastLocalPos.line) { return; }
2020-04-10 12:41:51 +00:00
// Take everything from the first mark ending after the pos
2020-04-14 15:11:47 +00:00
if (p.endLine > lastLocalPos.line || p.endCh >= lastLocalPos.ch) {
first.marks.splice(i);
2020-04-10 12:41:51 +00:00
return true;
}
});
2020-04-14 15:11:47 +00:00
if (first.marks.length) {
var toJoinMark = first.marks[first.marks.length - 1].slice();
toJoin = parseMark(toJoinMark);
}
2020-04-10 12:41:51 +00:00
console.info('to keep');
2020-04-14 15:11:47 +00:00
console.info(JSON.stringify(toKeep));
// Add the new markers to the result
Array.prototype.unshift.apply(toKeepEnd, toKeep);
// Fix their offset: compute added lines and added characters on the last line
// using the chainpad operation data (toInsert and toRemove)
var pos = SFCodeMirror.posToCursor(first.offset, content);
var removed = content.slice(first.offset, first.offset + first.toRemove);
var removedS = removed.split('\n');
var addedS = first.toInsert.split('\n');
var addLine = addedS.length - removedS.length;
var addCh = addedS[addedS.length - 1].length - removedS[removedS.length - 1].length;
if (addLine > 0) { addCh -= pos.ch; }
2020-04-10 12:41:51 +00:00
toKeepEnd.forEach(function (array) {
// Push to correct lines
array[1] += addLine;
if (typeof(array[4]) !== "undefined") { array[3] += addLine; }
// If they have markers on my end line, push their "ch"
2020-04-14 15:11:47 +00:00
if (array[1] === toJoin[1]) {
2020-04-10 12:41:51 +00:00
array[2] += addCh;
// If they have no end line, it means end line === start line,
// so we also push their end offset
if (!array[4] && array[3]) { array[3] += addCh; }
}
});
2020-04-14 15:11:47 +00:00
if (toKeep.length && toJoin) {
// Make sure the marks are joined correctly:
// fix the start position of the marks to keep
toKeepEnd[0][1] = toJoin.endLine;
toKeepEnd[0][2] = toJoin.endCh;
}
console.info(JSON.stringify(toJoin));
console.info(JSON.stringify(first.marks));
console.info(JSON.stringify(last.marks));
console.info(JSON.stringify(toKeepEnd));
2020-04-10 12:41:51 +00:00
};
2020-04-09 14:06:04 +00:00
var checkAuthors = function (userDoc) {
var chainpad = framework._.cpNfInner.chainpad;
var authDoc = JSON.parse(chainpad.getAuthDoc() || '{}');
if (!authDoc.content || !userDoc.content) { return; }
2020-04-14 15:11:47 +00:00
if (authDoc.content === userDoc.content) { return; } // No uncommitted work
2020-04-09 14:06:04 +00:00
if (!authormarks || !Array.isArray(authormarks.marks)) { return; }
2020-04-10 12:41:51 +00:00
var localDoc = CodeMirror.canonicalize(editor.getValue());
2020-04-14 15:11:47 +00:00
var commonParent = chainpad.getAuthBlock().getParent().getContent().doc;
console.log(chainpad);
console.log(commonParent);
var content = JSON.parse(commonParent || '{}').content || '';
2020-04-10 12:41:51 +00:00
// Their changes are the diff between my local doc (my local changes only)
// and the userDoc (my local changes + their changes pushed to the authdoc)
2020-04-14 15:11:47 +00:00
//var theirOps = ChainPad.Diff.diff(localDoc, userDoc.content);
var theirOps = ChainPad.Diff.diff(content, authDoc.content);
2020-04-10 12:41:51 +00:00
// My changes are the diff between my userDoc (my local changes + their changes)
// and the authDoc (their changes only)
2020-04-14 15:11:47 +00:00
//var myOps = ChainPad.Diff.diff(authDoc.content, userDoc.content);
var myOps = ChainPad.Diff.diff(content, localDoc);
if (!myOps.length || !theirOps.length) { return; }
2020-04-10 12:41:51 +00:00
2020-04-09 14:06:04 +00:00
// If I have uncommited content when receiving a remote patch, and they have
// pushed content to the same line as me, I need to update all the authormarks
// after their changes to push them by the length of the text I added
2020-04-14 15:11:47 +00:00
console.log(JSON.stringify(oldMarks.marks));
console.log(JSON.stringify(authDoc.authormarks.marks));
console.log(JSON.stringify(authormarks.marks));
2020-04-09 14:06:04 +00:00
console.warn(myOps);
console.warn(theirOps);
2020-04-10 12:41:51 +00:00
var marks = authormarks.marks;
2020-04-14 15:11:47 +00:00
var ops = {};
var myTotal = 0;
var theirTotal = 0;
var parseOp = function (me) {
return function (op) {
var size = (op.toInsert.length - op.toRemove);
/*
var pos = SFCodeMirror.posToCursor(op.offset, content);
var pos2 = SFCodeMirror.posToCursor(op.offset+size, content);
*/
ops[op.offset] = {
me: me,
offset: op.offset,
toInsert: op.toInsert,
toRemove: op.toRemove,
size: size,
/*
size: size,
startLine: pos.line,
startCh: pos.ch,
endLine: pos2.line,
endCh: pos2.ch,
addLine: pos2.line - pos.line,
addCh: pos2.ch - pos.ch,
*/
marks: (me ? (oldMarks && oldMarks.marks)
: (authDoc.authormarks && authDoc.authormarks.marks)) || [],
doc: me ? localDoc : authDoc.content
2020-04-10 12:41:51 +00:00
};
2020-04-14 15:11:47 +00:00
if (me) {
myTotal += size;
2020-04-10 12:41:51 +00:00
} else {
2020-04-14 15:11:47 +00:00
theirTotal += size;
2020-04-10 12:41:51 +00:00
}
2020-04-14 15:11:47 +00:00
};
};
myOps.forEach(parseOp(true));
theirOps.forEach(parseOp(false));
console.error(myTotal, theirTotal);
/*
theirOps.map(function (_op) {
var _pos = SFCodeMirror.posToCursor(_op.offset, content);
var _size = (_op.toInsert.length - _op.toRemove);
var _pos2 = SFCodeMirror.posToCursor(_op.offset+_size, content);
ops[_op.offset] = {
me: false,
offset: _op.offset,
size: _size,
startLine: _pos.line,
startCh: _pos.ch,
endLine: _pos2.line,
endCh: _pos2.ch,
marks: (authDoc.authormarks && authDoc.authormarks.marks) || [],
doc: authDoc.content
};
theirTotal += _size;
});
*/
var sorted = Object.keys(ops).map(Number);
sorted.sort().reverse();
console.log(sorted);
// We start from the end so that we don't have to fix the offsets everytime
var prev;
var toKeepEnd = [];
sorted.forEach(function (offset) {
var op = ops[offset];
// Not the same author? fix!
if (prev && prev.me !== op.me) {
prev.total = prev.me ? myTotal : theirTotal;
op.total = op.me ? myTotal : theirTotal;
fixMarks(op, prev, content, toKeepEnd);
}
if (op.me) { myTotal -= op.size }
else { theirTotal -= op.size }
prev = op;
2020-04-09 14:06:04 +00:00
});
2020-04-14 15:11:47 +00:00
var first = ops[sorted[sorted.length - 1]];
console.error(JSON.stringify(first.marks));
if (first) {
Array.prototype.unshift.apply(toKeepEnd, first.marks);
}
console.error(JSON.stringify(toKeepEnd));
authormarks.marks = toKeepEnd;
2020-04-09 14:06:04 +00:00
};
framework.onContentUpdate(function (newContent) {
var highlightMode = newContent.highlightMode;
if (highlightMode && highlightMode !== CodeMirror.highlightMode) {
CodeMirror.setMode(highlightMode, evModeChange.fire);
2017-09-05 09:35:15 +00:00
}
2020-04-07 14:08:28 +00:00
2020-04-14 15:11:47 +00:00
var chainpad = framework._.cpNfInner.chainpad;
console.error(chainpad._.authDoc);
console.warn(chainpad._.uncommitted);
console.error(authormarks.marks, oldMarks.marks);
2020-04-09 14:06:04 +00:00
if (newContent.authormarks) {
2020-04-10 12:41:51 +00:00
oldMarks = authormarks;
2020-04-09 14:06:04 +00:00
authormarks = newContent.authormarks;
if (!authormarks.marks) { authormarks.marks = []; }
if (!authormarks.authors) { authormarks.authors = {}; }
}
2020-04-07 14:08:28 +00:00
2020-04-09 14:06:04 +00:00
var chainpad = framework._.cpNfInner.chainpad;
var ops = ChainPad.Diff.diff(chainpad.getAuthDoc(), chainpad.getUserDoc());
if (ops.length) {
console.error(ops);
}
checkAuthors(newContent);
CodeMirror.contentUpdate(newContent); //, authormarks.marks, authormarksLocal);
previewPane.draw();
2020-04-10 12:41:51 +00:00
framework.localChange();
});
framework.setContentGetter(function () {
CodeMirror.removeCursors();
var content = CodeMirror.getContent();
content.highlightMode = CodeMirror.highlightMode;
previewPane.draw();
2020-04-07 14:08:28 +00:00
// get author marks
2020-04-09 14:06:04 +00:00
var authors = authormarks.authors || {};
var _marks = [];
2020-04-10 12:41:51 +00:00
var all = [];
var i = 0;
2020-04-07 14:08:28 +00:00
editor.getAllMarks().forEach(function (mark) {
var pos = mark.find();
2020-04-09 14:06:04 +00:00
var attributes = mark.attributes || {};
if (!pos || attributes['data-type'] !== 'authormark') { return; }
2020-04-10 12:41:51 +00:00
var uid = Number(attributes['data-uid']) || 0;
2020-04-09 14:06:04 +00:00
var author = authors[uid] || {};
2020-04-10 12:41:51 +00:00
all.forEach(function (obj,i) {
if (obj.uid !== uid) { return; }
if (obj.removed) { return; }
// Merge left
if (obj.pos.to.line === pos.from.line && obj.pos.to.ch === pos.from.ch) {
obj.removed = true;
_marks[obj.index] = undefined;
obj.mark.clear();
2020-04-09 14:06:04 +00:00
mark.clear();
2020-04-10 12:41:51 +00:00
mark = addMark(obj.pos.from, pos.to, uid);
pos.from = obj.pos.from;
return;
}
2020-04-10 12:41:51 +00:00
// Merge right
if (obj.pos.from.line === pos.to.line && obj.pos.from.ch === pos.to.ch) {
obj.removed = true;
_marks[obj.index] = undefined;
obj.mark.clear();
mark.clear();
mark = addMark(pos.from, obj.pos.to, uid);
pos.to = obj.pos.to;
}
});
2020-04-09 14:06:04 +00:00
var array = [uid, pos.from.line, pos.from.ch];
if (pos.from.line === pos.to.line && pos.to.ch > (pos.from.ch+1)) {
// If there is more than 1 character, add the "to" character
array.push(pos.to.ch);
} else if (pos.from.line !== pos.to.line) {
// If the mark is on more than one line, add the "to" line data
Array.prototype.push.apply(array, [pos.to.line, pos.to.ch]);
}
_marks.push(array);
2020-04-10 12:41:51 +00:00
all.push({
uid: uid,
2020-04-09 14:06:04 +00:00
pos: pos,
mark: mark,
2020-04-10 12:41:51 +00:00
index: i
});
i++;
2020-04-07 14:08:28 +00:00
});
2020-04-09 14:06:04 +00:00
_marks.sort(sortMarks);
2020-04-10 12:41:51 +00:00
authormarks.marks = _marks.filter(Boolean);
content.authormarks = authormarks;
2020-04-09 14:06:04 +00:00
//authormarksLocal = _marks.slice();
2020-04-07 14:08:28 +00:00
return content;
});
2017-09-05 09:35:15 +00:00
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
};
2018-12-04 16:18:42 +00:00
framework.onCursorUpdate(CodeMirror.setRemoteCursor);
framework.setCursorGetter(CodeMirror.getCursor);
editor.on('cursorActivity', updateCursor);
2018-12-04 16:18:42 +00:00
framework.onEditableChange(function () {
editor.setOption('readOnly', framework.isLocked() || framework.isReadOnly());
});
2017-09-05 09:35:15 +00:00
framework.setTitleRecommender(CodeMirror.getHeadingText);
2017-09-05 09:35:15 +00:00
2020-04-09 14:06:04 +00:00
var getMyAuthorId = function () {
var existing = Object.keys(authormarks.authors || {});
if (!common.isLoggedIn()) { return authorUid(existing); }
var userData = common.getMetadataMgr().getUserData();
var uid;
existing.some(function (id) {
var author = authormarks.authors[id] || {};
if (author.curvePublic !== userData.curvePublic) { return; }
uid = Number(id);
return true;
});
// XXX update my color?
return uid || authorUid(existing);
};
framework.onReady(function (newPad) {
2018-02-28 12:16:30 +00:00
editor.focus();
if (newPad && !CodeMirror.highlightMode) {
CodeMirror.setMode('gfm', evModeChange.fire);
//console.log("%s => %s", CodeMirror.highlightMode, CodeMirror.$language.val());
2017-09-05 09:35:15 +00:00
}
2020-04-09 14:06:04 +00:00
myAuthorId = getMyAuthorId();
console.warn(myAuthorId);
2017-09-05 09:35:15 +00:00
var fmConfig = {
dropArea: $('.CodeMirror'),
body: $('body'),
onUploaded: function (ev, data) {
2017-11-13 15:32:40 +00:00
var parsed = Hash.parsePadUrl(data.url);
2018-05-25 16:00:10 +00:00
var secret = Hash.getSecrets('file', parsed.hash, data.password);
2019-08-27 13:31:22 +00:00
var fileHost = privateData.fileHost || privateData.origin;
var src = fileHost + Hash.getBlobPathFromHex(secret.channel);
2018-05-25 16:00:10 +00:00
var key = Hash.encodeBase64(secret.keys.cryptKey);
var mt = '<media-tag src="' + src + '" data-crypto-key="cryptpad:' + key + '"></media-tag>';
2017-09-05 09:35:15 +00:00
editor.replaceSelection(mt);
}
};
common.createFileManager(fmConfig);
});
2017-09-05 09:35:15 +00:00
framework.onDefaultContentNeeded(function () {
2018-02-27 16:38:29 +00:00
editor.setValue(''); //Messages.codeInitialState);
});
2017-09-05 09:35:15 +00:00
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;
});
2017-09-05 09:35:15 +00:00
framework.setNormalizer(function (c) {
return {
content: c.content,
2020-04-07 14:08:28 +00:00
highlightMode: c.highlightMode,
authormarks: c.authormarks
};
});
2020-04-07 14:08:28 +00:00
editor.on('change', function( cm, change ) {
2020-04-09 14:06:04 +00:00
if (change.text !== undefined && (change.origin === "+input" || change.origin === "paste")) {
2020-04-07 14:08:28 +00:00
// add new author mark if text is added. marks from removed text are removed automatically
2020-04-09 14:06:04 +00:00
// If my text is inside an existing mark:
// * if it's my mark, do nothing
// * if it's someone else's mark, break it
// We can only have one author mark at a given position, but there may be
// another mark (cursor selection...) at this position so we use ".some"
var toSplit, abort;
editor.findMarksAt(change.from).some(function (mark) {
if (!mark.attributes) { return; }
if (mark.attributes['data-type'] !== 'authormark') { return; }
if (mark.attributes['data-uid'] !== myAuthorId) {
toSplit = {
mark: mark,
uid: mark.attributes['data-uid']
};
} else {
// This is our mark: abort to avoid making a new one
abort = true;
}
return true;
});
if (abort) { return void framework.localChange(); }
// Add my data to the doc if it's missing
if (!authormarks.authors[myAuthorId]) {
var userData = common.getMetadataMgr().getUserData();
authormarks.authors[myAuthorId] = {
name: userData.name,
curvePublic: userData.curvePublic,
color: userData.color
}
}
var to_add = {
line: change.from.line + change.text.length-1,
};
if (change.text.length > 1) {
2020-04-09 14:06:04 +00:00
// Multiple lines => take the length of the text added to the last line
to_add.ch = change.text[change.text.length-1].length;
} else {
2020-04-09 14:06:04 +00:00
// Single line => use the "from" position and add the length of the text
to_add.ch = change.from.ch + change.text[change.text.length-1].length;
}
if (toSplit && toSplit.mark && typeof(toSplit.uid) !== "undefined") {
// Break the other user's mark if needed
var _pos = toSplit.mark.find();
toSplit.mark.clear();
addMark(_pos.from, change.from, toSplit.uid); // their mark, 1st part
addMark(change.from, to_add, myAuthorId); // my mark
addMark(to_add, _pos.to, toSplit.uid); // their mark, 2nd part
} else {
// Add my mark
addMark(change.from, to_add, myAuthorId);
}
} else if (change.origin === "setValue") {
2020-04-07 14:08:28 +00:00
// on remote update: remove all marks, add new marks
editor.getAllMarks().forEach(function (marker) {
2020-04-09 14:06:04 +00:00
if (marker.attributes && marker.attributes['data-type'] === 'authormark') {
marker.clear();
}
});
2020-04-09 14:06:04 +00:00
authormarks.marks.forEach(function (mark) {
var from_line;
var to_line;
var from_ch;
var to_ch;
2020-04-09 14:06:04 +00:00
var uid = mark[0];
if (!authormarks.authors || !authormarks.authors[uid]) { return; }
var data = authormarks.authors[uid];
if (mark.length === 3) {
2020-04-07 18:49:36 +00:00
from_line = mark[1];
to_line = mark[1];
from_ch = mark[2];
to_ch = mark[2]+1;
} else if (mark.length === 4) {
2020-04-07 18:49:36 +00:00
from_line = mark[1];
to_line = mark[1];
from_ch = mark[2];
to_ch = mark[3];
2020-04-07 18:49:36 +00:00
} else if (mark.length === 5) {
from_line = mark[1];
to_line = mark[3];
from_ch = mark[2];
to_ch = mark[4];
}
2020-04-09 14:06:04 +00:00
addMark({
line: from_line, ch: from_ch
}, {
line: to_line, ch: to_ch
}, uid);
2020-04-07 14:08:28 +00:00
});
}
framework.localChange();
});
2017-09-05 09:35:15 +00:00
framework.start();
2018-11-22 13:45:12 +00:00
window.easyTest = function () {
var test = TypingTest.testCode(editor);
return test;
};
2017-09-05 09:35:15 +00:00
};
2017-10-24 16:49:58 +00:00
var getThumbnailContainer = function () {
var $preview = $('#cp-app-code-preview-content');
if ($preview.length && $preview.is(':visible')) {
return $preview[0];
}
};
2017-09-05 09:35:15 +00:00
var main = function () {
var CodeMirror;
var editor;
var framework;
2017-09-05 09:35:15 +00:00
nThen(function (waitFor) {
Framework.create({
toolbarContainer: '#cme_toolbox',
2017-10-24 16:49:58 +00:00
contentContainer: '#cp-app-code-editor',
2017-10-26 10:31:16 +00:00
thumbnail: {
getContainer: getThumbnailContainer,
filter: function (el, before) {
if (before) {
//$(el).parents().css('overflow', 'visible');
2017-10-26 10:31:16 +00:00
$(el).css('max-height', Math.max(600, $(el).width()) + 'px');
return;
}
$(el).parents().css('overflow', '');
2017-10-26 10:31:16 +00:00
$(el).css('max-height', '');
editor.refresh();
2017-10-26 10:31:16 +00:00
}
}
}, 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());
2017-09-05 09:35:15 +00:00
}).nThen(function (/*waitFor*/) {
framework._.sfCommon.isPresentUrl(function (err, val) {
andThen2(editor, CodeMirror, framework, val);
});
2017-09-05 09:35:15 +00:00
});
};
main();
});