mirror of
https://github.com/ether/etherpad-lite.git
synced 2025-04-22 00:16:15 -04:00
Added saved revisions support
This commit is contained in:
parent
9c9e5db72d
commit
f631e1e8ab
11 changed files with 77 additions and 520 deletions
|
@ -13,6 +13,7 @@ var padManager = require("./PadManager");
|
||||||
var padMessageHandler = require("../handler/PadMessageHandler");
|
var padMessageHandler = require("../handler/PadMessageHandler");
|
||||||
var readOnlyManager = require("./ReadOnlyManager");
|
var readOnlyManager = require("./ReadOnlyManager");
|
||||||
var crypto = require("crypto");
|
var crypto = require("crypto");
|
||||||
|
var randomString = require("../utils/randomstring");
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Copied from the Etherpad source code. It converts Windows line breaks to Unix line breaks and convert Tabs to spaces
|
* Copied from the Etherpad source code. It converts Windows line breaks to Unix line breaks and convert Tabs to spaces
|
||||||
|
@ -32,7 +33,7 @@ var Pad = function Pad(id) {
|
||||||
this.publicStatus = false;
|
this.publicStatus = false;
|
||||||
this.passwordHash = null;
|
this.passwordHash = null;
|
||||||
this.id = id;
|
this.id = id;
|
||||||
|
this.savedRevisions = [];
|
||||||
};
|
};
|
||||||
|
|
||||||
exports.Pad = Pad;
|
exports.Pad = Pad;
|
||||||
|
@ -466,6 +467,31 @@ Pad.prototype.isPasswordProtected = function isPasswordProtected() {
|
||||||
return this.passwordHash != null;
|
return this.passwordHash != null;
|
||||||
};
|
};
|
||||||
|
|
||||||
|
Pad.prototype.addSavedRevision = function addSavedRevision(revNum, savedById, label) {
|
||||||
|
//if this revision is already saved, return silently
|
||||||
|
for(var i in this.savedRevisions){
|
||||||
|
if(this.savedRevisions.revNum === revNum){
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
//build the saved revision object
|
||||||
|
var savedRevision = {};
|
||||||
|
savedRevision.revNum = revNum;
|
||||||
|
savedRevision.savedById = savedById;
|
||||||
|
savedRevision.label = label || "Revision " + revNum;
|
||||||
|
savedRevision.timestamp = new Date().getTime();
|
||||||
|
savedRevision.id = randomString(10);
|
||||||
|
|
||||||
|
//save this new saved revision
|
||||||
|
this.savedRevisions.push(savedRevision);
|
||||||
|
this.saveToDatabase();
|
||||||
|
};
|
||||||
|
|
||||||
|
Pad.prototype.getSavedRevisions = function getSavedRevisions() {
|
||||||
|
return this.savedRevisions;
|
||||||
|
};
|
||||||
|
|
||||||
/* Crypto helper methods */
|
/* Crypto helper methods */
|
||||||
|
|
||||||
function hash(password, salt)
|
function hash(password, salt)
|
||||||
|
|
|
@ -184,6 +184,11 @@ exports.handleMessage = function(client, message)
|
||||||
{
|
{
|
||||||
handleChatMessage(client, message);
|
handleChatMessage(client, message);
|
||||||
}
|
}
|
||||||
|
else if(message.type == "COLLABROOM" &&
|
||||||
|
message.data.type == "SAVE_REVISION")
|
||||||
|
{
|
||||||
|
handleSaveRevisionMessage(client, message);
|
||||||
|
}
|
||||||
else if(message.type == "COLLABROOM" &&
|
else if(message.type == "COLLABROOM" &&
|
||||||
message.data.type == "CLIENT_MESSAGE" &&
|
message.data.type == "CLIENT_MESSAGE" &&
|
||||||
message.data.payload.type == "suggestUserName")
|
message.data.payload.type == "suggestUserName")
|
||||||
|
@ -197,6 +202,23 @@ exports.handleMessage = function(client, message)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Handles a save revision message
|
||||||
|
* @param client the client that send this message
|
||||||
|
* @param message the message from the client
|
||||||
|
*/
|
||||||
|
function handleSaveRevisionMessage(client, message){
|
||||||
|
var padId = session2pad[client.id];
|
||||||
|
var userId = sessioninfos[client.id].author;
|
||||||
|
|
||||||
|
padManager.getPad(padId, function(err, pad)
|
||||||
|
{
|
||||||
|
if(ERR(err)) return;
|
||||||
|
|
||||||
|
pad.addSavedRevision(pad.head, userId);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Handles a Chat Message
|
* Handles a Chat Message
|
||||||
* @param client the client that send this message
|
* @param client the client that send this message
|
||||||
|
|
|
@ -165,6 +165,7 @@ function createTimesliderClientVars (padId, callback)
|
||||||
hooks: [],
|
hooks: [],
|
||||||
initialStyledContents: {}
|
initialStyledContents: {}
|
||||||
};
|
};
|
||||||
|
|
||||||
var pad;
|
var pad;
|
||||||
var initialChangesets = [];
|
var initialChangesets = [];
|
||||||
|
|
||||||
|
@ -179,6 +180,12 @@ function createTimesliderClientVars (padId, callback)
|
||||||
callback();
|
callback();
|
||||||
});
|
});
|
||||||
},
|
},
|
||||||
|
//get all saved revisions and add them
|
||||||
|
function(callback)
|
||||||
|
{
|
||||||
|
clientVars.savedRevisions = pad.getSavedRevisions();
|
||||||
|
callback();
|
||||||
|
},
|
||||||
//get all authors and add them to
|
//get all authors and add them to
|
||||||
function(callback)
|
function(callback)
|
||||||
{
|
{
|
||||||
|
|
|
@ -1065,6 +1065,9 @@ margin-top: 1px;
|
||||||
background-position: 0px -183px;
|
background-position: 0px -183px;
|
||||||
display: inline-block;
|
display: inline-block;
|
||||||
}
|
}
|
||||||
|
.buttonicon-savedRevision {
|
||||||
|
background-position: 0px -493px
|
||||||
|
}
|
||||||
|
|
||||||
#usericon
|
#usericon
|
||||||
{
|
{
|
||||||
|
|
Binary file not shown.
Before Width: | Height: | Size: 5.4 KiB After Width: | Height: | Size: 8.1 KiB |
BIN
static/img/star.png
Normal file
BIN
static/img/star.png
Normal file
Binary file not shown.
After Width: | Height: | Size: 3.2 KiB |
|
@ -42,7 +42,7 @@ var padeditbar = require('/pad_editbar').padeditbar;
|
||||||
var padeditor = require('/pad_editor').padeditor;
|
var padeditor = require('/pad_editor').padeditor;
|
||||||
var padimpexp = require('/pad_impexp').padimpexp;
|
var padimpexp = require('/pad_impexp').padimpexp;
|
||||||
var padmodals = require('/pad_modals').padmodals;
|
var padmodals = require('/pad_modals').padmodals;
|
||||||
var padsavedrevs = require('/pad_savedrevs').padsavedrevs;
|
var padsavedrevs = require('/pad_savedrevs');
|
||||||
var paduserlist = require('/pad_userlist').paduserlist;
|
var paduserlist = require('/pad_userlist').paduserlist;
|
||||||
var padutils = require('/pad_utils').padutils;
|
var padutils = require('/pad_utils').padutils;
|
||||||
|
|
||||||
|
@ -488,7 +488,7 @@ var pad = {
|
||||||
guestPolicy: pad.padOptions.guestPolicy
|
guestPolicy: pad.padOptions.guestPolicy
|
||||||
}, this);
|
}, this);
|
||||||
padimpexp.init(this);
|
padimpexp.init(this);
|
||||||
padsavedrevs.init(clientVars.initialRevisionList, this);
|
padsavedrevs.init(this);
|
||||||
|
|
||||||
padeditor.init(postAceInit, pad.padOptions.view || {}, this);
|
padeditor.init(postAceInit, pad.padOptions.view || {}, this);
|
||||||
|
|
||||||
|
|
|
@ -446,12 +446,6 @@ var paddocbar = (function()
|
||||||
enabled = false;
|
enabled = false;
|
||||||
self.render();
|
self.render();
|
||||||
},
|
},
|
||||||
handleResizePage: function()
|
|
||||||
{
|
|
||||||
// Side-step circular reference. This should be injected.
|
|
||||||
var padsavedrevs = require('/pad_savedrevs').padsavedrevs;
|
|
||||||
padsavedrevs.handleResizePage();
|
|
||||||
},
|
|
||||||
hideLaterIfNoOtherInteraction: function()
|
hideLaterIfNoOtherInteraction: function()
|
||||||
{
|
{
|
||||||
return padutils.getCancellableAction('hide-docbar-panel', function()
|
return padutils.getCancellableAction('hide-docbar-panel', function()
|
||||||
|
|
|
@ -22,7 +22,7 @@
|
||||||
|
|
||||||
var padutils = require('/pad_utils').padutils;
|
var padutils = require('/pad_utils').padutils;
|
||||||
var padeditor = require('/pad_editor').padeditor;
|
var padeditor = require('/pad_editor').padeditor;
|
||||||
var padsavedrevs = require('/pad_savedrevs').padsavedrevs;
|
var padsavedrevs = require('/pad_savedrevs');
|
||||||
|
|
||||||
function indexOf(array, value) {
|
function indexOf(array, value) {
|
||||||
for (var i = 0, ii = array.length; i < ii; i++) {
|
for (var i = 0, ii = array.length; i < ii; i++) {
|
||||||
|
@ -131,7 +131,7 @@ var padeditbar = (function()
|
||||||
{
|
{
|
||||||
self.toogleDropDown("importexport");
|
self.toogleDropDown("importexport");
|
||||||
}
|
}
|
||||||
else if (cmd == 'save')
|
else if (cmd == 'savedRevision')
|
||||||
{
|
{
|
||||||
padsavedrevs.saveNow();
|
padsavedrevs.saveNow();
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,11 +1,5 @@
|
||||||
/**
|
/**
|
||||||
* This code is mostly from the old Etherpad. Please help us to comment this code.
|
* Copyright 2012 Peter 'Pita' Martischka
|
||||||
* This helps other people to understand this code better and helps them to improve it.
|
|
||||||
* TL;DR COMMENTS ON THIS FILE ARE HIGHLY APPRECIATED
|
|
||||||
*/
|
|
||||||
|
|
||||||
/**
|
|
||||||
* Copyright 2009 Google Inc.
|
|
||||||
*
|
*
|
||||||
* Licensed under the Apache License, Version 2.0 (the "License");
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
* you may not use this file except in compliance with the License.
|
* you may not use this file except in compliance with the License.
|
||||||
|
@ -20,507 +14,13 @@
|
||||||
* limitations under the License.
|
* limitations under the License.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
var padutils = require('/pad_utils').padutils;
|
var pad;
|
||||||
var paddocbar = require('/pad_docbar').paddocbar;
|
|
||||||
|
|
||||||
var padsavedrevs = (function()
|
exports.saveNow = function(){
|
||||||
{
|
pad.collabClient.sendMessage({"type": "SAVE_REVISION"});
|
||||||
|
alert("This revision is now marked as a saved revision");
|
||||||
function reversedCopy(L)
|
|
||||||
{
|
|
||||||
var L2 = L.slice();
|
|
||||||
L2.reverse();
|
|
||||||
return L2;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
function makeRevisionBox(revisionInfo, rnum)
|
exports.init = function(_pad){
|
||||||
{
|
|
||||||
var box = $('<div class="srouterbox">' + '<div class="srinnerbox">' + '<a href="javascript:void(0)" class="srname"><!-- --></a>' + '<div class="sractions"><a class="srview" href="javascript:void(0)" target="_blank">view</a> | <a class="srrestore" href="javascript:void(0)">restore</a></div>' + '<div class="srtime"><!-- --></div>' + '<div class="srauthor"><!-- --></div>' + '<img class="srtwirly" src="static/img/misc/status-ball.gif">' + '</div></div>');
|
|
||||||
setBoxLabel(box, revisionInfo.label);
|
|
||||||
setBoxTimestamp(box, revisionInfo.timestamp);
|
|
||||||
box.find(".srauthor").html("by " + padutils.escapeHtml(revisionInfo.savedBy));
|
|
||||||
var viewLink = '/ep/pad/view/' + pad.getPadId() + '/' + revisionInfo.id;
|
|
||||||
box.find(".srview").attr('href', viewLink);
|
|
||||||
var restoreLink = 'javascript:void(require('+JSON.stringify(module.id)+').padsavedrevs.restoreRevision(' + JSON.stringify(rnum) + ');';
|
|
||||||
box.find(".srrestore").attr('href', restoreLink);
|
|
||||||
box.find(".srname").click(function(evt)
|
|
||||||
{
|
|
||||||
editRevisionLabel(rnum, box);
|
|
||||||
});
|
|
||||||
return box;
|
|
||||||
}
|
|
||||||
|
|
||||||
function setBoxLabel(box, label)
|
|
||||||
{
|
|
||||||
box.find(".srname").html(padutils.escapeHtml(label)).attr('title', label);
|
|
||||||
}
|
|
||||||
|
|
||||||
function setBoxTimestamp(box, timestamp)
|
|
||||||
{
|
|
||||||
box.find(".srtime").html(padutils.escapeHtml(
|
|
||||||
padutils.timediff(new Date(timestamp))));
|
|
||||||
}
|
|
||||||
|
|
||||||
function getNthBox(n)
|
|
||||||
{
|
|
||||||
return $("#savedrevisions .srouterbox").eq(n);
|
|
||||||
}
|
|
||||||
|
|
||||||
function editRevisionLabel(rnum, box)
|
|
||||||
{
|
|
||||||
var input = $('<input type="text" class="srnameedit"/>');
|
|
||||||
box.find(".srnameedit").remove(); // just in case
|
|
||||||
var label = box.find(".srname");
|
|
||||||
input.width(label.width());
|
|
||||||
input.height(label.height());
|
|
||||||
input.css('top', label.position().top);
|
|
||||||
input.css('left', label.position().left);
|
|
||||||
label.after(input);
|
|
||||||
label.css('opacity', 0);
|
|
||||||
|
|
||||||
function endEdit()
|
|
||||||
{
|
|
||||||
input.remove();
|
|
||||||
label.css('opacity', 1);
|
|
||||||
}
|
|
||||||
var rev = currentRevisionList[rnum];
|
|
||||||
var oldLabel = rev.label;
|
|
||||||
input.blur(function()
|
|
||||||
{
|
|
||||||
var newLabel = input.val();
|
|
||||||
if (newLabel && newLabel != oldLabel)
|
|
||||||
{
|
|
||||||
relabelRevision(rnum, newLabel);
|
|
||||||
}
|
|
||||||
endEdit();
|
|
||||||
});
|
|
||||||
input.val(rev.label).focus().select();
|
|
||||||
padutils.bindEnterAndEscape(input, function onEnter()
|
|
||||||
{
|
|
||||||
input.blur();
|
|
||||||
}, function onEscape()
|
|
||||||
{
|
|
||||||
input.val('').blur();
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
function relabelRevision(rnum, newLabel)
|
|
||||||
{
|
|
||||||
var rev = currentRevisionList[rnum];
|
|
||||||
$.ajax(
|
|
||||||
{
|
|
||||||
type: 'post',
|
|
||||||
url: '/ep/pad/saverevisionlabel',
|
|
||||||
data: {
|
|
||||||
userId: pad.getUserId(),
|
|
||||||
padId: pad.getPadId(),
|
|
||||||
revId: rev.id,
|
|
||||||
newLabel: newLabel
|
|
||||||
},
|
|
||||||
success: success,
|
|
||||||
error: error
|
|
||||||
});
|
|
||||||
|
|
||||||
function success(text)
|
|
||||||
{
|
|
||||||
var newRevisionList = JSON.parse(text);
|
|
||||||
self.newRevisionList(newRevisionList);
|
|
||||||
pad.sendClientMessage(
|
|
||||||
{
|
|
||||||
type: 'revisionLabel',
|
|
||||||
revisionList: reversedCopy(currentRevisionList),
|
|
||||||
savedBy: pad.getUserName(),
|
|
||||||
newLabel: newLabel
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
function error(e)
|
|
||||||
{
|
|
||||||
alert("Oops! There was an error saving that revision label. Please try again later.");
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
var currentRevisionList = [];
|
|
||||||
|
|
||||||
function setRevisionList(newRevisionList, noAnimation)
|
|
||||||
{
|
|
||||||
// deals with changed labels and new added revisions
|
|
||||||
for (var i = 0; i < currentRevisionList.length; i++)
|
|
||||||
{
|
|
||||||
var a = currentRevisionList[i];
|
|
||||||
var b = newRevisionList[i];
|
|
||||||
if (b.label != a.label)
|
|
||||||
{
|
|
||||||
setBoxLabel(getNthBox(i), b.label);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
for (var j = currentRevisionList.length; j < newRevisionList.length; j++)
|
|
||||||
{
|
|
||||||
var newBox = makeRevisionBox(newRevisionList[j], j);
|
|
||||||
$("#savedrevs-scrollinner").append(newBox);
|
|
||||||
newBox.css('left', j * REVISION_BOX_WIDTH);
|
|
||||||
}
|
|
||||||
var newOnes = (newRevisionList.length > currentRevisionList.length);
|
|
||||||
currentRevisionList = newRevisionList;
|
|
||||||
if (newOnes)
|
|
||||||
{
|
|
||||||
setDesiredScroll(getMaxScroll());
|
|
||||||
if (noAnimation)
|
|
||||||
{
|
|
||||||
setScroll(desiredScroll);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (!noAnimation)
|
|
||||||
{
|
|
||||||
var nameOfLast = currentRevisionList[currentRevisionList.length - 1].label;
|
|
||||||
displaySavedTip(nameOfLast);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
function refreshRevisionList()
|
|
||||||
{
|
|
||||||
for (var i = 0; i < currentRevisionList.length; i++)
|
|
||||||
{
|
|
||||||
var r = currentRevisionList[i];
|
|
||||||
var box = getNthBox(i);
|
|
||||||
setBoxTimestamp(box, r.timestamp);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
var savedTipAnimator = padutils.makeShowHideAnimator(function(state)
|
|
||||||
{
|
|
||||||
if (state == -1)
|
|
||||||
{
|
|
||||||
$("#revision-notifier").css('opacity', 0).css('display', 'block');
|
|
||||||
}
|
|
||||||
else if (state == 0)
|
|
||||||
{
|
|
||||||
$("#revision-notifier").css('opacity', 1);
|
|
||||||
}
|
|
||||||
else if (state == 1)
|
|
||||||
{
|
|
||||||
$("#revision-notifier").css('opacity', 0).css('display', 'none');
|
|
||||||
}
|
|
||||||
else if (state < 0)
|
|
||||||
{
|
|
||||||
$("#revision-notifier").css('opacity', 1);
|
|
||||||
}
|
|
||||||
else if (state > 0)
|
|
||||||
{
|
|
||||||
$("#revision-notifier").css('opacity', 1 - state);
|
|
||||||
}
|
|
||||||
}, false, 25, 300);
|
|
||||||
|
|
||||||
function displaySavedTip(text)
|
|
||||||
{
|
|
||||||
$("#revision-notifier .name").html(padutils.escapeHtml(text));
|
|
||||||
savedTipAnimator.show();
|
|
||||||
padutils.cancelActions("hide-revision-notifier");
|
|
||||||
var hideLater = padutils.getCancellableAction("hide-revision-notifier", function()
|
|
||||||
{
|
|
||||||
savedTipAnimator.hide();
|
|
||||||
});
|
|
||||||
window.setTimeout(hideLater, 3000);
|
|
||||||
}
|
|
||||||
|
|
||||||
var REVISION_BOX_WIDTH = 120;
|
|
||||||
var curScroll = 0; // distance between left of revisions and right of view
|
|
||||||
var desiredScroll = 0;
|
|
||||||
|
|
||||||
function getScrollWidth()
|
|
||||||
{
|
|
||||||
return REVISION_BOX_WIDTH * currentRevisionList.length;
|
|
||||||
}
|
|
||||||
|
|
||||||
function getViewportWidth()
|
|
||||||
{
|
|
||||||
return $("#savedrevs-scrollouter").width();
|
|
||||||
}
|
|
||||||
|
|
||||||
function getMinScroll()
|
|
||||||
{
|
|
||||||
return Math.min(getViewportWidth(), getScrollWidth());
|
|
||||||
}
|
|
||||||
|
|
||||||
function getMaxScroll()
|
|
||||||
{
|
|
||||||
return getScrollWidth();
|
|
||||||
}
|
|
||||||
|
|
||||||
function setScroll(newScroll)
|
|
||||||
{
|
|
||||||
curScroll = newScroll;
|
|
||||||
$("#savedrevs-scrollinner").css('right', newScroll);
|
|
||||||
updateScrollArrows();
|
|
||||||
}
|
|
||||||
|
|
||||||
function setDesiredScroll(newDesiredScroll, dontUpdate)
|
|
||||||
{
|
|
||||||
desiredScroll = Math.min(getMaxScroll(), Math.max(getMinScroll(), newDesiredScroll));
|
|
||||||
if (!dontUpdate)
|
|
||||||
{
|
|
||||||
updateScroll();
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
function updateScroll()
|
|
||||||
{
|
|
||||||
updateScrollArrows();
|
|
||||||
scrollAnimator.scheduleAnimation();
|
|
||||||
}
|
|
||||||
|
|
||||||
function updateScrollArrows()
|
|
||||||
{
|
|
||||||
$("#savedrevs-scrollleft").toggleClass("disabledscrollleft", desiredScroll <= getMinScroll());
|
|
||||||
$("#savedrevs-scrollright").toggleClass("disabledscrollright", desiredScroll >= getMaxScroll());
|
|
||||||
}
|
|
||||||
var scrollAnimator = padutils.makeAnimationScheduler(function()
|
|
||||||
{
|
|
||||||
setDesiredScroll(desiredScroll, true); // re-clamp
|
|
||||||
if (Math.abs(desiredScroll - curScroll) < 1)
|
|
||||||
{
|
|
||||||
setScroll(desiredScroll);
|
|
||||||
return false;
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
setScroll(curScroll + (desiredScroll - curScroll) * 0.5);
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
}, 50, 2);
|
|
||||||
|
|
||||||
var isSaving = false;
|
|
||||||
|
|
||||||
function setIsSaving(v)
|
|
||||||
{
|
|
||||||
isSaving = v;
|
|
||||||
rerenderButton();
|
|
||||||
}
|
|
||||||
|
|
||||||
function haveReachedRevLimit()
|
|
||||||
{
|
|
||||||
var mv = pad.getPrivilege('maxRevisions');
|
|
||||||
return (!(mv < 0 || mv > currentRevisionList.length));
|
|
||||||
}
|
|
||||||
|
|
||||||
function rerenderButton()
|
|
||||||
{
|
|
||||||
if (isSaving || (!pad.isFullyConnected()) || haveReachedRevLimit())
|
|
||||||
{
|
|
||||||
$("#savedrevs-savenow").css('opacity', 0.75);
|
|
||||||
}
|
|
||||||
else
|
|
||||||
{
|
|
||||||
$("#savedrevs-savenow").css('opacity', 1);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
var scrollRepeatTimer = null;
|
|
||||||
var scrollStartTime = 0;
|
|
||||||
|
|
||||||
function setScrollRepeatTimer(dir)
|
|
||||||
{
|
|
||||||
clearScrollRepeatTimer();
|
|
||||||
scrollStartTime = +new Date;
|
|
||||||
scrollRepeatTimer = window.setTimeout(function f()
|
|
||||||
{
|
|
||||||
if (!scrollRepeatTimer)
|
|
||||||
{
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
self.scroll(dir);
|
|
||||||
var scrollTime = (+new Date) - scrollStartTime;
|
|
||||||
var delay = (scrollTime > 2000 ? 50 : 300);
|
|
||||||
scrollRepeatTimer = window.setTimeout(f, delay);
|
|
||||||
}, 300);
|
|
||||||
$(document).bind('mouseup', clearScrollRepeatTimer);
|
|
||||||
}
|
|
||||||
|
|
||||||
function clearScrollRepeatTimer()
|
|
||||||
{
|
|
||||||
if (scrollRepeatTimer)
|
|
||||||
{
|
|
||||||
window.clearTimeout(scrollRepeatTimer);
|
|
||||||
scrollRepeatTimer = null;
|
|
||||||
}
|
|
||||||
$(document).unbind('mouseup', clearScrollRepeatTimer);
|
|
||||||
}
|
|
||||||
|
|
||||||
var pad = undefined;
|
|
||||||
var self = {
|
|
||||||
init: function(initialRevisions, _pad)
|
|
||||||
{
|
|
||||||
pad = _pad;
|
pad = _pad;
|
||||||
self.newRevisionList(initialRevisions, true);
|
|
||||||
|
|
||||||
$("#savedrevs-savenow").click(function()
|
|
||||||
{
|
|
||||||
self.saveNow();
|
|
||||||
});
|
|
||||||
$("#savedrevs-scrollleft").mousedown(function()
|
|
||||||
{
|
|
||||||
self.scroll('left');
|
|
||||||
setScrollRepeatTimer('left');
|
|
||||||
});
|
|
||||||
$("#savedrevs-scrollright").mousedown(function()
|
|
||||||
{
|
|
||||||
self.scroll('right');
|
|
||||||
setScrollRepeatTimer('right');
|
|
||||||
});
|
|
||||||
$("#savedrevs-close").click(function()
|
|
||||||
{
|
|
||||||
paddocbar.setShownPanel(null);
|
|
||||||
});
|
|
||||||
|
|
||||||
// update "saved n minutes ago" times
|
|
||||||
window.setInterval(function()
|
|
||||||
{
|
|
||||||
refreshRevisionList();
|
|
||||||
}, 60 * 1000);
|
|
||||||
},
|
|
||||||
restoreRevision: function(rnum)
|
|
||||||
{
|
|
||||||
var rev = currentRevisionList[rnum];
|
|
||||||
var warning = ("Restoring this revision will overwrite the current" + " text of the pad. " + "Are you sure you want to continue?");
|
|
||||||
var hidePanel = paddocbar.hideLaterIfNoOtherInteraction();
|
|
||||||
var box = getNthBox(rnum);
|
|
||||||
if (confirm(warning))
|
|
||||||
{
|
|
||||||
box.find(".srtwirly").show();
|
|
||||||
$.ajax(
|
|
||||||
{
|
|
||||||
type: 'get',
|
|
||||||
url: '/ep/pad/getrevisionatext',
|
|
||||||
data: {
|
|
||||||
padId: pad.getPadId(),
|
|
||||||
revId: rev.id
|
|
||||||
},
|
|
||||||
success: success,
|
|
||||||
error: error
|
|
||||||
});
|
|
||||||
}
|
}
|
||||||
|
|
||||||
function success(resultJson)
|
|
||||||
{
|
|
||||||
untwirl();
|
|
||||||
var result = JSON.parse(resultJson);
|
|
||||||
padeditor.restoreRevisionText(result);
|
|
||||||
window.setTimeout(function()
|
|
||||||
{
|
|
||||||
hidePanel();
|
|
||||||
}, 0);
|
|
||||||
}
|
|
||||||
|
|
||||||
function error(e)
|
|
||||||
{
|
|
||||||
untwirl();
|
|
||||||
alert("Oops! There was an error retreiving the text (revNum= " + rev.revNum + "; padId=" + pad.getPadId());
|
|
||||||
}
|
|
||||||
|
|
||||||
function untwirl()
|
|
||||||
{
|
|
||||||
box.find(".srtwirly").hide();
|
|
||||||
}
|
|
||||||
},
|
|
||||||
showReachedLimit: function()
|
|
||||||
{
|
|
||||||
alert("Sorry, you do not have privileges to save more than " + pad.getPrivilege('maxRevisions') + " revisions.");
|
|
||||||
},
|
|
||||||
newRevisionList: function(lst, noAnimation)
|
|
||||||
{
|
|
||||||
// server gives us list with newest first;
|
|
||||||
// we want chronological order
|
|
||||||
var L = reversedCopy(lst);
|
|
||||||
setRevisionList(L, noAnimation);
|
|
||||||
rerenderButton();
|
|
||||||
},
|
|
||||||
saveNow: function()
|
|
||||||
{
|
|
||||||
if (isSaving)
|
|
||||||
{
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
if (!pad.isFullyConnected())
|
|
||||||
{
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
if (haveReachedRevLimit())
|
|
||||||
{
|
|
||||||
self.showReachedLimit();
|
|
||||||
return;
|
|
||||||
}
|
|
||||||
setIsSaving(true);
|
|
||||||
var savedBy = pad.getUserName() || "unnamed";
|
|
||||||
pad.callWhenNotCommitting(submitSave);
|
|
||||||
|
|
||||||
function submitSave()
|
|
||||||
{
|
|
||||||
$.ajax(
|
|
||||||
{
|
|
||||||
type: 'post',
|
|
||||||
url: '/ep/pad/saverevision',
|
|
||||||
data: {
|
|
||||||
padId: pad.getPadId(),
|
|
||||||
savedBy: savedBy,
|
|
||||||
savedById: pad.getUserId(),
|
|
||||||
revNum: pad.getCollabRevisionNumber()
|
|
||||||
},
|
|
||||||
success: success,
|
|
||||||
error: error
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
function success(text)
|
|
||||||
{
|
|
||||||
setIsSaving(false);
|
|
||||||
var newRevisionList = JSON.parse(text);
|
|
||||||
self.newRevisionList(newRevisionList);
|
|
||||||
pad.sendClientMessage(
|
|
||||||
{
|
|
||||||
type: 'newRevisionList',
|
|
||||||
revisionList: newRevisionList,
|
|
||||||
savedBy: savedBy
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
function error(e)
|
|
||||||
{
|
|
||||||
setIsSaving(false);
|
|
||||||
alert("Oops! The server failed to save the revision. Please try again later.");
|
|
||||||
}
|
|
||||||
},
|
|
||||||
handleResizePage: function()
|
|
||||||
{
|
|
||||||
updateScrollArrows();
|
|
||||||
},
|
|
||||||
handleIsFullyConnected: function(isConnected)
|
|
||||||
{
|
|
||||||
rerenderButton();
|
|
||||||
},
|
|
||||||
scroll: function(dir)
|
|
||||||
{
|
|
||||||
var minScroll = getMinScroll();
|
|
||||||
var maxScroll = getMaxScroll();
|
|
||||||
if (dir == 'left')
|
|
||||||
{
|
|
||||||
if (desiredScroll > minScroll)
|
|
||||||
{
|
|
||||||
var n = Math.floor((desiredScroll - 1 - minScroll) / REVISION_BOX_WIDTH);
|
|
||||||
setDesiredScroll(Math.max(0, n) * REVISION_BOX_WIDTH + minScroll);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
else if (dir == 'right')
|
|
||||||
{
|
|
||||||
if (desiredScroll < maxScroll)
|
|
||||||
{
|
|
||||||
var n = Math.floor((maxScroll - desiredScroll - 1) / REVISION_BOX_WIDTH);
|
|
||||||
setDesiredScroll(maxScroll - Math.max(0, n) * REVISION_BOX_WIDTH);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
};
|
|
||||||
return self;
|
|
||||||
}());
|
|
||||||
|
|
||||||
exports.padsavedrevs = padsavedrevs;
|
|
||||||
|
|
|
@ -73,6 +73,11 @@
|
||||||
</li>
|
</li>
|
||||||
</ul>
|
</ul>
|
||||||
<ul id="menu_right">
|
<ul id="menu_right">
|
||||||
|
<li onClick="window.pad&&pad.editbarClick('savedRevision');return false;">
|
||||||
|
<a id="settingslink" title="Mark this revision as a saved revision">
|
||||||
|
<div class="buttonicon buttonicon-savedRevision"></div>
|
||||||
|
</a>
|
||||||
|
</li>
|
||||||
<li onClick="window.pad&&pad.editbarClick('settings');return false;">
|
<li onClick="window.pad&&pad.editbarClick('settings');return false;">
|
||||||
<a id="settingslink" title="Settings of this pad">
|
<a id="settingslink" title="Settings of this pad">
|
||||||
<div class="buttonicon buttonicon-settings"></div>
|
<div class="buttonicon buttonicon-settings"></div>
|
||||||
|
|
Loading…
Add table
Add a link
Reference in a new issue