2011-12-04 15:33:56 +00:00
|
|
|
/**
|
2013-11-23 19:39:52 +02:00
|
|
|
* This code is mostly from the old Etherpad. Please help us to comment this code.
|
2011-12-04 15:33:56 +00:00
|
|
|
* This helps other people to understand this code better and helps them to improve it.
|
|
|
|
* TL;DR COMMENTS ON THIS FILE ARE HIGHLY APPRECIATED
|
|
|
|
*/
|
|
|
|
|
2011-06-20 11:44:04 +01:00
|
|
|
/**
|
|
|
|
* Copyright 2009 Google Inc.
|
2011-07-07 18:59:34 +01:00
|
|
|
*
|
2011-06-20 11:44:04 +01:00
|
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
* you may not use this file except in compliance with the License.
|
|
|
|
* You may obtain a copy of the License at
|
2011-07-07 18:59:34 +01:00
|
|
|
*
|
2011-06-20 11:44:04 +01:00
|
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
2011-07-07 18:59:34 +01:00
|
|
|
*
|
2011-06-20 11:44:04 +01:00
|
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
|
|
* distributed under the License is distributed on an "AS-IS" BASIS,
|
|
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
* See the License for the specific language governing permissions and
|
|
|
|
* limitations under the License.
|
|
|
|
*/
|
|
|
|
|
2012-01-26 21:20:26 -08:00
|
|
|
// These parameters were global, now they are injected. A reference to the
|
|
|
|
// Timeslider controller would probably be more appropriate.
|
2012-03-17 13:36:42 +01:00
|
|
|
var _ = require('./underscore');
|
2012-04-23 14:47:07 +02:00
|
|
|
var padmodals = require('./pad_modals').padmodals;
|
2013-11-23 19:39:52 +02:00
|
|
|
require("./jquery.class");
|
2012-03-04 13:10:36 -08:00
|
|
|
|
2013-11-24 20:14:21 +02:00
|
|
|
/**
|
|
|
|
* A class for anything which can be hung off of the slider bar.
|
|
|
|
* I.e. this is for the handle, or saved-revisions (stars).
|
|
|
|
* TODO:
|
|
|
|
* - handle mousedowns/clicks
|
|
|
|
* - handle drags
|
|
|
|
*/
|
|
|
|
$.Class("SliderHandleUI",
|
|
|
|
{//statics
|
|
|
|
},
|
|
|
|
{//instance
|
|
|
|
/**
|
|
|
|
* Construct the SliderHandle.
|
|
|
|
* @param {SliderUI} slider The slider from which this handle will be hung.
|
|
|
|
* @param {Number} position The initial position for this handle.
|
|
|
|
*/
|
2013-11-28 18:48:11 +02:00
|
|
|
init: function (slider, value, type) {
|
|
|
|
console.log("New SliderHandle(%d, %s)", value, type);
|
2013-11-24 20:14:21 +02:00
|
|
|
this.slider = slider;
|
2013-11-28 18:48:11 +02:00
|
|
|
this.value = value;
|
2013-11-24 20:14:21 +02:00
|
|
|
//create the element:
|
2013-11-28 18:48:11 +02:00
|
|
|
this.element = $("<div class='ui-slider-handle'></div>");
|
|
|
|
if (type === "")
|
|
|
|
type = "handle";
|
|
|
|
this.element.addClass("ui-slider-handle-" + type);
|
|
|
|
this._mouseInit();
|
2013-11-24 20:14:21 +02:00
|
|
|
},
|
2013-11-28 18:48:11 +02:00
|
|
|
_mouseInit: function () {
|
|
|
|
this.element.on("mousedown.sliderhandle", null, this, function(event) {
|
|
|
|
console.log("sliderhandleui - mousedown")
|
|
|
|
})
|
2013-11-24 20:14:21 +02:00
|
|
|
},
|
|
|
|
}
|
|
|
|
);
|
|
|
|
|
|
|
|
SliderHandleUI("SliderSavedRevisionUI",
|
2013-11-28 18:48:11 +02:00
|
|
|
{//statics
|
|
|
|
},
|
|
|
|
{//instance
|
|
|
|
init: function(slider, revision) {
|
|
|
|
this._super(slider, revision.revNum, 'star');
|
|
|
|
this.revision = revision;
|
2013-11-24 20:14:21 +02:00
|
|
|
},
|
2013-11-28 18:48:11 +02:00
|
|
|
_mouseInit: function () {
|
|
|
|
this.element.on("mousedown.sliderhandle", null, this, function(event) {
|
|
|
|
console.log("SliderSavedRevisionUI - mousedown")
|
|
|
|
})
|
|
|
|
},
|
|
|
|
}
|
2013-11-24 20:14:21 +02:00
|
|
|
);
|
|
|
|
|
2013-11-28 18:48:11 +02:00
|
|
|
//TODO:
|
|
|
|
// - window resizing is currently broken!
|
|
|
|
// - keyboard events
|
2013-11-24 20:14:21 +02:00
|
|
|
$.Class("SliderUI",
|
2013-11-28 18:48:11 +02:00
|
|
|
{//statics
|
|
|
|
defaults: {
|
|
|
|
min: 0,
|
|
|
|
max: 100,
|
|
|
|
}
|
|
|
|
},
|
|
|
|
{//instance
|
|
|
|
init: function (element, options) {
|
|
|
|
this.options = $.extend({}, this.defaults, options);
|
|
|
|
this.element = element;
|
|
|
|
this.current_value = "value" in this.options ? this.options.value : 0;
|
|
|
|
this.handles = [];
|
|
|
|
this.attachHandle(new SliderHandleUI(this, this._current_value, 'handle'));
|
|
|
|
this._mouseInit();
|
|
|
|
|
|
|
|
// handle window resize
|
|
|
|
var _this = this;
|
|
|
|
$(window).resize(function() {
|
|
|
|
_this.render();
|
|
|
|
});
|
|
|
|
},
|
|
|
|
_getStep: function () {
|
|
|
|
return (this.element.width()) / (this.options.max * 1.0);
|
|
|
|
},
|
|
|
|
render: function () {
|
|
|
|
for(var h in this.handles) {
|
|
|
|
handle = this.handles[h];
|
|
|
|
handle.element.css('left', (handle.value * this._getStep()) );
|
|
|
|
}
|
|
|
|
},
|
|
|
|
// this internal version of _setValue should only be used to render
|
|
|
|
// when the handle changes position as a result of UI events handled
|
|
|
|
// by this slider.
|
|
|
|
_setValue: function (value) {
|
|
|
|
if (value < 0)
|
|
|
|
value = 0;
|
|
|
|
if (value > this.options.max)
|
|
|
|
value = this.options.max;
|
|
|
|
this.handles[0].value = value;
|
|
|
|
this.current_value = value;
|
|
|
|
this.render();
|
|
|
|
},
|
|
|
|
// this 'public' version of _setValue also triggers a change event
|
|
|
|
setValue: function(value) {
|
|
|
|
this._setValue(value);
|
|
|
|
this._trigger("change", value);
|
|
|
|
},
|
|
|
|
setMax: function (max) {
|
|
|
|
this.options.max = max;
|
|
|
|
this.render();
|
|
|
|
},
|
|
|
|
attachHandle: function (handle) {
|
|
|
|
this.handles.push(handle);
|
|
|
|
this.element.append(handle.element);
|
|
|
|
this.current_value = this.handles[0].value;
|
|
|
|
return handle;
|
|
|
|
},
|
|
|
|
_trigger: function (eventname, value) {
|
|
|
|
//TODO: implement trigger
|
|
|
|
console.log("triggering event: ", eventname);
|
|
|
|
if (eventname in this.options) {
|
|
|
|
this.options[eventname](value);
|
|
|
|
}
|
|
|
|
},
|
|
|
|
_mouseInit: function () {
|
|
|
|
// handle all mouse events for the slider and handles right here
|
|
|
|
var _this = this;
|
|
|
|
this.element.on("mousedown.slider", function (event) {
|
|
|
|
if (event.target == _this.element[0] || $(event.target).hasClass("ui-slider-handle")) {
|
|
|
|
// the click is on the slider bar itself.
|
|
|
|
var start_value = Math.floor((event.clientX-_this.element.offset().left) / _this._getStep());
|
|
|
|
console.log("sliderbar mousedown, value:", start_value);
|
|
|
|
if (_this.current_value != start_value)
|
|
|
|
_this._setValue(start_value);
|
|
|
|
|
|
|
|
$(document).on("mousemove.slider", function (event) {
|
|
|
|
var current_value = Math.floor((event.clientX-_this.element.offset().left) / _this._getStep());
|
|
|
|
console.log("sliderbar mousemove, value:", current_value);
|
|
|
|
// don't change the value if it hasn't actually changed!
|
|
|
|
if (_this.current_value != current_value) {
|
|
|
|
_this._setValue(current_value);
|
|
|
|
_this._trigger("slide", current_value);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
$(document).on("mouseup.slider", function (event) {
|
|
|
|
// make sure to get rid of the handlers on document,
|
|
|
|
// we don't need them after this 'slide' session is done.
|
|
|
|
$(document).off("mouseup.slider mousemove.slider");
|
|
|
|
var end_value = Math.floor((event.clientX-_this.element.offset().left) / _this._getStep());
|
|
|
|
console.log("sliderbar mouseup, value:", end_value);
|
|
|
|
// always change the value at mouseup
|
|
|
|
_this._setValue(end_value);
|
|
|
|
_this._trigger("change", end_value);
|
|
|
|
|
|
|
|
});
|
|
|
|
} else {
|
|
|
|
console.log("We shouldn't be here!")
|
|
|
|
console.log(event.target);
|
|
|
|
}
|
|
|
|
})
|
|
|
|
},
|
|
|
|
}
|
|
|
|
);
|
|
|
|
|
|
|
|
$.Class("RevisionSlider",
|
2013-11-24 20:14:21 +02:00
|
|
|
{//statics
|
|
|
|
},
|
|
|
|
{//instance
|
|
|
|
init: function (timeslider, root_element) {
|
|
|
|
this.timeslider = timeslider;
|
2013-12-02 17:43:39 +02:00
|
|
|
this.revision_number = this.timeslider.head_revision;
|
|
|
|
console.log("New RevisionSlider, head_revision = %d", this.revision_number);
|
2013-11-24 20:14:21 +02:00
|
|
|
// parse the various elements we need:
|
|
|
|
this.elements = {};
|
|
|
|
this.loadElements(root_element);
|
2013-11-28 18:48:11 +02:00
|
|
|
var _this = this;
|
|
|
|
this.slider = new SliderUI(this.elements['slider-bar'],
|
|
|
|
options = {
|
|
|
|
max: this.timeslider.head_revision,
|
|
|
|
change: function () { _this.onChange.apply(_this, arguments); },
|
|
|
|
slide: function () { _this.onSlide.apply(_this, arguments); },
|
|
|
|
});
|
2013-12-02 17:43:39 +02:00
|
|
|
this.loadSavedRevisionHandles();
|
|
|
|
|
|
|
|
this._mouseInit();
|
2013-11-28 18:48:11 +02:00
|
|
|
},
|
|
|
|
onChange: function (value) {
|
|
|
|
console.log("in change handler:", value);
|
|
|
|
},
|
|
|
|
onSlide: function (value) {
|
|
|
|
console.log("in slide handler:", value);
|
2013-11-24 20:14:21 +02:00
|
|
|
},
|
|
|
|
loadElements: function (root_element) {
|
|
|
|
this.elements['root'] = root_element;
|
|
|
|
//this.elements['slider-handle'] = root_element.first("#ui-slider-handle");
|
|
|
|
this.elements['slider-bar'] = root_element.find("#ui-slider-bar");
|
|
|
|
this.elements['slider'] = root_element.find("#timeslider-slider");
|
|
|
|
this.elements['button-left'] = root_element.find("#leftstep");
|
|
|
|
this.elements['button-right'] = root_element.find("#rightstep");
|
|
|
|
this.elements['button-play'] = root_element.find("#playpause_button");
|
|
|
|
this.elements['timestamp'] = root_element.find("#timer");
|
|
|
|
this.elements['revision-label'] = root_element.find("#revision_label");
|
|
|
|
this.elements['revision-date'] = root_element.find("#revision_date");
|
|
|
|
this.elements['authors'] = root_element.first("#authorsList");
|
|
|
|
},
|
2013-12-02 17:43:39 +02:00
|
|
|
loadSavedRevisionHandles: function () {
|
2013-11-24 20:14:21 +02:00
|
|
|
for (var r in this.timeslider.savedRevisions) {
|
|
|
|
var rev = this.timeslider.savedRevisions[r];
|
2013-11-28 18:48:11 +02:00
|
|
|
this.slider.attachHandle(new SliderSavedRevisionUI(this, rev));
|
2013-11-24 20:14:21 +02:00
|
|
|
};
|
|
|
|
},
|
|
|
|
goToRevision: function (revNum) {
|
|
|
|
//TODO: this should actually do an async jump to revision (with all the server fetching
|
|
|
|
//and changeset rendering that that implies), and perform the setPosition in a callback.
|
2013-12-02 17:43:39 +02:00
|
|
|
//TODO: we need some kind of callback for setting revision metadata.
|
|
|
|
//TODO: at some point we need to set window.location.hash
|
|
|
|
if (revNum > this.timeslider.head_revision)
|
2013-11-24 20:14:21 +02:00
|
|
|
revNum = this.timeslider.latest_revision;
|
2013-12-02 17:43:39 +02:00
|
|
|
if (revNum < 0)
|
|
|
|
revNum = 0;
|
2013-11-24 20:14:21 +02:00
|
|
|
console.log("GO TO REVISION", revNum)
|
2013-12-02 17:43:39 +02:00
|
|
|
this.elements["revision-label"].html(html10n.get("timeslider.version", { "version": revNum }));
|
2013-11-28 18:48:11 +02:00
|
|
|
this.slider.setValue(revNum);
|
2013-12-02 17:43:39 +02:00
|
|
|
this.revision_number = revNum;
|
|
|
|
//TODO: set the enabled/disabled for button-left and button-right
|
2013-11-24 20:14:21 +02:00
|
|
|
},
|
2013-12-02 17:43:39 +02:00
|
|
|
_mouseInit: function () {
|
|
|
|
var _this = this;
|
|
|
|
this.elements["button-left"].on("click", function (event) {
|
|
|
|
console.log("was :", _this.revision_number);
|
|
|
|
_this.goToRevision(_this.revision_number - 1);
|
|
|
|
});
|
|
|
|
|
|
|
|
this.elements["button-right"].on("click", function (event) {
|
|
|
|
_this.goToRevision(_this.revision_number + 1);
|
|
|
|
});
|
2013-11-24 20:14:21 +02:00
|
|
|
|
2013-12-02 17:43:39 +02:00
|
|
|
}
|
2013-11-24 20:14:21 +02:00
|
|
|
|
|
|
|
}
|
|
|
|
);
|
|
|
|
|
|
|
|
function loadBroadcastSliderJS(tsclient, fireWhenAllScriptsAreLoaded)
|
2011-06-20 11:44:04 +01:00
|
|
|
{
|
2012-01-26 21:20:26 -08:00
|
|
|
var BroadcastSlider;
|
2011-06-20 11:44:04 +01:00
|
|
|
|
2011-07-07 18:59:34 +01:00
|
|
|
(function()
|
2011-06-20 11:44:04 +01:00
|
|
|
{ // wrap this code in its own namespace
|
2013-11-24 20:14:21 +02:00
|
|
|
|
2013-11-28 18:48:11 +02:00
|
|
|
tsui = new RevisionSlider(tsclient, $("#timeslider-top"));
|
2013-11-24 20:14:21 +02:00
|
|
|
|
|
|
|
// if there was a revision specified in the 'location.hash', jump to it.
|
|
|
|
if (window.location.hash.length > 1) {
|
|
|
|
var rev = Number(window.location.hash.substr(1));
|
|
|
|
if(!isNaN(rev))
|
|
|
|
tsui.goToRevision(rev);
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2011-06-20 11:44:04 +01:00
|
|
|
var sliderLength = 1000;
|
|
|
|
var sliderPos = 0;
|
|
|
|
var sliderActive = false;
|
|
|
|
var slidercallbacks = [];
|
|
|
|
var savedRevisions = [];
|
|
|
|
var sliderPlaying = false;
|
2013-11-24 20:14:21 +02:00
|
|
|
clientVars = tsclient.clientVars;
|
2011-06-20 11:44:04 +01:00
|
|
|
|
|
|
|
function disableSelection(element)
|
|
|
|
{
|
2011-07-07 18:59:34 +01:00
|
|
|
element.onselectstart = function()
|
2011-06-20 11:44:04 +01:00
|
|
|
{
|
|
|
|
return false;
|
|
|
|
};
|
|
|
|
element.unselectable = "on";
|
|
|
|
element.style.MozUserSelect = "none";
|
|
|
|
element.style.cursor = "default";
|
|
|
|
}
|
2011-07-07 18:59:34 +01:00
|
|
|
var _callSliderCallbacks = function(newval)
|
2011-06-20 11:44:04 +01:00
|
|
|
{
|
|
|
|
sliderPos = newval;
|
|
|
|
for (var i = 0; i < slidercallbacks.length; i++)
|
|
|
|
{
|
|
|
|
slidercallbacks[i](newval);
|
|
|
|
}
|
2012-04-23 14:47:07 +02:00
|
|
|
}
|
2011-07-07 18:59:34 +01:00
|
|
|
var removeSavedRevision = function(position)
|
2011-06-20 11:44:04 +01:00
|
|
|
{
|
|
|
|
var element = $("div.star [pos=" + position + "]");
|
|
|
|
savedRevisions.remove(element);
|
|
|
|
element.remove();
|
|
|
|
return element;
|
2012-04-23 14:47:07 +02:00
|
|
|
};
|
2011-06-20 11:44:04 +01:00
|
|
|
|
|
|
|
/* Begin small 'API' */
|
|
|
|
|
|
|
|
function onSlider(callback)
|
|
|
|
{
|
|
|
|
slidercallbacks.push(callback);
|
|
|
|
}
|
|
|
|
|
|
|
|
function getSliderPosition()
|
|
|
|
{
|
|
|
|
return sliderPos;
|
|
|
|
}
|
|
|
|
|
|
|
|
function setSliderPosition(newpos)
|
|
|
|
{
|
|
|
|
newpos = Number(newpos);
|
|
|
|
if (newpos < 0 || newpos > sliderLength) return;
|
2013-02-06 23:35:58 +00:00
|
|
|
if(!newpos){
|
|
|
|
newpos = 0; // stops it from displaying NaN if newpos isn't set
|
|
|
|
}
|
2013-02-07 15:34:10 +01:00
|
|
|
window.location.hash = "#" + newpos;
|
2011-06-20 11:44:04 +01:00
|
|
|
$("#ui-slider-handle").css('left', newpos * ($("#ui-slider-bar").width() - 2) / (sliderLength * 1.0));
|
2011-07-07 18:59:34 +01:00
|
|
|
$("a.tlink").map(function()
|
2011-06-20 11:44:04 +01:00
|
|
|
{
|
|
|
|
$(this).attr('href', $(this).attr('thref').replace("%revision%", newpos));
|
|
|
|
});
|
2013-02-07 15:34:10 +01:00
|
|
|
|
2012-12-19 19:55:31 +01:00
|
|
|
$("#revision_label").html(html10n.get("timeslider.version", { "version": newpos}));
|
2011-06-20 11:44:04 +01:00
|
|
|
|
|
|
|
if (newpos == 0)
|
|
|
|
{
|
|
|
|
$("#leftstar").css('opacity', .5);
|
|
|
|
$("#leftstep").css('opacity', .5);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
$("#leftstar").css('opacity', 1);
|
|
|
|
$("#leftstep").css('opacity', 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (newpos == sliderLength)
|
|
|
|
{
|
|
|
|
$("#rightstar").css('opacity', .5);
|
|
|
|
$("#rightstep").css('opacity', .5);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
$("#rightstar").css('opacity', 1);
|
|
|
|
$("#rightstep").css('opacity', 1);
|
|
|
|
}
|
|
|
|
|
|
|
|
sliderPos = newpos;
|
|
|
|
_callSliderCallbacks(newpos);
|
|
|
|
}
|
|
|
|
|
|
|
|
function getSliderLength()
|
|
|
|
{
|
|
|
|
return sliderLength;
|
|
|
|
}
|
|
|
|
|
|
|
|
function setSliderLength(newlength)
|
|
|
|
{
|
|
|
|
sliderLength = newlength;
|
|
|
|
updateSliderElements();
|
|
|
|
}
|
|
|
|
|
|
|
|
// just take over the whole slider screen with a reconnect message
|
|
|
|
|
|
|
|
function showReconnectUI()
|
|
|
|
{
|
2012-07-14 15:54:44 +02:00
|
|
|
padmodals.showModal("disconnected");
|
2011-06-20 11:44:04 +01:00
|
|
|
}
|
|
|
|
|
2013-11-28 18:48:11 +02:00
|
|
|
//TODO: figure out what the hell this is for
|
2012-03-24 20:35:45 +01:00
|
|
|
var fixPadHeight = _.throttle(function(){
|
|
|
|
var height = $('#timeslider-top').height();
|
|
|
|
$('#editorcontainerbox').css({marginTop: height});
|
|
|
|
}, 600);
|
2013-11-23 19:39:52 +02:00
|
|
|
|
2011-06-20 11:44:04 +01:00
|
|
|
function setAuthors(authors)
|
|
|
|
{
|
2012-03-24 20:35:45 +01:00
|
|
|
var authorsList = $("#authorsList");
|
|
|
|
authorsList.empty();
|
2011-06-20 11:44:04 +01:00
|
|
|
var numAnonymous = 0;
|
|
|
|
var numNamed = 0;
|
2012-03-25 15:48:31 +02:00
|
|
|
var colorsAnonymous = [];
|
2012-03-17 13:36:42 +01:00
|
|
|
_.each(authors, function(author)
|
2011-06-20 11:44:04 +01:00
|
|
|
{
|
2012-03-25 15:48:31 +02:00
|
|
|
var authorColor = clientVars.colorPalette[author.colorId] || author.colorId;
|
2011-06-20 11:44:04 +01:00
|
|
|
if (author.name)
|
|
|
|
{
|
2012-03-24 20:35:45 +01:00
|
|
|
if (numNamed !== 0) authorsList.append(', ');
|
2013-11-23 19:39:52 +02:00
|
|
|
|
2012-03-25 15:48:31 +02:00
|
|
|
$('<span />')
|
|
|
|
.text(author.name || "unnamed")
|
|
|
|
.css('background-color', authorColor)
|
|
|
|
.addClass('author')
|
|
|
|
.appendTo(authorsList);
|
|
|
|
|
2011-06-20 11:44:04 +01:00
|
|
|
numNamed++;
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
numAnonymous++;
|
2012-03-25 15:48:31 +02:00
|
|
|
if(authorColor) colorsAnonymous.push(authorColor);
|
2011-06-20 11:44:04 +01:00
|
|
|
}
|
|
|
|
});
|
|
|
|
if (numAnonymous > 0)
|
|
|
|
{
|
2013-07-23 12:43:30 +02:00
|
|
|
var anonymousAuthorString = html10n.get("timeslider.unnamedauthors", { num: numAnonymous });
|
2013-11-23 19:39:52 +02:00
|
|
|
|
2012-03-24 20:35:45 +01:00
|
|
|
if (numNamed !== 0){
|
|
|
|
authorsList.append(' + ' + anonymousAuthorString);
|
|
|
|
} else {
|
|
|
|
authorsList.append(anonymousAuthorString);
|
|
|
|
}
|
2013-11-23 19:39:52 +02:00
|
|
|
|
2012-03-25 15:48:31 +02:00
|
|
|
if(colorsAnonymous.length > 0){
|
|
|
|
authorsList.append(' (');
|
|
|
|
_.each(colorsAnonymous, function(color, i){
|
2013-11-23 19:39:52 +02:00
|
|
|
if( i > 0 ) authorsList.append(' ');
|
2012-04-15 19:18:29 +02:00
|
|
|
$('<span> </span>')
|
2012-03-25 15:48:31 +02:00
|
|
|
.css('background-color', color)
|
|
|
|
.addClass('author author-anonymous')
|
|
|
|
.appendTo(authorsList);
|
|
|
|
});
|
|
|
|
authorsList.append(')');
|
|
|
|
}
|
2013-11-23 19:39:52 +02:00
|
|
|
|
2011-06-20 11:44:04 +01:00
|
|
|
}
|
|
|
|
if (authors.length == 0)
|
|
|
|
{
|
2012-12-19 19:55:31 +01:00
|
|
|
authorsList.append(html10n.get("timeslider.toolbar.authorsList"));
|
2011-06-20 11:44:04 +01:00
|
|
|
}
|
2013-11-23 19:39:52 +02:00
|
|
|
|
2012-03-24 20:35:45 +01:00
|
|
|
fixPadHeight();
|
2011-06-20 11:44:04 +01:00
|
|
|
}
|
|
|
|
|
2013-12-02 17:43:39 +02:00
|
|
|
//This API is in use by broadcast.js
|
|
|
|
//TODO: refactor broadcast.js to use RevisionSlider instead
|
2012-01-26 21:20:26 -08:00
|
|
|
BroadcastSlider = {
|
2011-06-20 11:44:04 +01:00
|
|
|
onSlider: onSlider,
|
|
|
|
getSliderPosition: getSliderPosition,
|
|
|
|
setSliderPosition: setSliderPosition,
|
|
|
|
getSliderLength: getSliderLength,
|
|
|
|
setSliderLength: setSliderLength,
|
2011-07-07 18:59:34 +01:00
|
|
|
isSliderActive: function()
|
2011-06-20 11:44:04 +01:00
|
|
|
{
|
|
|
|
return sliderActive;
|
|
|
|
},
|
|
|
|
playpause: playpause,
|
|
|
|
showReconnectUI: showReconnectUI,
|
|
|
|
setAuthors: setAuthors
|
|
|
|
}
|
|
|
|
|
|
|
|
function playButtonUpdater()
|
|
|
|
{
|
|
|
|
if (sliderPlaying)
|
|
|
|
{
|
|
|
|
if (getSliderPosition() + 1 > sliderLength)
|
|
|
|
{
|
|
|
|
$("#playpause_button_icon").toggleClass('pause');
|
|
|
|
sliderPlaying = false;
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
setSliderPosition(getSliderPosition() + 1);
|
|
|
|
|
|
|
|
setTimeout(playButtonUpdater, 100);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
function playpause()
|
|
|
|
{
|
|
|
|
$("#playpause_button_icon").toggleClass('pause');
|
|
|
|
|
|
|
|
if (!sliderPlaying)
|
|
|
|
{
|
|
|
|
if (getSliderPosition() == sliderLength) setSliderPosition(0);
|
|
|
|
sliderPlaying = true;
|
|
|
|
playButtonUpdater();
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
sliderPlaying = false;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// assign event handlers to html UI elements after page load
|
|
|
|
//$(window).load(function ()
|
2011-07-07 18:59:34 +01:00
|
|
|
fireWhenAllScriptsAreLoaded.push(function()
|
2011-06-20 11:44:04 +01:00
|
|
|
{
|
|
|
|
disableSelection($("#playpause_button")[0]);
|
|
|
|
disableSelection($("#timeslider")[0]);
|
2013-11-23 19:39:52 +02:00
|
|
|
|
2012-04-15 19:25:23 +02:00
|
|
|
$(document).keyup(function(e)
|
2011-06-20 11:44:04 +01:00
|
|
|
{
|
2012-04-15 19:25:23 +02:00
|
|
|
var code = -1;
|
|
|
|
if (!e) var e = window.event;
|
|
|
|
if (e.keyCode) code = e.keyCode;
|
|
|
|
else if (e.which) code = e.which;
|
|
|
|
|
|
|
|
if (code == 37)
|
|
|
|
{ // left
|
|
|
|
if (!e.shiftKey)
|
|
|
|
{
|
|
|
|
setSliderPosition(getSliderPosition() - 1);
|
2011-06-20 11:44:04 +01:00
|
|
|
}
|
2012-04-15 19:25:23 +02:00
|
|
|
else
|
2011-06-20 11:44:04 +01:00
|
|
|
{
|
2012-04-15 19:25:23 +02:00
|
|
|
var nextStar = 0; // default to first revision in document
|
|
|
|
for (var i = 0; i < savedRevisions.length; i++)
|
2011-06-20 11:44:04 +01:00
|
|
|
{
|
2012-04-15 19:25:23 +02:00
|
|
|
var pos = parseInt(savedRevisions[i].attr('pos'));
|
|
|
|
if (pos < getSliderPosition() && nextStar < pos) nextStar = pos;
|
2011-06-20 11:44:04 +01:00
|
|
|
}
|
2012-04-15 19:25:23 +02:00
|
|
|
setSliderPosition(nextStar);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
else if (code == 39)
|
|
|
|
{
|
|
|
|
if (!e.shiftKey)
|
|
|
|
{
|
|
|
|
setSliderPosition(getSliderPosition() + 1);
|
|
|
|
}
|
|
|
|
else
|
|
|
|
{
|
|
|
|
var nextStar = sliderLength; // default to last revision in document
|
|
|
|
for (var i = 0; i < savedRevisions.length; i++)
|
2011-06-20 11:44:04 +01:00
|
|
|
{
|
2012-04-15 19:25:23 +02:00
|
|
|
var pos = parseInt(savedRevisions[i].attr('pos'));
|
|
|
|
if (pos > getSliderPosition() && nextStar > pos) nextStar = pos;
|
2011-06-20 11:44:04 +01:00
|
|
|
}
|
2012-04-15 19:25:23 +02:00
|
|
|
setSliderPosition(nextStar);
|
2011-06-20 11:44:04 +01:00
|
|
|
}
|
2012-04-15 19:25:23 +02:00
|
|
|
}
|
|
|
|
else if (code == 32) playpause();
|
2011-06-20 11:44:04 +01:00
|
|
|
|
2012-04-15 19:25:23 +02:00
|
|
|
});
|
2013-11-23 19:39:52 +02:00
|
|
|
|
2011-06-20 11:44:04 +01:00
|
|
|
|
|
|
|
// play/pause toggling
|
2013-12-02 17:43:39 +02:00
|
|
|
$("XXXX#playpause_button").mousedown(function(evt)
|
2011-06-20 11:44:04 +01:00
|
|
|
{
|
|
|
|
var self = this;
|
|
|
|
|
2011-06-20 15:37:41 +01:00
|
|
|
$(self).css('background-image', 'url(/static/img/crushed_button_depressed.png)');
|
2011-07-07 18:59:34 +01:00
|
|
|
$(self).mouseup(function(evt2)
|
2011-06-20 11:44:04 +01:00
|
|
|
{
|
2011-06-20 15:37:41 +01:00
|
|
|
$(self).css('background-image', 'url(/static/img/crushed_button_undepressed.png)');
|
2011-06-20 11:44:04 +01:00
|
|
|
$(self).unbind('mouseup');
|
|
|
|
BroadcastSlider.playpause();
|
|
|
|
});
|
2011-07-07 18:59:34 +01:00
|
|
|
$(document).mouseup(function(evt2)
|
2011-06-20 11:44:04 +01:00
|
|
|
{
|
2011-06-20 15:37:41 +01:00
|
|
|
$(self).css('background-image', 'url(/static/img/crushed_button_undepressed.png)');
|
2011-06-20 11:44:04 +01:00
|
|
|
$(document).unbind('mouseup');
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
// next/prev saved revision and changeset
|
2013-12-02 17:43:39 +02:00
|
|
|
$('XXX.stepper').mousedown(function(evt)
|
2011-06-20 11:44:04 +01:00
|
|
|
{
|
|
|
|
var self = this;
|
|
|
|
var origcss = $(self).css('background-position');
|
|
|
|
if (!origcss)
|
|
|
|
{
|
|
|
|
origcss = $(self).css('background-position-x') + " " + $(self).css('background-position-y');
|
|
|
|
}
|
|
|
|
var origpos = parseInt(origcss.split(" ")[1]);
|
|
|
|
var newpos = (origpos - 43);
|
|
|
|
if (newpos < 0) newpos += 87;
|
|
|
|
|
|
|
|
var newcss = (origcss.split(" ")[0] + " " + newpos + "px");
|
|
|
|
if ($(self).css('opacity') != 1.0) newcss = origcss;
|
|
|
|
|
|
|
|
$(self).css('background-position', newcss)
|
|
|
|
|
2011-07-07 18:59:34 +01:00
|
|
|
$(self).mouseup(function(evt2)
|
2011-06-20 11:44:04 +01:00
|
|
|
{
|
|
|
|
$(self).css('background-position', origcss);
|
|
|
|
$(self).unbind('mouseup');
|
|
|
|
$(document).unbind('mouseup');
|
|
|
|
if ($(self).attr("id") == ("leftstep"))
|
|
|
|
{
|
|
|
|
setSliderPosition(getSliderPosition() - 1);
|
|
|
|
}
|
|
|
|
else if ($(self).attr("id") == ("rightstep"))
|
|
|
|
{
|
|
|
|
setSliderPosition(getSliderPosition() + 1);
|
|
|
|
}
|
|
|
|
else if ($(self).attr("id") == ("leftstar"))
|
|
|
|
{
|
|
|
|
var nextStar = 0; // default to first revision in document
|
|
|
|
for (var i = 0; i < savedRevisions.length; i++)
|
|
|
|
{
|
|
|
|
var pos = parseInt(savedRevisions[i].attr('pos'));
|
|
|
|
if (pos < getSliderPosition() && nextStar < pos) nextStar = pos;
|
|
|
|
}
|
|
|
|
setSliderPosition(nextStar);
|
|
|
|
}
|
|
|
|
else if ($(self).attr("id") == ("rightstar"))
|
|
|
|
{
|
|
|
|
var nextStar = sliderLength; // default to last revision in document
|
|
|
|
for (var i = 0; i < savedRevisions.length; i++)
|
|
|
|
{
|
|
|
|
var pos = parseInt(savedRevisions[i].attr('pos'));
|
|
|
|
if (pos > getSliderPosition() && nextStar > pos) nextStar = pos;
|
|
|
|
}
|
|
|
|
setSliderPosition(nextStar);
|
|
|
|
}
|
|
|
|
});
|
2011-07-07 18:59:34 +01:00
|
|
|
$(document).mouseup(function(evt2)
|
2011-06-20 11:44:04 +01:00
|
|
|
{
|
|
|
|
$(self).css('background-position', origcss);
|
|
|
|
$(self).unbind('mouseup');
|
|
|
|
$(document).unbind('mouseup');
|
|
|
|
});
|
|
|
|
})
|
|
|
|
|
|
|
|
if (clientVars)
|
|
|
|
{
|
2012-04-15 19:25:23 +02:00
|
|
|
$("#timeslider").show();
|
2013-11-23 19:39:52 +02:00
|
|
|
|
2011-06-20 11:44:04 +01:00
|
|
|
}
|
|
|
|
});
|
|
|
|
})();
|
|
|
|
|
2011-07-07 18:59:34 +01:00
|
|
|
BroadcastSlider.onSlider(function(loc)
|
2011-06-20 11:44:04 +01:00
|
|
|
{
|
|
|
|
$("#viewlatest").html(loc == BroadcastSlider.getSliderLength() ? "Viewing latest content" : "View latest content");
|
|
|
|
})
|
2012-01-26 21:20:26 -08:00
|
|
|
|
|
|
|
return BroadcastSlider;
|
2011-06-20 11:44:04 +01:00
|
|
|
}
|
2012-01-15 17:23:48 -08:00
|
|
|
|
|
|
|
exports.loadBroadcastSliderJS = loadBroadcastSliderJS;
|