2011-12-04 16:33:56 +01:00
/ * *
* This code is mostly from the old Etherpad . Please help us to comment this code .
* 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 12:44:04 +02:00
/ * *
* Copyright 2009 Google Inc .
2011-07-07 19:59:34 +02:00
*
2011-06-20 12:44:04 +02: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 19:59:34 +02:00
*
2011-06-20 12:44:04 +02:00
* http : //www.apache.org/licenses/LICENSE-2.0
2011-07-07 19:59:34 +02:00
*
2011-06-20 12:44:04 +02: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-03-07 02:27:03 +01:00
var makeCSSManager = require ( './cssmanager' ) . makeCSSManager ;
var domline = require ( './domline' ) . domline ;
2012-03-18 09:05:46 +01:00
var AttribPool = require ( './AttributePool' ) ;
2012-03-07 02:27:03 +01:00
var Changeset = require ( './Changeset' ) ;
var linestylefilter = require ( './linestylefilter' ) . linestylefilter ;
var colorutils = require ( './colorutils' ) . colorutils ;
2012-03-17 13:36:42 +01:00
var _ = require ( './underscore' ) ;
2012-01-16 05:16:11 +01:00
2012-01-27 06:20:26 +01:00
// These parameters were global, now they are injected. A reference to the
// Timeslider controller would probably be more appropriate.
function loadBroadcastJS ( socket , sendSocketMsg , fireWhenAllScriptsAreLoaded , BroadcastSlider )
2011-06-20 12:44:04 +02:00
{
2012-01-27 06:20:26 +01:00
var changesetLoader = undefined ;
2011-06-20 12:44:04 +02:00
// Below Array#indexOf code was direct pasted by AppJet/Etherpad, licence unknown. Possible source: http://www.tutorialspoint.com/javascript/array_indexof.htm
if ( ! Array . prototype . indexOf )
{
2011-07-07 19:59:34 +02:00
Array . prototype . indexOf = function ( elt /*, from*/ )
2011-06-20 12:44:04 +02:00
{
var len = this . length >>> 0 ;
var from = Number ( arguments [ 1 ] ) || 0 ;
from = ( from < 0 ) ? Math . ceil ( from ) : Math . floor ( from ) ;
if ( from < 0 ) from += len ;
for ( ; from < len ; from ++ )
{
if ( from in this && this [ from ] === elt ) return from ;
}
return - 1 ;
} ;
}
function debugLog ( )
{
try
{
2011-07-07 19:59:34 +02:00
if ( window . console ) console . log . apply ( console , arguments ) ;
2011-06-20 12:44:04 +02:00
}
catch ( e )
{
2011-07-07 19:59:34 +02:00
if ( window . console ) console . log ( "error printing: " , e ) ;
2011-06-20 12:44:04 +02:00
}
}
// for IE
if ( $ . browser . msie )
{
try
{
document . execCommand ( "BackgroundImageCache" , false , true ) ;
}
catch ( e )
{ }
}
2012-02-19 16:05:07 +01:00
2011-06-20 16:37:41 +02:00
//var socket;
2011-06-20 12:44:04 +02:00
var channelState = "DISCONNECTED" ;
var appLevelDisconnectReason = null ;
var padContents = {
2012-04-23 12:52:30 +02:00
currentRevision : clientVars . collab _client _vars . rev ,
currentTime : clientVars . collab _client _vars . time ,
currentLines : Changeset . splitTextLines ( clientVars . collab _client _vars . initialAttributedText . text ) ,
2011-06-20 12:44:04 +02:00
currentDivs : null ,
// to be filled in once the dom loads
2012-04-23 12:52:30 +02:00
apool : ( new AttribPool ( ) ) . fromJsonable ( clientVars . collab _client _vars . apool ) ,
2011-06-20 12:44:04 +02:00
alines : Changeset . splitAttributionLines (
2012-04-23 12:52:30 +02:00
clientVars . collab _client _vars . initialAttributedText . attribs , clientVars . collab _client _vars . initialAttributedText . text ) ,
2011-06-20 12:44:04 +02:00
// generates a jquery element containing HTML for a line
2011-07-07 19:59:34 +02:00
lineToElement : function ( line , aline )
2011-06-20 12:44:04 +02:00
{
var element = document . createElement ( "div" ) ;
var emptyLine = ( line == '\n' ) ;
var domInfo = domline . createDomLine ( ! emptyLine , true ) ;
linestylefilter . populateDomLine ( line , aline , this . apool , domInfo ) ;
domInfo . prepareForAdd ( ) ;
element . className = domInfo . node . className ;
element . innerHTML = domInfo . node . innerHTML ;
element . id = Math . random ( ) ;
return $ ( element ) ;
} ,
2011-07-07 19:59:34 +02:00
applySpliceToDivs : function ( start , numRemoved , newLines )
2011-06-20 12:44:04 +02:00
{
// remove spliced-out lines from DOM
for ( var i = start ; i < start + numRemoved && i < this . currentDivs . length ; i ++ )
{
debugLog ( "removing" , this . currentDivs [ i ] . attr ( 'id' ) ) ;
this . currentDivs [ i ] . remove ( ) ;
}
// remove spliced-out line divs from currentDivs array
this . currentDivs . splice ( start , numRemoved ) ;
var newDivs = [ ] ;
for ( var i = 0 ; i < newLines . length ; i ++ )
{
newDivs . push ( this . lineToElement ( newLines [ i ] , this . alines [ start + i ] ) ) ;
}
// grab the div just before the first one
var startDiv = this . currentDivs [ start - 1 ] || null ;
// insert the div elements into the correct place, in the correct order
for ( var i = 0 ; i < newDivs . length ; i ++ )
{
if ( startDiv )
{
startDiv . after ( newDivs [ i ] ) ;
}
else
{
$ ( "#padcontent" ) . prepend ( newDivs [ i ] ) ;
}
startDiv = newDivs [ i ] ;
}
// insert new divs into currentDivs array
newDivs . unshift ( 0 ) ; // remove 0 elements
newDivs . unshift ( start ) ;
this . currentDivs . splice . apply ( this . currentDivs , newDivs ) ;
return this ;
} ,
// splice the lines
2011-07-07 19:59:34 +02:00
splice : function ( start , numRemoved , newLinesVA )
2011-06-20 12:44:04 +02:00
{
2012-03-17 13:36:42 +01:00
var newLines = _ . map ( Array . prototype . slice . call ( arguments , 2 ) , function ( s ) {
2011-06-20 12:44:04 +02:00
return s ;
} ) ;
// apply this splice to the divs
this . applySpliceToDivs ( start , numRemoved , newLines ) ;
// call currentLines.splice, to keep the currentLines array up to date
newLines . unshift ( numRemoved ) ;
newLines . unshift ( start ) ;
this . currentLines . splice . apply ( this . currentLines , arguments ) ;
} ,
// returns the contents of the specified line I
2011-07-07 19:59:34 +02:00
get : function ( i )
2011-06-20 12:44:04 +02:00
{
return this . currentLines [ i ] ;
} ,
// returns the number of lines in the document
2011-07-07 19:59:34 +02:00
length : function ( )
2011-06-20 12:44:04 +02:00
{
return this . currentLines . length ;
} ,
2011-07-07 19:59:34 +02:00
getActiveAuthors : function ( )
2011-06-20 12:44:04 +02:00
{
var self = this ;
var authors = [ ] ;
var seenNums = { } ;
var alines = self . alines ;
for ( var i = 0 ; i < alines . length ; i ++ )
{
2011-07-07 19:59:34 +02:00
Changeset . eachAttribNumber ( alines [ i ] , function ( n )
2011-06-20 12:44:04 +02:00
{
if ( ! seenNums [ n ] )
{
seenNums [ n ] = true ;
if ( self . apool . getAttribKey ( n ) == 'author' )
{
var a = self . apool . getAttribValue ( n ) ;
if ( a )
{
authors . push ( a ) ;
}
}
}
} ) ;
}
authors . sort ( ) ;
return authors ;
}
} ;
function callCatchingErrors ( catcher , func )
{
try
{
wrapRecordingErrors ( catcher , func ) ( ) ;
}
catch ( e )
{ /*absorb*/
}
}
function wrapRecordingErrors ( catcher , func )
{
2011-07-07 19:59:34 +02:00
return function ( )
2011-06-20 12:44:04 +02:00
{
try
{
return func . apply ( this , Array . prototype . slice . call ( arguments ) ) ;
}
catch ( e )
{
// caughtErrors.push(e);
// caughtErrorCatchers.push(catcher);
// caughtErrorTimes.push(+new Date());
// console.dir({catcher: catcher, e: e});
debugLog ( e ) ; // TODO(kroo): added temporary, to catch errors
throw e ;
}
} ;
}
function loadedNewChangeset ( changesetForward , changesetBackward , revision , timeDelta )
{
var broadcasting = ( BroadcastSlider . getSliderPosition ( ) == revisionInfo . latest ) ;
debugLog ( "broadcasting:" , broadcasting , BroadcastSlider . getSliderPosition ( ) , revisionInfo . latest , revision ) ;
revisionInfo . addChangeset ( revision , revision + 1 , changesetForward , changesetBackward , timeDelta ) ;
BroadcastSlider . setSliderLength ( revisionInfo . latest ) ;
if ( broadcasting ) applyChangeset ( changesetForward , revision + 1 , false , timeDelta ) ;
}
2013-01-10 01:01:38 +01:00
/ *
2011-06-20 12:44:04 +02:00
At this point , we must be certain that the changeset really does map from
the current revision to the specified revision . Any mistakes here will
cause the whole slider to get out of sync .
* /
function applyChangeset ( changeset , revision , preventSliderMovement , timeDelta )
2013-01-10 01:01:38 +01:00
{
2011-06-20 12:44:04 +02:00
// disable the next 'gotorevision' call handled by a timeslider update
if ( ! preventSliderMovement )
{
goToRevisionIfEnabledCount ++ ;
BroadcastSlider . setSliderPosition ( revision ) ;
}
try
{
// must mutate attribution lines before text lines
Changeset . mutateAttributionLines ( changeset , padContents . alines , padContents . apool ) ;
}
catch ( e )
{
debugLog ( e ) ;
}
Changeset . mutateTextLines ( changeset , padContents ) ;
padContents . currentRevision = revision ;
2013-01-10 01:01:38 +01:00
padContents . currentTime += timeDelta * 1000 ;
2011-06-20 12:44:04 +02:00
debugLog ( 'Time Delta: ' , timeDelta )
updateTimer ( ) ;
2012-02-19 15:47:20 +01:00
2012-03-17 13:36:42 +01:00
var authors = _ . map ( padContents . getActiveAuthors ( ) , function ( name )
2011-06-20 12:44:04 +02:00
{
return authorData [ name ] ;
2012-02-19 15:47:20 +01:00
} ) ;
BroadcastSlider . setAuthors ( authors ) ;
2011-06-20 12:44:04 +02:00
}
function updateTimer ( )
{
2011-07-07 19:59:34 +02:00
var zpad = function ( str , length )
2011-06-20 12:44:04 +02:00
{
str = str + "" ;
while ( str . length < length )
str = '0' + str ;
return str ;
}
var date = new Date ( padContents . currentTime ) ;
2011-07-07 19:59:34 +02:00
var dateFormat = function ( )
2011-06-20 12:44:04 +02:00
{
var month = zpad ( date . getMonth ( ) + 1 , 2 ) ;
var day = zpad ( date . getDate ( ) , 2 ) ;
var year = ( date . getFullYear ( ) ) ;
var hours = zpad ( date . getHours ( ) , 2 ) ;
var minutes = zpad ( date . getMinutes ( ) , 2 ) ;
var seconds = zpad ( date . getSeconds ( ) , 2 ) ;
2012-12-19 19:55:31 +01:00
return ( html10n . get ( "timeslider.dateformat" , {
2012-11-23 06:20:09 +01:00
"day" : day ,
"month" : month ,
"year" : year ,
"hours" : hours ,
"minutes" : minutes ,
"seconds" : seconds
} ) ) ;
2011-06-20 12:44:04 +02:00
}
2011-07-07 19:59:34 +02:00
2011-06-20 12:44:04 +02:00
$ ( '#timer' ) . html ( dateFormat ( ) ) ;
2012-12-19 19:55:31 +01:00
var revisionDate = html10n . get ( "timeslider.saved" , {
2012-11-23 06:20:09 +01:00
"day" : date . getDate ( ) ,
"month" : [
2012-12-19 19:55:31 +01:00
html10n . get ( "timeslider.month.january" ) ,
html10n . get ( "timeslider.month.february" ) ,
html10n . get ( "timeslider.month.march" ) ,
html10n . get ( "timeslider.month.april" ) ,
html10n . get ( "timeslider.month.may" ) ,
html10n . get ( "timeslider.month.june" ) ,
html10n . get ( "timeslider.month.july" ) ,
html10n . get ( "timeslider.month.august" ) ,
html10n . get ( "timeslider.month.september" ) ,
html10n . get ( "timeslider.month.october" ) ,
html10n . get ( "timeslider.month.november" ) ,
html10n . get ( "timeslider.month.december" )
2012-11-23 06:20:09 +01:00
] [ date . getMonth ( ) ] ,
"year" : date . getFullYear ( )
} ) ;
2011-06-20 12:44:04 +02:00
$ ( '#revision_date' ) . html ( revisionDate )
}
2011-07-25 18:26:56 +02:00
updateTimer ( ) ;
2011-06-20 12:44:04 +02:00
function goToRevision ( newRevision )
{
padContents . targetRevision = newRevision ;
var self = this ;
var path = revisionInfo . getPath ( padContents . currentRevision , newRevision ) ;
debugLog ( 'newRev: ' , padContents . currentRevision , path ) ;
if ( path . status == 'complete' )
{
var cs = path . changesets ;
debugLog ( "status: complete, changesets: " , cs , "path:" , path ) ;
var changeset = cs [ 0 ] ;
var timeDelta = path . times [ 0 ] ;
for ( var i = 1 ; i < cs . length ; i ++ )
{
changeset = Changeset . compose ( changeset , cs [ i ] , padContents . apool ) ;
timeDelta += path . times [ i ] ;
}
if ( changeset ) applyChangeset ( changeset , path . rev , true , timeDelta ) ;
}
else if ( path . status == "partial" )
{
debugLog ( 'partial' ) ;
var sliderLocation = padContents . currentRevision ;
// callback is called after changeset information is pulled from server
// this may never get called, if the changeset has already been loaded
2011-07-07 19:59:34 +02:00
var update = function ( start , end )
2011-06-20 12:44:04 +02:00
{
// if we've called goToRevision in the time since, don't goToRevision
goToRevision ( padContents . targetRevision ) ;
} ;
// do our best with what we have...
var cs = path . changesets ;
var changeset = cs [ 0 ] ;
var timeDelta = path . times [ 0 ] ;
for ( var i = 1 ; i < cs . length ; i ++ )
{
changeset = Changeset . compose ( changeset , cs [ i ] , padContents . apool ) ;
timeDelta += path . times [ i ] ;
}
if ( changeset ) applyChangeset ( changeset , path . rev , true , timeDelta ) ;
2014-08-09 18:52:44 +02:00
// Loading changeset history for new revision
loadChangesetsForRevision ( newRevision , update ) ;
// Loading changeset history for old revision (to make diff between old and new revision)
loadChangesetsForRevision ( padContents . currentRevision - 1 ) ;
2011-06-20 12:44:04 +02:00
}
2012-02-19 15:47:20 +01:00
2012-03-17 13:36:42 +01:00
var authors = _ . map ( padContents . getActiveAuthors ( ) , function ( name ) {
2011-06-20 12:44:04 +02:00
return authorData [ name ] ;
2012-02-19 15:47:20 +01:00
} ) ;
BroadcastSlider . setAuthors ( authors ) ;
2011-06-20 12:44:04 +02:00
}
2014-08-09 18:52:44 +02:00
function loadChangesetsForRevision ( revision , callback ) {
if ( BroadcastSlider . getSliderLength ( ) > 10000 )
{
var start = ( Math . floor ( ( revision ) / 10000 ) * 10000 ) ; // revision 0 to 10
changesetLoader . queueUp ( start , 100 ) ;
}
if ( BroadcastSlider . getSliderLength ( ) > 1000 )
{
var start = ( Math . floor ( ( revision ) / 1000 ) * 1000 ) ; // (start from -1, go to 19) + 1
changesetLoader . queueUp ( start , 10 ) ;
}
start = ( Math . floor ( ( revision ) / 100 ) * 100 ) ;
changesetLoader . queueUp ( start , 1 , callback ) ;
}
2011-06-20 12:44:04 +02:00
2012-01-27 06:20:26 +01:00
changesetLoader = {
2011-06-20 12:44:04 +02:00
running : false ,
resolved : [ ] ,
requestQueue1 : [ ] ,
requestQueue2 : [ ] ,
requestQueue3 : [ ] ,
2011-06-20 16:37:41 +02:00
reqCallbacks : [ ] ,
2011-07-07 19:59:34 +02:00
queueUp : function ( revision , width , callback )
2011-06-20 12:44:04 +02:00
{
if ( revision < 0 ) revision = 0 ;
// if(changesetLoader.requestQueue.indexOf(revision) != -1)
// return; // already in the queue.
if ( changesetLoader . resolved . indexOf ( revision + "_" + width ) != - 1 ) return ; // already loaded from the server
changesetLoader . resolved . push ( revision + "_" + width ) ;
var requestQueue = width == 1 ? changesetLoader . requestQueue3 : width == 10 ? changesetLoader . requestQueue2 : changesetLoader . requestQueue1 ;
requestQueue . push (
{
'rev' : revision ,
'res' : width ,
'callback' : callback
} ) ;
if ( ! changesetLoader . running )
{
changesetLoader . running = true ;
setTimeout ( changesetLoader . loadFromQueue , 10 ) ;
}
} ,
2011-07-07 19:59:34 +02:00
loadFromQueue : function ( )
2011-06-20 12:44:04 +02:00
{
var self = changesetLoader ;
var requestQueue = self . requestQueue1 . length > 0 ? self . requestQueue1 : self . requestQueue2 . length > 0 ? self . requestQueue2 : self . requestQueue3 . length > 0 ? self . requestQueue3 : null ;
if ( ! requestQueue )
{
self . running = false ;
return ;
}
var request = requestQueue . pop ( ) ;
var granularity = request . res ;
var callback = request . callback ;
var start = request . rev ;
2011-07-07 19:59:34 +02:00
var requestID = Math . floor ( Math . random ( ) * 100000 ) ;
sendSocketMsg ( "CHANGESET_REQ" , {
"start" : start ,
"granularity" : granularity ,
"requestID" : requestID
} ) ;
2011-06-20 16:37:41 +02:00
self . reqCallbacks [ requestID ] = callback ;
} ,
2011-07-07 19:59:34 +02:00
handleSocketResponse : function ( message )
{
var self = changesetLoader ;
var start = message . data . start ;
var granularity = message . data . granularity ;
var callback = self . reqCallbacks [ message . data . requestID ] ;
delete self . reqCallbacks [ message . data . requestID ] ;
self . handleResponse ( message . data , start , granularity , callback ) ;
setTimeout ( self . loadFromQueue , 10 ) ;
2011-06-20 12:44:04 +02:00
} ,
2011-07-07 19:59:34 +02:00
handleResponse : function ( data , start , granularity , callback )
2011-06-20 12:44:04 +02:00
{
debugLog ( "response: " , data ) ;
var pool = ( new AttribPool ( ) ) . fromJsonable ( data . apool ) ;
for ( var i = 0 ; i < data . forwardsChangesets . length ; i ++ )
{
var astart = start + i * granularity - 1 ; // rev -1 is a blank single line
var aend = start + ( i + 1 ) * granularity - 1 ; // totalRevs is the most recent revision
if ( aend > data . actualEndNum - 1 ) aend = data . actualEndNum - 1 ;
2014-08-09 18:52:44 +02:00
//debugLog("adding changeset:", astart, aend);
2011-06-20 12:44:04 +02:00
var forwardcs = Changeset . moveOpsToNewPool ( data . forwardsChangesets [ i ] , pool , padContents . apool ) ;
var backwardcs = Changeset . moveOpsToNewPool ( data . backwardsChangesets [ i ] , pool , padContents . apool ) ;
revisionInfo . addChangeset ( astart , aend , forwardcs , backwardcs , data . timeDeltas [ i ] ) ;
}
if ( callback ) callback ( start - 1 , start + data . forwardsChangesets . length * granularity - 1 ) ;
2012-04-23 13:29:27 +02:00
} ,
handleMessageFromServer : function ( obj )
2011-06-20 12:44:04 +02:00
{
2012-04-23 13:29:27 +02:00
debugLog ( "handleMessage:" , arguments ) ;
2011-06-20 12:44:04 +02:00
2012-04-23 13:29:27 +02:00
if ( obj . type == "COLLABROOM" )
2011-06-20 12:44:04 +02:00
{
2012-04-23 13:29:27 +02:00
obj = obj . data ;
if ( obj . type == "NEW_CHANGES" )
{
debugLog ( obj ) ;
var changeset = Changeset . moveOpsToNewPool (
obj . changeset , ( new AttribPool ( ) ) . fromJsonable ( obj . apool ) , padContents . apool ) ;
var changesetBack = Changeset . inverse (
obj . changeset , padContents . currentLines , padContents . alines , padContents . apool ) ;
var changesetBack = Changeset . moveOpsToNewPool (
changesetBack , ( new AttribPool ( ) ) . fromJsonable ( obj . apool ) , padContents . apool ) ;
loadedNewChangeset ( changeset , changesetBack , obj . newRev - 1 , obj . timeDelta ) ;
}
else if ( obj . type == "NEW_AUTHORDATA" )
{
var authorMap = { } ;
authorMap [ obj . author ] = obj . data ;
receiveAuthorData ( authorMap ) ;
2011-06-20 12:44:04 +02:00
2012-04-23 13:29:27 +02:00
var authors = _ . map ( padContents . getActiveAuthors ( ) , function ( name ) {
return authorData [ name ] ;
} ) ;
2011-06-20 12:44:04 +02:00
2012-04-23 13:29:27 +02:00
BroadcastSlider . setAuthors ( authors ) ;
}
else if ( obj . type == "NEW_SAVEDREV" )
{
var savedRev = obj . savedRev ;
BroadcastSlider . addSavedRevision ( savedRev . revNum , savedRev ) ;
}
2011-06-20 12:44:04 +02:00
}
2012-04-23 13:29:27 +02:00
else if ( obj . type == "CHANGESET_REQ" )
2011-06-20 12:44:04 +02:00
{
2012-04-23 13:29:27 +02:00
changesetLoader . handleSocketResponse ( obj ) ;
2011-06-20 12:44:04 +02:00
}
2012-04-23 13:29:27 +02:00
else
2011-06-20 12:44:04 +02:00
{
2012-04-23 13:29:27 +02:00
debugLog ( "Unknown message type: " + obj . type ) ;
2011-06-20 12:44:04 +02:00
}
}
2012-04-23 13:29:27 +02:00
} ;
2011-06-20 12:44:04 +02:00
// to start upon window load, just push a function onto this array
//window['onloadFuncts'].push(setUpSocket);
//window['onloadFuncts'].push(function ()
2011-07-07 19:59:34 +02:00
fireWhenAllScriptsAreLoaded . push ( function ( )
2011-06-20 12:44:04 +02:00
{
// set up the currentDivs and DOM
padContents . currentDivs = [ ] ;
$ ( "#padcontent" ) . html ( "" ) ;
for ( var i = 0 ; i < padContents . currentLines . length ; i ++ )
{
var div = padContents . lineToElement ( padContents . currentLines [ i ] , padContents . alines [ i ] ) ;
padContents . currentDivs . push ( div ) ;
$ ( "#padcontent" ) . append ( div ) ;
}
debugLog ( padContents . currentDivs ) ;
} ) ;
// this is necessary to keep infinite loops of events firing,
// since goToRevision changes the slider position
var goToRevisionIfEnabledCount = 0 ;
2012-04-23 14:47:07 +02:00
var goToRevisionIfEnabled = function ( ) {
if ( goToRevisionIfEnabledCount > 0 )
2011-06-20 12:44:04 +02:00
{
2012-04-23 14:47:07 +02:00
goToRevisionIfEnabledCount -- ;
}
else
2011-06-20 12:44:04 +02:00
{
2012-04-23 14:47:07 +02:00
goToRevision . apply ( goToRevision , arguments ) ;
2011-06-20 12:44:04 +02:00
}
2012-04-23 14:47:07 +02:00
}
BroadcastSlider . onSlider ( goToRevisionIfEnabled ) ;
2011-06-20 12:44:04 +02:00
var dynamicCSS = makeCSSManager ( 'dynamicsyntax' ) ;
var authorData = { } ;
function receiveAuthorData ( newAuthorData )
{
for ( var author in newAuthorData )
{
var data = newAuthorData [ author ] ;
2011-08-21 19:53:30 +02:00
var bgcolor = typeof data . colorId == "number" ? clientVars . colorPalette [ data . colorId ] : data . colorId ;
2011-08-20 19:22:10 +02:00
if ( bgcolor && dynamicCSS )
2011-06-20 12:44:04 +02:00
{
2012-02-06 16:00:42 +01:00
var selector = dynamicCSS . selectorStyle ( '.' + linestylefilter . getAuthorClassName ( author ) ) ;
selector . backgroundColor = bgcolor
selector . color = ( colorutils . luminosity ( colorutils . css2triple ( bgcolor ) ) < 0.5 ) ? '#ffffff' : '#000000' ; //see ace2_inner.js for the other part
2011-06-20 12:44:04 +02:00
}
authorData [ author ] = data ;
}
}
2012-04-23 12:52:30 +02:00
receiveAuthorData ( clientVars . collab _client _vars . historicalAuthorData ) ;
2012-01-27 06:20:26 +01:00
return changesetLoader ;
2011-06-20 12:44:04 +02:00
}
2012-01-16 02:23:48 +01:00
exports . loadBroadcastJS = loadBroadcastJS ;