Make code2 the default code app
This commit is contained in:
parent
c0f14dea5e
commit
d2787e3eff
@ -10,7 +10,7 @@
|
|||||||
bottom: 10vh;
|
bottom: 10vh;
|
||||||
opacity: 0.9;
|
opacity: 0.9;
|
||||||
box-sizing: border-box;
|
box-sizing: border-box;
|
||||||
z-index: 10000;
|
z-index: 1000000;
|
||||||
display: none;
|
display: none;
|
||||||
#cp-fileupload-table {
|
#cp-fileupload-table {
|
||||||
width: 80vw;
|
width: 80vw;
|
||||||
|
|||||||
@ -18,6 +18,6 @@ html.cp-app-noscroll {
|
|||||||
}
|
}
|
||||||
|
|
||||||
body.cp-app-pad { @import "../../../pad/app-pad.less"; }
|
body.cp-app-pad { @import "../../../pad/app-pad.less"; }
|
||||||
body.cp-app-code { @import "../../../code2/app-code.less"; }
|
body.cp-app-code { @import "../../../code/app-code.less"; }
|
||||||
body.cp-app-filepicker { @import "../../../filepicker/app-filepicker.less"; }
|
body.cp-app-filepicker { @import "../../../filepicker/app-filepicker.less"; }
|
||||||
|
|
||||||
|
|||||||
@ -1,41 +1,39 @@
|
|||||||
<!DOCTYPE html>
|
<!DOCTYPE html>
|
||||||
<html class="cp code">
|
<html>
|
||||||
<head>
|
<head>
|
||||||
<title>CryptPad</title>
|
<title>CryptPad</title>
|
||||||
<meta content="text/html; charset=utf-8" http-equiv="content-type"/>
|
<meta content="text/html; charset=utf-8" http-equiv="content-type"/>
|
||||||
<meta name="viewport" content="width=device-width, initial-scale=1.0">
|
<meta name="viewport" content="width=device-width, initial-scale=1.0">
|
||||||
<meta name="referrer" content="no-referrer" />
|
<meta name="referrer" content="no-referrer" />
|
||||||
<script async data-bootload="/customize/template.js" data-main="/common/boot.js?ver=1.0" src="/bower_components/requirejs/require.js?ver=2.3.5"></script>
|
<script async data-bootload="main.js" data-main="/common/boot.js?ver=1.0" src="/bower_components/requirejs/require.js?ver=2.3.5"></script>
|
||||||
<style>
|
<style>
|
||||||
html, body {
|
html, body {
|
||||||
overflow-y: hidden;
|
margin: 0px;
|
||||||
}
|
|
||||||
#iframe-container {
|
|
||||||
position: fixed;
|
|
||||||
top: 0px;
|
|
||||||
bottom: 0px;
|
|
||||||
right: 0px;
|
|
||||||
left: 0px;
|
|
||||||
padding: 0px;
|
padding: 0px;
|
||||||
}
|
}
|
||||||
#pad-iframe {
|
#sbox-iframe {
|
||||||
|
position:fixed;
|
||||||
|
top:0px;
|
||||||
|
left:0px;
|
||||||
|
bottom:0px;
|
||||||
|
right:0px;
|
||||||
width:100%;
|
width:100%;
|
||||||
height:100%;
|
height:100%;
|
||||||
border:none;
|
border:none;
|
||||||
margin:0;
|
margin:0;
|
||||||
padding:0;
|
padding:0;
|
||||||
overflow:hidden;
|
overflow:hidden;
|
||||||
box-sizing: border-box;
|
|
||||||
}
|
}
|
||||||
/* We use !important here to override the 96% set to the element in DecorateToolbar.js
|
#sbox-filePicker-iframe {
|
||||||
when we enter fullscreen mode. It allows us to avoid changing the iframe's size in JS */
|
position: fixed;
|
||||||
#pad-iframe.fullscreen {
|
top:0; left:0;
|
||||||
top: 0px;
|
bottom:0; right:0;
|
||||||
height: 100% !important;
|
width:100%;
|
||||||
|
height: 100%;
|
||||||
|
border: 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
</style>
|
</style>
|
||||||
</head>
|
</head>
|
||||||
<body>
|
<body>
|
||||||
<div id="iframe-container">
|
<iframe id="sbox-iframe">
|
||||||
<iframe id="pad-iframe"></iframe><script src="/common/noscriptfix.js"></script>
|
|
||||||
</div>
|
|
||||||
|
|||||||
@ -1,16 +1,20 @@
|
|||||||
<!DOCTYPE html>
|
<!DOCTYPE html>
|
||||||
<html style="height: 100%;">
|
<html class="cp-app-noscroll">
|
||||||
<head>
|
<head>
|
||||||
<meta content="text/html; charset=utf-8" http-equiv="content-type"/>
|
<meta content="text/html; charset=utf-8" http-equiv="content-type"/>
|
||||||
<script src="/bower_components/jquery/dist/jquery.min.js"></script>
|
<script async data-bootload="/code/inner.js" data-main="/common/sframe-boot.js?ver=1.1" src="/bower_components/requirejs/require.js?ver=2.3.5"></script>
|
||||||
<script async data-bootload="inner.js" data-main="/common/boot.js?ver=1.0" src="/bower_components/requirejs/require.js?ver=2.3.5"></script>
|
<style>
|
||||||
<style> .loading-hidden { display: none; } </style>
|
.loading-hidden { display: none; }
|
||||||
|
#editor1 { display: none; }
|
||||||
|
</style>
|
||||||
</head>
|
</head>
|
||||||
<body class="loading-hidden">
|
<body class="cp-app-code">
|
||||||
<div id="cme_toolbox" class="toolbar-container"></div>
|
<div id="cme_toolbox" class="cp-toolbar-container"></div>
|
||||||
<div id="editorContainer">
|
<div id="cp-app-code-editor">
|
||||||
<textarea id="editor1" name="editor1"></textarea>
|
<textarea id="editor1" name="editor1"></textarea>
|
||||||
<div id="previewContainer"><div id="preview"></div></div>
|
<div id="cp-app-code-preview">
|
||||||
|
<div id="cp-app-code-preview-content"></div>
|
||||||
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</body>
|
</body>
|
||||||
</html>
|
</html>
|
||||||
|
|||||||
@ -1,13 +1,23 @@
|
|||||||
define([
|
define([
|
||||||
'jquery',
|
'jquery',
|
||||||
|
'/bower_components/chainpad-crypto/crypto.js',
|
||||||
|
'/bower_components/textpatcher/TextPatcher.js',
|
||||||
|
'/common/toolbar3.js',
|
||||||
|
'json.sortify',
|
||||||
|
'/bower_components/chainpad-json-validator/json-ot.js',
|
||||||
|
'/common/cryptpad-common.js',
|
||||||
|
'/common/cryptget.js',
|
||||||
|
'/common/diffMarked.js',
|
||||||
|
'/bower_components/nthen/index.js',
|
||||||
|
'/common/sframe-common.js',
|
||||||
|
'/api/config',
|
||||||
|
'/common/common-realtime.js',
|
||||||
|
|
||||||
'cm/lib/codemirror',
|
'cm/lib/codemirror',
|
||||||
|
|
||||||
'css!/bower_components/components-font-awesome/css/font-awesome.min.css',
|
|
||||||
'css!/bower_components/bootstrap/dist/css/bootstrap.min.css',
|
'css!/bower_components/bootstrap/dist/css/bootstrap.min.css',
|
||||||
'less!/code/code.less',
|
'css!/bower_components/components-font-awesome/css/font-awesome.min.css',
|
||||||
'less!/customize/src/less/toolbar.less',
|
'less!/customize/src/less2/main.less',
|
||||||
'less!/customize/src/less/cryptpad.less',
|
|
||||||
|
|
||||||
'css!cm/lib/codemirror.css',
|
'css!cm/lib/codemirror.css',
|
||||||
'css!cm/addon/dialog/dialog.css',
|
'css!cm/addon/dialog/dialog.css',
|
||||||
@ -34,7 +44,599 @@ define([
|
|||||||
'cm/addon/fold/markdown-fold',
|
'cm/addon/fold/markdown-fold',
|
||||||
'cm/addon/fold/comment-fold',
|
'cm/addon/fold/comment-fold',
|
||||||
'cm/addon/display/placeholder',
|
'cm/addon/display/placeholder',
|
||||||
], function ($, CMeditor) {
|
|
||||||
|
], function (
|
||||||
|
$,
|
||||||
|
Crypto,
|
||||||
|
TextPatcher,
|
||||||
|
Toolbar,
|
||||||
|
JSONSortify,
|
||||||
|
JsonOT,
|
||||||
|
Cryptpad,
|
||||||
|
Cryptget,
|
||||||
|
DiffMd,
|
||||||
|
nThen,
|
||||||
|
SFCommon,
|
||||||
|
ApiConfig,
|
||||||
|
CommonRealtime,
|
||||||
|
CMeditor)
|
||||||
|
{
|
||||||
window.CodeMirror = CMeditor;
|
window.CodeMirror = CMeditor;
|
||||||
$('.loading-hidden').removeClass('loading-hidden');
|
var Messages = Cryptpad.Messages;
|
||||||
|
|
||||||
|
var APP = window.APP = {
|
||||||
|
Cryptpad: Cryptpad,
|
||||||
|
};
|
||||||
|
|
||||||
|
var stringify = function (obj) {
|
||||||
|
return JSONSortify(obj);
|
||||||
|
};
|
||||||
|
|
||||||
|
var toolbar;
|
||||||
|
|
||||||
|
var onConnectError = function () {
|
||||||
|
Cryptpad.errorLoadingScreen(Messages.websocketError);
|
||||||
|
};
|
||||||
|
|
||||||
|
var andThen = function (editor, CodeMirror, common) {
|
||||||
|
var readOnly = false;
|
||||||
|
var cpNfInner;
|
||||||
|
var metadataMgr;
|
||||||
|
var $bar = $('#cme_toolbox');
|
||||||
|
|
||||||
|
var isHistoryMode = false;
|
||||||
|
|
||||||
|
var $contentContainer = $('#cp-app-code-editor');
|
||||||
|
var $previewContainer = $('#cp-app-code-preview');
|
||||||
|
var $preview = $('#cp-app-code-preview-content');
|
||||||
|
$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');
|
||||||
|
window.open(href);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
var setIndentation = APP.setIndentation = function (units, useTabs) {
|
||||||
|
if (typeof(units) !== 'number') { return; }
|
||||||
|
editor.setOption('indentUnit', units);
|
||||||
|
editor.setOption('tabSize', units);
|
||||||
|
editor.setOption('indentWithTabs', useTabs);
|
||||||
|
};
|
||||||
|
|
||||||
|
var indentKey = 'indentUnit';
|
||||||
|
var useTabsKey = 'indentWithTabs';
|
||||||
|
var updateIndentSettings = function () {
|
||||||
|
if (!metadataMgr) { return; }
|
||||||
|
var data = metadataMgr.getPrivateData().settings;
|
||||||
|
var indentUnit = data[indentKey];
|
||||||
|
var useTabs = data[useTabsKey];
|
||||||
|
setIndentation(
|
||||||
|
typeof(indentUnit) === 'number'? indentUnit: 2,
|
||||||
|
typeof(useTabs) === 'boolean'? useTabs: false);
|
||||||
|
};
|
||||||
|
|
||||||
|
var setEditable = APP.setEditable = function (bool) {
|
||||||
|
if (readOnly && bool) { return; }
|
||||||
|
editor.setOption('readOnly', !bool);
|
||||||
|
};
|
||||||
|
|
||||||
|
var Title;
|
||||||
|
|
||||||
|
var config = {
|
||||||
|
readOnly: readOnly,
|
||||||
|
transformFunction: JsonOT.validate,
|
||||||
|
// cryptpad debug logging (default is 1)
|
||||||
|
// logLevel: 0,
|
||||||
|
validateContent: function (content) {
|
||||||
|
try {
|
||||||
|
JSON.parse(content);
|
||||||
|
return true;
|
||||||
|
} catch (e) {
|
||||||
|
console.log("Failed to parse, rejecting patch");
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
var canonicalize = function (t) { return t.replace(/\r\n/g, '\n'); };
|
||||||
|
|
||||||
|
var setHistory = function (bool, update) {
|
||||||
|
isHistoryMode = bool;
|
||||||
|
setEditable(!bool);
|
||||||
|
if (!bool && update) {
|
||||||
|
config.onRemote();
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
CommonRealtime.onInfiniteSpinner(function () { setEditable(false); });
|
||||||
|
|
||||||
|
setEditable(false);
|
||||||
|
var initializing = true;
|
||||||
|
|
||||||
|
var stringifyInner = function (textValue) {
|
||||||
|
var obj = {
|
||||||
|
content: textValue,
|
||||||
|
metadata: metadataMgr.getMetadataLazy()
|
||||||
|
};
|
||||||
|
/* metadata: {
|
||||||
|
users: UserList.userData,
|
||||||
|
defaultTitle: Title.defaultTitle
|
||||||
|
}
|
||||||
|
};
|
||||||
|
if (!initializing) {
|
||||||
|
obj.metadata.title = Title.title;
|
||||||
|
}*/
|
||||||
|
// set mode too...
|
||||||
|
obj.highlightMode = CodeMirror.highlightMode;
|
||||||
|
|
||||||
|
// stringify the json and send it into chainpad
|
||||||
|
return stringify(obj);
|
||||||
|
};
|
||||||
|
|
||||||
|
var forceDrawPreview = function () {
|
||||||
|
try {
|
||||||
|
DiffMd.apply(DiffMd.render(editor.getValue()), $preview);
|
||||||
|
} catch (e) { console.error(e); }
|
||||||
|
};
|
||||||
|
|
||||||
|
var drawPreview = Cryptpad.throttle(function () {
|
||||||
|
if (CodeMirror.highlightMode !== 'markdown') { return; }
|
||||||
|
if (!$previewContainer.is(':visible')) { return; }
|
||||||
|
forceDrawPreview();
|
||||||
|
}, 150);
|
||||||
|
|
||||||
|
var onLocal = config.onLocal = function () {
|
||||||
|
if (initializing) { return; }
|
||||||
|
if (isHistoryMode) { return; }
|
||||||
|
if (readOnly) { return; }
|
||||||
|
|
||||||
|
editor.save();
|
||||||
|
|
||||||
|
drawPreview();
|
||||||
|
|
||||||
|
var textValue = canonicalize(CodeMirror.$textarea.val());
|
||||||
|
var shjson = stringifyInner(textValue);
|
||||||
|
|
||||||
|
APP.patchText(shjson);
|
||||||
|
|
||||||
|
if (APP.realtime.getUserDoc() !== shjson) {
|
||||||
|
console.error("realtime.getUserDoc() !== shjson");
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
var mediaTagModes = [
|
||||||
|
'markdown',
|
||||||
|
'html',
|
||||||
|
'htmlembedded',
|
||||||
|
'htmlmixed',
|
||||||
|
'index.html',
|
||||||
|
'php',
|
||||||
|
'velocity',
|
||||||
|
'xml',
|
||||||
|
];
|
||||||
|
|
||||||
|
var onModeChanged = function (mode) {
|
||||||
|
var $codeMirror = $('.CodeMirror');
|
||||||
|
window.clearTimeout(APP.previewTo);
|
||||||
|
$codeMirror.addClass('transition');
|
||||||
|
APP.previewTo = window.setTimeout(function () {
|
||||||
|
$codeMirror.removeClass('transition');
|
||||||
|
}, 500);
|
||||||
|
if (mediaTagModes.indexOf(mode) !== -1) {
|
||||||
|
APP.$mediaTagButton.show();
|
||||||
|
} else { APP.$mediaTagButton.hide(); }
|
||||||
|
|
||||||
|
if (mode === "markdown") {
|
||||||
|
APP.$previewButton.show();
|
||||||
|
common.getPadAttribute('previewMode', function (e, data) {
|
||||||
|
if (e) { return void console.error(e); }
|
||||||
|
if (data !== false) {
|
||||||
|
$previewContainer.show();
|
||||||
|
APP.$previewButton.addClass('active');
|
||||||
|
$codeMirror.removeClass('fullPage');
|
||||||
|
}
|
||||||
|
});
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
APP.$previewButton.hide();
|
||||||
|
$previewContainer.hide();
|
||||||
|
APP.$previewButton.removeClass('active');
|
||||||
|
$codeMirror.addClass('fullPage');
|
||||||
|
};
|
||||||
|
|
||||||
|
config.onInit = function (info) {
|
||||||
|
metadataMgr.onChangeLazy(updateIndentSettings);
|
||||||
|
updateIndentSettings();
|
||||||
|
|
||||||
|
readOnly = metadataMgr.getPrivateData().readOnly;
|
||||||
|
|
||||||
|
var titleCfg = { getHeadingText: CodeMirror.getHeadingText };
|
||||||
|
Title = common.createTitle(titleCfg, config.onLocal, common, metadataMgr);
|
||||||
|
|
||||||
|
var configTb = {
|
||||||
|
displayed: ['title', 'useradmin', 'spinner', 'share', 'userlist', 'newpad', 'limit'],
|
||||||
|
title: Title.getTitleConfig(),
|
||||||
|
metadataMgr: metadataMgr,
|
||||||
|
readOnly: readOnly,
|
||||||
|
ifrw: window,
|
||||||
|
realtime: info.realtime,
|
||||||
|
common: Cryptpad,
|
||||||
|
sfCommon: common,
|
||||||
|
$container: $bar,
|
||||||
|
$contentContainer: $contentContainer
|
||||||
|
};
|
||||||
|
toolbar = APP.toolbar = Toolbar.create(configTb);
|
||||||
|
Title.setToolbar(toolbar);
|
||||||
|
CodeMirror.init(config.onLocal, Title, toolbar);
|
||||||
|
|
||||||
|
var $rightside = toolbar.$rightside;
|
||||||
|
var $drawer = toolbar.$drawer;
|
||||||
|
|
||||||
|
/* add a history button */
|
||||||
|
var histConfig = {
|
||||||
|
onLocal: config.onLocal,
|
||||||
|
onRemote: config.onRemote,
|
||||||
|
setHistory: setHistory,
|
||||||
|
applyVal: function (val) {
|
||||||
|
var remoteDoc = JSON.parse(val || '{}').content;
|
||||||
|
editor.setValue(remoteDoc || '');
|
||||||
|
editor.save();
|
||||||
|
},
|
||||||
|
$toolbar: $bar
|
||||||
|
};
|
||||||
|
var $hist = common.createButton('history', true, {histConfig: histConfig});
|
||||||
|
$drawer.append($hist);
|
||||||
|
|
||||||
|
/* save as template */
|
||||||
|
if (!metadataMgr.getPrivateData().isTemplate) {
|
||||||
|
var templateObj = {
|
||||||
|
rt: info.realtime,
|
||||||
|
getTitle: function () { return metadataMgr.getMetadata().title; }
|
||||||
|
};
|
||||||
|
var $templateButton = common.createButton('template', true, templateObj);
|
||||||
|
$rightside.append($templateButton);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* add an export button */
|
||||||
|
var $export = common.createButton('export', true, {}, CodeMirror.exportText);
|
||||||
|
$drawer.append($export);
|
||||||
|
|
||||||
|
if (!readOnly) {
|
||||||
|
/* add an import button */
|
||||||
|
var $import = common.createButton('import', true, {}, CodeMirror.importText);
|
||||||
|
$drawer.append($import);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* add a forget button */
|
||||||
|
var forgetCb = function (err) {
|
||||||
|
if (err) { return; }
|
||||||
|
setEditable(false);
|
||||||
|
};
|
||||||
|
var $forgetPad = common.createButton('forget', true, {}, forgetCb);
|
||||||
|
$rightside.append($forgetPad);
|
||||||
|
|
||||||
|
var $previewButton = APP.$previewButton = common.createButton(null, true);
|
||||||
|
$previewButton.removeClass('fa-question').addClass('fa-eye');
|
||||||
|
$previewButton.attr('title', Messages.previewButtonTitle);
|
||||||
|
$previewButton.click(function () {
|
||||||
|
var $codeMirror = $('.CodeMirror');
|
||||||
|
window.clearTimeout(APP.previewTo);
|
||||||
|
$codeMirror.addClass('transition');
|
||||||
|
APP.previewTo = window.setTimeout(function () {
|
||||||
|
$codeMirror.removeClass('transition');
|
||||||
|
}, 500);
|
||||||
|
if (CodeMirror.highlightMode !== 'markdown') {
|
||||||
|
$previewContainer.show();
|
||||||
|
}
|
||||||
|
$previewContainer.toggle();
|
||||||
|
if ($previewContainer.is(':visible')) {
|
||||||
|
forceDrawPreview();
|
||||||
|
$codeMirror.removeClass('fullPage');
|
||||||
|
$previewButton.addClass('active');
|
||||||
|
common.setPadAttribute('previewMode', true, function (e) {
|
||||||
|
if (e) { return console.log(e); }
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
$codeMirror.addClass('fullPage');
|
||||||
|
$previewButton.removeClass('active');
|
||||||
|
common.setPadAttribute('previewMode', false, function (e) {
|
||||||
|
if (e) { return console.log(e); }
|
||||||
|
});
|
||||||
|
}
|
||||||
|
});
|
||||||
|
$rightside.append($previewButton);
|
||||||
|
|
||||||
|
if (!readOnly) {
|
||||||
|
CodeMirror.configureTheme(function () {
|
||||||
|
CodeMirror.configureLanguage(null, onModeChanged);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
CodeMirror.configureTheme();
|
||||||
|
}
|
||||||
|
|
||||||
|
var fileDialogCfg = {
|
||||||
|
onSelect: function (data) {
|
||||||
|
if (data.type === 'file') {
|
||||||
|
var mt = '<media-tag src="' + data.src + '" data-crypto-key="cryptpad:' + data.key + '"></media-tag>';
|
||||||
|
editor.replaceSelection(mt);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
common.initFilePicker(common, fileDialogCfg);
|
||||||
|
APP.$mediaTagButton = $('<button>', {
|
||||||
|
title: Messages.filePickerButton,
|
||||||
|
'class': 'cp-toolbar-rightside-button fa fa-picture-o',
|
||||||
|
style: 'font-size: 17px'
|
||||||
|
}).click(function () {
|
||||||
|
var pickerCfg = {
|
||||||
|
types: ['file'],
|
||||||
|
where: ['root']
|
||||||
|
};
|
||||||
|
common.openFilePicker(common, pickerCfg);
|
||||||
|
}).appendTo($rightside);
|
||||||
|
|
||||||
|
};
|
||||||
|
|
||||||
|
config.onReady = function (info) {
|
||||||
|
console.log('onready');
|
||||||
|
if (APP.realtime !== info.realtime) {
|
||||||
|
var realtime = APP.realtime = info.realtime;
|
||||||
|
APP.patchText = TextPatcher.create({
|
||||||
|
realtime: realtime,
|
||||||
|
//logging: true
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
var userDoc = APP.realtime.getUserDoc();
|
||||||
|
|
||||||
|
var isNew = false;
|
||||||
|
if (userDoc === "" || userDoc === "{}") { isNew = true; }
|
||||||
|
|
||||||
|
var newDoc = "";
|
||||||
|
if (userDoc !== "") {
|
||||||
|
var hjson = JSON.parse(userDoc);
|
||||||
|
|
||||||
|
if (hjson && hjson.metadata) {
|
||||||
|
metadataMgr.updateMetadata(hjson.metadata);
|
||||||
|
}
|
||||||
|
if (typeof (hjson) !== 'object' || Array.isArray(hjson) ||
|
||||||
|
(typeof(hjson.type) !== 'undefined' && hjson.type !== 'code')) {
|
||||||
|
var errorText = Messages.typeError;
|
||||||
|
Cryptpad.errorLoadingScreen(errorText);
|
||||||
|
throw new Error(errorText);
|
||||||
|
}
|
||||||
|
|
||||||
|
newDoc = hjson.content;
|
||||||
|
|
||||||
|
if (hjson.highlightMode) {
|
||||||
|
CodeMirror.setMode(hjson.highlightMode, onModeChanged);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
Title.updateTitle(Cryptpad.initialName || Title.defaultTitle);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!CodeMirror.highlightMode) {
|
||||||
|
CodeMirror.setMode('markdown', onModeChanged);
|
||||||
|
//console.log("%s => %s", CodeMirror.highlightMode, CodeMirror.$language.val());
|
||||||
|
}
|
||||||
|
|
||||||
|
// Update the user list (metadata) from the hyperjson
|
||||||
|
//Metadata.update(userDoc);
|
||||||
|
|
||||||
|
if (newDoc) {
|
||||||
|
editor.setValue(newDoc);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (Cryptpad.initialName && Title.isDefaultTitle()) {
|
||||||
|
Title.updateTitle(Cryptpad.initialName);
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
common.getPadAttribute('previewMode', function (e, data) {
|
||||||
|
if (e) { return void console.error(e); }
|
||||||
|
if (data === false && APP.$previewButton) {
|
||||||
|
APP.$previewButton.click();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
|
||||||
|
/*
|
||||||
|
// add the splitter
|
||||||
|
if (!$iframe.has('.cp-splitter').length) {
|
||||||
|
var $preview = $iframe.find('#previewContainer');
|
||||||
|
var splitter = $('<div>', {
|
||||||
|
'class': 'cp-splitter'
|
||||||
|
}).appendTo($preview);
|
||||||
|
|
||||||
|
$preview.on('scroll', function() {
|
||||||
|
splitter.css('top', $preview.scrollTop() + 'px');
|
||||||
|
});
|
||||||
|
|
||||||
|
var $target = $iframe.find('.CodeMirror');
|
||||||
|
|
||||||
|
splitter.on('mousedown', function (e) {
|
||||||
|
e.preventDefault();
|
||||||
|
var x = e.pageX;
|
||||||
|
var w = $target.width();
|
||||||
|
|
||||||
|
$iframe.on('mouseup mousemove', function handler(evt) {
|
||||||
|
if (evt.type === 'mouseup') {
|
||||||
|
$iframe.off('mouseup mousemove', handler);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
$target.css('width', (w - x + evt.pageX) + 'px');
|
||||||
|
});
|
||||||
|
});
|
||||||
|
}
|
||||||
|
*/
|
||||||
|
|
||||||
|
Cryptpad.removeLoadingScreen();
|
||||||
|
setEditable(!readOnly);
|
||||||
|
initializing = false;
|
||||||
|
|
||||||
|
onLocal(); // push local state to avoid parse errors later.
|
||||||
|
|
||||||
|
if (readOnly) {
|
||||||
|
config.onRemote();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (isNew) {
|
||||||
|
common.openTemplatePicker(common);
|
||||||
|
}
|
||||||
|
|
||||||
|
var fmConfig = {
|
||||||
|
dropArea: $('.CodeMirror'),
|
||||||
|
body: $('body'),
|
||||||
|
onUploaded: function (ev, data) {
|
||||||
|
//var cursor = editor.getCursor();
|
||||||
|
//var cleanName = data.name.replace(/[\[\]]/g, '');
|
||||||
|
//var text = '';
|
||||||
|
var parsed = Cryptpad.parsePadUrl(data.url);
|
||||||
|
var hexFileName = Cryptpad.base64ToHex(parsed.hashData.channel);
|
||||||
|
var src = '/blob/' + hexFileName.slice(0,2) + '/' + hexFileName;
|
||||||
|
var mt = '<media-tag src="' + src + '" data-crypto-key="cryptpad:' + parsed.hashData.key + '"></media-tag>';
|
||||||
|
editor.replaceSelection(mt);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
APP.FM = common.createFileManager(fmConfig);
|
||||||
|
};
|
||||||
|
|
||||||
|
config.onRemote = function () {
|
||||||
|
if (initializing) { return; }
|
||||||
|
if (isHistoryMode) { return; }
|
||||||
|
|
||||||
|
var oldDoc = canonicalize(CodeMirror.$textarea.val());
|
||||||
|
var shjson = APP.realtime.getUserDoc();
|
||||||
|
|
||||||
|
// Update the user list (metadata) from the hyperjson
|
||||||
|
//Metadata.update(shjson);
|
||||||
|
|
||||||
|
var hjson = JSON.parse(shjson);
|
||||||
|
var remoteDoc = hjson.content;
|
||||||
|
|
||||||
|
if (hjson.metadata) {
|
||||||
|
metadataMgr.updateMetadata(hjson.metadata);
|
||||||
|
}
|
||||||
|
|
||||||
|
var highlightMode = hjson.highlightMode;
|
||||||
|
if (highlightMode && highlightMode !== APP.highlightMode) {
|
||||||
|
CodeMirror.setMode(highlightMode, onModeChanged);
|
||||||
|
}
|
||||||
|
|
||||||
|
CodeMirror.setValueAndCursor(oldDoc, remoteDoc, TextPatcher);
|
||||||
|
drawPreview();
|
||||||
|
|
||||||
|
if (!readOnly) {
|
||||||
|
var textValue = canonicalize(CodeMirror.$textarea.val());
|
||||||
|
var shjson2 = stringifyInner(textValue);
|
||||||
|
if (shjson2 !== shjson) {
|
||||||
|
console.error("shjson2 !== shjson");
|
||||||
|
TextPatcher.log(shjson, TextPatcher.diff(shjson, shjson2));
|
||||||
|
APP.patchText(shjson2);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (oldDoc !== remoteDoc) { Cryptpad.notify(); }
|
||||||
|
};
|
||||||
|
|
||||||
|
config.onAbort = function () {
|
||||||
|
// inform of network disconnect
|
||||||
|
setEditable(false);
|
||||||
|
toolbar.failed();
|
||||||
|
Cryptpad.alert(Messages.common_connectionLost, undefined, true);
|
||||||
|
};
|
||||||
|
|
||||||
|
config.onConnectionChange = function (info) {
|
||||||
|
setEditable(info.state);
|
||||||
|
//toolbar.failed();
|
||||||
|
if (info.state) {
|
||||||
|
initializing = true;
|
||||||
|
//toolbar.reconnecting(info.myId);
|
||||||
|
Cryptpad.findOKButton().click();
|
||||||
|
} else {
|
||||||
|
Cryptpad.alert(Messages.common_connectionLost, undefined, true);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
config.onError = onConnectError;
|
||||||
|
|
||||||
|
cpNfInner = common.startRealtime(config);
|
||||||
|
metadataMgr = cpNfInner.metadataMgr;
|
||||||
|
|
||||||
|
editor.on('change', onLocal);
|
||||||
|
|
||||||
|
Cryptpad.onLogout(function () { setEditable(false); });
|
||||||
|
};
|
||||||
|
/*
|
||||||
|
var interval = 100;
|
||||||
|
var second = function (CM) {
|
||||||
|
Cryptpad.ready(function () {
|
||||||
|
andThen(CM);
|
||||||
|
Cryptpad.reportAppUsage();
|
||||||
|
});
|
||||||
|
Cryptpad.onError(function (info) {
|
||||||
|
if (info && info.type === "store") {
|
||||||
|
onConnectError();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
var first = function () {
|
||||||
|
if (ifrw.CodeMirror) {
|
||||||
|
second(ifrw.CodeMirror);
|
||||||
|
} else {
|
||||||
|
console.log("CodeMirror was not defined. Trying again in %sms", interval);
|
||||||
|
setTimeout(first, interval);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
first();*/
|
||||||
|
var CMEDITOR_CHECK_INTERVAL = 100;
|
||||||
|
var cmEditorAvailable = function (cb) {
|
||||||
|
var intr;
|
||||||
|
var check = function () {
|
||||||
|
if (window.CodeMirror) {
|
||||||
|
clearTimeout(intr);
|
||||||
|
cb(window.CodeMirror);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
intr = setInterval(function () {
|
||||||
|
console.log("CodeMirror was not defined. Trying again in %sms", CMEDITOR_CHECK_INTERVAL);
|
||||||
|
check();
|
||||||
|
}, CMEDITOR_CHECK_INTERVAL);
|
||||||
|
check();
|
||||||
|
};
|
||||||
|
var main = function () {
|
||||||
|
var CM;
|
||||||
|
var CodeMirror;
|
||||||
|
var editor;
|
||||||
|
var common;
|
||||||
|
|
||||||
|
nThen(function (waitFor) {
|
||||||
|
cmEditorAvailable(waitFor(function (cm) {
|
||||||
|
CM = cm;
|
||||||
|
}));
|
||||||
|
$(waitFor(function () {
|
||||||
|
Cryptpad.addLoadingScreen();
|
||||||
|
}));
|
||||||
|
SFCommon.create(waitFor(function (c) { APP.common = common = c; }));
|
||||||
|
}).nThen(function (/*waitFor*/) {
|
||||||
|
CodeMirror = Cryptpad.createCodemirror(window, Cryptpad, null, CM);
|
||||||
|
$('.CodeMirror').addClass('fullPage');
|
||||||
|
editor = CodeMirror.editor;
|
||||||
|
Cryptpad.onError(function (info) {
|
||||||
|
if (info && info.type === "store") {
|
||||||
|
onConnectError();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
andThen(editor, CodeMirror, common);
|
||||||
|
});
|
||||||
|
};
|
||||||
|
main();
|
||||||
});
|
});
|
||||||
|
|||||||
584
www/code/main.js
584
www/code/main.js
@ -1,559 +1,41 @@
|
|||||||
|
// Load #1, load as little as possible because we are in a race to get the loading screen up.
|
||||||
define([
|
define([
|
||||||
|
'/bower_components/nthen/index.js',
|
||||||
|
'/api/config',
|
||||||
'jquery',
|
'jquery',
|
||||||
'/bower_components/chainpad-crypto/crypto.js',
|
'/common/requireconfig.js',
|
||||||
'/bower_components/chainpad-netflux/chainpad-netflux.js',
|
'/common/sframe-common-outer.js'
|
||||||
'/bower_components/textpatcher/TextPatcher.js',
|
], function (nThen, ApiConfig, $, RequireConfig, SFCommonO) {
|
||||||
'/common/toolbar2.js',
|
var requireConfig = RequireConfig();
|
||||||
'json.sortify',
|
|
||||||
'/bower_components/chainpad-json-validator/json-ot.js',
|
|
||||||
'/common/cryptpad-common.js',
|
|
||||||
'/common/cryptget.js',
|
|
||||||
'/common/diffMarked.js',
|
|
||||||
|
|
||||||
'css!/bower_components/components-font-awesome/css/font-awesome.min.css',
|
// Loaded in load #2
|
||||||
'less!/customize/src/less/cryptpad.less'
|
nThen(function (waitFor) {
|
||||||
], function ($, Crypto, Realtime, TextPatcher, Toolbar, JSONSortify, JsonOT, Cryptpad,
|
$(waitFor());
|
||||||
Cryptget, DiffMd) {
|
}).nThen(function (waitFor) {
|
||||||
var Messages = Cryptpad.Messages;
|
var req = {
|
||||||
|
cfg: requireConfig,
|
||||||
var APP = window.APP = {
|
req: [ '/common/loading.js' ],
|
||||||
Cryptpad: Cryptpad,
|
pfx: window.location.origin
|
||||||
};
|
};
|
||||||
|
window.rc = requireConfig;
|
||||||
|
window.apiconf = ApiConfig;
|
||||||
|
$('#sbox-iframe').attr('src',
|
||||||
|
ApiConfig.httpSafeOrigin + '/code/inner.html?' + requireConfig.urlArgs +
|
||||||
|
'#' + encodeURIComponent(JSON.stringify(req)));
|
||||||
|
|
||||||
$(function () {
|
// This is a cheap trick to avoid loading sframe-channel in parallel with the
|
||||||
Cryptpad.addLoadingScreen();
|
// loading screen setup.
|
||||||
|
var done = waitFor();
|
||||||
var ifrw = APP.ifrw = $('#pad-iframe')[0].contentWindow;
|
var onMsg = function (msg) {
|
||||||
var stringify = function (obj) {
|
var data = JSON.parse(msg.data);
|
||||||
return JSONSortify(obj);
|
if (data.q !== 'READY') { return; }
|
||||||
|
window.removeEventListener('message', onMsg);
|
||||||
|
var _done = done;
|
||||||
|
done = function () { };
|
||||||
|
_done();
|
||||||
};
|
};
|
||||||
|
window.addEventListener('message', onMsg);
|
||||||
var toolbar;
|
}).nThen(function (/*waitFor*/) {
|
||||||
var editor;
|
SFCommonO.start();
|
||||||
|
|
||||||
var secret = Cryptpad.getSecrets();
|
|
||||||
var readOnly = secret.keys && !secret.keys.editKeyStr;
|
|
||||||
if (!secret.keys) {
|
|
||||||
secret.keys = secret.key;
|
|
||||||
}
|
|
||||||
|
|
||||||
var onConnectError = function () {
|
|
||||||
Cryptpad.errorLoadingScreen(Messages.websocketError);
|
|
||||||
};
|
|
||||||
|
|
||||||
var andThen = function (CMeditor) {
|
|
||||||
var $iframe = $('#pad-iframe').contents();
|
|
||||||
var $contentContainer = $iframe.find('#editorContainer');
|
|
||||||
var $previewContainer = $iframe.find('#previewContainer');
|
|
||||||
var $preview = $iframe.find('#preview');
|
|
||||||
$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');
|
|
||||||
window.open(href);
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
var CodeMirror = Cryptpad.createCodemirror(ifrw, Cryptpad, null, CMeditor);
|
|
||||||
$iframe.find('.CodeMirror').addClass('fullPage');
|
|
||||||
editor = CodeMirror.editor;
|
|
||||||
|
|
||||||
var setIndentation = APP.setIndentation = function (units, useTabs) {
|
|
||||||
if (typeof(units) !== 'number') { return; }
|
|
||||||
editor.setOption('indentUnit', units);
|
|
||||||
editor.setOption('tabSize', units);
|
|
||||||
editor.setOption('indentWithTabs', useTabs);
|
|
||||||
};
|
|
||||||
|
|
||||||
var indentKey = 'indentUnit';
|
|
||||||
var useTabsKey = 'indentWithTabs';
|
|
||||||
|
|
||||||
var proxy = Cryptpad.getProxy();
|
|
||||||
|
|
||||||
var updateIndentSettings = APP.updateIndentSettings = function () {
|
|
||||||
var indentUnit = proxy.settings[indentKey];
|
|
||||||
var useTabs = proxy.settings[useTabsKey];
|
|
||||||
setIndentation(
|
|
||||||
typeof(indentUnit) === 'number'? indentUnit: 2,
|
|
||||||
typeof(useTabs) === 'boolean'? useTabs: false);
|
|
||||||
};
|
|
||||||
|
|
||||||
proxy.on('change', ['settings', indentKey], updateIndentSettings);
|
|
||||||
proxy.on('change', ['settings', useTabsKey], updateIndentSettings);
|
|
||||||
|
|
||||||
var $bar = $('#pad-iframe')[0].contentWindow.$('#cme_toolbox');
|
|
||||||
|
|
||||||
var isHistoryMode = false;
|
|
||||||
|
|
||||||
var setEditable = APP.setEditable = function (bool) {
|
|
||||||
if (readOnly && bool) { return; }
|
|
||||||
editor.setOption('readOnly', !bool);
|
|
||||||
};
|
|
||||||
|
|
||||||
var Title;
|
|
||||||
var UserList;
|
|
||||||
var Metadata;
|
|
||||||
|
|
||||||
var config = {
|
|
||||||
initialState: '{}',
|
|
||||||
websocketURL: Cryptpad.getWebsocketURL(),
|
|
||||||
channel: secret.channel,
|
|
||||||
// our public key
|
|
||||||
validateKey: secret.keys.validateKey || undefined,
|
|
||||||
readOnly: readOnly,
|
|
||||||
crypto: Crypto.createEncryptor(secret.keys),
|
|
||||||
network: Cryptpad.getNetwork(),
|
|
||||||
transformFunction: JsonOT.validate,
|
|
||||||
};
|
|
||||||
|
|
||||||
var canonicalize = function (t) { return t.replace(/\r\n/g, '\n'); };
|
|
||||||
|
|
||||||
var setHistory = function (bool, update) {
|
|
||||||
isHistoryMode = bool;
|
|
||||||
setEditable(!bool);
|
|
||||||
if (!bool && update) {
|
|
||||||
config.onRemote();
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
var initializing = true;
|
|
||||||
|
|
||||||
var stringifyInner = function (textValue) {
|
|
||||||
var obj = {
|
|
||||||
content: textValue,
|
|
||||||
metadata: {
|
|
||||||
users: UserList.userData,
|
|
||||||
defaultTitle: Title.defaultTitle
|
|
||||||
}
|
|
||||||
};
|
|
||||||
if (!initializing) {
|
|
||||||
obj.metadata.title = Title.title;
|
|
||||||
}
|
|
||||||
// set mode too...
|
|
||||||
obj.highlightMode = CodeMirror.highlightMode;
|
|
||||||
|
|
||||||
// stringify the json and send it into chainpad
|
|
||||||
return stringify(obj);
|
|
||||||
};
|
|
||||||
|
|
||||||
var forceDrawPreview = function () {
|
|
||||||
try {
|
|
||||||
DiffMd.apply(DiffMd.render(editor.getValue()), $preview);
|
|
||||||
} catch (e) { console.error(e); }
|
|
||||||
};
|
|
||||||
|
|
||||||
var drawPreview = Cryptpad.throttle(function () {
|
|
||||||
if (CodeMirror.highlightMode !== 'markdown') { return; }
|
|
||||||
if (!$previewContainer.is(':visible')) { return; }
|
|
||||||
forceDrawPreview();
|
|
||||||
}, 150);
|
|
||||||
|
|
||||||
var onLocal = config.onLocal = function () {
|
|
||||||
if (initializing) { return; }
|
|
||||||
if (isHistoryMode) { return; }
|
|
||||||
if (readOnly) { return; }
|
|
||||||
|
|
||||||
editor.save();
|
|
||||||
|
|
||||||
drawPreview();
|
|
||||||
|
|
||||||
var textValue = canonicalize(CodeMirror.$textarea.val());
|
|
||||||
var shjson = stringifyInner(textValue);
|
|
||||||
|
|
||||||
APP.patchText(shjson);
|
|
||||||
|
|
||||||
if (APP.realtime.getUserDoc() !== shjson) {
|
|
||||||
console.error("realtime.getUserDoc() !== shjson");
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
var mediaTagModes = [
|
|
||||||
'markdown',
|
|
||||||
'html',
|
|
||||||
'htmlembedded',
|
|
||||||
'htmlmixed',
|
|
||||||
'index.html',
|
|
||||||
'php',
|
|
||||||
'velocity',
|
|
||||||
'xml',
|
|
||||||
];
|
|
||||||
|
|
||||||
var onModeChanged = function (mode) {
|
|
||||||
var $codeMirror = $iframe.find('.CodeMirror');
|
|
||||||
window.clearTimeout(APP.previewTo);
|
|
||||||
$codeMirror.addClass('transition');
|
|
||||||
APP.previewTo = window.setTimeout(function () {
|
|
||||||
$codeMirror.removeClass('transition');
|
|
||||||
}, 500);
|
|
||||||
if (mediaTagModes.indexOf(mode) !== -1) {
|
|
||||||
APP.$mediaTagButton.show();
|
|
||||||
} else { APP.$mediaTagButton.hide(); }
|
|
||||||
|
|
||||||
if (mode === "markdown") {
|
|
||||||
APP.$previewButton.show();
|
|
||||||
Cryptpad.getPadAttribute('previewMode', function (e, data) {
|
|
||||||
if (e) { return void console.error(e); }
|
|
||||||
if (data !== false) {
|
|
||||||
$previewContainer.show();
|
|
||||||
APP.$previewButton.addClass('active');
|
|
||||||
$codeMirror.removeClass('fullPage');
|
|
||||||
}
|
|
||||||
});
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
APP.$previewButton.hide();
|
|
||||||
$previewContainer.hide();
|
|
||||||
APP.$previewButton.removeClass('active');
|
|
||||||
$codeMirror.addClass('fullPage');
|
|
||||||
if (typeof(APP.updateIndentSettings) === 'function') {
|
|
||||||
APP.updateIndentSettings();
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
config.onInit = function (info) {
|
|
||||||
UserList = Cryptpad.createUserList(info, config.onLocal, Cryptget, Cryptpad);
|
|
||||||
|
|
||||||
var titleCfg = { getHeadingText: CodeMirror.getHeadingText };
|
|
||||||
Title = Cryptpad.createTitle(titleCfg, config.onLocal, Cryptpad);
|
|
||||||
|
|
||||||
Metadata = Cryptpad.createMetadata(UserList, Title, null, Cryptpad);
|
|
||||||
|
|
||||||
var configTb = {
|
|
||||||
displayed: ['title', 'useradmin', 'spinner', 'lag', 'state', 'share', 'userlist', 'newpad', 'limit', 'upgrade'],
|
|
||||||
userList: UserList.getToolbarConfig(),
|
|
||||||
share: {
|
|
||||||
secret: secret,
|
|
||||||
channel: info.channel
|
|
||||||
},
|
|
||||||
title: Title.getTitleConfig(),
|
|
||||||
common: Cryptpad,
|
|
||||||
readOnly: readOnly,
|
|
||||||
ifrw: ifrw,
|
|
||||||
realtime: info.realtime,
|
|
||||||
network: info.network,
|
|
||||||
$container: $bar,
|
|
||||||
$contentContainer: $contentContainer
|
|
||||||
};
|
|
||||||
toolbar = APP.toolbar = Toolbar.create(configTb);
|
|
||||||
|
|
||||||
Title.setToolbar(toolbar);
|
|
||||||
CodeMirror.init(config.onLocal, Title, toolbar);
|
|
||||||
|
|
||||||
var $rightside = toolbar.$rightside;
|
|
||||||
var $drawer = toolbar.$drawer;
|
|
||||||
|
|
||||||
var editHash;
|
|
||||||
if (!readOnly) {
|
|
||||||
editHash = Cryptpad.getEditHashFromKeys(info.channel, secret.keys);
|
|
||||||
}
|
|
||||||
|
|
||||||
/* add a history button */
|
|
||||||
var histConfig = {
|
|
||||||
onLocal: config.onLocal,
|
|
||||||
onRemote: config.onRemote,
|
|
||||||
setHistory: setHistory,
|
|
||||||
applyVal: function (val) {
|
|
||||||
var remoteDoc = JSON.parse(val || '{}').content;
|
|
||||||
editor.setValue(remoteDoc || '');
|
|
||||||
editor.save();
|
|
||||||
},
|
|
||||||
$toolbar: $bar
|
|
||||||
};
|
|
||||||
var $hist = Cryptpad.createButton('history', true, {histConfig: histConfig});
|
|
||||||
$drawer.append($hist);
|
|
||||||
|
|
||||||
/* save as template */
|
|
||||||
if (!Cryptpad.isTemplate(window.location.href)) {
|
|
||||||
var templateObj = {
|
|
||||||
rt: info.realtime,
|
|
||||||
Crypt: Cryptget,
|
|
||||||
getTitle: Title.getTitle
|
|
||||||
};
|
|
||||||
var $templateButton = Cryptpad.createButton('template', true, templateObj);
|
|
||||||
$rightside.append($templateButton);
|
|
||||||
}
|
|
||||||
|
|
||||||
/* add an export button */
|
|
||||||
var $export = Cryptpad.createButton('export', true, {}, CodeMirror.exportText);
|
|
||||||
$drawer.append($export);
|
|
||||||
|
|
||||||
if (!readOnly) {
|
|
||||||
/* add an import button */
|
|
||||||
var $import = Cryptpad.createButton('import', true, {}, CodeMirror.importText);
|
|
||||||
$drawer.append($import);
|
|
||||||
}
|
|
||||||
|
|
||||||
/* add a forget button */
|
|
||||||
var forgetCb = function (err) {
|
|
||||||
if (err) { return; }
|
|
||||||
setEditable(false);
|
|
||||||
};
|
|
||||||
var $forgetPad = Cryptpad.createButton('forget', true, {}, forgetCb);
|
|
||||||
$rightside.append($forgetPad);
|
|
||||||
|
|
||||||
var fileDialogCfg = {
|
|
||||||
$body: $iframe.find('body'),
|
|
||||||
onSelect: function (href) {
|
|
||||||
var parsed = Cryptpad.parsePadUrl(href);
|
|
||||||
var hexFileName = Cryptpad.base64ToHex(parsed.hashData.channel);
|
|
||||||
var src = '/blob/' + hexFileName.slice(0,2) + '/' + hexFileName;
|
|
||||||
var mt = '<media-tag src="' + src + '" data-crypto-key="cryptpad:' + parsed.hashData.key + '"></media-tag>';
|
|
||||||
editor.replaceSelection(mt);
|
|
||||||
},
|
|
||||||
data: APP
|
|
||||||
};
|
|
||||||
APP.$mediaTagButton = $('<button>', {
|
|
||||||
title: Messages.filePickerButton,
|
|
||||||
'class': 'rightside-button fa fa-picture-o',
|
|
||||||
style: 'font-size: 17px'
|
|
||||||
}).click(function () {
|
|
||||||
Cryptpad.createFileDialog(fileDialogCfg);
|
|
||||||
}).appendTo($rightside);
|
|
||||||
|
|
||||||
var $previewButton = APP.$previewButton = Cryptpad.createButton(null, true);
|
|
||||||
$previewButton.removeClass('fa-question').addClass('fa-eye');
|
|
||||||
$previewButton.attr('title', Messages.previewButtonTitle);
|
|
||||||
$previewButton.click(function () {
|
|
||||||
var $codeMirror = $iframe.find('.CodeMirror');
|
|
||||||
window.clearTimeout(APP.previewTo);
|
|
||||||
$codeMirror.addClass('transition');
|
|
||||||
APP.previewTo = window.setTimeout(function () {
|
|
||||||
$codeMirror.removeClass('transition');
|
|
||||||
}, 500);
|
|
||||||
if (CodeMirror.highlightMode !== 'markdown') {
|
|
||||||
$previewContainer.show();
|
|
||||||
}
|
|
||||||
$previewContainer.toggle();
|
|
||||||
if ($previewContainer.is(':visible')) {
|
|
||||||
forceDrawPreview();
|
|
||||||
$codeMirror.removeClass('fullPage');
|
|
||||||
Cryptpad.setPadAttribute('previewMode', true, function (e) {
|
|
||||||
if (e) { return console.log(e); }
|
|
||||||
});
|
|
||||||
$previewButton.addClass('active');
|
|
||||||
} else {
|
|
||||||
$codeMirror.addClass('fullPage');
|
|
||||||
$previewButton.removeClass('active');
|
|
||||||
Cryptpad.setPadAttribute('previewMode', false, function (e) {
|
|
||||||
if (e) { return console.log(e); }
|
|
||||||
});
|
|
||||||
}
|
|
||||||
});
|
|
||||||
$rightside.append($previewButton);
|
|
||||||
|
|
||||||
if (!readOnly) {
|
|
||||||
CodeMirror.configureTheme(function () {
|
|
||||||
CodeMirror.configureLanguage(null, onModeChanged);
|
|
||||||
});
|
|
||||||
}
|
|
||||||
else {
|
|
||||||
CodeMirror.configureTheme();
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
// set the hash
|
|
||||||
if (!readOnly) { Cryptpad.replaceHash(editHash); }
|
|
||||||
};
|
|
||||||
|
|
||||||
config.onReady = function (info) {
|
|
||||||
if (APP.realtime !== info.realtime) {
|
|
||||||
var realtime = APP.realtime = info.realtime;
|
|
||||||
APP.patchText = TextPatcher.create({
|
|
||||||
realtime: realtime,
|
|
||||||
//logging: true
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
var userDoc = APP.realtime.getUserDoc();
|
|
||||||
|
|
||||||
var isNew = false;
|
|
||||||
if (userDoc === "" || userDoc === "{}") { isNew = true; }
|
|
||||||
|
|
||||||
var newDoc = "";
|
|
||||||
if(userDoc !== "") {
|
|
||||||
var hjson = JSON.parse(userDoc);
|
|
||||||
|
|
||||||
if (typeof (hjson) !== 'object' || Array.isArray(hjson) ||
|
|
||||||
(typeof(hjson.type) !== 'undefined' && hjson.type !== 'code')) {
|
|
||||||
var errorText = Messages.typeError;
|
|
||||||
Cryptpad.errorLoadingScreen(errorText);
|
|
||||||
throw new Error(errorText);
|
|
||||||
}
|
|
||||||
|
|
||||||
newDoc = hjson.content;
|
|
||||||
|
|
||||||
if (hjson.highlightMode) {
|
|
||||||
CodeMirror.setMode(hjson.highlightMode, onModeChanged);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!CodeMirror.highlightMode) {
|
|
||||||
CodeMirror.setMode('markdown', onModeChanged);
|
|
||||||
console.log("%s => %s", CodeMirror.highlightMode, CodeMirror.$language.val());
|
|
||||||
}
|
|
||||||
|
|
||||||
// Update the user list (metadata) from the hyperjson
|
|
||||||
Metadata.update(userDoc);
|
|
||||||
|
|
||||||
if (newDoc) {
|
|
||||||
editor.setValue(newDoc);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (Cryptpad.initialName && Title.isDefaultTitle()) {
|
|
||||||
Title.updateTitle(Cryptpad.initialName);
|
|
||||||
}
|
|
||||||
|
|
||||||
Cryptpad.getPadAttribute('previewMode', function (e, data) {
|
|
||||||
if (e) { return void console.error(e); }
|
|
||||||
if (data === false && APP.$previewButton) {
|
|
||||||
APP.$previewButton.click();
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
/*
|
|
||||||
// add the splitter
|
|
||||||
if (!$iframe.has('.cp-splitter').length) {
|
|
||||||
var $preview = $iframe.find('#previewContainer');
|
|
||||||
var splitter = $('<div>', {
|
|
||||||
'class': 'cp-splitter'
|
|
||||||
}).appendTo($preview);
|
|
||||||
|
|
||||||
$preview.on('scroll', function() {
|
|
||||||
splitter.css('top', $preview.scrollTop() + 'px');
|
|
||||||
});
|
|
||||||
|
|
||||||
var $target = $iframe.find('.CodeMirror');
|
|
||||||
|
|
||||||
splitter.on('mousedown', function (e) {
|
|
||||||
e.preventDefault();
|
|
||||||
var x = e.pageX;
|
|
||||||
var w = $target.width();
|
|
||||||
|
|
||||||
$iframe.on('mouseup mousemove', function handler(evt) {
|
|
||||||
if (evt.type === 'mouseup') {
|
|
||||||
$iframe.off('mouseup mousemove', handler);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
$target.css('width', (w - x + evt.pageX) + 'px');
|
|
||||||
});
|
|
||||||
});
|
|
||||||
}
|
|
||||||
*/
|
|
||||||
|
|
||||||
Cryptpad.removeLoadingScreen();
|
|
||||||
setEditable(true);
|
|
||||||
initializing = false;
|
|
||||||
|
|
||||||
onLocal(); // push local state to avoid parse errors later.
|
|
||||||
|
|
||||||
if (readOnly) {
|
|
||||||
config.onRemote();
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
UserList.getLastName(toolbar.$userNameButton, isNew);
|
|
||||||
var fmConfig = {
|
|
||||||
dropArea: $iframe.find('.CodeMirror'),
|
|
||||||
body: $iframe.find('body'),
|
|
||||||
onUploaded: function (ev, data) {
|
|
||||||
//var cursor = editor.getCursor();
|
|
||||||
//var cleanName = data.name.replace(/[\[\]]/g, '');
|
|
||||||
//var text = '';
|
|
||||||
var parsed = Cryptpad.parsePadUrl(data.url);
|
|
||||||
var hexFileName = Cryptpad.base64ToHex(parsed.hashData.channel);
|
|
||||||
var src = '/blob/' + hexFileName.slice(0,2) + '/' + hexFileName;
|
|
||||||
var mt = '<media-tag src="' + src + '" data-crypto-key="cryptpad:' + parsed.hashData.key + '"></media-tag>';
|
|
||||||
editor.replaceSelection(mt);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
APP.FM = Cryptpad.createFileManager(fmConfig);
|
|
||||||
};
|
|
||||||
|
|
||||||
config.onRemote = function () {
|
|
||||||
if (initializing) { return; }
|
|
||||||
if (isHistoryMode) { return; }
|
|
||||||
|
|
||||||
var oldDoc = canonicalize(CodeMirror.$textarea.val());
|
|
||||||
var shjson = APP.realtime.getUserDoc();
|
|
||||||
|
|
||||||
// Update the user list (metadata) from the hyperjson
|
|
||||||
Metadata.update(shjson);
|
|
||||||
|
|
||||||
var hjson = JSON.parse(shjson);
|
|
||||||
var remoteDoc = hjson.content;
|
|
||||||
|
|
||||||
var highlightMode = hjson.highlightMode;
|
|
||||||
if (highlightMode && highlightMode !== APP.highlightMode) {
|
|
||||||
CodeMirror.setMode(highlightMode, onModeChanged);
|
|
||||||
}
|
|
||||||
|
|
||||||
CodeMirror.setValueAndCursor(oldDoc, remoteDoc, TextPatcher);
|
|
||||||
drawPreview();
|
|
||||||
|
|
||||||
if (!readOnly) {
|
|
||||||
var textValue = canonicalize(CodeMirror.$textarea.val());
|
|
||||||
var shjson2 = stringifyInner(textValue);
|
|
||||||
if (shjson2 !== shjson) {
|
|
||||||
console.error("shjson2 !== shjson");
|
|
||||||
TextPatcher.log(shjson, TextPatcher.diff(shjson, shjson2));
|
|
||||||
APP.patchText(shjson2);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if (oldDoc !== remoteDoc) { Cryptpad.notify(); }
|
|
||||||
};
|
|
||||||
|
|
||||||
config.onAbort = function () {
|
|
||||||
// inform of network disconnect
|
|
||||||
setEditable(false);
|
|
||||||
toolbar.failed();
|
|
||||||
Cryptpad.alert(Messages.common_connectionLost, undefined, true);
|
|
||||||
};
|
|
||||||
|
|
||||||
config.onConnectionChange = function (info) {
|
|
||||||
setEditable(info.state);
|
|
||||||
toolbar.failed();
|
|
||||||
if (info.state) {
|
|
||||||
initializing = true;
|
|
||||||
toolbar.reconnecting(info.myId);
|
|
||||||
Cryptpad.findOKButton().click();
|
|
||||||
} else {
|
|
||||||
Cryptpad.alert(Messages.common_connectionLost, undefined, true);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
config.onError = onConnectError;
|
|
||||||
|
|
||||||
APP.realtime = Realtime.start(config);
|
|
||||||
|
|
||||||
editor.on('change', onLocal);
|
|
||||||
|
|
||||||
Cryptpad.onLogout(function () { setEditable(false); });
|
|
||||||
};
|
|
||||||
|
|
||||||
var interval = 100;
|
|
||||||
var second = function (CM) {
|
|
||||||
Cryptpad.ready(function () {
|
|
||||||
andThen(CM);
|
|
||||||
Cryptpad.reportAppUsage();
|
|
||||||
});
|
|
||||||
Cryptpad.onError(function (info) {
|
|
||||||
if (info && info.type === "store") {
|
|
||||||
onConnectError();
|
|
||||||
}
|
|
||||||
});
|
|
||||||
};
|
|
||||||
|
|
||||||
var first = function () {
|
|
||||||
if (ifrw.CodeMirror) {
|
|
||||||
second(ifrw.CodeMirror);
|
|
||||||
} else {
|
|
||||||
console.log("CodeMirror was not defined. Trying again in %sms", interval);
|
|
||||||
setTimeout(first, interval);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
first();
|
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|||||||
@ -1,39 +0,0 @@
|
|||||||
<!DOCTYPE html>
|
|
||||||
<html>
|
|
||||||
<head>
|
|
||||||
<title>CryptPad</title>
|
|
||||||
<meta content="text/html; charset=utf-8" http-equiv="content-type"/>
|
|
||||||
<meta name="viewport" content="width=device-width, initial-scale=1.0">
|
|
||||||
<meta name="referrer" content="no-referrer" />
|
|
||||||
<script async data-bootload="main.js" data-main="/common/boot.js?ver=1.0" src="/bower_components/requirejs/require.js?ver=2.3.5"></script>
|
|
||||||
<style>
|
|
||||||
html, body {
|
|
||||||
margin: 0px;
|
|
||||||
padding: 0px;
|
|
||||||
}
|
|
||||||
#sbox-iframe {
|
|
||||||
position:fixed;
|
|
||||||
top:0px;
|
|
||||||
left:0px;
|
|
||||||
bottom:0px;
|
|
||||||
right:0px;
|
|
||||||
width:100%;
|
|
||||||
height:100%;
|
|
||||||
border:none;
|
|
||||||
margin:0;
|
|
||||||
padding:0;
|
|
||||||
overflow:hidden;
|
|
||||||
}
|
|
||||||
#sbox-filePicker-iframe {
|
|
||||||
position: fixed;
|
|
||||||
top:0; left:0;
|
|
||||||
bottom:0; right:0;
|
|
||||||
width:100%;
|
|
||||||
height: 100%;
|
|
||||||
border: 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
</style>
|
|
||||||
</head>
|
|
||||||
<body>
|
|
||||||
<iframe id="sbox-iframe">
|
|
||||||
@ -1,18 +0,0 @@
|
|||||||
<!DOCTYPE html>
|
|
||||||
<html class="cp-app-noscroll">
|
|
||||||
<head>
|
|
||||||
<meta content="text/html; charset=utf-8" http-equiv="content-type"/>
|
|
||||||
<script async data-bootload="/code2/inner.js" data-main="/common/sframe-boot.js?ver=1.1" src="/bower_components/requirejs/require.js?ver=2.3.5"></script>
|
|
||||||
<style> .loading-hidden { display: none; } </style>
|
|
||||||
</head>
|
|
||||||
<body class="cp-app-code">
|
|
||||||
<div id="cme_toolbox" class="cp-toolbar-container"></div>
|
|
||||||
<div id="cp-app-code-editor">
|
|
||||||
<textarea id="editor1" name="editor1"></textarea>
|
|
||||||
<div id="cp-app-code-preview">
|
|
||||||
<div id="cp-app-code-preview-content"></div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</body>
|
|
||||||
</html>
|
|
||||||
|
|
||||||
@ -1,632 +0,0 @@
|
|||||||
define([
|
|
||||||
'jquery',
|
|
||||||
'/bower_components/chainpad-crypto/crypto.js',
|
|
||||||
'/bower_components/textpatcher/TextPatcher.js',
|
|
||||||
'/common/toolbar3.js',
|
|
||||||
'json.sortify',
|
|
||||||
'/bower_components/chainpad-json-validator/json-ot.js',
|
|
||||||
'/common/cryptpad-common.js',
|
|
||||||
'/common/cryptget.js',
|
|
||||||
'/common/diffMarked.js',
|
|
||||||
'/bower_components/nthen/index.js',
|
|
||||||
'/common/sframe-common.js',
|
|
||||||
'/api/config',
|
|
||||||
'/common/common-realtime.js',
|
|
||||||
|
|
||||||
'cm/lib/codemirror',
|
|
||||||
|
|
||||||
'css!/bower_components/bootstrap/dist/css/bootstrap.min.css',
|
|
||||||
'css!/bower_components/components-font-awesome/css/font-awesome.min.css',
|
|
||||||
'less!/customize/src/less2/main.less',
|
|
||||||
|
|
||||||
'css!cm/lib/codemirror.css',
|
|
||||||
'css!cm/addon/dialog/dialog.css',
|
|
||||||
'css!cm/addon/fold/foldgutter.css',
|
|
||||||
|
|
||||||
'cm/mode/markdown/markdown',
|
|
||||||
'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',
|
|
||||||
|
|
||||||
], function (
|
|
||||||
$,
|
|
||||||
Crypto,
|
|
||||||
TextPatcher,
|
|
||||||
Toolbar,
|
|
||||||
JSONSortify,
|
|
||||||
JsonOT,
|
|
||||||
Cryptpad,
|
|
||||||
Cryptget,
|
|
||||||
DiffMd,
|
|
||||||
nThen,
|
|
||||||
SFCommon,
|
|
||||||
ApiConfig,
|
|
||||||
CommonRealtime,
|
|
||||||
CMeditor)
|
|
||||||
{
|
|
||||||
window.CodeMirror = CMeditor;
|
|
||||||
var Messages = Cryptpad.Messages;
|
|
||||||
|
|
||||||
var APP = window.APP = {
|
|
||||||
Cryptpad: Cryptpad,
|
|
||||||
};
|
|
||||||
|
|
||||||
var stringify = function (obj) {
|
|
||||||
return JSONSortify(obj);
|
|
||||||
};
|
|
||||||
|
|
||||||
var toolbar;
|
|
||||||
|
|
||||||
var onConnectError = function () {
|
|
||||||
Cryptpad.errorLoadingScreen(Messages.websocketError);
|
|
||||||
};
|
|
||||||
|
|
||||||
var andThen = function (editor, CodeMirror, common) {
|
|
||||||
var readOnly = false;
|
|
||||||
var cpNfInner;
|
|
||||||
var metadataMgr;
|
|
||||||
var $bar = $('#cme_toolbox');
|
|
||||||
|
|
||||||
var isHistoryMode = false;
|
|
||||||
|
|
||||||
var $contentContainer = $('#cp-app-code-editor');
|
|
||||||
var $previewContainer = $('#cp-app-code-preview');
|
|
||||||
var $preview = $('#cp-app-code-preview-content');
|
|
||||||
$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');
|
|
||||||
window.open(href);
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
var setIndentation = APP.setIndentation = function (units, useTabs) {
|
|
||||||
if (typeof(units) !== 'number') { return; }
|
|
||||||
editor.setOption('indentUnit', units);
|
|
||||||
editor.setOption('tabSize', units);
|
|
||||||
editor.setOption('indentWithTabs', useTabs);
|
|
||||||
};
|
|
||||||
|
|
||||||
var indentKey = 'indentUnit';
|
|
||||||
var useTabsKey = 'indentWithTabs';
|
|
||||||
var updateIndentSettings = function () {
|
|
||||||
if (!metadataMgr) { return; }
|
|
||||||
var data = metadataMgr.getPrivateData().settings;
|
|
||||||
var indentUnit = data[indentKey];
|
|
||||||
var useTabs = data[useTabsKey];
|
|
||||||
setIndentation(
|
|
||||||
typeof(indentUnit) === 'number'? indentUnit: 2,
|
|
||||||
typeof(useTabs) === 'boolean'? useTabs: false);
|
|
||||||
};
|
|
||||||
|
|
||||||
var setEditable = APP.setEditable = function (bool) {
|
|
||||||
if (readOnly && bool) { return; }
|
|
||||||
editor.setOption('readOnly', !bool);
|
|
||||||
};
|
|
||||||
|
|
||||||
var Title;
|
|
||||||
|
|
||||||
var config = {
|
|
||||||
readOnly: readOnly,
|
|
||||||
transformFunction: JsonOT.validate,
|
|
||||||
// cryptpad debug logging (default is 1)
|
|
||||||
// logLevel: 0,
|
|
||||||
validateContent: function (content) {
|
|
||||||
try {
|
|
||||||
JSON.parse(content);
|
|
||||||
return true;
|
|
||||||
} catch (e) {
|
|
||||||
console.log("Failed to parse, rejecting patch");
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
var canonicalize = function (t) { return t.replace(/\r\n/g, '\n'); };
|
|
||||||
|
|
||||||
var setHistory = function (bool, update) {
|
|
||||||
isHistoryMode = bool;
|
|
||||||
setEditable(!bool);
|
|
||||||
if (!bool && update) {
|
|
||||||
config.onRemote();
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
CommonRealtime.onInfiniteSpinner(function () { setEditable(false); });
|
|
||||||
|
|
||||||
setEditable(false);
|
|
||||||
var initializing = true;
|
|
||||||
|
|
||||||
var stringifyInner = function (textValue) {
|
|
||||||
var obj = {
|
|
||||||
content: textValue,
|
|
||||||
metadata: metadataMgr.getMetadataLazy()
|
|
||||||
};
|
|
||||||
/* metadata: {
|
|
||||||
users: UserList.userData,
|
|
||||||
defaultTitle: Title.defaultTitle
|
|
||||||
}
|
|
||||||
};
|
|
||||||
if (!initializing) {
|
|
||||||
obj.metadata.title = Title.title;
|
|
||||||
}*/
|
|
||||||
// set mode too...
|
|
||||||
obj.highlightMode = CodeMirror.highlightMode;
|
|
||||||
|
|
||||||
// stringify the json and send it into chainpad
|
|
||||||
return stringify(obj);
|
|
||||||
};
|
|
||||||
|
|
||||||
var forceDrawPreview = function () {
|
|
||||||
try {
|
|
||||||
DiffMd.apply(DiffMd.render(editor.getValue()), $preview);
|
|
||||||
} catch (e) { console.error(e); }
|
|
||||||
};
|
|
||||||
|
|
||||||
var drawPreview = Cryptpad.throttle(function () {
|
|
||||||
if (CodeMirror.highlightMode !== 'markdown') { return; }
|
|
||||||
if (!$previewContainer.is(':visible')) { return; }
|
|
||||||
forceDrawPreview();
|
|
||||||
}, 150);
|
|
||||||
|
|
||||||
var onLocal = config.onLocal = function () {
|
|
||||||
if (initializing) { return; }
|
|
||||||
if (isHistoryMode) { return; }
|
|
||||||
if (readOnly) { return; }
|
|
||||||
|
|
||||||
editor.save();
|
|
||||||
|
|
||||||
drawPreview();
|
|
||||||
|
|
||||||
var textValue = canonicalize(CodeMirror.$textarea.val());
|
|
||||||
var shjson = stringifyInner(textValue);
|
|
||||||
|
|
||||||
APP.patchText(shjson);
|
|
||||||
|
|
||||||
if (APP.realtime.getUserDoc() !== shjson) {
|
|
||||||
console.error("realtime.getUserDoc() !== shjson");
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
var mediaTagModes = [
|
|
||||||
'markdown',
|
|
||||||
'html',
|
|
||||||
'htmlembedded',
|
|
||||||
'htmlmixed',
|
|
||||||
'index.html',
|
|
||||||
'php',
|
|
||||||
'velocity',
|
|
||||||
'xml',
|
|
||||||
];
|
|
||||||
|
|
||||||
var onModeChanged = function (mode) {
|
|
||||||
var $codeMirror = $('.CodeMirror');
|
|
||||||
window.clearTimeout(APP.previewTo);
|
|
||||||
$codeMirror.addClass('transition');
|
|
||||||
APP.previewTo = window.setTimeout(function () {
|
|
||||||
$codeMirror.removeClass('transition');
|
|
||||||
}, 500);
|
|
||||||
if (mediaTagModes.indexOf(mode) !== -1) {
|
|
||||||
APP.$mediaTagButton.show();
|
|
||||||
} else { APP.$mediaTagButton.hide(); }
|
|
||||||
|
|
||||||
if (mode === "markdown") {
|
|
||||||
APP.$previewButton.show();
|
|
||||||
common.getPadAttribute('previewMode', function (e, data) {
|
|
||||||
if (e) { return void console.error(e); }
|
|
||||||
if (data !== false) {
|
|
||||||
$previewContainer.show();
|
|
||||||
APP.$previewButton.addClass('active');
|
|
||||||
$codeMirror.removeClass('fullPage');
|
|
||||||
}
|
|
||||||
});
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
APP.$previewButton.hide();
|
|
||||||
$previewContainer.hide();
|
|
||||||
APP.$previewButton.removeClass('active');
|
|
||||||
$codeMirror.addClass('fullPage');
|
|
||||||
};
|
|
||||||
|
|
||||||
config.onInit = function (info) {
|
|
||||||
metadataMgr.onChangeLazy(updateIndentSettings);
|
|
||||||
updateIndentSettings();
|
|
||||||
|
|
||||||
readOnly = metadataMgr.getPrivateData().readOnly;
|
|
||||||
|
|
||||||
var titleCfg = { getHeadingText: CodeMirror.getHeadingText };
|
|
||||||
Title = common.createTitle(titleCfg, config.onLocal, common, metadataMgr);
|
|
||||||
|
|
||||||
var configTb = {
|
|
||||||
displayed: ['title', 'useradmin', 'spinner', 'share', 'userlist', 'newpad', 'limit'],
|
|
||||||
title: Title.getTitleConfig(),
|
|
||||||
metadataMgr: metadataMgr,
|
|
||||||
readOnly: readOnly,
|
|
||||||
ifrw: window,
|
|
||||||
realtime: info.realtime,
|
|
||||||
common: Cryptpad,
|
|
||||||
sfCommon: common,
|
|
||||||
$container: $bar,
|
|
||||||
$contentContainer: $contentContainer
|
|
||||||
};
|
|
||||||
toolbar = APP.toolbar = Toolbar.create(configTb);
|
|
||||||
Title.setToolbar(toolbar);
|
|
||||||
CodeMirror.init(config.onLocal, Title, toolbar);
|
|
||||||
|
|
||||||
var $rightside = toolbar.$rightside;
|
|
||||||
var $drawer = toolbar.$drawer;
|
|
||||||
|
|
||||||
/* add a history button */
|
|
||||||
var histConfig = {
|
|
||||||
onLocal: config.onLocal,
|
|
||||||
onRemote: config.onRemote,
|
|
||||||
setHistory: setHistory,
|
|
||||||
applyVal: function (val) {
|
|
||||||
var remoteDoc = JSON.parse(val || '{}').content;
|
|
||||||
editor.setValue(remoteDoc || '');
|
|
||||||
editor.save();
|
|
||||||
},
|
|
||||||
$toolbar: $bar
|
|
||||||
};
|
|
||||||
var $hist = common.createButton('history', true, {histConfig: histConfig});
|
|
||||||
$drawer.append($hist);
|
|
||||||
|
|
||||||
/* save as template */
|
|
||||||
if (!metadataMgr.getPrivateData().isTemplate) {
|
|
||||||
var templateObj = {
|
|
||||||
rt: info.realtime,
|
|
||||||
getTitle: function () { return metadataMgr.getMetadata().title; }
|
|
||||||
};
|
|
||||||
var $templateButton = common.createButton('template', true, templateObj);
|
|
||||||
$rightside.append($templateButton);
|
|
||||||
}
|
|
||||||
|
|
||||||
/* add an export button */
|
|
||||||
var $export = common.createButton('export', true, {}, CodeMirror.exportText);
|
|
||||||
$drawer.append($export);
|
|
||||||
|
|
||||||
if (!readOnly) {
|
|
||||||
/* add an import button */
|
|
||||||
var $import = common.createButton('import', true, {}, CodeMirror.importText);
|
|
||||||
$drawer.append($import);
|
|
||||||
}
|
|
||||||
|
|
||||||
/* add a forget button */
|
|
||||||
var forgetCb = function (err) {
|
|
||||||
if (err) { return; }
|
|
||||||
setEditable(false);
|
|
||||||
};
|
|
||||||
var $forgetPad = common.createButton('forget', true, {}, forgetCb);
|
|
||||||
$rightside.append($forgetPad);
|
|
||||||
|
|
||||||
var $previewButton = APP.$previewButton = common.createButton(null, true);
|
|
||||||
$previewButton.removeClass('fa-question').addClass('fa-eye');
|
|
||||||
$previewButton.attr('title', Messages.previewButtonTitle);
|
|
||||||
$previewButton.click(function () {
|
|
||||||
var $codeMirror = $('.CodeMirror');
|
|
||||||
window.clearTimeout(APP.previewTo);
|
|
||||||
$codeMirror.addClass('transition');
|
|
||||||
APP.previewTo = window.setTimeout(function () {
|
|
||||||
$codeMirror.removeClass('transition');
|
|
||||||
}, 500);
|
|
||||||
if (CodeMirror.highlightMode !== 'markdown') {
|
|
||||||
$previewContainer.show();
|
|
||||||
}
|
|
||||||
$previewContainer.toggle();
|
|
||||||
if ($previewContainer.is(':visible')) {
|
|
||||||
forceDrawPreview();
|
|
||||||
$codeMirror.removeClass('fullPage');
|
|
||||||
$previewButton.addClass('active');
|
|
||||||
common.setPadAttribute('previewMode', true, function (e) {
|
|
||||||
if (e) { return console.log(e); }
|
|
||||||
});
|
|
||||||
} else {
|
|
||||||
$codeMirror.addClass('fullPage');
|
|
||||||
$previewButton.removeClass('active');
|
|
||||||
common.setPadAttribute('previewMode', false, function (e) {
|
|
||||||
if (e) { return console.log(e); }
|
|
||||||
});
|
|
||||||
}
|
|
||||||
});
|
|
||||||
$rightside.append($previewButton);
|
|
||||||
|
|
||||||
if (!readOnly) {
|
|
||||||
CodeMirror.configureTheme(function () {
|
|
||||||
CodeMirror.configureLanguage(null, onModeChanged);
|
|
||||||
});
|
|
||||||
}
|
|
||||||
else {
|
|
||||||
CodeMirror.configureTheme();
|
|
||||||
}
|
|
||||||
|
|
||||||
var fileDialogCfg = {
|
|
||||||
onSelect: function (data) {
|
|
||||||
var mt = '<media-tag src="' + data.src + '" data-crypto-key="cryptpad:' + data.key + '"></media-tag>';
|
|
||||||
editor.replaceSelection(mt);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
common.initFilePicker(common, fileDialogCfg);
|
|
||||||
APP.$mediaTagButton = $('<button>', {
|
|
||||||
title: Messages.filePickerButton,
|
|
||||||
'class': 'cp-toolbar-rightside-button fa fa-picture-o',
|
|
||||||
style: 'font-size: 17px'
|
|
||||||
}).click(function () {
|
|
||||||
common.openFilePicker(common);
|
|
||||||
}).appendTo($rightside);
|
|
||||||
|
|
||||||
};
|
|
||||||
|
|
||||||
config.onReady = function (info) {
|
|
||||||
console.log('onready');
|
|
||||||
if (APP.realtime !== info.realtime) {
|
|
||||||
var realtime = APP.realtime = info.realtime;
|
|
||||||
APP.patchText = TextPatcher.create({
|
|
||||||
realtime: realtime,
|
|
||||||
//logging: true
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
var userDoc = APP.realtime.getUserDoc();
|
|
||||||
|
|
||||||
var isNew = false;
|
|
||||||
if (userDoc === "" || userDoc === "{}") { isNew = true; }
|
|
||||||
|
|
||||||
var newDoc = "";
|
|
||||||
if (userDoc !== "") {
|
|
||||||
var hjson = JSON.parse(userDoc);
|
|
||||||
|
|
||||||
if (hjson && hjson.metadata) {
|
|
||||||
metadataMgr.updateMetadata(hjson.metadata);
|
|
||||||
}
|
|
||||||
if (typeof (hjson) !== 'object' || Array.isArray(hjson) ||
|
|
||||||
(typeof(hjson.type) !== 'undefined' && hjson.type !== 'code')) {
|
|
||||||
var errorText = Messages.typeError;
|
|
||||||
Cryptpad.errorLoadingScreen(errorText);
|
|
||||||
throw new Error(errorText);
|
|
||||||
}
|
|
||||||
|
|
||||||
newDoc = hjson.content;
|
|
||||||
|
|
||||||
if (hjson.highlightMode) {
|
|
||||||
CodeMirror.setMode(hjson.highlightMode, onModeChanged);
|
|
||||||
}
|
|
||||||
} else {
|
|
||||||
Title.updateTitle(Cryptpad.initialName || Title.defaultTitle);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!CodeMirror.highlightMode) {
|
|
||||||
CodeMirror.setMode('markdown', onModeChanged);
|
|
||||||
//console.log("%s => %s", CodeMirror.highlightMode, CodeMirror.$language.val());
|
|
||||||
}
|
|
||||||
|
|
||||||
// Update the user list (metadata) from the hyperjson
|
|
||||||
//Metadata.update(userDoc);
|
|
||||||
|
|
||||||
if (newDoc) {
|
|
||||||
editor.setValue(newDoc);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (Cryptpad.initialName && Title.isDefaultTitle()) {
|
|
||||||
Title.updateTitle(Cryptpad.initialName);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
common.getPadAttribute('previewMode', function (e, data) {
|
|
||||||
if (e) { return void console.error(e); }
|
|
||||||
if (data === false && APP.$previewButton) {
|
|
||||||
APP.$previewButton.click();
|
|
||||||
}
|
|
||||||
});
|
|
||||||
|
|
||||||
|
|
||||||
/*
|
|
||||||
// add the splitter
|
|
||||||
if (!$iframe.has('.cp-splitter').length) {
|
|
||||||
var $preview = $iframe.find('#previewContainer');
|
|
||||||
var splitter = $('<div>', {
|
|
||||||
'class': 'cp-splitter'
|
|
||||||
}).appendTo($preview);
|
|
||||||
|
|
||||||
$preview.on('scroll', function() {
|
|
||||||
splitter.css('top', $preview.scrollTop() + 'px');
|
|
||||||
});
|
|
||||||
|
|
||||||
var $target = $iframe.find('.CodeMirror');
|
|
||||||
|
|
||||||
splitter.on('mousedown', function (e) {
|
|
||||||
e.preventDefault();
|
|
||||||
var x = e.pageX;
|
|
||||||
var w = $target.width();
|
|
||||||
|
|
||||||
$iframe.on('mouseup mousemove', function handler(evt) {
|
|
||||||
if (evt.type === 'mouseup') {
|
|
||||||
$iframe.off('mouseup mousemove', handler);
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
$target.css('width', (w - x + evt.pageX) + 'px');
|
|
||||||
});
|
|
||||||
});
|
|
||||||
}
|
|
||||||
*/
|
|
||||||
|
|
||||||
Cryptpad.removeLoadingScreen();
|
|
||||||
setEditable(!readOnly);
|
|
||||||
initializing = false;
|
|
||||||
|
|
||||||
onLocal(); // push local state to avoid parse errors later.
|
|
||||||
|
|
||||||
if (readOnly) {
|
|
||||||
config.onRemote();
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
//UserList.getLastName(toolbar.$userNameButton, isNew);
|
|
||||||
|
|
||||||
var fmConfig = {
|
|
||||||
dropArea: $('.CodeMirror'),
|
|
||||||
body: $('body'),
|
|
||||||
onUploaded: function (ev, data) {
|
|
||||||
//var cursor = editor.getCursor();
|
|
||||||
//var cleanName = data.name.replace(/[\[\]]/g, '');
|
|
||||||
//var text = '';
|
|
||||||
var parsed = Cryptpad.parsePadUrl(data.url);
|
|
||||||
var hexFileName = Cryptpad.base64ToHex(parsed.hashData.channel);
|
|
||||||
var src = '/blob/' + hexFileName.slice(0,2) + '/' + hexFileName;
|
|
||||||
var mt = '<media-tag src="' + src + '" data-crypto-key="cryptpad:' + parsed.hashData.key + '"></media-tag>';
|
|
||||||
editor.replaceSelection(mt);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
APP.FM = common.createFileManager(fmConfig);
|
|
||||||
};
|
|
||||||
|
|
||||||
config.onRemote = function () {
|
|
||||||
if (initializing) { return; }
|
|
||||||
if (isHistoryMode) { return; }
|
|
||||||
|
|
||||||
var oldDoc = canonicalize(CodeMirror.$textarea.val());
|
|
||||||
var shjson = APP.realtime.getUserDoc();
|
|
||||||
|
|
||||||
// Update the user list (metadata) from the hyperjson
|
|
||||||
//Metadata.update(shjson);
|
|
||||||
|
|
||||||
var hjson = JSON.parse(shjson);
|
|
||||||
var remoteDoc = hjson.content;
|
|
||||||
|
|
||||||
if (hjson.metadata) {
|
|
||||||
metadataMgr.updateMetadata(hjson.metadata);
|
|
||||||
}
|
|
||||||
|
|
||||||
var highlightMode = hjson.highlightMode;
|
|
||||||
if (highlightMode && highlightMode !== APP.highlightMode) {
|
|
||||||
CodeMirror.setMode(highlightMode, onModeChanged);
|
|
||||||
}
|
|
||||||
|
|
||||||
CodeMirror.setValueAndCursor(oldDoc, remoteDoc, TextPatcher);
|
|
||||||
drawPreview();
|
|
||||||
|
|
||||||
if (!readOnly) {
|
|
||||||
var textValue = canonicalize(CodeMirror.$textarea.val());
|
|
||||||
var shjson2 = stringifyInner(textValue);
|
|
||||||
if (shjson2 !== shjson) {
|
|
||||||
console.error("shjson2 !== shjson");
|
|
||||||
TextPatcher.log(shjson, TextPatcher.diff(shjson, shjson2));
|
|
||||||
APP.patchText(shjson2);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
if (oldDoc !== remoteDoc) { Cryptpad.notify(); }
|
|
||||||
};
|
|
||||||
|
|
||||||
config.onAbort = function () {
|
|
||||||
// inform of network disconnect
|
|
||||||
setEditable(false);
|
|
||||||
toolbar.failed();
|
|
||||||
Cryptpad.alert(Messages.common_connectionLost, undefined, true);
|
|
||||||
};
|
|
||||||
|
|
||||||
config.onConnectionChange = function (info) {
|
|
||||||
setEditable(info.state);
|
|
||||||
//toolbar.failed();
|
|
||||||
if (info.state) {
|
|
||||||
initializing = true;
|
|
||||||
//toolbar.reconnecting(info.myId);
|
|
||||||
Cryptpad.findOKButton().click();
|
|
||||||
} else {
|
|
||||||
Cryptpad.alert(Messages.common_connectionLost, undefined, true);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
config.onError = onConnectError;
|
|
||||||
|
|
||||||
cpNfInner = common.startRealtime(config);
|
|
||||||
metadataMgr = cpNfInner.metadataMgr;
|
|
||||||
|
|
||||||
editor.on('change', onLocal);
|
|
||||||
|
|
||||||
Cryptpad.onLogout(function () { setEditable(false); });
|
|
||||||
};
|
|
||||||
/*
|
|
||||||
var interval = 100;
|
|
||||||
var second = function (CM) {
|
|
||||||
Cryptpad.ready(function () {
|
|
||||||
andThen(CM);
|
|
||||||
Cryptpad.reportAppUsage();
|
|
||||||
});
|
|
||||||
Cryptpad.onError(function (info) {
|
|
||||||
if (info && info.type === "store") {
|
|
||||||
onConnectError();
|
|
||||||
}
|
|
||||||
});
|
|
||||||
};
|
|
||||||
|
|
||||||
var first = function () {
|
|
||||||
if (ifrw.CodeMirror) {
|
|
||||||
second(ifrw.CodeMirror);
|
|
||||||
} else {
|
|
||||||
console.log("CodeMirror was not defined. Trying again in %sms", interval);
|
|
||||||
setTimeout(first, interval);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
first();*/
|
|
||||||
var CMEDITOR_CHECK_INTERVAL = 100;
|
|
||||||
var cmEditorAvailable = function (cb) {
|
|
||||||
var intr;
|
|
||||||
var check = function () {
|
|
||||||
if (window.CodeMirror) {
|
|
||||||
clearTimeout(intr);
|
|
||||||
cb(window.CodeMirror);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
intr = setInterval(function () {
|
|
||||||
console.log("CodeMirror was not defined. Trying again in %sms", CMEDITOR_CHECK_INTERVAL);
|
|
||||||
check();
|
|
||||||
}, CMEDITOR_CHECK_INTERVAL);
|
|
||||||
check();
|
|
||||||
};
|
|
||||||
var main = function () {
|
|
||||||
var CM;
|
|
||||||
var CodeMirror;
|
|
||||||
var editor;
|
|
||||||
var common;
|
|
||||||
|
|
||||||
nThen(function (waitFor) {
|
|
||||||
cmEditorAvailable(waitFor(function (cm) {
|
|
||||||
CM = cm;
|
|
||||||
}));
|
|
||||||
$(waitFor(function () {
|
|
||||||
Cryptpad.addLoadingScreen();
|
|
||||||
}));
|
|
||||||
SFCommon.create(waitFor(function (c) { APP.common = common = c; }));
|
|
||||||
}).nThen(function (/*waitFor*/) {
|
|
||||||
CodeMirror = Cryptpad.createCodemirror(window, Cryptpad, null, CM);
|
|
||||||
$('.CodeMirror').addClass('fullPage');
|
|
||||||
editor = CodeMirror.editor;
|
|
||||||
Cryptpad.onError(function (info) {
|
|
||||||
if (info && info.type === "store") {
|
|
||||||
onConnectError();
|
|
||||||
}
|
|
||||||
});
|
|
||||||
andThen(editor, CodeMirror, common);
|
|
||||||
});
|
|
||||||
};
|
|
||||||
main();
|
|
||||||
});
|
|
||||||
@ -1,41 +0,0 @@
|
|||||||
// Load #1, load as little as possible because we are in a race to get the loading screen up.
|
|
||||||
define([
|
|
||||||
'/bower_components/nthen/index.js',
|
|
||||||
'/api/config',
|
|
||||||
'jquery',
|
|
||||||
'/common/requireconfig.js',
|
|
||||||
'/common/sframe-common-outer.js'
|
|
||||||
], function (nThen, ApiConfig, $, RequireConfig, SFCommonO) {
|
|
||||||
var requireConfig = RequireConfig();
|
|
||||||
|
|
||||||
// Loaded in load #2
|
|
||||||
nThen(function (waitFor) {
|
|
||||||
$(waitFor());
|
|
||||||
}).nThen(function (waitFor) {
|
|
||||||
var req = {
|
|
||||||
cfg: requireConfig,
|
|
||||||
req: [ '/common/loading.js' ],
|
|
||||||
pfx: window.location.origin
|
|
||||||
};
|
|
||||||
window.rc = requireConfig;
|
|
||||||
window.apiconf = ApiConfig;
|
|
||||||
$('#sbox-iframe').attr('src',
|
|
||||||
ApiConfig.httpSafeOrigin + '/code2/inner.html?' + requireConfig.urlArgs +
|
|
||||||
'#' + encodeURIComponent(JSON.stringify(req)));
|
|
||||||
|
|
||||||
// This is a cheap trick to avoid loading sframe-channel in parallel with the
|
|
||||||
// loading screen setup.
|
|
||||||
var done = waitFor();
|
|
||||||
var onMsg = function (msg) {
|
|
||||||
var data = JSON.parse(msg.data);
|
|
||||||
if (data.q !== 'READY') { return; }
|
|
||||||
window.removeEventListener('message', onMsg);
|
|
||||||
var _done = done;
|
|
||||||
done = function () { };
|
|
||||||
_done();
|
|
||||||
};
|
|
||||||
window.addEventListener('message', onMsg);
|
|
||||||
}).nThen(function (/*waitFor*/) {
|
|
||||||
SFCommonO.start();
|
|
||||||
});
|
|
||||||
});
|
|
||||||
@ -556,6 +556,16 @@ define([
|
|||||||
});
|
});
|
||||||
common.findOKButton().text(Messages.cancelButton);
|
common.findOKButton().text(Messages.cancelButton);
|
||||||
};
|
};
|
||||||
|
// Secure iframes
|
||||||
|
common.useTemplate = function (href, Crypt, cb) {
|
||||||
|
var parsed = parsePadUrl(href);
|
||||||
|
if(!parsed) { throw new Error("Cannot get template hash"); }
|
||||||
|
Crypt.get(parsed.hash, function (err, val) {
|
||||||
|
if (err) { throw new Error(err); }
|
||||||
|
var p = parsePadUrl(window.location.href);
|
||||||
|
Crypt.put(p.hash, val, cb);
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
// STORAGE
|
// STORAGE
|
||||||
/* fetch and migrate your pad history from the store */
|
/* fetch and migrate your pad history from the store */
|
||||||
@ -735,17 +745,19 @@ define([
|
|||||||
return list;
|
return list;
|
||||||
};
|
};
|
||||||
// Needed for the secure filepicker app
|
// Needed for the secure filepicker app
|
||||||
common.getSecureFilesList = function (cb) {
|
common.getSecureFilesList = function (filter, cb) {
|
||||||
var store = common.getStore();
|
var store = common.getStore();
|
||||||
if (!store) { return void cb("Store is not ready"); }
|
if (!store) { return void cb("Store is not ready"); }
|
||||||
var proxy = store.getProxy();
|
var proxy = store.getProxy();
|
||||||
var fo = proxy.fo;
|
var fo = proxy.fo;
|
||||||
var list = {};
|
var list = {};
|
||||||
var hashes = [];
|
var hashes = [];
|
||||||
fo.getFiles([fo.ROOT]).forEach(function (id) {
|
var types = filter.types;
|
||||||
|
var where = filter.where;
|
||||||
|
fo.getFiles(where).forEach(function (id) {
|
||||||
var data = fo.getFileData(id);
|
var data = fo.getFileData(id);
|
||||||
var parsed = parsePadUrl(data.href);
|
var parsed = parsePadUrl(data.href);
|
||||||
if ((parsed.type === 'file' || parsed.type === 'media')
|
if ((!types || types.length === 0 || types.indexOf(parsed.type) !== -1)
|
||||||
&& hashes.indexOf(parsed.hash) === -1) {
|
&& hashes.indexOf(parsed.hash) === -1) {
|
||||||
hashes.push(parsed.hash);
|
hashes.push(parsed.hash);
|
||||||
list[id] = data;
|
list[id] = data;
|
||||||
|
|||||||
@ -73,6 +73,7 @@ define(['json.sortify'], function (Sortify) {
|
|||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
|
console.log('here register');
|
||||||
sframeChan.on('EV_METADATA_UPDATE', function (ev) {
|
sframeChan.on('EV_METADATA_UPDATE', function (ev) {
|
||||||
meta = ev;
|
meta = ev;
|
||||||
if (ev.priv) {
|
if (ev.priv) {
|
||||||
|
|||||||
@ -46,7 +46,9 @@ define([
|
|||||||
$('<td>').text(Messages.cancel).appendTo($thead);
|
$('<td>').text(Messages.cancel).appendTo($thead);
|
||||||
|
|
||||||
var createTableContainer = function ($body) {
|
var createTableContainer = function ($body) {
|
||||||
|
console.log($body);
|
||||||
File.$container = $('<div>', { id: 'cp-fileupload' }).append($table).appendTo($body);
|
File.$container = $('<div>', { id: 'cp-fileupload' }).append($table).appendTo($body);
|
||||||
|
console.log('done');
|
||||||
return File.$container;
|
return File.$container;
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -62,6 +64,25 @@ define([
|
|||||||
return data;
|
return data;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
var sframeChan = common.getSframeChannel();
|
||||||
|
var onError = $.noop,
|
||||||
|
onComplete = $.noop,
|
||||||
|
updateProgress = $.noop,
|
||||||
|
onPending = $.noop;
|
||||||
|
sframeChan.on('EV_FILE_UPLOAD_STATE', function (data) {
|
||||||
|
if (data.error) {
|
||||||
|
return void onError(data.error);
|
||||||
|
}
|
||||||
|
if (data.complete && data.href) {
|
||||||
|
return void onComplete(data.href);
|
||||||
|
}
|
||||||
|
if (typeof data.progress !== "undefined") {
|
||||||
|
return void updateProgress(data.progress);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
sframeChan.on('Q_CANCEL_PENDING_FILE_UPLOAD', function (data, cb) {
|
||||||
|
onPending(cb);
|
||||||
|
});
|
||||||
var upload = function (file) {
|
var upload = function (file) {
|
||||||
var blob = file.blob; // This is not a blob but an array buffer
|
var blob = file.blob; // This is not a blob but an array buffer
|
||||||
var u8 = new Uint8Array(blob);
|
var u8 = new Uint8Array(blob);
|
||||||
@ -80,16 +101,14 @@ define([
|
|||||||
var $pc = $row.find('.cp-fileupload-table-progress');
|
var $pc = $row.find('.cp-fileupload-table-progress');
|
||||||
var $link = $row.find('.cp-fileupload-table-link');
|
var $link = $row.find('.cp-fileupload-table-link');
|
||||||
|
|
||||||
var sframeChan = common.getSframeChannel();
|
updateProgress = function (progressValue) {
|
||||||
|
|
||||||
var updateProgress = function (progressValue) {
|
|
||||||
$pv.text(Math.round(progressValue*100)/100 + '%');
|
$pv.text(Math.round(progressValue*100)/100 + '%');
|
||||||
$pb.css({
|
$pb.css({
|
||||||
width: (progressValue/100)*$pc.width()+'px'
|
width: (progressValue/100)*$pc.width()+'px'
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
var onComplete = function (href) {
|
onComplete = function (href) {
|
||||||
$link.attr('href', href)
|
$link.attr('href', href)
|
||||||
.click(function (e) {
|
.click(function (e) {
|
||||||
e.preventDefault();
|
e.preventDefault();
|
||||||
@ -108,7 +127,7 @@ define([
|
|||||||
queue.next();
|
queue.next();
|
||||||
};
|
};
|
||||||
|
|
||||||
var onError = function (e) {
|
onError = function (e) {
|
||||||
queue.inProgress = false;
|
queue.inProgress = false;
|
||||||
queue.next();
|
queue.next();
|
||||||
if (e === 'TOO_LARGE') {
|
if (e === 'TOO_LARGE') {
|
||||||
@ -123,24 +142,10 @@ define([
|
|||||||
return void Cryptpad.alert(Messages.upload_serverError);
|
return void Cryptpad.alert(Messages.upload_serverError);
|
||||||
};
|
};
|
||||||
|
|
||||||
var onPending = function (cb) {
|
onPending = function (cb) {
|
||||||
Cryptpad.confirm(Messages.upload_uploadPending, cb);
|
Cryptpad.confirm(Messages.upload_uploadPending, cb);
|
||||||
};
|
};
|
||||||
|
|
||||||
sframeChan.on('EV_FILE_UPLOAD_STATE', function (data) {
|
|
||||||
if (data.error) {
|
|
||||||
return void onError(data.error);
|
|
||||||
}
|
|
||||||
if (data.complete && data.href) {
|
|
||||||
return void onComplete(data.href);
|
|
||||||
}
|
|
||||||
if (typeof data.progress !== "undefined") {
|
|
||||||
return void updateProgress(data.progress);
|
|
||||||
}
|
|
||||||
});
|
|
||||||
sframeChan.on('Q_CANCEL_PENDING_FILE_UPLOAD', function (data, cb) {
|
|
||||||
onPending(cb);
|
|
||||||
});
|
|
||||||
file.noStore = config.noStore;
|
file.noStore = config.noStore;
|
||||||
try {
|
try {
|
||||||
file.blob = Nacl.util.encodeBase64(u8);
|
file.blob = Nacl.util.encodeBase64(u8);
|
||||||
@ -168,7 +173,8 @@ define([
|
|||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
if (queue.inProgress) { return; }
|
if (queue.inProgress) { return; }
|
||||||
File.$container.show();
|
// setTimeout to fix a firefox error 'NS_ERROR_NOT_AVAILABLE'
|
||||||
|
window.setTimeout(function () { File.$container.show(); });
|
||||||
var file = queue.queue.shift();
|
var file = queue.queue.shift();
|
||||||
upload(file);
|
upload(file);
|
||||||
};
|
};
|
||||||
@ -177,7 +183,8 @@ define([
|
|||||||
obj.id = id;
|
obj.id = id;
|
||||||
queue.queue.push(obj);
|
queue.queue.push(obj);
|
||||||
|
|
||||||
$table.show();
|
// setTimeout to fix a firefox error 'NS_ERROR_NOT_AVAILABLE'
|
||||||
|
window.setTimeout(function () { $table.show(); });
|
||||||
var estimate = FileCrypto.computeEncryptedSize(obj.blob.byteLength, obj.metadata);
|
var estimate = FileCrypto.computeEncryptedSize(obj.blob.byteLength, obj.metadata);
|
||||||
|
|
||||||
var $progressBar = $('<div>', {'class':'cp-fileupload-table-progress-container'});
|
var $progressBar = $('<div>', {'class':'cp-fileupload-table-progress-container'});
|
||||||
|
|||||||
@ -261,69 +261,6 @@ define([
|
|||||||
return $userAdmin;
|
return $userAdmin;
|
||||||
};
|
};
|
||||||
|
|
||||||
// createFileDialog can only be used in filepicker due to access rights restrictions
|
|
||||||
UI.createFileDialog = function (cfg) {
|
|
||||||
var common = cfg.common;
|
|
||||||
var sframeChan = common.getSframeChannel();
|
|
||||||
var updateContainer;
|
|
||||||
var hideFileDialog = function () {
|
|
||||||
sframeChan.event('EV_FILE_PICKER_CLOSE');
|
|
||||||
};
|
|
||||||
// Create modal
|
|
||||||
var $blockContainer = Cryptpad.createModal({
|
|
||||||
id: 'cp-filepicker-dialog',
|
|
||||||
$body: cfg.$body,
|
|
||||||
onClose: hideFileDialog
|
|
||||||
}).show();
|
|
||||||
// Set the fixed content
|
|
||||||
var $block = $blockContainer.find('.cp-modal');
|
|
||||||
var $description = $('<p>').text(Messages.filePicker_description);
|
|
||||||
$block.append($description);
|
|
||||||
var $filter = $('<p>', {'class': 'cp-modal-form'}).appendTo($block);
|
|
||||||
var to;
|
|
||||||
$('<input>', {
|
|
||||||
type: 'text',
|
|
||||||
'class': 'cp-filepicker-filter',
|
|
||||||
'placeholder': Messages.filePicker_filter
|
|
||||||
}).appendTo($filter).on('keypress', function () {
|
|
||||||
if (to) { window.clearTimeout(to); }
|
|
||||||
to = window.setTimeout(updateContainer, 300);
|
|
||||||
});
|
|
||||||
$filter.append(common.createButton('upload', false, cfg.data, function () {
|
|
||||||
hideFileDialog();
|
|
||||||
}));
|
|
||||||
var $container = $('<span>', {'class': 'cp-filepicker-content'}).appendTo($block);
|
|
||||||
// Update the files list when needed
|
|
||||||
updateContainer = function () {
|
|
||||||
$container.html('');
|
|
||||||
var filter = $filter.find('.cp-filepicker-filter').val().trim();
|
|
||||||
var todo = function (err, list) {
|
|
||||||
if (err) { return void console.error(err); }
|
|
||||||
Object.keys(list).forEach(function (id) {
|
|
||||||
var data = list[id];
|
|
||||||
var name = data.title || '?';
|
|
||||||
if (filter && name.toLowerCase().indexOf(filter.toLowerCase()) === -1) {
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
var $span = $('<span>', {
|
|
||||||
'class': 'cp-filepicker-content-element',
|
|
||||||
'title': name,
|
|
||||||
}).appendTo($container);
|
|
||||||
$span.append(Cryptpad.getFileIcon(data));
|
|
||||||
$span.append(name);
|
|
||||||
$span.click(function () {
|
|
||||||
if (typeof cfg.onSelect === "function") { cfg.onSelect(data.href); }
|
|
||||||
hideFileDialog();
|
|
||||||
});
|
|
||||||
});
|
|
||||||
};
|
|
||||||
common.getFilesList(todo);
|
|
||||||
};
|
|
||||||
updateContainer();
|
|
||||||
sframeChan.on('EV_FILE_PICKER_REFRESH', updateContainer);
|
|
||||||
};
|
|
||||||
|
|
||||||
|
|
||||||
UI.initFilePicker = function (common, cfg) {
|
UI.initFilePicker = function (common, cfg) {
|
||||||
var onSelect = cfg.onSelect || $.noop;
|
var onSelect = cfg.onSelect || $.noop;
|
||||||
var sframeChan = common.getSframeChannel();
|
var sframeChan = common.getSframeChannel();
|
||||||
@ -331,9 +268,35 @@ define([
|
|||||||
onSelect(data);
|
onSelect(data);
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
UI.openFilePicker = function (common) {
|
UI.openFilePicker = function (common, types) {
|
||||||
var sframeChan = common.getSframeChannel();
|
var sframeChan = common.getSframeChannel();
|
||||||
sframeChan.event("EV_FILE_PICKER_OPEN");
|
sframeChan.event("EV_FILE_PICKER_OPEN", types);
|
||||||
|
};
|
||||||
|
|
||||||
|
UI.openTemplatePicker = function (common) {
|
||||||
|
var metadataMgr = common.getMetadataMgr();
|
||||||
|
var type = metadataMgr.getMetadataLazy().type;
|
||||||
|
var first = true; // We can only pick a template once (for a new document)
|
||||||
|
var fileDialogCfg = {
|
||||||
|
onSelect: function (data) {
|
||||||
|
if (data.type === type && first) {
|
||||||
|
Cryptpad.addLoadingScreen(null, true);
|
||||||
|
var sframeChan = common.getSframeChannel();
|
||||||
|
sframeChan.query('Q_TEMPLATE_USE', data.href, function () {
|
||||||
|
first = false;
|
||||||
|
Cryptpad.removeLoadingScreen();
|
||||||
|
common.feedback('TEMPLATE_USED');
|
||||||
|
});
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
common.initFilePicker(common, fileDialogCfg);
|
||||||
|
var pickerCfg = {
|
||||||
|
types: [type],
|
||||||
|
where: ['template']
|
||||||
|
};
|
||||||
|
common.openFilePicker(common, pickerCfg);
|
||||||
};
|
};
|
||||||
|
|
||||||
return UI;
|
return UI;
|
||||||
|
|||||||
@ -255,7 +255,7 @@ define([
|
|||||||
});
|
});
|
||||||
|
|
||||||
var FP = {};
|
var FP = {};
|
||||||
var initFilePicker = function () {
|
var initFilePicker = function (types) {
|
||||||
var config = {};
|
var config = {};
|
||||||
config.onFilePicked = function (data) {
|
config.onFilePicked = function (data) {
|
||||||
sframeChan.event('EV_FILE_PICKED', data);
|
sframeChan.event('EV_FILE_PICKED', data);
|
||||||
@ -264,16 +264,21 @@ define([
|
|||||||
FP.$iframe.hide();
|
FP.$iframe.hide();
|
||||||
};
|
};
|
||||||
config.onFileUpload = onFileUpload;
|
config.onFileUpload = onFileUpload;
|
||||||
|
config.types = types;
|
||||||
if (!FP.$iframe) {
|
if (!FP.$iframe) {
|
||||||
FP.$iframe = $('<iframe>', {id: 'sbox-filePicker-iframe'}).appendTo($('body'));
|
FP.$iframe = $('<iframe>', {id: 'sbox-filePicker-iframe'}).appendTo($('body'));
|
||||||
FP.picker = FilePicker.create(config);
|
FP.picker = FilePicker.create(config);
|
||||||
} else {
|
} else {
|
||||||
FP.$iframe.show();
|
FP.$iframe.show();
|
||||||
FP.picker.refresh();
|
FP.picker.refresh(types);
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
sframeChan.on('EV_FILE_PICKER_OPEN', function () {
|
sframeChan.on('EV_FILE_PICKER_OPEN', function (data) {
|
||||||
initFilePicker();
|
initFilePicker(data);
|
||||||
|
});
|
||||||
|
|
||||||
|
sframeChan.on('Q_TEMPLATE_USE', function (href, cb) {
|
||||||
|
Cryptpad.useTemplate(href, Cryptget, cb);
|
||||||
});
|
});
|
||||||
|
|
||||||
CpNfOuter.start({
|
CpNfOuter.start({
|
||||||
|
|||||||
@ -55,9 +55,9 @@ define([
|
|||||||
// UI
|
// UI
|
||||||
funcs.createUserAdminMenu = UI.createUserAdminMenu;
|
funcs.createUserAdminMenu = UI.createUserAdminMenu;
|
||||||
funcs.displayAvatar = UI.displayAvatar;
|
funcs.displayAvatar = UI.displayAvatar;
|
||||||
funcs.createFileDialog = UI.createFileDialog;
|
|
||||||
funcs.initFilePicker = UI.initFilePicker;
|
funcs.initFilePicker = UI.initFilePicker;
|
||||||
funcs.openFilePicker = UI.openFilePicker;
|
funcs.openFilePicker = UI.openFilePicker;
|
||||||
|
funcs.openTemplatePicker = UI.openTemplatePicker;
|
||||||
|
|
||||||
// History
|
// History
|
||||||
funcs.getHistory = function (config) { return History.create(funcs, config); };
|
funcs.getHistory = function (config) { return History.create(funcs, config); };
|
||||||
@ -341,8 +341,8 @@ define([
|
|||||||
|
|
||||||
|
|
||||||
// Can, only be called by the filepicker app
|
// Can, only be called by the filepicker app
|
||||||
funcs.getFilesList = function (cb) {
|
funcs.getFilesList = function (types, cb) {
|
||||||
ctx.sframeChan.query('Q_GET_FILES_LIST', null, function (err, data) {
|
ctx.sframeChan.query('Q_GET_FILES_LIST', types, function (err, data) {
|
||||||
cb(err || data.error, data.data);
|
cb(err || data.error, data.data);
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
|
|||||||
@ -100,6 +100,9 @@ define({
|
|||||||
// Get all the files from the drive to display them in a file picker secure app
|
// Get all the files from the drive to display them in a file picker secure app
|
||||||
'Q_GET_FILES_LIST': true,
|
'Q_GET_FILES_LIST': true,
|
||||||
|
|
||||||
|
// Template picked, replace the content of the pad
|
||||||
|
'Q_TEMPLATE_USE': true,
|
||||||
|
|
||||||
// File upload queries and events
|
// File upload queries and events
|
||||||
'Q_UPLOAD_FILE': true,
|
'Q_UPLOAD_FILE': true,
|
||||||
'EV_FILE_UPLOAD_STATE': true,
|
'EV_FILE_UPLOAD_STATE': true,
|
||||||
|
|||||||
@ -6,6 +6,7 @@ define([
|
|||||||
'/common/cryptpad-common.js',
|
'/common/cryptpad-common.js',
|
||||||
'/bower_components/nthen/index.js',
|
'/bower_components/nthen/index.js',
|
||||||
'/common/sframe-common.js',
|
'/common/sframe-common.js',
|
||||||
|
'json.sortify',
|
||||||
|
|
||||||
'css!/bower_components/bootstrap/dist/css/bootstrap.min.css',
|
'css!/bower_components/bootstrap/dist/css/bootstrap.min.css',
|
||||||
'css!/bower_components/components-font-awesome/css/font-awesome.min.css',
|
'css!/bower_components/components-font-awesome/css/font-awesome.min.css',
|
||||||
@ -17,7 +18,8 @@ define([
|
|||||||
JsonOT,
|
JsonOT,
|
||||||
Cryptpad,
|
Cryptpad,
|
||||||
nThen,
|
nThen,
|
||||||
SFCommon)
|
SFCommon,
|
||||||
|
Sortify)
|
||||||
{
|
{
|
||||||
var Messages = Cryptpad.Messages;
|
var Messages = Cryptpad.Messages;
|
||||||
|
|
||||||
@ -30,18 +32,31 @@ define([
|
|||||||
};
|
};
|
||||||
|
|
||||||
var andThen = function (common) {
|
var andThen = function (common) {
|
||||||
//var metadataMgr = common.getMetadataMgr();
|
var metadataMgr = common.getMetadataMgr();
|
||||||
var $body = $('body');
|
var $body = $('body');
|
||||||
var sframeChan = common.getSframeChannel();
|
var sframeChan = common.getSframeChannel();
|
||||||
|
var filters = metadataMgr.getPrivateData().types;
|
||||||
|
|
||||||
|
var hideFileDialog = function () {
|
||||||
|
sframeChan.event('EV_FILE_PICKER_CLOSE');
|
||||||
|
};
|
||||||
var onFilePicked = function (data) {
|
var onFilePicked = function (data) {
|
||||||
var parsed = Cryptpad.parsePadUrl(data.url);
|
var parsed = Cryptpad.parsePadUrl(data.url);
|
||||||
|
hideFileDialog();
|
||||||
|
if (parsed.type === 'file') {
|
||||||
var hexFileName = Cryptpad.base64ToHex(parsed.hashData.channel);
|
var hexFileName = Cryptpad.base64ToHex(parsed.hashData.channel);
|
||||||
var src = '/blob/' + hexFileName.slice(0,2) + '/' + hexFileName;
|
var src = '/blob/' + hexFileName.slice(0,2) + '/' + hexFileName;
|
||||||
sframeChan.event("EV_FILE_PICKED", {
|
sframeChan.event("EV_FILE_PICKED", {
|
||||||
|
type: parsed.type,
|
||||||
src: src,
|
src: src,
|
||||||
key: parsed.hashData.key
|
key: parsed.hashData.key
|
||||||
});
|
});
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
sframeChan.event("EV_FILE_PICKED", {
|
||||||
|
type: parsed.type,
|
||||||
|
href: data.url,
|
||||||
|
});
|
||||||
};
|
};
|
||||||
|
|
||||||
// File uploader
|
// File uploader
|
||||||
@ -61,12 +76,9 @@ define([
|
|||||||
var data = {
|
var data = {
|
||||||
FM: APP.FM
|
FM: APP.FM
|
||||||
};
|
};
|
||||||
var createFileDialog = function (cfg) {
|
|
||||||
var sframeChan = common.getSframeChannel();
|
|
||||||
var updateContainer;
|
var updateContainer;
|
||||||
var hideFileDialog = function () {
|
var createFileDialog = function () {
|
||||||
sframeChan.event('EV_FILE_PICKER_CLOSE');
|
var types = filters.types;
|
||||||
};
|
|
||||||
// Create modal
|
// Create modal
|
||||||
var $blockContainer = Cryptpad.createModal({
|
var $blockContainer = Cryptpad.createModal({
|
||||||
id: 'cp-filepicker-dialog',
|
id: 'cp-filepicker-dialog',
|
||||||
@ -75,8 +87,16 @@ define([
|
|||||||
}).show();
|
}).show();
|
||||||
// Set the fixed content
|
// Set the fixed content
|
||||||
var $block = $blockContainer.find('.cp-modal');
|
var $block = $blockContainer.find('.cp-modal');
|
||||||
var $description = $('<p>').text(Messages.filePicker_description);
|
|
||||||
|
// Description
|
||||||
|
var text = Messages.filePicker_description;
|
||||||
|
if (types.length === 1 && types[0] !== 'file') {
|
||||||
|
// Should be Templates
|
||||||
|
text = Messages.selectTemplate;
|
||||||
|
}
|
||||||
|
var $description = $('<p>').text(text);
|
||||||
$block.append($description);
|
$block.append($description);
|
||||||
|
|
||||||
var $filter = $('<p>', {'class': 'cp-modal-form'}).appendTo($block);
|
var $filter = $('<p>', {'class': 'cp-modal-form'}).appendTo($block);
|
||||||
var to;
|
var to;
|
||||||
$('<input>', {
|
$('<input>', {
|
||||||
@ -87,9 +107,12 @@ define([
|
|||||||
if (to) { window.clearTimeout(to); }
|
if (to) { window.clearTimeout(to); }
|
||||||
to = window.setTimeout(updateContainer, 300);
|
to = window.setTimeout(updateContainer, 300);
|
||||||
});
|
});
|
||||||
$filter.append(common.createButton('upload', false, data, function () {
|
|
||||||
hideFileDialog();
|
//If file, display the upload button
|
||||||
}));
|
if (types.indexOf('file') !== -1) {
|
||||||
|
$filter.append(common.createButton('upload', false, data));
|
||||||
|
}
|
||||||
|
|
||||||
var $container = $('<span>', {'class': 'cp-filepicker-content'}).appendTo($block);
|
var $container = $('<span>', {'class': 'cp-filepicker-content'}).appendTo($block);
|
||||||
// Update the files list when needed
|
// Update the files list when needed
|
||||||
updateContainer = function () {
|
updateContainer = function () {
|
||||||
@ -111,15 +134,23 @@ define([
|
|||||||
$span.append(name);
|
$span.append(name);
|
||||||
$span.click(function () {
|
$span.click(function () {
|
||||||
if (typeof onSelect === "function") { onSelect(data.href); }
|
if (typeof onSelect === "function") { onSelect(data.href); }
|
||||||
hideFileDialog();
|
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
common.getFilesList(todo);
|
common.getFilesList(filters, todo);
|
||||||
};
|
};
|
||||||
updateContainer();
|
updateContainer();
|
||||||
sframeChan.on('EV_FILE_PICKER_REFRESH', updateContainer);
|
|
||||||
};
|
};
|
||||||
|
sframeChan.on('EV_FILE_PICKER_REFRESH', function (newFilters) {
|
||||||
|
console.log(Sortify(filters));
|
||||||
|
console.log(Sortify(newFilters));
|
||||||
|
if (Sortify(filters) !== Sortify(newFilters)) {
|
||||||
|
$body.html('');
|
||||||
|
filters = newFilters;
|
||||||
|
return void createFileDialog();
|
||||||
|
}
|
||||||
|
updateContainer();
|
||||||
|
});
|
||||||
createFileDialog();
|
createFileDialog();
|
||||||
|
|
||||||
Cryptpad.removeLoadingScreen();
|
Cryptpad.removeLoadingScreen();
|
||||||
@ -134,12 +165,19 @@ define([
|
|||||||
}));
|
}));
|
||||||
SFCommon.create(waitFor(function (c) { APP.common = common = c; }));
|
SFCommon.create(waitFor(function (c) { APP.common = common = c; }));
|
||||||
}).nThen(function (/*waitFor*/) {
|
}).nThen(function (/*waitFor*/) {
|
||||||
|
var metadataMgr = common.getMetadataMgr();
|
||||||
Cryptpad.onError(function (info) {
|
Cryptpad.onError(function (info) {
|
||||||
if (info && info.type === "store") {
|
if (info && info.type === "store") {
|
||||||
onConnectError();
|
onConnectError();
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
if (metadataMgr.getMetadataLazy() !== 'uninitialized') {
|
||||||
andThen(common);
|
andThen(common);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
metadataMgr.onChange(function () {
|
||||||
|
andThen(common);
|
||||||
|
});
|
||||||
});
|
});
|
||||||
};
|
};
|
||||||
main();
|
main();
|
||||||
|
|||||||
@ -84,7 +84,8 @@ define([
|
|||||||
pathname: window.location.pathname,
|
pathname: window.location.pathname,
|
||||||
feedbackAllowed: Cryptpad.isFeedbackAllowed(),
|
feedbackAllowed: Cryptpad.isFeedbackAllowed(),
|
||||||
friends: proxy.friends || {},
|
friends: proxy.friends || {},
|
||||||
settings: proxy.settings || {}
|
settings: proxy.settings || {},
|
||||||
|
types: config.types
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
@ -117,9 +118,9 @@ define([
|
|||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
sframeChan.on('Q_GET_FILES_LIST', function (data, cb) {
|
sframeChan.on('Q_GET_FILES_LIST', function (types, cb) {
|
||||||
console.error("TODO: make sure Q_GET_FILES_LIST is only available from filepicker");
|
console.error("TODO: make sure Q_GET_FILES_LIST is only available from filepicker");
|
||||||
Cryptpad.getSecureFilesList(function (err, data) {
|
Cryptpad.getSecureFilesList(types, function (err, data) {
|
||||||
cb({
|
cb({
|
||||||
error: err,
|
error: err,
|
||||||
data: data
|
data: data
|
||||||
@ -128,21 +129,19 @@ define([
|
|||||||
});
|
});
|
||||||
|
|
||||||
sframeChan.on('EV_FILE_PICKER_CLOSE', function () {
|
sframeChan.on('EV_FILE_PICKER_CLOSE', function () {
|
||||||
console.log('close file picker;..');
|
|
||||||
config.onClose();
|
config.onClose();
|
||||||
});
|
});
|
||||||
sframeChan.on('EV_FILE_PICKED', function (data) {
|
sframeChan.on('EV_FILE_PICKED', function (data) {
|
||||||
config.onFilePicked(data);
|
config.onFilePicked(data);
|
||||||
console.log(data);
|
|
||||||
});
|
});
|
||||||
sframeChan.on('Q_UPLOAD_FILE', function (data, cb) {
|
sframeChan.on('Q_UPLOAD_FILE', function (data, cb) {
|
||||||
config.onFileUpload(sframeChan, data, cb);
|
config.onFileUpload(sframeChan, data, cb);
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
var refresh = function () {
|
var refresh = function (types) {
|
||||||
if (!sframeChan) { return; }
|
if (!sframeChan) { return; }
|
||||||
sframeChan.event('EV_FILE_PICKER_REFRESH');
|
sframeChan.event('EV_FILE_PICKER_REFRESH', types);
|
||||||
};
|
};
|
||||||
return {
|
return {
|
||||||
refresh: refresh
|
refresh: refresh
|
||||||
|
|||||||
41
www/oldcode/index.html
Normal file
41
www/oldcode/index.html
Normal file
@ -0,0 +1,41 @@
|
|||||||
|
<!DOCTYPE html>
|
||||||
|
<html class="cp code">
|
||||||
|
<head>
|
||||||
|
<title>CryptPad</title>
|
||||||
|
<meta content="text/html; charset=utf-8" http-equiv="content-type"/>
|
||||||
|
<meta name="viewport" content="width=device-width, initial-scale=1.0">
|
||||||
|
<meta name="referrer" content="no-referrer" />
|
||||||
|
<script async data-bootload="/customize/template.js" data-main="/common/boot.js?ver=1.0" src="/bower_components/requirejs/require.js?ver=2.3.5"></script>
|
||||||
|
<style>
|
||||||
|
html, body {
|
||||||
|
overflow-y: hidden;
|
||||||
|
}
|
||||||
|
#iframe-container {
|
||||||
|
position: fixed;
|
||||||
|
top: 0px;
|
||||||
|
bottom: 0px;
|
||||||
|
right: 0px;
|
||||||
|
left: 0px;
|
||||||
|
padding: 0px;
|
||||||
|
}
|
||||||
|
#pad-iframe {
|
||||||
|
width:100%;
|
||||||
|
height:100%;
|
||||||
|
border:none;
|
||||||
|
margin:0;
|
||||||
|
padding:0;
|
||||||
|
overflow:hidden;
|
||||||
|
box-sizing: border-box;
|
||||||
|
}
|
||||||
|
/* We use !important here to override the 96% set to the element in DecorateToolbar.js
|
||||||
|
when we enter fullscreen mode. It allows us to avoid changing the iframe's size in JS */
|
||||||
|
#pad-iframe.fullscreen {
|
||||||
|
top: 0px;
|
||||||
|
height: 100% !important;
|
||||||
|
}
|
||||||
|
</style>
|
||||||
|
</head>
|
||||||
|
<body>
|
||||||
|
<div id="iframe-container">
|
||||||
|
<iframe id="pad-iframe"></iframe><script src="/common/noscriptfix.js"></script>
|
||||||
|
</div>
|
||||||
17
www/oldcode/inner.html
Normal file
17
www/oldcode/inner.html
Normal file
@ -0,0 +1,17 @@
|
|||||||
|
<!DOCTYPE html>
|
||||||
|
<html style="height: 100%;">
|
||||||
|
<head>
|
||||||
|
<meta content="text/html; charset=utf-8" http-equiv="content-type"/>
|
||||||
|
<script src="/bower_components/jquery/dist/jquery.min.js"></script>
|
||||||
|
<script async data-bootload="inner.js" data-main="/common/boot.js?ver=1.0" src="/bower_components/requirejs/require.js?ver=2.3.5"></script>
|
||||||
|
<style> .loading-hidden { display: none; } </style>
|
||||||
|
</head>
|
||||||
|
<body class="loading-hidden">
|
||||||
|
<div id="cme_toolbox" class="toolbar-container"></div>
|
||||||
|
<div id="editorContainer">
|
||||||
|
<textarea id="editor1" name="editor1"></textarea>
|
||||||
|
<div id="previewContainer"><div id="preview"></div></div>
|
||||||
|
</div>
|
||||||
|
</body>
|
||||||
|
</html>
|
||||||
|
|
||||||
40
www/oldcode/inner.js
Normal file
40
www/oldcode/inner.js
Normal file
@ -0,0 +1,40 @@
|
|||||||
|
define([
|
||||||
|
'jquery',
|
||||||
|
|
||||||
|
'cm/lib/codemirror',
|
||||||
|
|
||||||
|
'css!/bower_components/components-font-awesome/css/font-awesome.min.css',
|
||||||
|
'css!/bower_components/bootstrap/dist/css/bootstrap.min.css',
|
||||||
|
'less!/code/code.less',
|
||||||
|
'less!/customize/src/less/toolbar.less',
|
||||||
|
'less!/customize/src/less/cryptpad.less',
|
||||||
|
|
||||||
|
'css!cm/lib/codemirror.css',
|
||||||
|
'css!cm/addon/dialog/dialog.css',
|
||||||
|
'css!cm/addon/fold/foldgutter.css',
|
||||||
|
|
||||||
|
'cm/mode/markdown/markdown',
|
||||||
|
'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',
|
||||||
|
], function ($, CMeditor) {
|
||||||
|
window.CodeMirror = CMeditor;
|
||||||
|
$('.loading-hidden').removeClass('loading-hidden');
|
||||||
|
});
|
||||||
559
www/oldcode/main.js
Normal file
559
www/oldcode/main.js
Normal file
@ -0,0 +1,559 @@
|
|||||||
|
define([
|
||||||
|
'jquery',
|
||||||
|
'/bower_components/chainpad-crypto/crypto.js',
|
||||||
|
'/bower_components/chainpad-netflux/chainpad-netflux.js',
|
||||||
|
'/bower_components/textpatcher/TextPatcher.js',
|
||||||
|
'/common/toolbar2.js',
|
||||||
|
'json.sortify',
|
||||||
|
'/bower_components/chainpad-json-validator/json-ot.js',
|
||||||
|
'/common/cryptpad-common.js',
|
||||||
|
'/common/cryptget.js',
|
||||||
|
'/common/diffMarked.js',
|
||||||
|
|
||||||
|
'css!/bower_components/components-font-awesome/css/font-awesome.min.css',
|
||||||
|
'less!/customize/src/less/cryptpad.less'
|
||||||
|
], function ($, Crypto, Realtime, TextPatcher, Toolbar, JSONSortify, JsonOT, Cryptpad,
|
||||||
|
Cryptget, DiffMd) {
|
||||||
|
var Messages = Cryptpad.Messages;
|
||||||
|
|
||||||
|
var APP = window.APP = {
|
||||||
|
Cryptpad: Cryptpad,
|
||||||
|
};
|
||||||
|
|
||||||
|
$(function () {
|
||||||
|
Cryptpad.addLoadingScreen();
|
||||||
|
|
||||||
|
var ifrw = APP.ifrw = $('#pad-iframe')[0].contentWindow;
|
||||||
|
var stringify = function (obj) {
|
||||||
|
return JSONSortify(obj);
|
||||||
|
};
|
||||||
|
|
||||||
|
var toolbar;
|
||||||
|
var editor;
|
||||||
|
|
||||||
|
var secret = Cryptpad.getSecrets();
|
||||||
|
var readOnly = secret.keys && !secret.keys.editKeyStr;
|
||||||
|
if (!secret.keys) {
|
||||||
|
secret.keys = secret.key;
|
||||||
|
}
|
||||||
|
|
||||||
|
var onConnectError = function () {
|
||||||
|
Cryptpad.errorLoadingScreen(Messages.websocketError);
|
||||||
|
};
|
||||||
|
|
||||||
|
var andThen = function (CMeditor) {
|
||||||
|
var $iframe = $('#pad-iframe').contents();
|
||||||
|
var $contentContainer = $iframe.find('#editorContainer');
|
||||||
|
var $previewContainer = $iframe.find('#previewContainer');
|
||||||
|
var $preview = $iframe.find('#preview');
|
||||||
|
$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');
|
||||||
|
window.open(href);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
var CodeMirror = Cryptpad.createCodemirror(ifrw, Cryptpad, null, CMeditor);
|
||||||
|
$iframe.find('.CodeMirror').addClass('fullPage');
|
||||||
|
editor = CodeMirror.editor;
|
||||||
|
|
||||||
|
var setIndentation = APP.setIndentation = function (units, useTabs) {
|
||||||
|
if (typeof(units) !== 'number') { return; }
|
||||||
|
editor.setOption('indentUnit', units);
|
||||||
|
editor.setOption('tabSize', units);
|
||||||
|
editor.setOption('indentWithTabs', useTabs);
|
||||||
|
};
|
||||||
|
|
||||||
|
var indentKey = 'indentUnit';
|
||||||
|
var useTabsKey = 'indentWithTabs';
|
||||||
|
|
||||||
|
var proxy = Cryptpad.getProxy();
|
||||||
|
|
||||||
|
var updateIndentSettings = APP.updateIndentSettings = function () {
|
||||||
|
var indentUnit = proxy.settings[indentKey];
|
||||||
|
var useTabs = proxy.settings[useTabsKey];
|
||||||
|
setIndentation(
|
||||||
|
typeof(indentUnit) === 'number'? indentUnit: 2,
|
||||||
|
typeof(useTabs) === 'boolean'? useTabs: false);
|
||||||
|
};
|
||||||
|
|
||||||
|
proxy.on('change', ['settings', indentKey], updateIndentSettings);
|
||||||
|
proxy.on('change', ['settings', useTabsKey], updateIndentSettings);
|
||||||
|
|
||||||
|
var $bar = $('#pad-iframe')[0].contentWindow.$('#cme_toolbox');
|
||||||
|
|
||||||
|
var isHistoryMode = false;
|
||||||
|
|
||||||
|
var setEditable = APP.setEditable = function (bool) {
|
||||||
|
if (readOnly && bool) { return; }
|
||||||
|
editor.setOption('readOnly', !bool);
|
||||||
|
};
|
||||||
|
|
||||||
|
var Title;
|
||||||
|
var UserList;
|
||||||
|
var Metadata;
|
||||||
|
|
||||||
|
var config = {
|
||||||
|
initialState: '{}',
|
||||||
|
websocketURL: Cryptpad.getWebsocketURL(),
|
||||||
|
channel: secret.channel,
|
||||||
|
// our public key
|
||||||
|
validateKey: secret.keys.validateKey || undefined,
|
||||||
|
readOnly: readOnly,
|
||||||
|
crypto: Crypto.createEncryptor(secret.keys),
|
||||||
|
network: Cryptpad.getNetwork(),
|
||||||
|
transformFunction: JsonOT.validate,
|
||||||
|
};
|
||||||
|
|
||||||
|
var canonicalize = function (t) { return t.replace(/\r\n/g, '\n'); };
|
||||||
|
|
||||||
|
var setHistory = function (bool, update) {
|
||||||
|
isHistoryMode = bool;
|
||||||
|
setEditable(!bool);
|
||||||
|
if (!bool && update) {
|
||||||
|
config.onRemote();
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
var initializing = true;
|
||||||
|
|
||||||
|
var stringifyInner = function (textValue) {
|
||||||
|
var obj = {
|
||||||
|
content: textValue,
|
||||||
|
metadata: {
|
||||||
|
users: UserList.userData,
|
||||||
|
defaultTitle: Title.defaultTitle
|
||||||
|
}
|
||||||
|
};
|
||||||
|
if (!initializing) {
|
||||||
|
obj.metadata.title = Title.title;
|
||||||
|
}
|
||||||
|
// set mode too...
|
||||||
|
obj.highlightMode = CodeMirror.highlightMode;
|
||||||
|
|
||||||
|
// stringify the json and send it into chainpad
|
||||||
|
return stringify(obj);
|
||||||
|
};
|
||||||
|
|
||||||
|
var forceDrawPreview = function () {
|
||||||
|
try {
|
||||||
|
DiffMd.apply(DiffMd.render(editor.getValue()), $preview);
|
||||||
|
} catch (e) { console.error(e); }
|
||||||
|
};
|
||||||
|
|
||||||
|
var drawPreview = Cryptpad.throttle(function () {
|
||||||
|
if (CodeMirror.highlightMode !== 'markdown') { return; }
|
||||||
|
if (!$previewContainer.is(':visible')) { return; }
|
||||||
|
forceDrawPreview();
|
||||||
|
}, 150);
|
||||||
|
|
||||||
|
var onLocal = config.onLocal = function () {
|
||||||
|
if (initializing) { return; }
|
||||||
|
if (isHistoryMode) { return; }
|
||||||
|
if (readOnly) { return; }
|
||||||
|
|
||||||
|
editor.save();
|
||||||
|
|
||||||
|
drawPreview();
|
||||||
|
|
||||||
|
var textValue = canonicalize(CodeMirror.$textarea.val());
|
||||||
|
var shjson = stringifyInner(textValue);
|
||||||
|
|
||||||
|
APP.patchText(shjson);
|
||||||
|
|
||||||
|
if (APP.realtime.getUserDoc() !== shjson) {
|
||||||
|
console.error("realtime.getUserDoc() !== shjson");
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
var mediaTagModes = [
|
||||||
|
'markdown',
|
||||||
|
'html',
|
||||||
|
'htmlembedded',
|
||||||
|
'htmlmixed',
|
||||||
|
'index.html',
|
||||||
|
'php',
|
||||||
|
'velocity',
|
||||||
|
'xml',
|
||||||
|
];
|
||||||
|
|
||||||
|
var onModeChanged = function (mode) {
|
||||||
|
var $codeMirror = $iframe.find('.CodeMirror');
|
||||||
|
window.clearTimeout(APP.previewTo);
|
||||||
|
$codeMirror.addClass('transition');
|
||||||
|
APP.previewTo = window.setTimeout(function () {
|
||||||
|
$codeMirror.removeClass('transition');
|
||||||
|
}, 500);
|
||||||
|
if (mediaTagModes.indexOf(mode) !== -1) {
|
||||||
|
APP.$mediaTagButton.show();
|
||||||
|
} else { APP.$mediaTagButton.hide(); }
|
||||||
|
|
||||||
|
if (mode === "markdown") {
|
||||||
|
APP.$previewButton.show();
|
||||||
|
Cryptpad.getPadAttribute('previewMode', function (e, data) {
|
||||||
|
if (e) { return void console.error(e); }
|
||||||
|
if (data !== false) {
|
||||||
|
$previewContainer.show();
|
||||||
|
APP.$previewButton.addClass('active');
|
||||||
|
$codeMirror.removeClass('fullPage');
|
||||||
|
}
|
||||||
|
});
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
APP.$previewButton.hide();
|
||||||
|
$previewContainer.hide();
|
||||||
|
APP.$previewButton.removeClass('active');
|
||||||
|
$codeMirror.addClass('fullPage');
|
||||||
|
if (typeof(APP.updateIndentSettings) === 'function') {
|
||||||
|
APP.updateIndentSettings();
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
config.onInit = function (info) {
|
||||||
|
UserList = Cryptpad.createUserList(info, config.onLocal, Cryptget, Cryptpad);
|
||||||
|
|
||||||
|
var titleCfg = { getHeadingText: CodeMirror.getHeadingText };
|
||||||
|
Title = Cryptpad.createTitle(titleCfg, config.onLocal, Cryptpad);
|
||||||
|
|
||||||
|
Metadata = Cryptpad.createMetadata(UserList, Title, null, Cryptpad);
|
||||||
|
|
||||||
|
var configTb = {
|
||||||
|
displayed: ['title', 'useradmin', 'spinner', 'lag', 'state', 'share', 'userlist', 'newpad', 'limit', 'upgrade'],
|
||||||
|
userList: UserList.getToolbarConfig(),
|
||||||
|
share: {
|
||||||
|
secret: secret,
|
||||||
|
channel: info.channel
|
||||||
|
},
|
||||||
|
title: Title.getTitleConfig(),
|
||||||
|
common: Cryptpad,
|
||||||
|
readOnly: readOnly,
|
||||||
|
ifrw: ifrw,
|
||||||
|
realtime: info.realtime,
|
||||||
|
network: info.network,
|
||||||
|
$container: $bar,
|
||||||
|
$contentContainer: $contentContainer
|
||||||
|
};
|
||||||
|
toolbar = APP.toolbar = Toolbar.create(configTb);
|
||||||
|
|
||||||
|
Title.setToolbar(toolbar);
|
||||||
|
CodeMirror.init(config.onLocal, Title, toolbar);
|
||||||
|
|
||||||
|
var $rightside = toolbar.$rightside;
|
||||||
|
var $drawer = toolbar.$drawer;
|
||||||
|
|
||||||
|
var editHash;
|
||||||
|
if (!readOnly) {
|
||||||
|
editHash = Cryptpad.getEditHashFromKeys(info.channel, secret.keys);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* add a history button */
|
||||||
|
var histConfig = {
|
||||||
|
onLocal: config.onLocal,
|
||||||
|
onRemote: config.onRemote,
|
||||||
|
setHistory: setHistory,
|
||||||
|
applyVal: function (val) {
|
||||||
|
var remoteDoc = JSON.parse(val || '{}').content;
|
||||||
|
editor.setValue(remoteDoc || '');
|
||||||
|
editor.save();
|
||||||
|
},
|
||||||
|
$toolbar: $bar
|
||||||
|
};
|
||||||
|
var $hist = Cryptpad.createButton('history', true, {histConfig: histConfig});
|
||||||
|
$drawer.append($hist);
|
||||||
|
|
||||||
|
/* save as template */
|
||||||
|
if (!Cryptpad.isTemplate(window.location.href)) {
|
||||||
|
var templateObj = {
|
||||||
|
rt: info.realtime,
|
||||||
|
Crypt: Cryptget,
|
||||||
|
getTitle: Title.getTitle
|
||||||
|
};
|
||||||
|
var $templateButton = Cryptpad.createButton('template', true, templateObj);
|
||||||
|
$rightside.append($templateButton);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* add an export button */
|
||||||
|
var $export = Cryptpad.createButton('export', true, {}, CodeMirror.exportText);
|
||||||
|
$drawer.append($export);
|
||||||
|
|
||||||
|
if (!readOnly) {
|
||||||
|
/* add an import button */
|
||||||
|
var $import = Cryptpad.createButton('import', true, {}, CodeMirror.importText);
|
||||||
|
$drawer.append($import);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* add a forget button */
|
||||||
|
var forgetCb = function (err) {
|
||||||
|
if (err) { return; }
|
||||||
|
setEditable(false);
|
||||||
|
};
|
||||||
|
var $forgetPad = Cryptpad.createButton('forget', true, {}, forgetCb);
|
||||||
|
$rightside.append($forgetPad);
|
||||||
|
|
||||||
|
var fileDialogCfg = {
|
||||||
|
$body: $iframe.find('body'),
|
||||||
|
onSelect: function (href) {
|
||||||
|
var parsed = Cryptpad.parsePadUrl(href);
|
||||||
|
var hexFileName = Cryptpad.base64ToHex(parsed.hashData.channel);
|
||||||
|
var src = '/blob/' + hexFileName.slice(0,2) + '/' + hexFileName;
|
||||||
|
var mt = '<media-tag src="' + src + '" data-crypto-key="cryptpad:' + parsed.hashData.key + '"></media-tag>';
|
||||||
|
editor.replaceSelection(mt);
|
||||||
|
},
|
||||||
|
data: APP
|
||||||
|
};
|
||||||
|
APP.$mediaTagButton = $('<button>', {
|
||||||
|
title: Messages.filePickerButton,
|
||||||
|
'class': 'rightside-button fa fa-picture-o',
|
||||||
|
style: 'font-size: 17px'
|
||||||
|
}).click(function () {
|
||||||
|
Cryptpad.createFileDialog(fileDialogCfg);
|
||||||
|
}).appendTo($rightside);
|
||||||
|
|
||||||
|
var $previewButton = APP.$previewButton = Cryptpad.createButton(null, true);
|
||||||
|
$previewButton.removeClass('fa-question').addClass('fa-eye');
|
||||||
|
$previewButton.attr('title', Messages.previewButtonTitle);
|
||||||
|
$previewButton.click(function () {
|
||||||
|
var $codeMirror = $iframe.find('.CodeMirror');
|
||||||
|
window.clearTimeout(APP.previewTo);
|
||||||
|
$codeMirror.addClass('transition');
|
||||||
|
APP.previewTo = window.setTimeout(function () {
|
||||||
|
$codeMirror.removeClass('transition');
|
||||||
|
}, 500);
|
||||||
|
if (CodeMirror.highlightMode !== 'markdown') {
|
||||||
|
$previewContainer.show();
|
||||||
|
}
|
||||||
|
$previewContainer.toggle();
|
||||||
|
if ($previewContainer.is(':visible')) {
|
||||||
|
forceDrawPreview();
|
||||||
|
$codeMirror.removeClass('fullPage');
|
||||||
|
Cryptpad.setPadAttribute('previewMode', true, function (e) {
|
||||||
|
if (e) { return console.log(e); }
|
||||||
|
});
|
||||||
|
$previewButton.addClass('active');
|
||||||
|
} else {
|
||||||
|
$codeMirror.addClass('fullPage');
|
||||||
|
$previewButton.removeClass('active');
|
||||||
|
Cryptpad.setPadAttribute('previewMode', false, function (e) {
|
||||||
|
if (e) { return console.log(e); }
|
||||||
|
});
|
||||||
|
}
|
||||||
|
});
|
||||||
|
$rightside.append($previewButton);
|
||||||
|
|
||||||
|
if (!readOnly) {
|
||||||
|
CodeMirror.configureTheme(function () {
|
||||||
|
CodeMirror.configureLanguage(null, onModeChanged);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
CodeMirror.configureTheme();
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
// set the hash
|
||||||
|
if (!readOnly) { Cryptpad.replaceHash(editHash); }
|
||||||
|
};
|
||||||
|
|
||||||
|
config.onReady = function (info) {
|
||||||
|
if (APP.realtime !== info.realtime) {
|
||||||
|
var realtime = APP.realtime = info.realtime;
|
||||||
|
APP.patchText = TextPatcher.create({
|
||||||
|
realtime: realtime,
|
||||||
|
//logging: true
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
var userDoc = APP.realtime.getUserDoc();
|
||||||
|
|
||||||
|
var isNew = false;
|
||||||
|
if (userDoc === "" || userDoc === "{}") { isNew = true; }
|
||||||
|
|
||||||
|
var newDoc = "";
|
||||||
|
if(userDoc !== "") {
|
||||||
|
var hjson = JSON.parse(userDoc);
|
||||||
|
|
||||||
|
if (typeof (hjson) !== 'object' || Array.isArray(hjson) ||
|
||||||
|
(typeof(hjson.type) !== 'undefined' && hjson.type !== 'code')) {
|
||||||
|
var errorText = Messages.typeError;
|
||||||
|
Cryptpad.errorLoadingScreen(errorText);
|
||||||
|
throw new Error(errorText);
|
||||||
|
}
|
||||||
|
|
||||||
|
newDoc = hjson.content;
|
||||||
|
|
||||||
|
if (hjson.highlightMode) {
|
||||||
|
CodeMirror.setMode(hjson.highlightMode, onModeChanged);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!CodeMirror.highlightMode) {
|
||||||
|
CodeMirror.setMode('markdown', onModeChanged);
|
||||||
|
console.log("%s => %s", CodeMirror.highlightMode, CodeMirror.$language.val());
|
||||||
|
}
|
||||||
|
|
||||||
|
// Update the user list (metadata) from the hyperjson
|
||||||
|
Metadata.update(userDoc);
|
||||||
|
|
||||||
|
if (newDoc) {
|
||||||
|
editor.setValue(newDoc);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (Cryptpad.initialName && Title.isDefaultTitle()) {
|
||||||
|
Title.updateTitle(Cryptpad.initialName);
|
||||||
|
}
|
||||||
|
|
||||||
|
Cryptpad.getPadAttribute('previewMode', function (e, data) {
|
||||||
|
if (e) { return void console.error(e); }
|
||||||
|
if (data === false && APP.$previewButton) {
|
||||||
|
APP.$previewButton.click();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
/*
|
||||||
|
// add the splitter
|
||||||
|
if (!$iframe.has('.cp-splitter').length) {
|
||||||
|
var $preview = $iframe.find('#previewContainer');
|
||||||
|
var splitter = $('<div>', {
|
||||||
|
'class': 'cp-splitter'
|
||||||
|
}).appendTo($preview);
|
||||||
|
|
||||||
|
$preview.on('scroll', function() {
|
||||||
|
splitter.css('top', $preview.scrollTop() + 'px');
|
||||||
|
});
|
||||||
|
|
||||||
|
var $target = $iframe.find('.CodeMirror');
|
||||||
|
|
||||||
|
splitter.on('mousedown', function (e) {
|
||||||
|
e.preventDefault();
|
||||||
|
var x = e.pageX;
|
||||||
|
var w = $target.width();
|
||||||
|
|
||||||
|
$iframe.on('mouseup mousemove', function handler(evt) {
|
||||||
|
if (evt.type === 'mouseup') {
|
||||||
|
$iframe.off('mouseup mousemove', handler);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
$target.css('width', (w - x + evt.pageX) + 'px');
|
||||||
|
});
|
||||||
|
});
|
||||||
|
}
|
||||||
|
*/
|
||||||
|
|
||||||
|
Cryptpad.removeLoadingScreen();
|
||||||
|
setEditable(true);
|
||||||
|
initializing = false;
|
||||||
|
|
||||||
|
onLocal(); // push local state to avoid parse errors later.
|
||||||
|
|
||||||
|
if (readOnly) {
|
||||||
|
config.onRemote();
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
UserList.getLastName(toolbar.$userNameButton, isNew);
|
||||||
|
var fmConfig = {
|
||||||
|
dropArea: $iframe.find('.CodeMirror'),
|
||||||
|
body: $iframe.find('body'),
|
||||||
|
onUploaded: function (ev, data) {
|
||||||
|
//var cursor = editor.getCursor();
|
||||||
|
//var cleanName = data.name.replace(/[\[\]]/g, '');
|
||||||
|
//var text = '';
|
||||||
|
var parsed = Cryptpad.parsePadUrl(data.url);
|
||||||
|
var hexFileName = Cryptpad.base64ToHex(parsed.hashData.channel);
|
||||||
|
var src = '/blob/' + hexFileName.slice(0,2) + '/' + hexFileName;
|
||||||
|
var mt = '<media-tag src="' + src + '" data-crypto-key="cryptpad:' + parsed.hashData.key + '"></media-tag>';
|
||||||
|
editor.replaceSelection(mt);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
APP.FM = Cryptpad.createFileManager(fmConfig);
|
||||||
|
};
|
||||||
|
|
||||||
|
config.onRemote = function () {
|
||||||
|
if (initializing) { return; }
|
||||||
|
if (isHistoryMode) { return; }
|
||||||
|
|
||||||
|
var oldDoc = canonicalize(CodeMirror.$textarea.val());
|
||||||
|
var shjson = APP.realtime.getUserDoc();
|
||||||
|
|
||||||
|
// Update the user list (metadata) from the hyperjson
|
||||||
|
Metadata.update(shjson);
|
||||||
|
|
||||||
|
var hjson = JSON.parse(shjson);
|
||||||
|
var remoteDoc = hjson.content;
|
||||||
|
|
||||||
|
var highlightMode = hjson.highlightMode;
|
||||||
|
if (highlightMode && highlightMode !== APP.highlightMode) {
|
||||||
|
CodeMirror.setMode(highlightMode, onModeChanged);
|
||||||
|
}
|
||||||
|
|
||||||
|
CodeMirror.setValueAndCursor(oldDoc, remoteDoc, TextPatcher);
|
||||||
|
drawPreview();
|
||||||
|
|
||||||
|
if (!readOnly) {
|
||||||
|
var textValue = canonicalize(CodeMirror.$textarea.val());
|
||||||
|
var shjson2 = stringifyInner(textValue);
|
||||||
|
if (shjson2 !== shjson) {
|
||||||
|
console.error("shjson2 !== shjson");
|
||||||
|
TextPatcher.log(shjson, TextPatcher.diff(shjson, shjson2));
|
||||||
|
APP.patchText(shjson2);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (oldDoc !== remoteDoc) { Cryptpad.notify(); }
|
||||||
|
};
|
||||||
|
|
||||||
|
config.onAbort = function () {
|
||||||
|
// inform of network disconnect
|
||||||
|
setEditable(false);
|
||||||
|
toolbar.failed();
|
||||||
|
Cryptpad.alert(Messages.common_connectionLost, undefined, true);
|
||||||
|
};
|
||||||
|
|
||||||
|
config.onConnectionChange = function (info) {
|
||||||
|
setEditable(info.state);
|
||||||
|
toolbar.failed();
|
||||||
|
if (info.state) {
|
||||||
|
initializing = true;
|
||||||
|
toolbar.reconnecting(info.myId);
|
||||||
|
Cryptpad.findOKButton().click();
|
||||||
|
} else {
|
||||||
|
Cryptpad.alert(Messages.common_connectionLost, undefined, true);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
config.onError = onConnectError;
|
||||||
|
|
||||||
|
APP.realtime = Realtime.start(config);
|
||||||
|
|
||||||
|
editor.on('change', onLocal);
|
||||||
|
|
||||||
|
Cryptpad.onLogout(function () { setEditable(false); });
|
||||||
|
};
|
||||||
|
|
||||||
|
var interval = 100;
|
||||||
|
var second = function (CM) {
|
||||||
|
Cryptpad.ready(function () {
|
||||||
|
andThen(CM);
|
||||||
|
Cryptpad.reportAppUsage();
|
||||||
|
});
|
||||||
|
Cryptpad.onError(function (info) {
|
||||||
|
if (info && info.type === "store") {
|
||||||
|
onConnectError();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
var first = function () {
|
||||||
|
if (ifrw.CodeMirror) {
|
||||||
|
second(ifrw.CodeMirror);
|
||||||
|
} else {
|
||||||
|
console.log("CodeMirror was not defined. Trying again in %sms", interval);
|
||||||
|
setTimeout(first, interval);
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
first();
|
||||||
|
});
|
||||||
|
});
|
||||||
Loading…
x
Reference in New Issue
Block a user