From 92ea03d7d9ed7fea77c55cd7c71ede8b85bfec36 Mon Sep 17 00:00:00 2001 From: yflory Date: Fri, 21 Apr 2017 17:31:47 +0200 Subject: [PATCH] View and restore the history of a pad --- customize.dist/src/less/toolbar.less | 11 ++ customize.dist/toolbar.css | 9 ++ customize.dist/translations/messages.fr.js | 11 ++ customize.dist/translations/messages.js | 11 ++ www/code/main.js | 56 +++++-- www/common/common-history.js | 172 +++++++++++++++------ www/common/cryptpad-common.js | 16 +- www/pad/main.js | 50 +++++- www/slide/main.js | 49 +++++- 9 files changed, 314 insertions(+), 71 deletions(-) diff --git a/customize.dist/src/less/toolbar.less b/customize.dist/src/less/toolbar.less index 5ff104c46..5e4f74dcf 100644 --- a/customize.dist/src/less/toolbar.less +++ b/customize.dist/src/less/toolbar.less @@ -422,6 +422,17 @@ display: inline-block; input { width: 50px; } } + .gotoInput { + vertical-align: middle; + } +} +.cke_toolbox .cryptpad-toolbar-history { + input.gotoInput { + background: white; + height: 20px; + padding: 3px 3px; + border-radius: 5px; + } } .cryptpad-spinner { height: 16px; diff --git a/customize.dist/toolbar.css b/customize.dist/toolbar.css index 5c9fd2094..1165c6df4 100644 --- a/customize.dist/toolbar.css +++ b/customize.dist/toolbar.css @@ -488,6 +488,15 @@ .cryptpad-toolbar-history .goto input { width: 50px; } +.cryptpad-toolbar-history .gotoInput { + vertical-align: middle; +} +.cke_toolbox .cryptpad-toolbar-history input.gotoInput { + background: white; + height: 20px; + padding: 3px 3px; + border-radius: 5px; +} .cryptpad-spinner { height: 16px; width: 16px; diff --git a/customize.dist/translations/messages.fr.js b/customize.dist/translations/messages.fr.js index 6d8b3899a..f6e9bc23d 100644 --- a/customize.dist/translations/messages.fr.js +++ b/customize.dist/translations/messages.fr.js @@ -115,6 +115,17 @@ define(function () { out.cancel = "Annuler"; out.cancelButton = 'Annuler (Echap)'; + out.historyButton = "Afficher l'historique du document"; + out.history_next = "Voir la version suivante"; + out.history_prev = "Voir la version précédente"; + out.history_goTo = "Voir la version sélectionnée"; + out.history_close = "Retour"; + out.history_closeTitle = "Fermer l'historique"; + out.history_restore = "Restaurer"; + out.history_restoreTitle = "Restaurer la version du document sélectionnée"; + out.history_restorePrompt = "Êtes-vous sûr de vouloir remplacer la version actuelle du document par la version affichée ?"; + out.history_restoreDone = "Document restauré"; + // Polls out.poll_title = "Sélecteur de date Zero Knowledge"; diff --git a/customize.dist/translations/messages.js b/customize.dist/translations/messages.js index 66f771033..035560d0e 100644 --- a/customize.dist/translations/messages.js +++ b/customize.dist/translations/messages.js @@ -117,6 +117,17 @@ define(function () { out.cancel = "Cancel"; out.cancelButton = 'Cancel (esc)'; + out.historyButton = "Display the document history"; + out.history_next = "Go to the next version"; + out.history_prev = "Go to the previous version"; + out.history_goTo = "Go to the selected version"; + out.history_close = "Back"; + out.history_closeTitle = "Close the history"; + out.history_restore = "Restore"; + out.history_restoreTitle = "Restore the selected version of the document"; + out.history_restorePrompt = "Are you sure you want to replace the current version of the document by the displayed one?"; + out.history_restoreDone = "Document restored"; + // Polls out.poll_title = "Zero Knowledge Date Picker"; diff --git a/www/code/main.js b/www/code/main.js index 4a03efd09..5f55fadae 100644 --- a/www/code/main.js +++ b/www/code/main.js @@ -54,6 +54,8 @@ define([ var defaultName = Cryptpad.getDefaultName(parsedHash); var initialState = Messages.codeInitialState; + var isHistoryMode = false; + var editor = module.editor = CMeditor.fromTextArea($textarea[0], { lineNumbers: true, lineWrapping: true, @@ -164,6 +166,14 @@ define([ 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 isDefaultTitle = function () { var parsed = Cryptpad.parsePadUrl(window.location.href); return Cryptpad.isDefaultName(parsed, document.title); @@ -191,6 +201,7 @@ define([ var onLocal = config.onLocal = function () { if (initializing) { return; } + if (isHistoryMode) { return; } if (readOnly) { return; } editor.save(); @@ -372,7 +383,7 @@ define([ var onInit = config.onInit = function (info) { userList = info.userList; - var config = { + var configTb = { displayed: ['useradmin', 'spinner', 'lag', 'state', 'share', 'userlist', 'newpad'], userData: userData, readOnly: readOnly, @@ -388,8 +399,7 @@ define([ }, common: Cryptpad }; - if (readOnly) {delete config.changeNameID; } - toolbar = module.toolbar = Toolbar.create($bar, info.myID, info.realtime, info.getLag, userList, config); + toolbar = module.toolbar = Toolbar.create($bar, info.myID, info.realtime, info.getLag, userList, configTb); var $rightside = $bar.find('.' + Toolbar.constants.rightside); var $userBlock = $bar.find('.' + Toolbar.constants.username); @@ -402,19 +412,36 @@ define([ editHash = Cryptpad.getEditHashFromKeys(info.channel, secret.keys); } - /* add an export button */ - var $hist = Cryptpad.createButton(); - var historyRender = function () {}; - var historyClose = function () { - // TODO: enable onlocal, onremote... (or at least the display part) + /* add a history button */ + var histConfig = {}; + histConfig.onRender = function (val) { + if (typeof val === "undefined") { return; } + try { + var hjson = JSON.parse(val || '{}'); + var remoteDoc = hjson.content; + editor.setValue(remoteDoc || ''); + editor.save(); + } catch (e) { + // Probably a parse error + console.error(e); + } }; - var historyTodo = function (hist) { - hist.display($bar, historyRender, historyClose); + histConfig.onClose = function () { + // Close button clicked + setHistory(false, true); }; - $hist.removeClass('fa-question').addClass('fa-history').click(function () { - // TODO: disable onlocal, onremote... - Cryptpad.getHistory(historyTodo); - }); + histConfig.onRevert = function () { + // Revert button clicked + setHistory(false, false); + config.onLocal(); + config.onRemote(); + }; + histConfig.onReady = function () { + // Called when the history is loaded and the UI displayed + setHistory(true); + }; + histConfig.$toolbar = $bar; + var $hist = Cryptpad.createButton('history', true, {histConfig: histConfig}); $rightside.append($hist); /* save as template */ @@ -663,6 +690,7 @@ define([ var onRemote = config.onRemote = function () { if (initializing) { return; } + if (isHistoryMode) { return; } var scroll = editor.getScrollInfo(); var oldDoc = canonicalize($textarea.val()); diff --git a/www/common/common-history.js b/www/common/common-history.js index d790ab553..c7d250bce 100644 --- a/www/common/common-history.js +++ b/www/common/common-history.js @@ -9,6 +9,17 @@ define([ var History = {}; + var getStates = function (rt) { + var states = []; + var b = rt.getAuthBlock(); + if (b) { states.unshift(b.getContent().doc); } + while (b.getParent()) { + b = b.getParent(); + states.unshift(b.getContent().doc); + } + return states; + }; + /* TODO * Implement GET_FULL_HISTORY serverside * All the history messages should be ['FULL_HISTORY', wc.id, msg] @@ -23,22 +34,22 @@ define([ var wcId = common.hrefToHexChannelId(window.location.href); var createRealtime = function(chan) { - console.log(ChainPad); return ChainPad.create({ userName: 'history', initialState: '', transformFunction: JsonOT.validate, - logLevel: 0 + logLevel: 1, + noPrune: true }); }; var realtime = createRealtime(); - var secret = Cryptpad.getSecrets(); + var secret = common.getSecrets(); var crypto = Crypto.createEncryptor(secret.keys); var to = window.setTimeout(function () { cb('[GET_FULL_HISTORY_TIMEOUT]'); - }, 3000); + }, 30000); var parse = function (msg) { try { @@ -50,104 +61,169 @@ define([ var onMsg = function (msg) { var parsed = parse(msg); if (parsed[0] === 'FULL_HISTORY_END') { + console.log('END'); window.clearTimeout(to); cb(null, realtime); return; } if (parsed[0] !== 'FULL_HISTORY') { return; } - var msg = parsed[1]; - var decryptedMsg = crypto.decrypt(msg, secret.keys.validateKey); - realtime.message(decryptedMsg); + msg = parsed[1][4]; + if (msg) { + msg = msg.replace(/^cp\|/, ''); + var decryptedMsg = crypto.decrypt(msg, secret.keys.validateKey); + realtime.message(decryptedMsg); + } }; network.on('message', function (msg, sender) { onMsg(msg); }); - network.sendto(hkn, JSON.stringify(['GET_FULL_HISTORY', wcId])); + network.sendto(hkn, JSON.stringify(['GET_FULL_HISTORY', wcId, secret.keys.validateKey])); }; - var create = History.create = function (common, cb) { - var exp = {}; + var create = History.create = function (common, config) { + if (!config.$toolbar) { return void console.error("config.$toolbar is undefined");} + var $toolbar = config.$toolbar; + var noFunc = function () {}; + var render = config.onRender || noFunc; + var onClose = config.onClose || noFunc; + var onRevert = config.onRevert || noFunc; + var onReady = config.onReady || noFunc; - var states = exp.states = ['a', 'b', 'c']; - var c = exp.current = states.length - 1; - console.log(c); + var Messages = common.Messages; + + var realtime; + + var states = []; //getStates(rt); //['a', 'b', 'c']; + var c = states.length - 1; + + var $hist = $toolbar.find('.cryptpad-toolbar-history'); + var $left = $toolbar.find('.cryptpad-toolbar-leftside'); + var $right = $toolbar.find('.cryptpad-toolbar-rightside'); var onUpdate; - var update = exp.update = function () { - states = []; + var update = function () { + if (!realtime) { return []; } + states = getStates(realtime); if (typeof onUpdate === "function") { onUpdate(); } return states; }; - var get = exp.get = function (i) { + // Get the content of the selected version, and change the version number + var get = function (i) { i = parseInt(i); - console.log('getting', i); - if (typeof(i) !== "number" || i < 0 || i > states.length - 1) { return; } - var hash = states[i]; + if (isNaN(i)) { return; } + if (i < 0) { i = 0; } + if (i > states.length - 1) { i = states.length - 1; } + var val = states[i]; c = i; if (typeof onUpdate === "function") { onUpdate(); } - return ''; + $hist.find('.next, .previous').show(); + if (c === states.length - 1) { $hist.find('.next').hide(); } + if (c === 0) { $hist.find('.previous').hide(); } + return val || ''; }; - var getNext = exp.getNext = function () { - if (c < states.length - 1) { return get(++c); } + var getNext = function (step) { + return typeof step === "number" ? get(c + step) : get(c + 1); }; - var getPrevious = exp.getPrevious = function () { - if (c > 0) { return get(--c); } + var getPrevious = function (step) { + return typeof step === "number" ? get(c - step) : get(c - 1); }; - var display = exp.display = function ($toolbar, render, onClose) { - var $hist = $toolbar.find('.cryptpad-toolbar-history').html('').show(); - var $left = $toolbar.find('.cryptpad-toolbar-leftside').hide(); - var $right = $toolbar.find('.cryptpad-toolbar-rightside').hide(); - - var $prev =$('