node-webodf
Version:
WebODF - JavaScript Document Engine http://webodf.org/
190 lines (169 loc) • 5.92 kB
JavaScript
/**
* Copyright (C) 2012-2013 KO GmbH <copyright@kogmbh.com>
*
* @licstart
* This file is part of WebODF.
*
* WebODF is free software: you can redistribute it and/or modify it
* under the terms of the GNU Affero General Public License (GNU AGPL)
* as published by the Free Software Foundation, either version 3 of
* the License, or (at your option) any later version.
*
* WebODF is distributed in the hope that it will be useful, but
* WITHOUT ANY WARRANTY; without even the implied warranty of
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
* GNU Affero General Public License for more details.
*
* You should have received a copy of the GNU Affero General Public License
* along with WebODF. If not, see <http://www.gnu.org/licenses/>.
* @licend
*
* @source: http://www.webodf.org/
* @source: https://github.com/kogmbh/WebODF/
*/
/*global core, gui, runtime*/
/**
* @constructor
* @implements {core.Destroyable}
* @param {!ops.EditInfo} editInfo marker takes ownership
* @param {boolean} initialVisibility Sets the initial edit info marker visibility
*/
gui.EditInfoMarker = function EditInfoMarker(editInfo, initialVisibility) {
"use strict";
var self = this,
/**@type{!Element}*/
editInfoNode,
/**@type{!gui.EditInfoHandle}*/
handle,
/**@type{!HTMLDivElement}*/
marker,
editinfons = 'urn:webodf:names:editinfo',
decayTimer0,
decayTimer1,
decayTimer2,
decayTimeStep = 10000; // 10 seconds
/**
* Runs and returns a timer that sets the marker's opacity
* to the specified value after the specified delay.
* @param {!number} opacity
* @param {!number} delay
* @return {!number}
*/
function applyDecay(opacity, delay) {
return runtime.setTimeout(function () {
marker.style.opacity = opacity;
}, delay);
}
/**
* Stops the specified timer
* @param {!number} timerId
*/
function deleteDecay(timerId) {
runtime.clearTimeout(timerId);
}
/**
* @param {!string} memberid
*/
function setLastAuthor(memberid) {
marker.setAttributeNS(editinfons, 'editinfo:memberid', memberid);
}
/**
* @param {!string} memberid
* @param {!Date} timestamp
*/
this.addEdit = function (memberid, timestamp) {
var age = (Date.now() - timestamp);
editInfo.addEdit(memberid, timestamp);
handle.setEdits(editInfo.getSortedEdits());
setLastAuthor(memberid);
// Since a new edit has arrived, stop decaying for the old edits
deleteDecay(decayTimer1);
deleteDecay(decayTimer2);
// Decide the decay path:
// this decides the initial opacity and subsequent decays to apply
// depending on the age of the edit (for example the edit might have arrived
// here a long time after it was added to the master session. We don't want
// an opaque marker in that case, we would want it to start with a lower opacity
// and decay accordingly further, if possible.
if (age < decayTimeStep) {
decayTimer0 = applyDecay(1, 0);
decayTimer1 = applyDecay(0.5, decayTimeStep - age);
decayTimer2 = applyDecay(0.2, decayTimeStep * 2 - age);
} else if (age >= decayTimeStep && age < decayTimeStep * 2) {
decayTimer0 = applyDecay(0.5, 0);
decayTimer2 = applyDecay(0.2, decayTimeStep * 2 - age);
} else {
decayTimer0 = applyDecay(0.2, 0);
}
};
this.getEdits = function () {
return editInfo.getEdits();
};
this.clearEdits = function () {
editInfo.clearEdits();
handle.setEdits([]);
if (marker.hasAttributeNS(editinfons, 'editinfo:memberid')) {
marker.removeAttributeNS(editinfons, 'editinfo:memberid');
}
};
this.getEditInfo = function () {
return editInfo;
};
/**
* Shows the edit information marker
*/
this.show = function () {
marker.style.display = 'block';
};
/**
* Hides the edit information marker
*/
this.hide = function () {
self.hideHandle();
// edit decays are not cleared as the marker should be properly
// faded if it is re-shown
marker.style.display = 'none';
};
this.showHandle = function () {
handle.show();
};
this.hideHandle = function () {
handle.hide();
};
/**
* @param {!function(!Error=)} callback, passing an error object in case of error
* @return {undefined}
*/
this.destroy = function (callback) {
deleteDecay(decayTimer0);
deleteDecay(decayTimer1);
deleteDecay(decayTimer2);
editInfoNode.removeChild(marker);
handle.destroy(function (err) {
if (err) {
callback(err);
} else {
editInfo.destroy(callback);
}
});
};
function init() {
var dom = editInfo.getOdtDocument().getDOMDocument(),
htmlns = dom.documentElement.namespaceURI;
marker = /**@type{!HTMLDivElement}*/(dom.createElementNS(htmlns, "div"));
marker.setAttribute("class", "editInfoMarker");
marker.onmouseover = function () {
self.showHandle();
};
marker.onmouseout = function () {
self.hideHandle();
};
editInfoNode = editInfo.getNode();
editInfoNode.appendChild(marker);
handle = new gui.EditInfoHandle(editInfoNode);
if (!initialVisibility) {
self.hide();
}
}
init();
};