Add correct options to annotviz.html fo 3rd day and gulp dist server
authorrougeronj
Fri, 23 Jan 2015 23:19:45 +0100
changeset 142 e424eed32f72
parent 141 e6892a30d454
child 143 95104e98d78d
Add correct options to annotviz.html fo 3rd day and gulp dist server
annot-server/static/js/annotviz.js
annot-server/static/js/annotviz.min.js
annot-server/webapp/templates/annotviz.html
--- a/annot-server/static/js/annotviz.js	Fri Jan 23 18:18:33 2015 +0100
+++ b/annot-server/static/js/annotviz.js	Fri Jan 23 23:19:45 2015 +0100
@@ -83,7 +83,7 @@
     var marginX = 15;
     for(var style in annotStyles) {
     	if (annotStyles[style].wordWrap === true){
-    		annotStyles[style].wordWrapWidth = this.widthRoll - marginX; 
+    		annotStyles[style].wordWrapWidth = this.widthRoll - marginX;
     	}
     }
     var started = false;
@@ -92,7 +92,7 @@
     var stageView = opts.stageView;
     var waitInterval;
     var wait = 0;
-    
+
     stageView.registerComponent(this);
 
     var isHidden = function(child) {
@@ -132,7 +132,7 @@
             resColor = colorsDef.defaultColor || DEFAULT_ANNOT_COLOR;
         }
         return resColor;
-    }
+    };
 
     this.addAnnot = function(category, text, user, color, ts){
 
@@ -146,9 +146,9 @@
 	            .beginFill(colorHex)
 	            .drawRect(x, y, 10, 3)
 	            .endFill();
-	
+
 	        this.container.addChild(graphics);
-	
+
 	        var textHeight = 0;
 	        var catLabel = new PIXI.Text(
 	            category,
@@ -158,7 +158,7 @@
 	        catLabel.y = y - 23;
 	        this.container.addChild(catLabel);
 	        textHeight += (catLabel.height - 23 + 2);
-	
+
 	        if(text) {
 	            var catText = new PIXI.Text(text, annotStyles.text);
 	            catText.x = x + marginX;
@@ -166,13 +166,13 @@
 	            this.container.addChild(catText);
 	            textHeight += (catText.height + 2);
 	        }
-	
+
 	        var catUser = new PIXI.Text(user, annotStyles.user);
 	        catUser.x = x + marginX;
 	        catUser.y = y + textHeight;
 	        this.container.addChild(catUser);
 	        textHeight += (catUser.height + 8);
-	        
+
 	        if (this.ignoreAnnots === true){
 		        wait = textHeight / this.pixelsPerSecond;
 		        waitInterval = setInterval(function() {_this.refreshWait();}, 1000);
@@ -202,7 +202,7 @@
         var diff = (this.startTs - Date.now())/1000;
         this.moveTo(diff);
     };
-    
+
     this.refreshWait = function(){
     	wait -= 1;
     	if (wait < 0){
@@ -229,7 +229,7 @@
         });
 
     };
-    
+
 
     this.start = function() {
         if(!started) {
@@ -302,29 +302,29 @@
     this.intervalWidth = opts.intervalWidth;
     this.maxCellHeight = opts.maxCellHeight;
     this.annotCategories = opts.annotCategories;
-	this.startTs = options.startTs || Date.now();
+    this.startTs = options.startTs || Date.now();
     this.showClockGraphics = opts.showClockGraphics;
     this.archive = opts.archive;
-    
+
     this.circleX = opts.circleX || (this.width/2);
     this.circleY = opts.circleY || (this.height/2);
     this.radius = opts.radius;
     var perimeter = 2*Math.PI* this.radius;
     this.intervalDuration = (this.intervalWidth * this.duration / perimeter);
-    
+
     var channel = opts.channel;
     var eventCode = opts.eventCode;
     var serverUrl = opts.serverUrl;
     var maxPages = opts.maxPages;
-    
+
     var totalIndex = Math.floor( perimeter/this.intervalWidth);
 
-    this.cells = []
+    this.cells = [];
     for (var i=0; i<(perimeter/this.intervalWidth) ; i++){
-    	this.cells[i] = [];
-    	this.cells[i].i = i;
-    	this.cells[i].totalAnnots = 0;
-    	this.cells[i].categories = {};
+        this.cells[i] = [];
+        this.cells[i].i = i;
+        this.cells[i].totalAnnots = 0;
+        this.cells[i].categories = {};
     }
 
     var ws = opts.ws;
@@ -333,18 +333,21 @@
     //draw the base - circle and line to locate the scene
     var graphics = new PIXI.Graphics();
     graphics.lineStyle(2, 0x646464)
-    	.drawCircle(this.circleX, this.circleY, this.radius - 3)
-    	.endFill()
+        .drawCircle(this.circleX, this.circleY, this.radius - 3)
+        .endFill();
     this.container.addChild(graphics);
 
     stageView.registerComponent(this);
 
-	var loadArchives = function() {
+    var loadArchives = function() {
         //start timeBegin end startTime
         //query -> need channel + eventCode
         //iterate over data fill cells
         var startTs = _this.timeBegin;
-        var endTs = _this.startTs;
+        var endTs = Math.min(_this.timeEnd,_this.startTs);
+
+        console.log("START TS",new Date(startTs).toISOString());
+        console.log("END TS",new Date(endTs).toISOString());
 
         var url = serverUrl + '/p/api/v1/annotation';
         var filters = [
@@ -360,7 +363,7 @@
         var currentPage = 1;
 
         var processResFunction = function(res) {
-
+            //console.log("RES archive", res);
             if(res) {
                 var data = res.target.json;
                 /*jshint -W069 */
@@ -383,73 +386,78 @@
     //Add Annotation to the TimeLine
     this.addAnnot = function(data){
 
-    	var ts = Date.parse(data.ts);
-    	var colorsDef;
-    	_(this.annotCategories).eachRight(function(cdef) {
+        var ts = Date.parse(data.ts);
+        var colorsDef;
+        _(this.annotCategories).eachRight(function(cdef) {
             if(cdef.ts < ts) {
                 colorsDef = cdef;
                 return false;
             }
         });
 
-    	if (this.timeEnd > ts){
-	    	var i = Math.floor((ts - this.timeBegin)/(1000*this.intervalDuration));
+        if(!colorsDef) {
+            return;
+        }
 
-	    	if (typeof(this.cells[i].graphics) === 'undefined'){
-	    		this.initCell(this.cells[i], colorsDef);
-	    	}
+        if (this.timeEnd > ts){
+            var i = Math.floor((ts - this.timeBegin)/(1000*this.intervalDuration));
+
+            if (typeof(this.cells[i].graphics) === 'undefined'){
+                this.initCell(this.cells[i], colorsDef);
+            }
 
-	    	if (typeof(colorsDef.colors[data.content.category.code]) !== 'undefined'){
-	    		var annotCode = data.content.category.code;
-	    	} else {
-	    		var annotCode = 'default';
-	    	}
+            var annotCode;
+            if (typeof(colorsDef.colors[data.content.category.code]) !== 'undefined'){
+                annotCode = data.content.category.code;
+            } else {
+                annotCode = 'default';
+            }
 
-			this.cells[i].categories[annotCode].count += 1;
-			this.cells[i].totalAnnots +=1;
-			this.redrawCell(this.cells[i], colorsDef);
-    	}
+            this.cells[i].categories[annotCode].count += 1;
+            this.cells[i].totalAnnots +=1;
+            this.redrawCell(this.cells[i], colorsDef);
+        }
     };
 
     this.initClockGraphics = function() {
-	    var tBeg = new PIXI.Text(Utils.formatTime(this.timeBegin), { font: '12pt Gothic Standard', fill: '#646464' });
-	    tBeg.x = this.circleX + 15;
-	    tBeg.y = this.circleY - this.radius - this.maxCellHeight - 10;
-	    this.container.addChild(tBeg);
+        var tBeg = new PIXI.Text(Utils.formatTime(this.timeBegin), { font: '12pt Gothic Standard', fill: '#646464' });
+        tBeg.x = this.circleX + 15;
+        tBeg.y = this.circleY - this.radius - this.maxCellHeight - 10;
+        this.container.addChild(tBeg);
 
-	    var tEnd = new PIXI.Text(Utils.formatTime(this.timeEnd), { font: '12pt Gothic Standard', fill: '#646464' });
-	    tEnd.x = this.circleX - 15 - tEnd.width;
-	    tEnd.y = this.circleY - this.radius - this.maxCellHeight - 10;
-	    this.container.addChild(tEnd);
+        var tEnd = new PIXI.Text(Utils.formatTime(this.timeEnd), { font: '12pt Gothic Standard', fill: '#646464' });
+        tEnd.x = this.circleX - 15 - tEnd.width;
+        tEnd.y = this.circleY - this.radius - this.maxCellHeight - 10;
+        this.container.addChild(tEnd);
 
-	    var t15 = new PIXI.Text(Utils.formatTime(((this.timeEnd - this.timeBegin)/4) + this.timeBegin), { font: '12pt Gothic Standard', fill: '#646464' });
-	    t15.x = this.circleX + this.radius + this.maxCellHeight + 10 ;
-	    t15.y = this.circleY - t15.height;
-	    t15.rotation = Math.PI /2;
-	    this.container.addChild(t15);
+        var t15 = new PIXI.Text(Utils.formatTime(((this.timeEnd - this.timeBegin)/4) + this.timeBegin), { font: '12pt Gothic Standard', fill: '#646464' });
+        t15.x = this.circleX + this.radius + this.maxCellHeight + 10 ;
+        t15.y = this.circleY - t15.height;
+        t15.rotation = Math.PI /2;
+        this.container.addChild(t15);
 
-	    var t30 = new PIXI.Text(Utils.formatTime(((this.timeEnd - this.timeBegin)/2) + this.timeBegin), { font: '12pt Gothic Standard', fill: '#646464' });
-	    t30.x = this.circleX - t30.width/2;
-	    t30.y = this.circleY + this.radius + this.maxCellHeight - 2;
-	    this.container.addChild(t30);
+        var t30 = new PIXI.Text(Utils.formatTime(((this.timeEnd - this.timeBegin)/2) + this.timeBegin), { font: '12pt Gothic Standard', fill: '#646464' });
+        t30.x = this.circleX - t30.width/2;
+        t30.y = this.circleY + this.radius + this.maxCellHeight - 2;
+        this.container.addChild(t30);
 
-	    var t45 = new PIXI.Text(Utils.formatTime(((this.timeEnd - this.timeBegin)*3/4) + this.timeBegin), { font: '12pt Gothic Standard', fill: '#646464' });
-	    t45.x = this.circleX - this.radius - this.maxCellHeight - 10 ;
-	    t45.y = this.circleY + t15.height;
-	    t45.rotation = -Math.PI/2;
-	    this.container.addChild(t45);
-	    
-	    var lineV = new PIXI.Graphics();
-	    lineV.lineStyle(1, 0x646464)
-	    	.moveTo(this.circleX, this.circleY - (this.radius/3)/2)
-	    	.lineTo(this.circleX, this.circleY - this.radius - this.maxCellHeight - 10)
-    	.endFill();
-	    this.container.addChild(lineV);
-    }
+        var t45 = new PIXI.Text(Utils.formatTime(((this.timeEnd - this.timeBegin)*3/4) + this.timeBegin), { font: '12pt Gothic Standard', fill: '#646464' });
+        t45.x = this.circleX - this.radius - this.maxCellHeight - 10 ;
+        t45.y = this.circleY + t15.height;
+        t45.rotation = -Math.PI/2;
+        this.container.addChild(t45);
+
+        var lineV = new PIXI.Graphics();
+        lineV.lineStyle(1, 0x646464)
+            .moveTo(this.circleX, this.circleY - (this.radius/3)/2)
+            .lineTo(this.circleX, this.circleY - this.radius - this.maxCellHeight - 10)
+            .endFill();
+        this.container.addChild(lineV);
+    };
 
     //Draw the cellule
     this.redrawCell = function(cell, colorsDef){
-    	var y = 0;
+        var y = 0;
 
         //Check if total height is higher than Max Cell Height
         var heightStep;
@@ -459,50 +467,50 @@
             heightStep = this.intervalHeight;
         }
 
-    	//Draw the rect depending on the height step calculated
-    	for (var i=0; i< colorsDef.order.length; i++){
-    		var currentCode = colorsDef.order[i];
-			cell.graphics.beginFill(cell.categories[currentCode].color.replace("#", "0x"))
-    			.drawRect(0, y, this.intervalWidth-1, -cell.categories[currentCode].count * heightStep)
-    			.endFill();
-    		y -= cell.categories[currentCode].count*heightStep;
-    	}
-    }
+        //Draw the rect depending on the height step calculated
+        for (var i=0; i< colorsDef.order.length; i++){
+            var currentCode = colorsDef.order[i];
+            cell.graphics.beginFill(cell.categories[currentCode].color.replace('#', '0x'))
+                .drawRect(0, y, this.intervalWidth-1, -cell.categories[currentCode].count * heightStep)
+                .endFill();
+            y -= cell.categories[currentCode].count*heightStep;
+        }
+    };
 
     this.initCell = function(cell, colorsDef){
-    	cell.graphics = new PIXI.Graphics();
-    	cell.graphics.position.x = this.circleX + this.radius * Math.sin(cell.i*(360/totalIndex)*(Math.PI/180));
-    	cell.graphics.position.y = this.circleY - this.radius * Math.cos(cell.i*(360/totalIndex)*(Math.PI/180));
-    	cell.graphics.rotation = (cell.i)*(360/totalIndex)*(Math.PI/180) + (360/(totalIndex*2))*(Math.PI/180);
-    	this.container.addChild(cell.graphics);
+        cell.graphics = new PIXI.Graphics();
+        cell.graphics.position.x = this.circleX + this.radius * Math.sin(cell.i*(360/totalIndex)*(Math.PI/180));
+        cell.graphics.position.y = this.circleY - this.radius * Math.cos(cell.i*(360/totalIndex)*(Math.PI/180));
+        cell.graphics.rotation = (cell.i)*(360/totalIndex)*(Math.PI/180) + (360/(totalIndex*2))*(Math.PI/180);
+        this.container.addChild(cell.graphics);
 
-    	for (var category in colorsDef.colors){
-    		cell.categories[category] = {
-				"count": 0,
-				"color": colorsDef.colors[category]
-    		};
-    	}
-    	if (typeof(cell.categories['default']) === 'undefined'){
-			cell.categories['default'] = {
-				"count": 0,
-				"color": colorsDef.defaultColor
-			}
-    	}
-    }
+        for (var category in colorsDef.colors){
+            cell.categories[category] = {
+                'count': 0,
+                'color': colorsDef.colors[category]
+            };
+        }
+        if (typeof(cell.categories['default']) === 'undefined'){
+            cell.categories['default'] = {
+                'count': 0,
+                'color': colorsDef.defaultColor
+            };
+        }
+    };
 
     this.init = function() {
-    	if (!this.archive){
-	        ws.message(function(data) {
-	            _this.addAnnot(data);
-	        });
-    	}
+        if (!this.archive){
+            ws.message(function(data) {
+                _this.addAnnot(data);
+            });
+        }
 
-    	if (this.showClockGraphics){this.initClockGraphics();}
+        if (this.showClockGraphics){this.initClockGraphics();}
     };
 
 
     this.start = function() {
-    	this.startTs = Date.now();
+        this.startTs = Date.now();
         loadArchives();
     };
 
@@ -517,7 +525,7 @@
 }
 
 module.exports = {
-	AnnotsTimeLine: AnnotsTimeLine
+    AnnotsTimeLine: AnnotsTimeLine
 };
 
 },{"./utils.js":9,"lodash":"lodash","pixi":"pixi"}],4:[function(require,module,exports){
@@ -545,33 +553,42 @@
 };
 
 function AnnotsVizView(options){
-	var _this = this;
+    var _this = this;
     var opts = _(options).defaults(defaultOptions).value();
 
     this.container = new PIXI.DisplayObjectContainer();
     this.container.x = opts.xInit;
     this.container.y = opts.yInit;
-	this.width = opts.width;
-	this.height= opts.height;
+    this.width = opts.width;
+    this.height= opts.height;
     this.timeBegin = opts.timeBegin;
     this.timeEnd = opts.timeEnd;
+
     this.annotCategories = [];
-    
-    Utils.getAnnotCategories(opts.urlCategories, this.annotCategories);
+    this.annotCategoriesDay2 = [];
+    this.annotCategoriesDay1 = [];
+
+    var wsPianoroll = opts.wsPianoroll;
+    var wsAnnot = opts.wsAnnot;
+    var stageView = opts.stageView;
+    var currentTime = Date.now() + 3600*1000;
+    var channel = opts.channel;
+    var serverUrl = opts.serverUrl;
     
-	var wsPianoroll = opts.wsPianoroll;
-	var wsAnnot = opts.wsAnnot;
-	var stageView = opts.stageView;
-	var currentTime = Date.now() + 3600*1000;
-	var eventCode = opts.eventCode;
-    var channel = opts.channel;
+    var eventCode = opts.eventCode;
+    var eventCodeSessionDay2 = opts.eventCodeSessionDay2;
+	var eventCodeSessionDay1 = opts.eventCodeSessionDay1;
 
-	stageView.registerComponent(this);
+	Utils.getAnnotCategories(opts.urlCategories + eventCode, this.annotCategories);
+    Utils.getAnnotCategories(opts.urlCategories + eventCodeSessionDay2, this.annotCategoriesDay2);
+    Utils.getAnnotCategories(opts.urlCategories + eventCodeSessionDay1, this.annotCategoriesDay1);
 
-	var timeLineDay2 = new AnnotsTimeLine.AnnotsTimeLine({
+    stageView.registerComponent(this);
+
+    var currentDay = new AnnotsTimeLine.AnnotsTimeLine({
     	stageView : stageView,
         logger: logger,
-        ws: new annotviz.WsWrapper(wsUriAnnotation, logger),
+        ws: wsAnnot,
         xInit: 0,
         yInit: 0,
         width: 1024 - 200 - 200,
@@ -584,45 +601,64 @@
         radius: 200,
         eventCode: eventCode,
         channel: channel,
+        serverUrl: serverUrl,
         annotCategories: this.annotCategories
     });
 	
-    //Archive day 1
+	//Archive day 2
+    var timeLineDay2 = new AnnotsTimeLine.AnnotsTimeLine({
+    	stageView : stageView,
+        xInit: 0,
+        yInit: 0,
+        width: 1024 - 200 - 200,
+        height: 768-200,
+        timeBegin: Date.parse("2015-01-23T09:30:00+01:00"),
+        timeEnd: Date.parse("2015-01-23T18:30:00+01:00"),
+        circleX: currentDay.circleX,
+        circleY: currentDay.circleY,
+        intervalWidth: (currentDay.radius*2/3)* currentDay.intervalWidth / currentDay.radius,
+        intervalHeight: (currentDay.intervalHeight * (currentDay.radius - currentDay.radius*2/3))/ currentDay.maxCellHeight,
+        maxCellHeight: (currentDay.radius - currentDay.radius*2/3)/2,
+        radius: currentDay.radius*2/3,
+        annotCategories: this.annotCategoriesDay2,
+        eventCode: eventCodeSessionDay2,
+        channel: channel,
+        serverUrl: serverUrl,
+        showClockGraphics:false,
+        archive: true
+    });
+	
+	//Archive day 1
     var timeLineDay1 = new AnnotsTimeLine.AnnotsTimeLine({
     	stageView : stageView,
-        archive: true,
         xInit: 0,
         yInit: 0,
         width: 1024 - 200 - 200,
         height: 768-200,
         timeBegin: Date.parse("2015-01-22T09:30:00+01:00"),
         timeEnd: Date.parse("2015-01-22T18:30:00+01:00"),
-        circleX: timeLineDay2.circleX,
-        circleY: timeLineDay2.circleY,
-        intervalWidth: (timeLineDay2.radius*2/3)* timeLineDay2.intervalWidth / timeLineDay2.radius,
-        intervalHeight: (timeLineDay2.intervalHeight * (timeLineDay2.radius - timeLineDay2.radius*2/3))/ timeLineDay2.maxCellHeight,
-        maxCellHeight: (timeLineDay2.radius - timeLineDay2.radius*2/3)/4,
-        radius: timeLineDay2.radius*2/3,
-        annotCategories: this.annotCategories,
-        eventCode: eventCode,
+        circleX: currentDay.circleX,
+        circleY: currentDay.circleY,
+        intervalWidth: (currentDay.radius*2/3)* currentDay.intervalWidth / currentDay.radius,
+        intervalHeight: (currentDay.intervalHeight * (currentDay.radius - currentDay.radius/3))/ currentDay.maxCellHeight,
+        maxCellHeight: (currentDay.radius*2/3 - currentDay.radius/3)/4,
+        radius: currentDay.radius/3,
+        annotCategories: this.annotCategoriesDay1,
+        eventCode: eventCodeSessionDay1,
         channel: channel,
-        showClockGraphics:false
+        serverUrl: serverUrl,
+        showClockGraphics:false,
+        archive: true,
     });
-    
+
     var currentTimeText = new PIXI.Text("-- : -- : --", { font: '18pt Gothic Standard', fill: '#646464' });
     currentTimeText.x = timeLineDay2.circleX - currentTimeText.width/2;
     currentTimeText.y = timeLineDay2.circleY - currentTimeText.height/2;
     this.container.addChild(currentTimeText);
 
-    var timeLineDay3 = new PIXI.Graphics();
-    timeLineDay3.lineStyle(1, 0x646464)
-    	.drawCircle(timeLineDay2.circleX, timeLineDay2.circleY, timeLineDay2.radius/3)
-    	.endFill()
-    this.container.addChild(timeLineDay3);
-    
-	var doubleRollH = new DoubleRoll.DoubleRoll({
+    var doubleRollH = new DoubleRoll.DoubleRoll({
         stageView : stageView,
-    	logger: logger,
+        logger: logger,
         ws: wsPianoroll,
         yInit: (this.height - 200),
         sceneHeight: 200,
@@ -636,9 +672,9 @@
         ]
     });
 
-	var doubleRollV = new DoubleRoll.DoubleRoll({
+    var doubleRollV = new DoubleRoll.DoubleRoll({
         stageView : stageView,
-    	logger: logger,
+        logger: logger,
         ws: wsPianoroll,
         orientation: 'vertical',
         sceneHeight: 768-200,
@@ -652,8 +688,8 @@
         ]
     });
 
-	var annotsRoll = new AnnotsRoll.AnnotsRoll({
-    	stageView : stageView,
+    var annotsRoll = new AnnotsRoll.AnnotsRoll({
+        stageView : stageView,
         logger: logger,
         ws: wsAnnot,
         parentContainer: doubleRollV.stage,
@@ -667,31 +703,31 @@
         annotColors: this.annotCategories
     });
 
-	var limiters = new PIXI.Graphics()
-		.lineStyle(1, 0x646464)
-		.moveTo(annotsRoll.container.x, annotsRoll.container.y)
-		.lineTo(annotsRoll.container.x, annotsRoll.container.y - annotsRoll.height)
-		.moveTo(annotsRoll.container.x + annotsRoll.widthRoll, annotsRoll.container.y)
-		.lineTo(annotsRoll.container.x + annotsRoll.widthRoll, annotsRoll.container.y - annotsRoll.height)
-		.moveTo(0, this.height - 200)
-		.lineTo(this.width, this.height - 200)
-		.drawRect(0, 0, this.width -1, this.height -1)
-		.beginFill(0xECECEC)
-		.drawRect(1024 - 200, 0, 200, 768-200)
-		.endFill();
-	this.container.addChild(limiters);
+    var limiters = new PIXI.Graphics()
+        .lineStyle(1, 0x646464)
+        .moveTo(annotsRoll.container.x, annotsRoll.container.y)
+        .lineTo(annotsRoll.container.x, annotsRoll.container.y - annotsRoll.height)
+        .moveTo(annotsRoll.container.x + annotsRoll.widthRoll, annotsRoll.container.y)
+        .lineTo(annotsRoll.container.x + annotsRoll.widthRoll, annotsRoll.container.y - annotsRoll.height)
+        .moveTo(0, this.height - 200)
+        .lineTo(this.width, this.height - 200)
+        .drawRect(0, 0, this.width -1, this.height -1)
+        .beginFill(0xECECEC)
+        .drawRect(1024 - 200, 0, 200, 768-200)
+        .endFill();
+    this.container.addChild(limiters);
 
-	this.init = function(){
-	};
+    this.init = function(){
+    };
 
-	this.updateTime = function(){
+    this.updateTime = function(){
         currentTimeText.setText(Utils.formatTime(Date.now()));
     };
 
     var refreshTimeInterval;
-    
-	this.start = function() {
-		refreshTimeInterval = setInterval(function() {_this.updateTime();}, 1000);
+
+    this.start = function() {
+        refreshTimeInterval = setInterval(function() {_this.updateTime();}, 1000);
     };
 
     this.refresh = function() {
@@ -705,7 +741,7 @@
 }
 
 module.exports = {
-	AnnotsVizView: AnnotsVizView
+    AnnotsVizView: AnnotsVizView
 };
 
 },{"./annotsroll.js":2,"./annotstimeline.js":3,"./doubleroll.js":5,"./utils.js":9,"lodash":"lodash","pixi":"pixi"}],5:[function(require,module,exports){
@@ -716,14 +752,12 @@
 * Take a look at http://browserify.org/ for more info
 */
 
-/* global document: false */
-
 'use strict';
 
 
 var PIXI = require('pixi');
 var _ = require('lodash');
-var PianoRoll = require('./pianoroll.js');
+var PianoRoll = require('./pianoroll');
 
 var defaultConfig = {
     orientation: 'horizontal',
@@ -774,7 +808,6 @@
     this.noteColors = opts.noteColors;
 
     var noteHeight = opts.noteHeight;
-    var sceneBgColor = opts.sceneBgColor;
     var sceneHeight = opts.sceneHeight || _(opts.pianorolls).reduce(function(s,p) { return s + p.height; }, 0);
     var timeWidth = opts.timeWidth;
     var lineInterval = opts.lineInterval;
@@ -792,7 +825,7 @@
     this.container = new PIXI.DisplayObjectContainer();
     this.container.x = Math.floor(sceneWidth*zeroShift);
     this.container.y = 0;
-    
+
     stageView.registerComponent(this);
 
     var pianorollList = [];
@@ -877,7 +910,7 @@
     };
 
     this.stop = function() {
-    	
+
         pianorollList.forEach(function(c) {
             c.stop();
         });
@@ -899,7 +932,7 @@
     DoubleRoll: DoubleRoll
 };
 
-},{"./pianoroll.js":7,"lodash":"lodash","pixi":"pixi"}],6:[function(require,module,exports){
+},{"./pianoroll":7,"lodash":"lodash","pixi":"pixi"}],6:[function(require,module,exports){
 /**
 * js/wswrapper.js
 *
@@ -937,7 +970,7 @@
         if(doLog) {
             console.log(msg);
         }
-    }
+    };
 
 }
 
@@ -1056,7 +1089,7 @@
                 }
             }
             else {
-                noteDuration = Date.now() - ts;
+                noteDuration = Math.abs(Date.now() - ts);
                 this.noteDict[channel][note] = { ts: ts, velocity: velocity, sessionTs: sessionTs};
             }
         }
@@ -1216,13 +1249,13 @@
     var sceneHeight = opts.sceneHeight;
     var canvasContainer = opts.canvasContainer;
     var timeContainer = [];
-    var components = []; 
-    
+    var components = [];
+
     //create an new instance of a pixi stage
     this.stage = new PIXI.Stage(sceneBgColor);
     //create a renderer instance.
     var renderer = PIXI.autoDetectRenderer(sceneWidth, sceneHeight);
-    	
+
     this.init = function() {
 
         if(typeof(canvasContainer) === 'string') {
@@ -1233,16 +1266,16 @@
         }
 
         canvasContainer.appendChild(renderer.view);
-        
+
         components.forEach(function(c){
     		c.init();
     	});
     };
-    
+
     this.registerTimeContainer = function(container) {
     	timeContainer.push(container);
     };
-    
+
     this.registerComponent = function(component) {
     	components.push(component);
     	this.stage.addChild(component.container);
@@ -1263,7 +1296,7 @@
         if(!externalRefresh) {
             refreshInterval = setInterval(function() {_this.refresh();}, 1000/this.framerate);
         }
-        
+
         components.forEach(function(c){
     		c.start();
     	});
@@ -1273,7 +1306,6 @@
         if(!externalRefresh) {
             clearInterval(refreshInterval);
         }
-        clearInterval(refreshTimeInterval);
         
         components.forEach(function(c){
     		c.stop();
@@ -1302,6 +1334,8 @@
 * basic tools
 *
 */
+/*jshint bitwise: false*/
+/*jshint camelcase: false */
 
 'use strict';
 
@@ -1310,12 +1344,12 @@
 var moment = require('moment');
 
 function formatTime (ts) {
-	return moment(ts).format("HH:mm:ss");
+    return moment(ts).format('HH:mm:ss');
 }
 
 function colorToHex(c) {
-	var m = /rgba?\(\s*(\d+)\s*,\s*(\d+)\s*,\s*(\d+)/.exec(c);
-	return m ? '#' + (1 << 24 | m[1] << 16 | m[2] << 8 | m[3]).toString(16).substr(1) : c;
+    var m = /rgba?\(\s*(\d+)\s*,\s*(\d+)\s*,\s*(\d+)/.exec(c);
+    return m ? '#' + (1 << 24 | m[1] << 16 | m[2] << 8 | m[3]).toString(16).substr(1) : c;
 }
 
 function getAnnotCategories(urlCategories, annotCategories) {
@@ -1326,35 +1360,35 @@
         var data = res.target.json;
 
         while(annotCategories.length > 0) {
-        	annotCategories.pop();
+            annotCategories.pop();
         }
 
         data.sessions.forEach(function(session) {
             var annotCat = {
                 ts: session.start_ts === null ? new Date(0) : Date.parse(session.start_ts),
                 colors: {}
-            }
+            };
             var categoriesJson = session.categories_json;
             annotCat.order = categoriesJson.order;
             if (typeof(annotCat.order['default']) === 'undefined'){
-            	annotCat.order.push('default');
+                annotCat.order.push('default');
             }
-			var catList = _.clone(categoriesJson.categories);
-			while(catList.length > 0) {
-				var cat = catList.pop();
-				if(cat.code) {
-					annotCat.colors[cat.code] = colorToHex(cat.color);
-				}
-				if(cat.subcategories) {
-					catList = catList.concat(cat.subcategories);
-				}
-			}
+            var catList = _.clone(categoriesJson.categories);
+            while(catList.length > 0) {
+                var cat = catList.pop();
+                if(cat.code) {
+                    annotCat.colors[cat.code] = colorToHex(cat.color);
+                }
+                if(cat.subcategories) {
+                    catList = catList.concat(cat.subcategories);
+                }
+            }
             categoriesJson.categories.forEach(function(cat) {
-				if(cat.code) {
-                	annotCat.colors[cat.code] = colorToHex(cat.color);
-				}
+                if(cat.code) {
+                    annotCat.colors[cat.code] = colorToHex(cat.color);
+                }
             });
-            annotCat.defaultColor = categoriesJson.defaultColor || "#536991";
+            annotCat.defaultColor = categoriesJson.defaultColor || '#536991';
             annotCategories.push(annotCat);
         });
     });
@@ -1364,9 +1398,9 @@
 }
 
 module.exports = {
-	formatTime: formatTime,
-	getAnnotCategories: getAnnotCategories,
-	colorToHex: colorToHex
+    formatTime: formatTime,
+    getAnnotCategories: getAnnotCategories,
+    colorToHex: colorToHex
 };
 
 },{"lodash":"lodash","moment":"moment","pixi":"pixi"}],10:[function(require,module,exports){
@@ -1426,4 +1460,4 @@
 
 },{}]},{},[1])(1)
 });
-//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["node_modules/browserify/node_modules/browser-pack/_prelude.js","./app/js/main.js","/Users/ymh/dev/projects/mons/dev/client/annotviz/app/js/annotsroll.js","/Users/ymh/dev/projects/mons/dev/client/annotviz/app/js/annotstimeline.js","/Users/ymh/dev/projects/mons/dev/client/annotviz/app/js/annotsvizview.js","/Users/ymh/dev/projects/mons/dev/client/annotviz/app/js/doubleroll.js","/Users/ymh/dev/projects/mons/dev/client/annotviz/app/js/logger.js","/Users/ymh/dev/projects/mons/dev/client/annotviz/app/js/pianoroll.js","/Users/ymh/dev/projects/mons/dev/client/annotviz/app/js/stageview.js","/Users/ymh/dev/projects/mons/dev/client/annotviz/app/js/utils.js","/Users/ymh/dev/projects/mons/dev/client/annotviz/app/js/wswrapper.js"],"names":[],"mappings":"AAAA;ACAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC9BA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC/NA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACvQA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC1LA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC7LA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC7CA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACpOA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACrHA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACxEA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA","file":"generated.js","sourceRoot":"","sourcesContent":["(function e(t,n,r){function s(o,u){if(!n[o]){if(!t[o]){var a=typeof require==\"function\"&&require;if(!u&&a)return a(o,!0);if(i)return i(o,!0);var f=new Error(\"Cannot find module '\"+o+\"'\");throw f.code=\"MODULE_NOT_FOUND\",f}var l=n[o]={exports:{}};t[o][0].call(l.exports,function(e){var n=t[o][1][e];return s(n?n:e)},l,l.exports,e,t,n,r)}return n[o].exports}var i=typeof require==\"function\"&&require;for(var o=0;o<r.length;o++)s(r[o]);return s})","/**\n * scripts/main.js\n *\n * This is the starting point for your application.\n * Take a look at http://browserify.org/ for more info\n */\n\n'use strict';\n\nvar doubleroll = require('./doubleroll');\nvar annotsroll = require('./annotsroll');\nvar annotstimeline = require('./annotstimeline');\nvar annotsvizview = require('./annotsvizview');\nvar stageview = require('./stageview');\nvar wswrapper = require('./wswrapper');\nvar logger = require('./logger');\nvar utils = require('./utils');\n\nvar _ = require('lodash');\n\nmodule.exports = _({})\n    .extend(doubleroll)\n    .extend(annotsroll)\n    .extend(annotstimeline)\n    .extend(annotsvizview)\n    .extend(stageview)\n    .extend(wswrapper)\n    .extend(logger)\n    .extend(utils)\n    .value();\n","/**\n* js/annotsRoll.js\n*\n* annotsRoll basic component\n*\n*/\n\n'use strict';\n\nvar PIXI = require('pixi');\nvar _ = require('lodash');\n\nvar DEFAULT_ANNOT_COLOR = '#bababa';\n\nvar defaultAnnotStyles = {\n    'label': { font: '16pt Arial Bold', fill: '#65A954', wordWrap: true},\n    'text' : { font: '12pt Arial Regular', fill: '#444444', wordWrap: true},\n    'user' : { font: '14pt Arial regular', fill: '#666666' },\n};\n\nvar defaultOptions = {\n    externalRefresh: false,\n    defaultColor: DEFAULT_ANNOT_COLOR,\n    annotStyles: defaultAnnotStyles,\n    ignoreAnnots:false\n};\n\nfunction AnnotsRoll(options) {\n\n//parentContainer, xInit, yInit, width, height, widthRoll, pixelsPerSecond, annotColors\n    var _this = this;\n    var opts = _(options).defaults(defaultOptions).value();\n\n\n    this.container = new PIXI.DisplayObjectContainer();\n    this.container.x = opts.xInit;\n    this.container.y = opts.yInit;\n    this.container.width = opts.width;\n\n    this.height = opts.height;\n    this.width = opts.width;\n    this.widthRoll = opts.widthRoll;\n    this.pixelsPerSecond = opts.pixelsPerSecond;\n    this.annotColors = opts.annotColors;\n    this.startTs = opts.startTs || Date.now();\n    this.ignoreAnnots = opts.ignoreAnnots;\n\n    var yInit = opts.yInit;\n    var annotStyles = _(opts.annotStyles).defaults(defaultAnnotStyles).value();\n    var marginX = 15;\n    for(var style in annotStyles) {\n    \tif (annotStyles[style].wordWrap === true){\n    \t\tannotStyles[style].wordWrapWidth = this.widthRoll - marginX; \n    \t}\n    }\n    var started = false;\n    var ws = opts.ws;\n    var externalRefresh = opts.externalRefresh;\n    var stageView = opts.stageView;\n    var waitInterval;\n    var wait = 0;\n    \n    stageView.registerComponent(this);\n\n    var isHidden = function(child) {\n        // TODO: the origin point is an approximation. Should refine this\n        var globalPos = child.toGlobal(new PIXI.Point(0,0));\n        return ((globalPos.x + child.width) < 0) || ((globalPos.y + child.height) < 0) ;\n    };\n\n    this.addAnnots = function(data) {\n\n        //var title = data.content.category.label;\n        //var user = data.content.user;\n        //Test cat and color\n        //var colorAnnot = 0x65A954;\n        var category = data.content.category.label,\n            text     = data.content.text,\n            user     = data.content.user,\n            ts       = Date.parse(data.ts),\n            color    = data.content.color || this.getColor(ts, data.content.category.code);\n\n        this.addAnnot(category, text, user, color, ts);\n    };\n\n    this.getColor = function(ts, code) {\n        var colorsDef;\n        _(this.annotColors).eachRight(function(cdef) {\n            if(cdef.ts < ts) {\n                colorsDef = cdef.colors;\n                return false;\n            }\n        });\n        var resColor;\n        if(colorsDef) {\n            resColor = colorsDef[code];\n        }\n        if(!resColor) {\n            resColor = colorsDef.defaultColor || DEFAULT_ANNOT_COLOR;\n        }\n        return resColor;\n    }\n\n    this.addAnnot = function(category, text, user, color, ts){\n\n        var x = 0;\n        var y = (ts-this.startTs) * this.pixelsPerSecond / 1000 + yInit;\n\n        var colorHex = parseInt(color.replace(/^#/, ''), 16);\n\n        if (wait === 0){\n\t        var graphics = new PIXI.Graphics()\n\t            .beginFill(colorHex)\n\t            .drawRect(x, y, 10, 3)\n\t            .endFill();\n\t\n\t        this.container.addChild(graphics);\n\t\n\t        var textHeight = 0;\n\t        var catLabel = new PIXI.Text(\n\t            category,\n\t            _(annotStyles.label).extend({fill: color}).value()\n\t        );\n\t        catLabel.x = x + marginX;\n\t        catLabel.y = y - 23;\n\t        this.container.addChild(catLabel);\n\t        textHeight += (catLabel.height - 23 + 2);\n\t\n\t        if(text) {\n\t            var catText = new PIXI.Text(text, annotStyles.text);\n\t            catText.x = x + marginX;\n\t            catText.y = y + textHeight;\n\t            this.container.addChild(catText);\n\t            textHeight += (catText.height + 2);\n\t        }\n\t\n\t        var catUser = new PIXI.Text(user, annotStyles.user);\n\t        catUser.x = x + marginX;\n\t        catUser.y = y + textHeight;\n\t        this.container.addChild(catUser);\n\t        textHeight += (catUser.height + 8);\n\t        \n\t        if (this.ignoreAnnots === true){\n\t\t        wait = textHeight / this.pixelsPerSecond;\n\t\t        waitInterval = setInterval(function() {_this.refreshWait();}, 1000);\n\t        }\n        }\n\n        this.addAnnotLine(colorHex, y);\n    };\n\n    this.addAnnotLine = function(color, y) {\n        var x = this.widthRoll;\n\n\n        var graphics = new PIXI.Graphics()\n            .beginFill(color)\n            .drawRect(x, y, this.width - x, 3)\n            .endFill();\n\n        this.container.addChild(graphics);\n    };\n\n    this.moveTo = function(diffTime){\n    \tthis.container.y = Math.floor(diffTime*this.pixelsPerSecond);\n    };\n\n    this.move = this.refresh = function() {\n        var diff = (this.startTs - Date.now())/1000;\n        this.moveTo(diff);\n    };\n    \n    this.refreshWait = function(){\n    \twait -= 1;\n    \tif (wait < 0){\n    \t\twait = 0;\n    \t\tclearInterval(waitInterval);\n    \t}\n    };\n\n    this.removePassedObjets = function(){\n        var childrenToRemove = [];\n        _(_this.container.children).forEach(function(child) {\n            return typeof(child) === 'undefined' ||\n                (isHidden(child) && childrenToRemove.push(child));\n        });\n        childrenToRemove.forEach(function(child) {\n            _this.container.removeChild(child);\n        });\n    };\n\n    this.init = function() {\n\n        ws.message(function(data) {\n            _this.addAnnots(data);\n        });\n\n    };\n    \n\n    this.start = function() {\n        if(!started) {\n            this.startTs = Date.now();\n            started = true;\n        }\n        this.cleanInterval = setInterval(function () { _this.removePassedObjets(); }, 1000 * this.height / this.pixelsPerSecond );\n        if(!externalRefresh) {\n            this.refreshInterval = setInterval(function() {_this.move();}, 1000/this.framerate);\n        }\n    };\n\n    this.stop = function() {\n        clearInterval(this.cleanInterval);\n        if(!externalRefresh) {\n            clearInterval(this.refreshInterval);\n        }\n    };\n\n}\n\nmodule.exports = {\n    AnnotsRoll: AnnotsRoll,\n};\n","/**\n* js/annotstimeline\n*\n* annotstimeline basic component\n*\n*/\n\n'use strict';\n\nvar PIXI = require('pixi');\nvar Utils = require('./utils.js');\nvar _ = require('lodash');\n\nvar defaultOptions = {\n    logger: undefined,\n    intervalWidth: 10,\n    intervalHeight: 5,\n    maxCellHeight: 200,\n    radius: 300,\n    serverUrl: 'http://127.0.0.1:8080',\n    channel: 'ANNOT',\n    maxPages: 1000,\n    showClockGraphics: true,\n    archive: false\n};\n\n\nfunction AnnotsTimeLine(options){\n    var _this = this;\n    var opts = _(options).defaults(defaultOptions).value();\n\n    this.container = new PIXI.DisplayObjectContainer();\n    this.container.x = opts.xInit;\n    this.container.y = opts.yInit;\n    this.container.width = opts.width;\n    this.container.height = opts.height;\n\n    this.timeBegin = opts.timeBegin;\n    this.timeEnd = opts.timeEnd;\n    this.duration = (this.timeEnd - this.timeBegin)/1000;\n    this.width = opts.width;\n    this.height = opts.height;\n    this.intervalHeight = opts.intervalHeight;\n    this.intervalWidth = opts.intervalWidth;\n    this.maxCellHeight = opts.maxCellHeight;\n    this.annotCategories = opts.annotCategories;\n\tthis.startTs = options.startTs || Date.now();\n    this.showClockGraphics = opts.showClockGraphics;\n    this.archive = opts.archive;\n    \n    this.circleX = opts.circleX || (this.width/2);\n    this.circleY = opts.circleY || (this.height/2);\n    this.radius = opts.radius;\n    var perimeter = 2*Math.PI* this.radius;\n    this.intervalDuration = (this.intervalWidth * this.duration / perimeter);\n    \n    var channel = opts.channel;\n    var eventCode = opts.eventCode;\n    var serverUrl = opts.serverUrl;\n    var maxPages = opts.maxPages;\n    \n    var totalIndex = Math.floor( perimeter/this.intervalWidth);\n\n    this.cells = []\n    for (var i=0; i<(perimeter/this.intervalWidth) ; i++){\n    \tthis.cells[i] = [];\n    \tthis.cells[i].i = i;\n    \tthis.cells[i].totalAnnots = 0;\n    \tthis.cells[i].categories = {};\n    }\n\n    var ws = opts.ws;\n    var stageView = opts.stageView;\n\n    //draw the base - circle and line to locate the scene\n    var graphics = new PIXI.Graphics();\n    graphics.lineStyle(2, 0x646464)\n    \t.drawCircle(this.circleX, this.circleY, this.radius - 3)\n    \t.endFill()\n    this.container.addChild(graphics);\n\n    stageView.registerComponent(this);\n\n\tvar loadArchives = function() {\n        //start timeBegin end startTime\n        //query -> need channel + eventCode\n        //iterate over data fill cells\n        var startTs = _this.timeBegin;\n        var endTs = _this.startTs;\n\n        var url = serverUrl + '/p/api/v1/annotation';\n        var filters = [\n            { name: 'ts', op: '>', val: new Date(startTs).toISOString()}, //start\n            { name: 'ts', op: '<=', val: new Date(endTs).toISOString()}, //end\n            { name: 'channel', op: '==', val: channel}, //channel\n            { name: 'event_code', op: '==', val: eventCode} //eventcode\n        ];\n\n        url = url + '?q=' + JSON.stringify({filters:filters});\n\n        var totalPage = 1;\n        var currentPage = 1;\n\n        var processResFunction = function(res) {\n\n            if(res) {\n                var data = res.target.json;\n                /*jshint -W069 */\n                totalPage = Math.min(maxPages,parseInt(data['total_pages']));\n                data.objects.forEach(function(annotation) {\n                    _this.addAnnot(annotation);\n                });\n            }\n            if(currentPage <= totalPage) {\n                var jsonLoader = new PIXI.JsonLoader(url+'&page='+currentPage, true);\n                jsonLoader.on('loaded', processResFunction);\n                jsonLoader.load();\n                currentPage++;\n            }\n        };\n        processResFunction();\n\n    };\n\n    //Add Annotation to the TimeLine\n    this.addAnnot = function(data){\n\n    \tvar ts = Date.parse(data.ts);\n    \tvar colorsDef;\n    \t_(this.annotCategories).eachRight(function(cdef) {\n            if(cdef.ts < ts) {\n                colorsDef = cdef;\n                return false;\n            }\n        });\n\n    \tif (this.timeEnd > ts){\n\t    \tvar i = Math.floor((ts - this.timeBegin)/(1000*this.intervalDuration));\n\n\t    \tif (typeof(this.cells[i].graphics) === 'undefined'){\n\t    \t\tthis.initCell(this.cells[i], colorsDef);\n\t    \t}\n\n\t    \tif (typeof(colorsDef.colors[data.content.category.code]) !== 'undefined'){\n\t    \t\tvar annotCode = data.content.category.code;\n\t    \t} else {\n\t    \t\tvar annotCode = 'default';\n\t    \t}\n\n\t\t\tthis.cells[i].categories[annotCode].count += 1;\n\t\t\tthis.cells[i].totalAnnots +=1;\n\t\t\tthis.redrawCell(this.cells[i], colorsDef);\n    \t}\n    };\n\n    this.initClockGraphics = function() {\n\t    var tBeg = new PIXI.Text(Utils.formatTime(this.timeBegin), { font: '12pt Gothic Standard', fill: '#646464' });\n\t    tBeg.x = this.circleX + 15;\n\t    tBeg.y = this.circleY - this.radius - this.maxCellHeight - 10;\n\t    this.container.addChild(tBeg);\n\n\t    var tEnd = new PIXI.Text(Utils.formatTime(this.timeEnd), { font: '12pt Gothic Standard', fill: '#646464' });\n\t    tEnd.x = this.circleX - 15 - tEnd.width;\n\t    tEnd.y = this.circleY - this.radius - this.maxCellHeight - 10;\n\t    this.container.addChild(tEnd);\n\n\t    var t15 = new PIXI.Text(Utils.formatTime(((this.timeEnd - this.timeBegin)/4) + this.timeBegin), { font: '12pt Gothic Standard', fill: '#646464' });\n\t    t15.x = this.circleX + this.radius + this.maxCellHeight + 10 ;\n\t    t15.y = this.circleY - t15.height;\n\t    t15.rotation = Math.PI /2;\n\t    this.container.addChild(t15);\n\n\t    var t30 = new PIXI.Text(Utils.formatTime(((this.timeEnd - this.timeBegin)/2) + this.timeBegin), { font: '12pt Gothic Standard', fill: '#646464' });\n\t    t30.x = this.circleX - t30.width/2;\n\t    t30.y = this.circleY + this.radius + this.maxCellHeight - 2;\n\t    this.container.addChild(t30);\n\n\t    var t45 = new PIXI.Text(Utils.formatTime(((this.timeEnd - this.timeBegin)*3/4) + this.timeBegin), { font: '12pt Gothic Standard', fill: '#646464' });\n\t    t45.x = this.circleX - this.radius - this.maxCellHeight - 10 ;\n\t    t45.y = this.circleY + t15.height;\n\t    t45.rotation = -Math.PI/2;\n\t    this.container.addChild(t45);\n\t    \n\t    var lineV = new PIXI.Graphics();\n\t    lineV.lineStyle(1, 0x646464)\n\t    \t.moveTo(this.circleX, this.circleY - (this.radius/3)/2)\n\t    \t.lineTo(this.circleX, this.circleY - this.radius - this.maxCellHeight - 10)\n    \t.endFill();\n\t    this.container.addChild(lineV);\n    }\n\n    //Draw the cellule\n    this.redrawCell = function(cell, colorsDef){\n    \tvar y = 0;\n\n        //Check if total height is higher than Max Cell Height\n        var heightStep;\n        if ((cell.totalAnnots*this.intervalHeight) > this.maxCellHeight){\n            heightStep = this.maxCellHeight/cell.totalAnnots;\n        } else {\n            heightStep = this.intervalHeight;\n        }\n\n    \t//Draw the rect depending on the height step calculated\n    \tfor (var i=0; i< colorsDef.order.length; i++){\n    \t\tvar currentCode = colorsDef.order[i];\n\t\t\tcell.graphics.beginFill(cell.categories[currentCode].color.replace(\"#\", \"0x\"))\n    \t\t\t.drawRect(0, y, this.intervalWidth-1, -cell.categories[currentCode].count * heightStep)\n    \t\t\t.endFill();\n    \t\ty -= cell.categories[currentCode].count*heightStep;\n    \t}\n    }\n\n    this.initCell = function(cell, colorsDef){\n    \tcell.graphics = new PIXI.Graphics();\n    \tcell.graphics.position.x = this.circleX + this.radius * Math.sin(cell.i*(360/totalIndex)*(Math.PI/180));\n    \tcell.graphics.position.y = this.circleY - this.radius * Math.cos(cell.i*(360/totalIndex)*(Math.PI/180));\n    \tcell.graphics.rotation = (cell.i)*(360/totalIndex)*(Math.PI/180) + (360/(totalIndex*2))*(Math.PI/180);\n    \tthis.container.addChild(cell.graphics);\n\n    \tfor (var category in colorsDef.colors){\n    \t\tcell.categories[category] = {\n\t\t\t\t\"count\": 0,\n\t\t\t\t\"color\": colorsDef.colors[category]\n    \t\t};\n    \t}\n    \tif (typeof(cell.categories['default']) === 'undefined'){\n\t\t\tcell.categories['default'] = {\n\t\t\t\t\"count\": 0,\n\t\t\t\t\"color\": colorsDef.defaultColor\n\t\t\t}\n    \t}\n    }\n\n    this.init = function() {\n    \tif (!this.archive){\n\t        ws.message(function(data) {\n\t            _this.addAnnot(data);\n\t        });\n    \t}\n\n    \tif (this.showClockGraphics){this.initClockGraphics();}\n    };\n\n\n    this.start = function() {\n    \tthis.startTs = Date.now();\n        loadArchives();\n    };\n\n    this.refresh = function() {\n\n    };\n\n    this.stop = function(){\n    };\n\n    return this;\n}\n\nmodule.exports = {\n\tAnnotsTimeLine: AnnotsTimeLine\n};\n","/**\n* js/annotsvizview.js\n*\n* This is the starting point for your application.\n* Take a look at http://browserify.org/ for more info\n*/\n\n'use strict';\n\nvar PIXI = require('pixi');\nvar _ = require('lodash');\nvar DoubleRoll = require('./doubleroll.js');\nvar AnnotsTimeLine = require('./annotstimeline.js');\nvar AnnotsRoll = require('./annotsroll.js');\nvar Utils = require('./utils.js');\n\nvar defaultOptions = {\n    xInit: 0,\n    yInit: 0,\n    width: 1024,\n    height: 768\n};\n\nfunction AnnotsVizView(options){\n\tvar _this = this;\n    var opts = _(options).defaults(defaultOptions).value();\n\n    this.container = new PIXI.DisplayObjectContainer();\n    this.container.x = opts.xInit;\n    this.container.y = opts.yInit;\n\tthis.width = opts.width;\n\tthis.height= opts.height;\n    this.timeBegin = opts.timeBegin;\n    this.timeEnd = opts.timeEnd;\n    this.annotCategories = [];\n    \n    Utils.getAnnotCategories(opts.urlCategories, this.annotCategories);\n    \n\tvar wsPianoroll = opts.wsPianoroll;\n\tvar wsAnnot = opts.wsAnnot;\n\tvar stageView = opts.stageView;\n\tvar currentTime = Date.now() + 3600*1000;\n\tvar eventCode = opts.eventCode;\n    var channel = opts.channel;\n\n\tstageView.registerComponent(this);\n\n\tvar timeLineDay2 = new AnnotsTimeLine.AnnotsTimeLine({\n    \tstageView : stageView,\n        logger: logger,\n        ws: new annotviz.WsWrapper(wsUriAnnotation, logger),\n        xInit: 0,\n        yInit: 0,\n        width: 1024 - 200 - 200,\n        height: 768-200,\n        timeBegin: this.timeBegin,\n        timeEnd: this.timeEnd,\n        intervalWidth: 6,\n        intervalHeight: 10,\n        maxCellHeight: 70,\n        radius: 200,\n        eventCode: eventCode,\n        channel: channel,\n        annotCategories: this.annotCategories\n    });\n\t\n    //Archive day 1\n    var timeLineDay1 = new AnnotsTimeLine.AnnotsTimeLine({\n    \tstageView : stageView,\n        archive: true,\n        xInit: 0,\n        yInit: 0,\n        width: 1024 - 200 - 200,\n        height: 768-200,\n        timeBegin: Date.parse(\"2015-01-22T09:30:00+01:00\"),\n        timeEnd: Date.parse(\"2015-01-22T18:30:00+01:00\"),\n        circleX: timeLineDay2.circleX,\n        circleY: timeLineDay2.circleY,\n        intervalWidth: (timeLineDay2.radius*2/3)* timeLineDay2.intervalWidth / timeLineDay2.radius,\n        intervalHeight: (timeLineDay2.intervalHeight * (timeLineDay2.radius - timeLineDay2.radius*2/3))/ timeLineDay2.maxCellHeight,\n        maxCellHeight: (timeLineDay2.radius - timeLineDay2.radius*2/3)/4,\n        radius: timeLineDay2.radius*2/3,\n        annotCategories: this.annotCategories,\n        eventCode: eventCode,\n        channel: channel,\n        showClockGraphics:false\n    });\n    \n    var currentTimeText = new PIXI.Text(\"-- : -- : --\", { font: '18pt Gothic Standard', fill: '#646464' });\n    currentTimeText.x = timeLineDay2.circleX - currentTimeText.width/2;\n    currentTimeText.y = timeLineDay2.circleY - currentTimeText.height/2;\n    this.container.addChild(currentTimeText);\n\n    var timeLineDay3 = new PIXI.Graphics();\n    timeLineDay3.lineStyle(1, 0x646464)\n    \t.drawCircle(timeLineDay2.circleX, timeLineDay2.circleY, timeLineDay2.radius/3)\n    \t.endFill()\n    this.container.addChild(timeLineDay3);\n    \n\tvar doubleRollH = new DoubleRoll.DoubleRoll({\n        stageView : stageView,\n    \tlogger: logger,\n        ws: wsPianoroll,\n        yInit: (this.height - 200),\n        sceneHeight: 200,\n        pianorolls : [\n            {\n                height: 200,\n                timeWidth: 10,\n                lineInterval: 5000,\n                noteHeight: 10\n            },\n        ]\n    });\n\n\tvar doubleRollV = new DoubleRoll.DoubleRoll({\n        stageView : stageView,\n    \tlogger: logger,\n        ws: wsPianoroll,\n        orientation: 'vertical',\n        sceneHeight: 768-200,\n        pianorolls : [\n            {\n                height: 200,\n                timeWidth: 60,\n                lineInterval: 5000,\n                noteHeight: 5,\n            },\n        ]\n    });\n\n\tvar annotsRoll = new AnnotsRoll.AnnotsRoll({\n    \tstageView : stageView,\n        logger: logger,\n        ws: wsAnnot,\n        parentContainer: doubleRollV.stage,\n        xInit: 1024 - 200 - 200,\n        yInit: 768-200,\n        width: 200 + 200,\n        height: 768-200,\n        widthRoll: 200,\n        framerate: doubleRollV.framerate,\n        pixelsPerSecond: Math.floor(1024 / 60),\n        annotColors: this.annotCategories\n    });\n\n\tvar limiters = new PIXI.Graphics()\n\t\t.lineStyle(1, 0x646464)\n\t\t.moveTo(annotsRoll.container.x, annotsRoll.container.y)\n\t\t.lineTo(annotsRoll.container.x, annotsRoll.container.y - annotsRoll.height)\n\t\t.moveTo(annotsRoll.container.x + annotsRoll.widthRoll, annotsRoll.container.y)\n\t\t.lineTo(annotsRoll.container.x + annotsRoll.widthRoll, annotsRoll.container.y - annotsRoll.height)\n\t\t.moveTo(0, this.height - 200)\n\t\t.lineTo(this.width, this.height - 200)\n\t\t.drawRect(0, 0, this.width -1, this.height -1)\n\t\t.beginFill(0xECECEC)\n\t\t.drawRect(1024 - 200, 0, 200, 768-200)\n\t\t.endFill();\n\tthis.container.addChild(limiters);\n\n\tthis.init = function(){\n\t};\n\n\tthis.updateTime = function(){\n        currentTimeText.setText(Utils.formatTime(Date.now()));\n    };\n\n    var refreshTimeInterval;\n    \n\tthis.start = function() {\n\t\trefreshTimeInterval = setInterval(function() {_this.updateTime();}, 1000);\n    };\n\n    this.refresh = function() {\n    };\n\n    this.stop = function(){\n    };\n\n    return this;\n\n}\n\nmodule.exports = {\n\tAnnotsVizView: AnnotsVizView\n};\n","/**\n* scripts/doubleroll.js\n*\n* This is the starting point for your application.\n* Take a look at http://browserify.org/ for more info\n*/\n\n/* global document: false */\n\n'use strict';\n\n\nvar PIXI = require('pixi');\nvar _ = require('lodash');\nvar PianoRoll = require('./pianoroll.js');\n\nvar defaultConfig = {\n    orientation: 'horizontal',\n    logger: undefined,\n    sceneWidth: 1024,\n    pianorolls : [\n      {\n        height: 435,\n        timeWidth: 10,\n        lineInterval: 5000,\n        noteHeight: undefined\n      },\n      {\n        height: 645,\n        timeWidth: 60,\n        lineInterval: 5000,\n        noteHeight: undefined\n      },\n    ],\n    framerate: 25,\n    offsetMusic: false,\n    sceneBgColor: 0xFFFFFF,\n    lineColor: 0x444444,\n    lineFillColor: 0xFFFF00,\n    noteColors: [0xB90000, 0x4BDD71, 0xAF931E, 0x1C28BA, 0x536991],\n    noteHeight: undefined,\n    zeroShift: 0.9,\n    timeWidth: 60,\n    lineInterval: 5000,\n//    wsUri: undefined,\n//    eventCode: undefined\n\n};\n\nfunction DoubleRoll(options) {\n\n    var _this = this;\n    var opts = _(options).defaults(defaultConfig).value();\n\n    var orientation = opts.orientation;\n    var isHorizontal = (orientation !== 'vertical');\n\n    this.logger = opts.logger;\n    this.lineColor = opts.lineColor;\n    this.lineFillColor = opts.lineFillColor;\n    this.framerate = opts.framerate;\n    this.offsetMusic = opts.offsetMusic;\n    this.noteColors = opts.noteColors;\n\n    var noteHeight = opts.noteHeight;\n    var sceneBgColor = opts.sceneBgColor;\n    var sceneHeight = opts.sceneHeight || _(opts.pianorolls).reduce(function(s,p) { return s + p.height; }, 0);\n    var timeWidth = opts.timeWidth;\n    var lineInterval = opts.lineInterval;\n    var offsetMusic = opts.offsetMusic;\n\n    var sceneWidth = opts.sceneWidth;\n    var stageView = opts.stageView;\n\n    var zeroShift = opts.zeroShift;\n\n    var ws = opts.ws;\n\n    var colorsReg = {};\n\n    this.container = new PIXI.DisplayObjectContainer();\n    this.container.x = Math.floor(sceneWidth*zeroShift);\n    this.container.y = 0;\n    \n    stageView.registerComponent(this);\n\n    var pianorollList = [];\n\n    var pianorollOptions = {\n        parentContainer: this.container,\n        orientation: orientation,\n        xInit: 0,\n        width: sceneWidth,\n        noteColors: this.noteColors,\n        colorsReg: colorsReg,\n        lineColor: this.lineColor,\n        lineInterval: lineInterval,\n        offsetMusic: offsetMusic,\n    };\n\n    var yInit = opts.yInit || 0;\n    var linesDown = true;\n    _(opts.pianorolls).forEach(function(prDef, i) {\n        var prNoteHeight = noteHeight || prDef.noteHeight || prDef.height / 128;\n        var prTimeWidth = prDef.timeWidth || timeWidth;\n        pianorollList.push(new PianoRoll(_({\n            yInit: yInit,\n            height: prDef.height,\n            linesDown: linesDown,\n            pixelsPerSecond: Math.floor(sceneWidth / prTimeWidth),\n            noteHeight: prNoteHeight,\n            lineInterval: prDef.lineInterval\n        }).defaults(pianorollOptions).value()));\n        yInit += prDef.height;\n        linesDown = !linesDown;\n\n        if(i<(opts.pianorolls.length-1)) {\n            var lineGraphics = new PIXI.Graphics()\n                .beginFill(_this.lineFillColor)\n                .lineStyle(1, _this.lineColor)\n                .moveTo(Math.floor(sceneWidth*zeroShift), yInit)\n                .lineTo(-sceneWidth - Math.floor(sceneWidth*zeroShift), yInit)\n                .endFill();\n            _this.container.addChild(lineGraphics);\n        }\n    });\n\n    if(!isHorizontal) {\n        this.container.rotation = Math.PI/2;\n        this.container.y = sceneHeight;\n        this.container.x = sceneWidth;\n    }\n\n\n    this.init = function() {\n\n    \tws.message(function(data) {\n            _this.addNotes(data);\n        });\n\n    };\n\n\n    this.addNotes = function(data) {\n\n        pianorollList.forEach(function(c) {\n            c.addNoteRaw(data);\n        });\n    };\n\n    this.refresh = function() {\n        pianorollList.forEach(function(c) {\n            c.move();\n        });\n    };\n\n    // Init page and intervals\n    var startTs;\n\n    this.start = function() {\n\n        startTs = Date.now();\n        pianorollList.forEach(function(c) {\n            c.start();\n        });\n    };\n\n    this.stop = function() {\n    \t\n        pianorollList.forEach(function(c) {\n            c.stop();\n        });\n    };\n\n\n    this.log = function(m) {\n        if(this.logger) {\n            this.logger.log(m);\n        }\n    };\n\n\n\n    return this;\n}\n\nmodule.exports = {\n    DoubleRoll: DoubleRoll\n};\n","/**\n* js/wswrapper.js\n*\n* simple logger service\n*\n*/\n\n/* global document: false */\n\n'use strict';\n\nfunction HtmlLogger(doLog, container) {\n\n    var logContainer = container;\n    if(typeof(container) === 'string') {\n        logContainer = document.getElementById(container);\n    }\n    if(!doLog) {\n        document.body.removeChild(logContainer);\n        logContainer = undefined;\n    }\n\n\n    this.log = function(msg) {\n        if(doLog && logContainer) {\n            logContainer.innerHTML += msg + '\\n';\n            logContainer.scrollTop = logContainer.scrollHeight;\n        }\n    };\n}\n\nfunction ConsoleLogger(doLog) {\n\n    this.log = function(msg) {\n        if(doLog) {\n            console.log(msg);\n        }\n    }\n\n}\n\nmodule.exports = {\n    HtmlLogger: HtmlLogger,\n    ConsoleLogger: ConsoleLogger\n};\n","/**\n* js/pianoroll.js\n*\n* pianoroll basic component\n*\n*/\n\n'use strict';\n\n\nvar PIXI = require('pixi');\nvar randomColor = require('randomColor');\nvar _ = require('lodash');\n\nvar NTP_EPOCH_DELTA = 2208988800; //c.f. RFC 868\n\nfunction PianoRoll(options) {\n    var _this = this;\n    this.container = new PIXI.DisplayObjectContainer();\n    this.container.x = options.xInit;\n    this.container.y = options.yInit;\n    options.parentContainer.addChild(this.container);\n\n    var orientation = options.orientation;\n    var isHorizontal = (orientation !== 'vertical');\n\n    this.linesDown = options.linesDown;\n    this.height = options.height;\n    this.pixelsPerSecond = options.pixelsPerSecond;\n    this.width = options.width;\n    this.noteColors = options.noteColors;\n    this.colorsReg = options.colorsReg || {};\n    this.lineColor = options.lineColor;\n    this.lineInterval = options.lineInterval;\n    this.offsetMusic = options.offsetMusic || false;\n    this.noteHeight = options.noteHeight;\n    this.noteDict = {};\n    this.startTs = options.startTs || Date.now();\n\n    var started = false;\n\n    var isHidden = function(child) {\n        // TODO: the origin point is an approximation. Should refine this\n        var globalPos = child.toGlobal(new PIXI.Point(0,0));\n        return ((globalPos.x + child.width) < 0) || ((globalPos.y + child.height) < 0) ;\n    };\n\n    //TODO: I do not like the \"regColor\" object. This should not be global, but local\n    this.getColor = function(canal) {\n        var color = this.colorsReg[canal];\n        if(typeof(color) === 'undefined') {\n            var colorsRegSize = Object.keys(this.colorsReg).length;\n            if(colorsRegSize < this.noteColors.length) {\n                color = this.colorsReg[canal] = this.noteColors[colorsRegSize];\n            }\n            else {\n                color = this.colorsReg[canal] = parseInt(randomColor({ luminosity: 'light', hue: 'random', format:'hex'}).replace(/^#/, ''), 16);\n            }\n        }\n        return color;\n    };\n\n    this.getNoteRect = function(x, y, color, alpha, width, height) {\n        var graphics = new PIXI.Graphics();\n        graphics.beginFill(color, alpha);\n        graphics.drawRect(0, 0, width, height);\n        graphics.endFill();\n        graphics.x = x;\n        graphics.y = y;\n        graphics.width = width;\n        graphics.height = height;\n        return graphics;\n    };\n\n    this.addNoteRaw = function(data) {\n        var note = data.content[3];\n        var velocity = data.content[4];\n        var ts = (data.content[0] - NTP_EPOCH_DELTA)*1000;\n        var channel = data.content[2];\n        var sessionTs = data.content[1];\n\n        this.addNote(note, ts, sessionTs, velocity, channel, 0);\n    };\n\n    this.addNote = function(note, startTime, sessionTs, velocity, channel, duration) {\n\n        var ts = startTime;\n        if(this.offsetMusic) {\n            ts = this.startTs + sessionTs;\n        }\n\n        var noteDuration = duration;\n        var noteVelocity = velocity;\n        var graphics;\n        if(!duration) {\n            if(typeof this.noteDict[channel]==='undefined'){\n                this.noteDict[channel] = {};\n            }\n            if(velocity===0) {\n                if(typeof this.noteDict[channel][note] !== 'undefined') {\n                    var noteDef = this.noteDict[channel][note];\n                    delete this.noteDict[channel][note];\n                    noteDuration = sessionTs - noteDef.sessionTs;\n                    graphics = noteDef.graphics;\n                    noteVelocity = noteDef.velocity;\n                    ts = noteDef.ts;\n                }\n            }\n            else {\n                noteDuration = Date.now() - ts;\n                this.noteDict[channel][note] = { ts: ts, velocity: velocity, sessionTs: sessionTs};\n            }\n        }\n\n\n        if(!this.offsetMusic || velocity===0) {\n\n            var width = noteDuration * this.pixelsPerSecond / 1000;\n            if(!graphics) {\n                var x = (ts-this.startTs) * this.pixelsPerSecond / 1000;\n                if((x+width) <  (Math.abs(this.container.x) - this.width)) {\n                    // not visible. do nothing\n                    return;\n                }\n                var y = Math.floor((128-note+0.5) * this.height / 128 - (this.noteHeight/2));\n                var color = this.getColor(channel);\n                var alpha = (noteVelocity / 128);\n\n                graphics = this.getNoteRect(x, y, color, alpha, width, this.noteHeight);\n                this.container.addChild(graphics);\n            }\n            else {\n                graphics.width = width;\n            }\n\n            if(!duration && velocity) {\n                this.noteDict[channel][note].graphics = graphics;\n            }\n        }\n    };\n\n    this.addLine = function(ts){\n\n        if(typeof(ts) === 'undefined') {\n            ts = new Date();\n        }\n        var x = -this.container.x;\n        var y = this.linesDown ? this.height - 20 : 0;\n\n        var graphics = new PIXI.Graphics()\n            .beginFill(0xFFFF00)\n            .lineStyle(1, this.lineColor)\n            .moveTo(0, 0)\n            .lineTo(0, 20)\n            .endFill();\n        graphics.x = x;\n        graphics.y = y;\n        this.container.addChild(graphics);\n        // Add text\n        //var totalSec = lineNb * this.lineInterval / 1000;\n        var hours = ts.getHours();\n        var minutes =ts.getMinutes();\n        var seconds = ts.getSeconds();\n        var timeStr = (hours < 10 ? '0' + hours : hours) + ':' + (minutes < 10 ? '0' + minutes : minutes) + ':' + (seconds  < 10 ? '0' + seconds : seconds);\n\n        var fontObj = { font: '10pt Arial', fill: '#444444' };\n        var t = new PIXI.Text(timeStr, fontObj);\n        if(isHorizontal) {\n            t.x = x + 2;\n            t.y = this.linesDown ? this.height - 15 : 2;\n        }\n        else {\n            t.rotation = -Math.PI/2;\n            t.x = x ;\n            t.y = this.linesDown ? this.height - 2 : t.width + 2;\n        }\n        this.container.addChild(t);\n    };\n\n    this.moveTo = function(diffTime){\n        var oldX = this.container.x;\n        this.container.x = Math.floor(diffTime*this.pixelsPerSecond);\n        var deltaX = Math.abs(oldX-this.container.x);\n        _.forOwn(this.noteDict, function(channelDict) {\n            _.forOwn(channelDict, function(noteDef) {\n                if(noteDef.graphics) {\n                    noteDef.graphics.width = noteDef.graphics.width + deltaX;\n                }\n            });\n        });\n    };\n\n    this.move = function() {\n        var diff = (this.startTs - Date.now())/1000;\n        this.moveTo(diff);\n    };\n\n    this.removePassedObjets = function(){\n        var childrenToRemove = [];\n        _(_this.container.children).forEach(function(child) {\n            return typeof(child) === 'undefined' ||\n                (isHidden(child) && childrenToRemove.push(child));\n        });\n        childrenToRemove.forEach(function(child) {\n            _this.container.removeChild(child);\n        });\n    };\n\n    this.start = function() {\n        if(!started) {\n            this.startTs = Date.now();\n            this.addLine();\n            started = true;\n        }\n        this.verticalLinesInterval = setInterval(function() { _this.addLine(); }, this.lineInterval);\n        this.cleanInterval = setInterval(function () { _this.removePassedObjets(); }, 1000 * this.width / this.pixelsPerSecond );\n    };\n\n    this.stop = function() {\n        //window.clearInterval(this.moveInterval);\n        clearInterval(this.verticalLinesInterval);\n        clearInterval(this.cleanInterval);\n    };\n\n\n}\n\nmodule.exports = PianoRoll;\n","/**\n* scripts/stageview.js\n*\n* This is the starting point for your application.\n* Take a look at http://browserify.org/ for more info\n*/\n\n/* global document: false */\n\n'use strict';\n\n\nvar PIXI = require('pixi');\nvar _ = require('lodash');\n\nvar defaultConfig = {\n    externalRefresh: false,\n    logger: undefined,\n    sceneWidth: 1024,\n    sceneHeight: 768,\n    framerate: 25,\n    sceneBgColor: 0xFFFFFF,\n    canvasContainer: 'canvasContainer',\n};\n\nfunction StageView(options) {\n\n    var _this = this;\n    var opts = _(options).defaults(defaultConfig).value();\n\n    var externalRefresh = opts.externalRefresh;\n\n    this.logger = opts.logger;\n    this.framerate = opts.framerate;\n    var sceneBgColor = opts.sceneBgColor;\n    var sceneWidth = opts.sceneWidth;\n    var sceneHeight = opts.sceneHeight;\n    var canvasContainer = opts.canvasContainer;\n    var timeContainer = [];\n    var components = []; \n    \n    //create an new instance of a pixi stage\n    this.stage = new PIXI.Stage(sceneBgColor);\n    //create a renderer instance.\n    var renderer = PIXI.autoDetectRenderer(sceneWidth, sceneHeight);\n    \t\n    this.init = function() {\n\n        if(typeof(canvasContainer) === 'string') {\n            canvasContainer = document.getElementById(canvasContainer);\n        }\n        if(typeof(timeContainer) === 'string') {\n            timeContainer = document.getElementById(timeContainer);\n        }\n\n        canvasContainer.appendChild(renderer.view);\n        \n        components.forEach(function(c){\n    \t\tc.init();\n    \t});\n    };\n    \n    this.registerTimeContainer = function(container) {\n    \ttimeContainer.push(container);\n    };\n    \n    this.registerComponent = function(component) {\n    \tcomponents.push(component);\n    \tthis.stage.addChild(component.container);\n    };\n\n    this.refresh = function() {\n    \tcomponents.forEach(function(c){\n    \t\tc.refresh();\n    \t});\n        renderer.render(this.stage);\n    };\n\n    // Init page and intervals\n    var refreshInterval;\n\n    this.start = function() {\n\n        if(!externalRefresh) {\n            refreshInterval = setInterval(function() {_this.refresh();}, 1000/this.framerate);\n        }\n        \n        components.forEach(function(c){\n    \t\tc.start();\n    \t});\n    };\n\n    this.stop = function() {\n        if(!externalRefresh) {\n            clearInterval(refreshInterval);\n        }\n        clearInterval(refreshTimeInterval);\n        \n        components.forEach(function(c){\n    \t\tc.stop();\n    \t});\n    };\n\n\n    this.log = function(m) {\n        if(this.logger) {\n            this.logger.log(m);\n        }\n    };\n\n\n    return this;\n}\n\nmodule.exports = {\n    StageView: StageView\n};\n","/**\n* js/utils.js\n*\n* basic tools\n*\n*/\n\n'use strict';\n\nvar PIXI = require('pixi');\nvar _ = require('lodash');\nvar moment = require('moment');\n\nfunction formatTime (ts) {\n\treturn moment(ts).format(\"HH:mm:ss\");\n}\n\nfunction colorToHex(c) {\n\tvar m = /rgba?\\(\\s*(\\d+)\\s*,\\s*(\\d+)\\s*,\\s*(\\d+)/.exec(c);\n\treturn m ? '#' + (1 << 24 | m[1] << 16 | m[2] << 8 | m[3]).toString(16).substr(1) : c;\n}\n\nfunction getAnnotCategories(urlCategories, annotCategories) {\n\n    var jsonLoader = new PIXI.JsonLoader(urlCategories, true);\n\n    jsonLoader.on('loaded', function(res) {\n        var data = res.target.json;\n\n        while(annotCategories.length > 0) {\n        \tannotCategories.pop();\n        }\n\n        data.sessions.forEach(function(session) {\n            var annotCat = {\n                ts: session.start_ts === null ? new Date(0) : Date.parse(session.start_ts),\n                colors: {}\n            }\n            var categoriesJson = session.categories_json;\n            annotCat.order = categoriesJson.order;\n            if (typeof(annotCat.order['default']) === 'undefined'){\n            \tannotCat.order.push('default');\n            }\n\t\t\tvar catList = _.clone(categoriesJson.categories);\n\t\t\twhile(catList.length > 0) {\n\t\t\t\tvar cat = catList.pop();\n\t\t\t\tif(cat.code) {\n\t\t\t\t\tannotCat.colors[cat.code] = colorToHex(cat.color);\n\t\t\t\t}\n\t\t\t\tif(cat.subcategories) {\n\t\t\t\t\tcatList = catList.concat(cat.subcategories);\n\t\t\t\t}\n\t\t\t}\n            categoriesJson.categories.forEach(function(cat) {\n\t\t\t\tif(cat.code) {\n                \tannotCat.colors[cat.code] = colorToHex(cat.color);\n\t\t\t\t}\n            });\n            annotCat.defaultColor = categoriesJson.defaultColor || \"#536991\";\n            annotCategories.push(annotCat);\n        });\n    });\n\n    jsonLoader.load();\n\n}\n\nmodule.exports = {\n\tformatTime: formatTime,\n\tgetAnnotCategories: getAnnotCategories,\n\tcolorToHex: colorToHex\n};\n","/**\n* js/wswrapper.js\n*\n* simple webservice wrapper to register callbacks on onmessage\n*\n*/\n\n/* global WebSocket: false */\n\n'use strict';\n\nfunction WsWrapper(wsurl, logger) {\n\n    var url = wsurl;\n    var sock = new WebSocket(url);\n    var loggerObj = logger;\n\n    var log = function(msg) {\n        if(loggerObj) {\n            loggerObj.log(msg);\n        }\n    };\n\n    var handlers = [];\n\n    sock.onopen = function() {\n        log('Connected to ' + url);\n    };\n\n    sock.onclose = function(e) {\n        log('Connection closed (wasClean = ' + e.wasClean + ', code = ' + e.code + ', reason = \\'' + e.reason + '\\')');\n        sock = null;\n    };\n\n    sock.onmessage = function(e) {\n        log('received ' + e.data);\n        var data = JSON.parse(e.data);\n        handlers.forEach(function(handler) {\n            handler(data);\n        });\n    };\n\n    this.message = function(handler) {\n        if(handler) {\n            handlers.push(handler);\n        }\n    };\n\n}\n\nmodule.exports = {\n    WsWrapper: WsWrapper\n};\n"]}
+//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["node_modules/browserify/node_modules/browser-pack/_prelude.js","./app/js/main.js","/Users/rougeronj/workspace/mons/client/annotviz/app/js/annotsroll.js","/Users/rougeronj/workspace/mons/client/annotviz/app/js/annotstimeline.js","/Users/rougeronj/workspace/mons/client/annotviz/app/js/annotsvizview.js","/Users/rougeronj/workspace/mons/client/annotviz/app/js/doubleroll.js","/Users/rougeronj/workspace/mons/client/annotviz/app/js/logger.js","/Users/rougeronj/workspace/mons/client/annotviz/app/js/pianoroll.js","/Users/rougeronj/workspace/mons/client/annotviz/app/js/stageview.js","/Users/rougeronj/workspace/mons/client/annotviz/app/js/utils.js","/Users/rougeronj/workspace/mons/client/annotviz/app/js/wswrapper.js"],"names":[],"mappings":"AAAA;ACAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC9BA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC/NA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC/QA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACtNA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC1LA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC7CA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACpOA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACpHA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC1EA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA","file":"generated.js","sourceRoot":"","sourcesContent":["(function e(t,n,r){function s(o,u){if(!n[o]){if(!t[o]){var a=typeof require==\"function\"&&require;if(!u&&a)return a(o,!0);if(i)return i(o,!0);var f=new Error(\"Cannot find module '\"+o+\"'\");throw f.code=\"MODULE_NOT_FOUND\",f}var l=n[o]={exports:{}};t[o][0].call(l.exports,function(e){var n=t[o][1][e];return s(n?n:e)},l,l.exports,e,t,n,r)}return n[o].exports}var i=typeof require==\"function\"&&require;for(var o=0;o<r.length;o++)s(r[o]);return s})","/**\n * scripts/main.js\n *\n * This is the starting point for your application.\n * Take a look at http://browserify.org/ for more info\n */\n\n'use strict';\n\nvar doubleroll = require('./doubleroll');\nvar annotsroll = require('./annotsroll');\nvar annotstimeline = require('./annotstimeline');\nvar annotsvizview = require('./annotsvizview');\nvar stageview = require('./stageview');\nvar wswrapper = require('./wswrapper');\nvar logger = require('./logger');\nvar utils = require('./utils');\n\nvar _ = require('lodash');\n\nmodule.exports = _({})\n    .extend(doubleroll)\n    .extend(annotsroll)\n    .extend(annotstimeline)\n    .extend(annotsvizview)\n    .extend(stageview)\n    .extend(wswrapper)\n    .extend(logger)\n    .extend(utils)\n    .value();\n","/**\n* js/annotsRoll.js\n*\n* annotsRoll basic component\n*\n*/\n\n'use strict';\n\nvar PIXI = require('pixi');\nvar _ = require('lodash');\n\nvar DEFAULT_ANNOT_COLOR = '#bababa';\n\nvar defaultAnnotStyles = {\n    'label': { font: '16pt Arial Bold', fill: '#65A954', wordWrap: true},\n    'text' : { font: '12pt Arial Regular', fill: '#444444', wordWrap: true},\n    'user' : { font: '14pt Arial regular', fill: '#666666' },\n};\n\nvar defaultOptions = {\n    externalRefresh: false,\n    defaultColor: DEFAULT_ANNOT_COLOR,\n    annotStyles: defaultAnnotStyles,\n    ignoreAnnots:false\n};\n\nfunction AnnotsRoll(options) {\n\n//parentContainer, xInit, yInit, width, height, widthRoll, pixelsPerSecond, annotColors\n    var _this = this;\n    var opts = _(options).defaults(defaultOptions).value();\n\n\n    this.container = new PIXI.DisplayObjectContainer();\n    this.container.x = opts.xInit;\n    this.container.y = opts.yInit;\n    this.container.width = opts.width;\n\n    this.height = opts.height;\n    this.width = opts.width;\n    this.widthRoll = opts.widthRoll;\n    this.pixelsPerSecond = opts.pixelsPerSecond;\n    this.annotColors = opts.annotColors;\n    this.startTs = opts.startTs || Date.now();\n    this.ignoreAnnots = opts.ignoreAnnots;\n\n    var yInit = opts.yInit;\n    var annotStyles = _(opts.annotStyles).defaults(defaultAnnotStyles).value();\n    var marginX = 15;\n    for(var style in annotStyles) {\n    \tif (annotStyles[style].wordWrap === true){\n    \t\tannotStyles[style].wordWrapWidth = this.widthRoll - marginX;\n    \t}\n    }\n    var started = false;\n    var ws = opts.ws;\n    var externalRefresh = opts.externalRefresh;\n    var stageView = opts.stageView;\n    var waitInterval;\n    var wait = 0;\n\n    stageView.registerComponent(this);\n\n    var isHidden = function(child) {\n        // TODO: the origin point is an approximation. Should refine this\n        var globalPos = child.toGlobal(new PIXI.Point(0,0));\n        return ((globalPos.x + child.width) < 0) || ((globalPos.y + child.height) < 0) ;\n    };\n\n    this.addAnnots = function(data) {\n\n        //var title = data.content.category.label;\n        //var user = data.content.user;\n        //Test cat and color\n        //var colorAnnot = 0x65A954;\n        var category = data.content.category.label,\n            text     = data.content.text,\n            user     = data.content.user,\n            ts       = Date.parse(data.ts),\n            color    = data.content.color || this.getColor(ts, data.content.category.code);\n\n        this.addAnnot(category, text, user, color, ts);\n    };\n\n    this.getColor = function(ts, code) {\n        var colorsDef;\n        _(this.annotColors).eachRight(function(cdef) {\n            if(cdef.ts < ts) {\n                colorsDef = cdef.colors;\n                return false;\n            }\n        });\n        var resColor;\n        if(colorsDef) {\n            resColor = colorsDef[code];\n        }\n        if(!resColor) {\n            resColor = colorsDef.defaultColor || DEFAULT_ANNOT_COLOR;\n        }\n        return resColor;\n    };\n\n    this.addAnnot = function(category, text, user, color, ts){\n\n        var x = 0;\n        var y = (ts-this.startTs) * this.pixelsPerSecond / 1000 + yInit;\n\n        var colorHex = parseInt(color.replace(/^#/, ''), 16);\n\n        if (wait === 0){\n\t        var graphics = new PIXI.Graphics()\n\t            .beginFill(colorHex)\n\t            .drawRect(x, y, 10, 3)\n\t            .endFill();\n\n\t        this.container.addChild(graphics);\n\n\t        var textHeight = 0;\n\t        var catLabel = new PIXI.Text(\n\t            category,\n\t            _(annotStyles.label).extend({fill: color}).value()\n\t        );\n\t        catLabel.x = x + marginX;\n\t        catLabel.y = y - 23;\n\t        this.container.addChild(catLabel);\n\t        textHeight += (catLabel.height - 23 + 2);\n\n\t        if(text) {\n\t            var catText = new PIXI.Text(text, annotStyles.text);\n\t            catText.x = x + marginX;\n\t            catText.y = y + textHeight;\n\t            this.container.addChild(catText);\n\t            textHeight += (catText.height + 2);\n\t        }\n\n\t        var catUser = new PIXI.Text(user, annotStyles.user);\n\t        catUser.x = x + marginX;\n\t        catUser.y = y + textHeight;\n\t        this.container.addChild(catUser);\n\t        textHeight += (catUser.height + 8);\n\n\t        if (this.ignoreAnnots === true){\n\t\t        wait = textHeight / this.pixelsPerSecond;\n\t\t        waitInterval = setInterval(function() {_this.refreshWait();}, 1000);\n\t        }\n        }\n\n        this.addAnnotLine(colorHex, y);\n    };\n\n    this.addAnnotLine = function(color, y) {\n        var x = this.widthRoll;\n\n\n        var graphics = new PIXI.Graphics()\n            .beginFill(color)\n            .drawRect(x, y, this.width - x, 3)\n            .endFill();\n\n        this.container.addChild(graphics);\n    };\n\n    this.moveTo = function(diffTime){\n    \tthis.container.y = Math.floor(diffTime*this.pixelsPerSecond);\n    };\n\n    this.move = this.refresh = function() {\n        var diff = (this.startTs - Date.now())/1000;\n        this.moveTo(diff);\n    };\n\n    this.refreshWait = function(){\n    \twait -= 1;\n    \tif (wait < 0){\n    \t\twait = 0;\n    \t\tclearInterval(waitInterval);\n    \t}\n    };\n\n    this.removePassedObjets = function(){\n        var childrenToRemove = [];\n        _(_this.container.children).forEach(function(child) {\n            return typeof(child) === 'undefined' ||\n                (isHidden(child) && childrenToRemove.push(child));\n        });\n        childrenToRemove.forEach(function(child) {\n            _this.container.removeChild(child);\n        });\n    };\n\n    this.init = function() {\n\n        ws.message(function(data) {\n            _this.addAnnots(data);\n        });\n\n    };\n\n\n    this.start = function() {\n        if(!started) {\n            this.startTs = Date.now();\n            started = true;\n        }\n        this.cleanInterval = setInterval(function () { _this.removePassedObjets(); }, 1000 * this.height / this.pixelsPerSecond );\n        if(!externalRefresh) {\n            this.refreshInterval = setInterval(function() {_this.move();}, 1000/this.framerate);\n        }\n    };\n\n    this.stop = function() {\n        clearInterval(this.cleanInterval);\n        if(!externalRefresh) {\n            clearInterval(this.refreshInterval);\n        }\n    };\n\n}\n\nmodule.exports = {\n    AnnotsRoll: AnnotsRoll,\n};\n","/**\n* js/annotstimeline\n*\n* annotstimeline basic component\n*\n*/\n\n'use strict';\n\nvar PIXI = require('pixi');\nvar Utils = require('./utils.js');\nvar _ = require('lodash');\n\nvar defaultOptions = {\n    logger: undefined,\n    intervalWidth: 10,\n    intervalHeight: 5,\n    maxCellHeight: 200,\n    radius: 300,\n    serverUrl: 'http://127.0.0.1:8080',\n    channel: 'ANNOT',\n    maxPages: 1000,\n    showClockGraphics: true,\n    archive: false\n};\n\n\nfunction AnnotsTimeLine(options){\n    var _this = this;\n    var opts = _(options).defaults(defaultOptions).value();\n\n    this.container = new PIXI.DisplayObjectContainer();\n    this.container.x = opts.xInit;\n    this.container.y = opts.yInit;\n    this.container.width = opts.width;\n    this.container.height = opts.height;\n\n    this.timeBegin = opts.timeBegin;\n    this.timeEnd = opts.timeEnd;\n    this.duration = (this.timeEnd - this.timeBegin)/1000;\n    this.width = opts.width;\n    this.height = opts.height;\n    this.intervalHeight = opts.intervalHeight;\n    this.intervalWidth = opts.intervalWidth;\n    this.maxCellHeight = opts.maxCellHeight;\n    this.annotCategories = opts.annotCategories;\n    this.startTs = options.startTs || Date.now();\n    this.showClockGraphics = opts.showClockGraphics;\n    this.archive = opts.archive;\n\n    this.circleX = opts.circleX || (this.width/2);\n    this.circleY = opts.circleY || (this.height/2);\n    this.radius = opts.radius;\n    var perimeter = 2*Math.PI* this.radius;\n    this.intervalDuration = (this.intervalWidth * this.duration / perimeter);\n\n    var channel = opts.channel;\n    var eventCode = opts.eventCode;\n    var serverUrl = opts.serverUrl;\n    var maxPages = opts.maxPages;\n\n    var totalIndex = Math.floor( perimeter/this.intervalWidth);\n\n    this.cells = [];\n    for (var i=0; i<(perimeter/this.intervalWidth) ; i++){\n        this.cells[i] = [];\n        this.cells[i].i = i;\n        this.cells[i].totalAnnots = 0;\n        this.cells[i].categories = {};\n    }\n\n    var ws = opts.ws;\n    var stageView = opts.stageView;\n\n    //draw the base - circle and line to locate the scene\n    var graphics = new PIXI.Graphics();\n    graphics.lineStyle(2, 0x646464)\n        .drawCircle(this.circleX, this.circleY, this.radius - 3)\n        .endFill();\n    this.container.addChild(graphics);\n\n    stageView.registerComponent(this);\n\n    var loadArchives = function() {\n        //start timeBegin end startTime\n        //query -> need channel + eventCode\n        //iterate over data fill cells\n        var startTs = _this.timeBegin;\n        var endTs = Math.min(_this.timeEnd,_this.startTs);\n\n        console.log(\"START TS\",new Date(startTs).toISOString());\n        console.log(\"END TS\",new Date(endTs).toISOString());\n\n        var url = serverUrl + '/p/api/v1/annotation';\n        var filters = [\n            { name: 'ts', op: '>', val: new Date(startTs).toISOString()}, //start\n            { name: 'ts', op: '<=', val: new Date(endTs).toISOString()}, //end\n            { name: 'channel', op: '==', val: channel}, //channel\n            { name: 'event_code', op: '==', val: eventCode} //eventcode\n        ];\n\n        url = url + '?q=' + JSON.stringify({filters:filters});\n\n        var totalPage = 1;\n        var currentPage = 1;\n\n        var processResFunction = function(res) {\n            //console.log(\"RES archive\", res);\n            if(res) {\n                var data = res.target.json;\n                /*jshint -W069 */\n                totalPage = Math.min(maxPages,parseInt(data['total_pages']));\n                data.objects.forEach(function(annotation) {\n                    _this.addAnnot(annotation);\n                });\n            }\n            if(currentPage <= totalPage) {\n                var jsonLoader = new PIXI.JsonLoader(url+'&page='+currentPage, true);\n                jsonLoader.on('loaded', processResFunction);\n                jsonLoader.load();\n                currentPage++;\n            }\n        };\n        processResFunction();\n\n    };\n\n    //Add Annotation to the TimeLine\n    this.addAnnot = function(data){\n\n        var ts = Date.parse(data.ts);\n        var colorsDef;\n        _(this.annotCategories).eachRight(function(cdef) {\n            if(cdef.ts < ts) {\n                colorsDef = cdef;\n                return false;\n            }\n        });\n\n        if(!colorsDef) {\n            return;\n        }\n\n        if (this.timeEnd > ts){\n            var i = Math.floor((ts - this.timeBegin)/(1000*this.intervalDuration));\n\n            if (typeof(this.cells[i].graphics) === 'undefined'){\n                this.initCell(this.cells[i], colorsDef);\n            }\n\n            var annotCode;\n            if (typeof(colorsDef.colors[data.content.category.code]) !== 'undefined'){\n                annotCode = data.content.category.code;\n            } else {\n                annotCode = 'default';\n            }\n\n            this.cells[i].categories[annotCode].count += 1;\n            this.cells[i].totalAnnots +=1;\n            this.redrawCell(this.cells[i], colorsDef);\n        }\n    };\n\n    this.initClockGraphics = function() {\n        var tBeg = new PIXI.Text(Utils.formatTime(this.timeBegin), { font: '12pt Gothic Standard', fill: '#646464' });\n        tBeg.x = this.circleX + 15;\n        tBeg.y = this.circleY - this.radius - this.maxCellHeight - 10;\n        this.container.addChild(tBeg);\n\n        var tEnd = new PIXI.Text(Utils.formatTime(this.timeEnd), { font: '12pt Gothic Standard', fill: '#646464' });\n        tEnd.x = this.circleX - 15 - tEnd.width;\n        tEnd.y = this.circleY - this.radius - this.maxCellHeight - 10;\n        this.container.addChild(tEnd);\n\n        var t15 = new PIXI.Text(Utils.formatTime(((this.timeEnd - this.timeBegin)/4) + this.timeBegin), { font: '12pt Gothic Standard', fill: '#646464' });\n        t15.x = this.circleX + this.radius + this.maxCellHeight + 10 ;\n        t15.y = this.circleY - t15.height;\n        t15.rotation = Math.PI /2;\n        this.container.addChild(t15);\n\n        var t30 = new PIXI.Text(Utils.formatTime(((this.timeEnd - this.timeBegin)/2) + this.timeBegin), { font: '12pt Gothic Standard', fill: '#646464' });\n        t30.x = this.circleX - t30.width/2;\n        t30.y = this.circleY + this.radius + this.maxCellHeight - 2;\n        this.container.addChild(t30);\n\n        var t45 = new PIXI.Text(Utils.formatTime(((this.timeEnd - this.timeBegin)*3/4) + this.timeBegin), { font: '12pt Gothic Standard', fill: '#646464' });\n        t45.x = this.circleX - this.radius - this.maxCellHeight - 10 ;\n        t45.y = this.circleY + t15.height;\n        t45.rotation = -Math.PI/2;\n        this.container.addChild(t45);\n\n        var lineV = new PIXI.Graphics();\n        lineV.lineStyle(1, 0x646464)\n            .moveTo(this.circleX, this.circleY - (this.radius/3)/2)\n            .lineTo(this.circleX, this.circleY - this.radius - this.maxCellHeight - 10)\n            .endFill();\n        this.container.addChild(lineV);\n    };\n\n    //Draw the cellule\n    this.redrawCell = function(cell, colorsDef){\n        var y = 0;\n\n        //Check if total height is higher than Max Cell Height\n        var heightStep;\n        if ((cell.totalAnnots*this.intervalHeight) > this.maxCellHeight){\n            heightStep = this.maxCellHeight/cell.totalAnnots;\n        } else {\n            heightStep = this.intervalHeight;\n        }\n\n        //Draw the rect depending on the height step calculated\n        for (var i=0; i< colorsDef.order.length; i++){\n            var currentCode = colorsDef.order[i];\n            cell.graphics.beginFill(cell.categories[currentCode].color.replace('#', '0x'))\n                .drawRect(0, y, this.intervalWidth-1, -cell.categories[currentCode].count * heightStep)\n                .endFill();\n            y -= cell.categories[currentCode].count*heightStep;\n        }\n    };\n\n    this.initCell = function(cell, colorsDef){\n        cell.graphics = new PIXI.Graphics();\n        cell.graphics.position.x = this.circleX + this.radius * Math.sin(cell.i*(360/totalIndex)*(Math.PI/180));\n        cell.graphics.position.y = this.circleY - this.radius * Math.cos(cell.i*(360/totalIndex)*(Math.PI/180));\n        cell.graphics.rotation = (cell.i)*(360/totalIndex)*(Math.PI/180) + (360/(totalIndex*2))*(Math.PI/180);\n        this.container.addChild(cell.graphics);\n\n        for (var category in colorsDef.colors){\n            cell.categories[category] = {\n                'count': 0,\n                'color': colorsDef.colors[category]\n            };\n        }\n        if (typeof(cell.categories['default']) === 'undefined'){\n            cell.categories['default'] = {\n                'count': 0,\n                'color': colorsDef.defaultColor\n            };\n        }\n    };\n\n    this.init = function() {\n        if (!this.archive){\n            ws.message(function(data) {\n                _this.addAnnot(data);\n            });\n        }\n\n        if (this.showClockGraphics){this.initClockGraphics();}\n    };\n\n\n    this.start = function() {\n        this.startTs = Date.now();\n        loadArchives();\n    };\n\n    this.refresh = function() {\n\n    };\n\n    this.stop = function(){\n    };\n\n    return this;\n}\n\nmodule.exports = {\n    AnnotsTimeLine: AnnotsTimeLine\n};\n","/**\n* js/annotsvizview.js\n*\n* This is the starting point for your application.\n* Take a look at http://browserify.org/ for more info\n*/\n\n'use strict';\n\nvar PIXI = require('pixi');\nvar _ = require('lodash');\nvar DoubleRoll = require('./doubleroll.js');\nvar AnnotsTimeLine = require('./annotstimeline.js');\nvar AnnotsRoll = require('./annotsroll.js');\nvar Utils = require('./utils.js');\n\nvar defaultOptions = {\n    xInit: 0,\n    yInit: 0,\n    width: 1024,\n    height: 768\n};\n\nfunction AnnotsVizView(options){\n    var _this = this;\n    var opts = _(options).defaults(defaultOptions).value();\n\n    this.container = new PIXI.DisplayObjectContainer();\n    this.container.x = opts.xInit;\n    this.container.y = opts.yInit;\n    this.width = opts.width;\n    this.height= opts.height;\n    this.timeBegin = opts.timeBegin;\n    this.timeEnd = opts.timeEnd;\n\n    this.annotCategories = [];\n    this.annotCategoriesDay2 = [];\n    this.annotCategoriesDay1 = [];\n\n    var wsPianoroll = opts.wsPianoroll;\n    var wsAnnot = opts.wsAnnot;\n    var stageView = opts.stageView;\n    var currentTime = Date.now() + 3600*1000;\n    var channel = opts.channel;\n    var serverUrl = opts.serverUrl;\n    \n    var eventCode = opts.eventCode;\n    var eventCodeSessionDay2 = opts.eventCodeSessionDay2;\n\tvar eventCodeSessionDay1 = opts.eventCodeSessionDay1;\n\n\tUtils.getAnnotCategories(opts.urlCategories + eventCode, this.annotCategories);\n    Utils.getAnnotCategories(opts.urlCategories + eventCodeSessionDay2, this.annotCategoriesDay2);\n    Utils.getAnnotCategories(opts.urlCategories + eventCodeSessionDay1, this.annotCategoriesDay1);\n\n    stageView.registerComponent(this);\n\n    var currentDay = new AnnotsTimeLine.AnnotsTimeLine({\n    \tstageView : stageView,\n        logger: logger,\n        ws: wsAnnot,\n        xInit: 0,\n        yInit: 0,\n        width: 1024 - 200 - 200,\n        height: 768-200,\n        timeBegin: this.timeBegin,\n        timeEnd: this.timeEnd,\n        intervalWidth: 6,\n        intervalHeight: 10,\n        maxCellHeight: 70,\n        radius: 200,\n        eventCode: eventCode,\n        channel: channel,\n        serverUrl: serverUrl,\n        annotCategories: this.annotCategories\n    });\n\t\n\t//Archive day 2\n    var timeLineDay2 = new AnnotsTimeLine.AnnotsTimeLine({\n    \tstageView : stageView,\n        xInit: 0,\n        yInit: 0,\n        width: 1024 - 200 - 200,\n        height: 768-200,\n        timeBegin: Date.parse(\"2015-01-23T09:30:00+01:00\"),\n        timeEnd: Date.parse(\"2015-01-23T18:30:00+01:00\"),\n        circleX: currentDay.circleX,\n        circleY: currentDay.circleY,\n        intervalWidth: (currentDay.radius*2/3)* currentDay.intervalWidth / currentDay.radius,\n        intervalHeight: (currentDay.intervalHeight * (currentDay.radius - currentDay.radius*2/3))/ currentDay.maxCellHeight,\n        maxCellHeight: (currentDay.radius - currentDay.radius*2/3)/2,\n        radius: currentDay.radius*2/3,\n        annotCategories: this.annotCategoriesDay2,\n        eventCode: eventCodeSessionDay2,\n        channel: channel,\n        serverUrl: serverUrl,\n        showClockGraphics:false,\n        archive: true\n    });\n\t\n\t//Archive day 1\n    var timeLineDay1 = new AnnotsTimeLine.AnnotsTimeLine({\n    \tstageView : stageView,\n        xInit: 0,\n        yInit: 0,\n        width: 1024 - 200 - 200,\n        height: 768-200,\n        timeBegin: Date.parse(\"2015-01-22T09:30:00+01:00\"),\n        timeEnd: Date.parse(\"2015-01-22T18:30:00+01:00\"),\n        circleX: currentDay.circleX,\n        circleY: currentDay.circleY,\n        intervalWidth: (currentDay.radius*2/3)* currentDay.intervalWidth / currentDay.radius,\n        intervalHeight: (currentDay.intervalHeight * (currentDay.radius - currentDay.radius/3))/ currentDay.maxCellHeight,\n        maxCellHeight: (currentDay.radius*2/3 - currentDay.radius/3)/4,\n        radius: currentDay.radius/3,\n        annotCategories: this.annotCategoriesDay1,\n        eventCode: eventCodeSessionDay1,\n        channel: channel,\n        serverUrl: serverUrl,\n        showClockGraphics:false,\n        archive: true,\n    });\n\n    var currentTimeText = new PIXI.Text(\"-- : -- : --\", { font: '18pt Gothic Standard', fill: '#646464' });\n    currentTimeText.x = timeLineDay2.circleX - currentTimeText.width/2;\n    currentTimeText.y = timeLineDay2.circleY - currentTimeText.height/2;\n    this.container.addChild(currentTimeText);\n\n    var doubleRollH = new DoubleRoll.DoubleRoll({\n        stageView : stageView,\n        logger: logger,\n        ws: wsPianoroll,\n        yInit: (this.height - 200),\n        sceneHeight: 200,\n        pianorolls : [\n            {\n                height: 200,\n                timeWidth: 10,\n                lineInterval: 5000,\n                noteHeight: 10\n            },\n        ]\n    });\n\n    var doubleRollV = new DoubleRoll.DoubleRoll({\n        stageView : stageView,\n        logger: logger,\n        ws: wsPianoroll,\n        orientation: 'vertical',\n        sceneHeight: 768-200,\n        pianorolls : [\n            {\n                height: 200,\n                timeWidth: 60,\n                lineInterval: 5000,\n                noteHeight: 5,\n            },\n        ]\n    });\n\n    var annotsRoll = new AnnotsRoll.AnnotsRoll({\n        stageView : stageView,\n        logger: logger,\n        ws: wsAnnot,\n        parentContainer: doubleRollV.stage,\n        xInit: 1024 - 200 - 200,\n        yInit: 768-200,\n        width: 200 + 200,\n        height: 768-200,\n        widthRoll: 200,\n        framerate: doubleRollV.framerate,\n        pixelsPerSecond: Math.floor(1024 / 60),\n        annotColors: this.annotCategories\n    });\n\n    var limiters = new PIXI.Graphics()\n        .lineStyle(1, 0x646464)\n        .moveTo(annotsRoll.container.x, annotsRoll.container.y)\n        .lineTo(annotsRoll.container.x, annotsRoll.container.y - annotsRoll.height)\n        .moveTo(annotsRoll.container.x + annotsRoll.widthRoll, annotsRoll.container.y)\n        .lineTo(annotsRoll.container.x + annotsRoll.widthRoll, annotsRoll.container.y - annotsRoll.height)\n        .moveTo(0, this.height - 200)\n        .lineTo(this.width, this.height - 200)\n        .drawRect(0, 0, this.width -1, this.height -1)\n        .beginFill(0xECECEC)\n        .drawRect(1024 - 200, 0, 200, 768-200)\n        .endFill();\n    this.container.addChild(limiters);\n\n    this.init = function(){\n    };\n\n    this.updateTime = function(){\n        currentTimeText.setText(Utils.formatTime(Date.now()));\n    };\n\n    var refreshTimeInterval;\n\n    this.start = function() {\n        refreshTimeInterval = setInterval(function() {_this.updateTime();}, 1000);\n    };\n\n    this.refresh = function() {\n    };\n\n    this.stop = function(){\n    };\n\n    return this;\n\n}\n\nmodule.exports = {\n    AnnotsVizView: AnnotsVizView\n};\n","/**\n* scripts/doubleroll.js\n*\n* This is the starting point for your application.\n* Take a look at http://browserify.org/ for more info\n*/\n\n'use strict';\n\n\nvar PIXI = require('pixi');\nvar _ = require('lodash');\nvar PianoRoll = require('./pianoroll');\n\nvar defaultConfig = {\n    orientation: 'horizontal',\n    logger: undefined,\n    sceneWidth: 1024,\n    pianorolls : [\n      {\n        height: 435,\n        timeWidth: 10,\n        lineInterval: 5000,\n        noteHeight: undefined\n      },\n      {\n        height: 645,\n        timeWidth: 60,\n        lineInterval: 5000,\n        noteHeight: undefined\n      },\n    ],\n    framerate: 25,\n    offsetMusic: false,\n    sceneBgColor: 0xFFFFFF,\n    lineColor: 0x444444,\n    lineFillColor: 0xFFFF00,\n    noteColors: [0xB90000, 0x4BDD71, 0xAF931E, 0x1C28BA, 0x536991],\n    noteHeight: undefined,\n    zeroShift: 0.9,\n    timeWidth: 60,\n    lineInterval: 5000,\n//    wsUri: undefined,\n//    eventCode: undefined\n\n};\n\nfunction DoubleRoll(options) {\n\n    var _this = this;\n    var opts = _(options).defaults(defaultConfig).value();\n\n    var orientation = opts.orientation;\n    var isHorizontal = (orientation !== 'vertical');\n\n    this.logger = opts.logger;\n    this.lineColor = opts.lineColor;\n    this.lineFillColor = opts.lineFillColor;\n    this.framerate = opts.framerate;\n    this.offsetMusic = opts.offsetMusic;\n    this.noteColors = opts.noteColors;\n\n    var noteHeight = opts.noteHeight;\n    var sceneHeight = opts.sceneHeight || _(opts.pianorolls).reduce(function(s,p) { return s + p.height; }, 0);\n    var timeWidth = opts.timeWidth;\n    var lineInterval = opts.lineInterval;\n    var offsetMusic = opts.offsetMusic;\n\n    var sceneWidth = opts.sceneWidth;\n    var stageView = opts.stageView;\n\n    var zeroShift = opts.zeroShift;\n\n    var ws = opts.ws;\n\n    var colorsReg = {};\n\n    this.container = new PIXI.DisplayObjectContainer();\n    this.container.x = Math.floor(sceneWidth*zeroShift);\n    this.container.y = 0;\n\n    stageView.registerComponent(this);\n\n    var pianorollList = [];\n\n    var pianorollOptions = {\n        parentContainer: this.container,\n        orientation: orientation,\n        xInit: 0,\n        width: sceneWidth,\n        noteColors: this.noteColors,\n        colorsReg: colorsReg,\n        lineColor: this.lineColor,\n        lineInterval: lineInterval,\n        offsetMusic: offsetMusic,\n    };\n\n    var yInit = opts.yInit || 0;\n    var linesDown = true;\n    _(opts.pianorolls).forEach(function(prDef, i) {\n        var prNoteHeight = noteHeight || prDef.noteHeight || prDef.height / 128;\n        var prTimeWidth = prDef.timeWidth || timeWidth;\n        pianorollList.push(new PianoRoll(_({\n            yInit: yInit,\n            height: prDef.height,\n            linesDown: linesDown,\n            pixelsPerSecond: Math.floor(sceneWidth / prTimeWidth),\n            noteHeight: prNoteHeight,\n            lineInterval: prDef.lineInterval\n        }).defaults(pianorollOptions).value()));\n        yInit += prDef.height;\n        linesDown = !linesDown;\n\n        if(i<(opts.pianorolls.length-1)) {\n            var lineGraphics = new PIXI.Graphics()\n                .beginFill(_this.lineFillColor)\n                .lineStyle(1, _this.lineColor)\n                .moveTo(Math.floor(sceneWidth*zeroShift), yInit)\n                .lineTo(-sceneWidth - Math.floor(sceneWidth*zeroShift), yInit)\n                .endFill();\n            _this.container.addChild(lineGraphics);\n        }\n    });\n\n    if(!isHorizontal) {\n        this.container.rotation = Math.PI/2;\n        this.container.y = sceneHeight;\n        this.container.x = sceneWidth;\n    }\n\n\n    this.init = function() {\n\n    \tws.message(function(data) {\n            _this.addNotes(data);\n        });\n\n    };\n\n\n    this.addNotes = function(data) {\n\n        pianorollList.forEach(function(c) {\n            c.addNoteRaw(data);\n        });\n    };\n\n    this.refresh = function() {\n        pianorollList.forEach(function(c) {\n            c.move();\n        });\n    };\n\n    // Init page and intervals\n    var startTs;\n\n    this.start = function() {\n\n        startTs = Date.now();\n        pianorollList.forEach(function(c) {\n            c.start();\n        });\n    };\n\n    this.stop = function() {\n\n        pianorollList.forEach(function(c) {\n            c.stop();\n        });\n    };\n\n\n    this.log = function(m) {\n        if(this.logger) {\n            this.logger.log(m);\n        }\n    };\n\n\n\n    return this;\n}\n\nmodule.exports = {\n    DoubleRoll: DoubleRoll\n};\n","/**\n* js/wswrapper.js\n*\n* simple logger service\n*\n*/\n\n/* global document: false */\n\n'use strict';\n\nfunction HtmlLogger(doLog, container) {\n\n    var logContainer = container;\n    if(typeof(container) === 'string') {\n        logContainer = document.getElementById(container);\n    }\n    if(!doLog) {\n        document.body.removeChild(logContainer);\n        logContainer = undefined;\n    }\n\n\n    this.log = function(msg) {\n        if(doLog && logContainer) {\n            logContainer.innerHTML += msg + '\\n';\n            logContainer.scrollTop = logContainer.scrollHeight;\n        }\n    };\n}\n\nfunction ConsoleLogger(doLog) {\n\n    this.log = function(msg) {\n        if(doLog) {\n            console.log(msg);\n        }\n    };\n\n}\n\nmodule.exports = {\n    HtmlLogger: HtmlLogger,\n    ConsoleLogger: ConsoleLogger\n};\n","/**\n* js/pianoroll.js\n*\n* pianoroll basic component\n*\n*/\n\n'use strict';\n\n\nvar PIXI = require('pixi');\nvar randomColor = require('randomColor');\nvar _ = require('lodash');\n\nvar NTP_EPOCH_DELTA = 2208988800; //c.f. RFC 868\n\nfunction PianoRoll(options) {\n    var _this = this;\n    this.container = new PIXI.DisplayObjectContainer();\n    this.container.x = options.xInit;\n    this.container.y = options.yInit;\n    options.parentContainer.addChild(this.container);\n\n    var orientation = options.orientation;\n    var isHorizontal = (orientation !== 'vertical');\n\n    this.linesDown = options.linesDown;\n    this.height = options.height;\n    this.pixelsPerSecond = options.pixelsPerSecond;\n    this.width = options.width;\n    this.noteColors = options.noteColors;\n    this.colorsReg = options.colorsReg || {};\n    this.lineColor = options.lineColor;\n    this.lineInterval = options.lineInterval;\n    this.offsetMusic = options.offsetMusic || false;\n    this.noteHeight = options.noteHeight;\n    this.noteDict = {};\n    this.startTs = options.startTs || Date.now();\n\n    var started = false;\n\n    var isHidden = function(child) {\n        // TODO: the origin point is an approximation. Should refine this\n        var globalPos = child.toGlobal(new PIXI.Point(0,0));\n        return ((globalPos.x + child.width) < 0) || ((globalPos.y + child.height) < 0) ;\n    };\n\n    //TODO: I do not like the \"regColor\" object. This should not be global, but local\n    this.getColor = function(canal) {\n        var color = this.colorsReg[canal];\n        if(typeof(color) === 'undefined') {\n            var colorsRegSize = Object.keys(this.colorsReg).length;\n            if(colorsRegSize < this.noteColors.length) {\n                color = this.colorsReg[canal] = this.noteColors[colorsRegSize];\n            }\n            else {\n                color = this.colorsReg[canal] = parseInt(randomColor({ luminosity: 'light', hue: 'random', format:'hex'}).replace(/^#/, ''), 16);\n            }\n        }\n        return color;\n    };\n\n    this.getNoteRect = function(x, y, color, alpha, width, height) {\n        var graphics = new PIXI.Graphics();\n        graphics.beginFill(color, alpha);\n        graphics.drawRect(0, 0, width, height);\n        graphics.endFill();\n        graphics.x = x;\n        graphics.y = y;\n        graphics.width = width;\n        graphics.height = height;\n        return graphics;\n    };\n\n    this.addNoteRaw = function(data) {\n        var note = data.content[3];\n        var velocity = data.content[4];\n        var ts = (data.content[0] - NTP_EPOCH_DELTA)*1000;\n        var channel = data.content[2];\n        var sessionTs = data.content[1];\n\n        this.addNote(note, ts, sessionTs, velocity, channel, 0);\n    };\n\n    this.addNote = function(note, startTime, sessionTs, velocity, channel, duration) {\n\n        var ts = startTime;\n        if(this.offsetMusic) {\n            ts = this.startTs + sessionTs;\n        }\n\n        var noteDuration = duration;\n        var noteVelocity = velocity;\n        var graphics;\n        if(!duration) {\n            if(typeof this.noteDict[channel]==='undefined'){\n                this.noteDict[channel] = {};\n            }\n            if(velocity===0) {\n                if(typeof this.noteDict[channel][note] !== 'undefined') {\n                    var noteDef = this.noteDict[channel][note];\n                    delete this.noteDict[channel][note];\n                    noteDuration = sessionTs - noteDef.sessionTs;\n                    graphics = noteDef.graphics;\n                    noteVelocity = noteDef.velocity;\n                    ts = noteDef.ts;\n                }\n            }\n            else {\n                noteDuration = Math.abs(Date.now() - ts);\n                this.noteDict[channel][note] = { ts: ts, velocity: velocity, sessionTs: sessionTs};\n            }\n        }\n\n\n        if(!this.offsetMusic || velocity===0) {\n\n            var width = noteDuration * this.pixelsPerSecond / 1000;\n            if(!graphics) {\n                var x = (ts-this.startTs) * this.pixelsPerSecond / 1000;\n                if((x+width) <  (Math.abs(this.container.x) - this.width)) {\n                    // not visible. do nothing\n                    return;\n                }\n                var y = Math.floor((128-note+0.5) * this.height / 128 - (this.noteHeight/2));\n                var color = this.getColor(channel);\n                var alpha = (noteVelocity / 128);\n\n                graphics = this.getNoteRect(x, y, color, alpha, width, this.noteHeight);\n                this.container.addChild(graphics);\n            }\n            else {\n                graphics.width = width;\n            }\n\n            if(!duration && velocity) {\n                this.noteDict[channel][note].graphics = graphics;\n            }\n        }\n    };\n\n    this.addLine = function(ts){\n\n        if(typeof(ts) === 'undefined') {\n            ts = new Date();\n        }\n        var x = -this.container.x;\n        var y = this.linesDown ? this.height - 20 : 0;\n\n        var graphics = new PIXI.Graphics()\n            .beginFill(0xFFFF00)\n            .lineStyle(1, this.lineColor)\n            .moveTo(0, 0)\n            .lineTo(0, 20)\n            .endFill();\n        graphics.x = x;\n        graphics.y = y;\n        this.container.addChild(graphics);\n        // Add text\n        //var totalSec = lineNb * this.lineInterval / 1000;\n        var hours = ts.getHours();\n        var minutes =ts.getMinutes();\n        var seconds = ts.getSeconds();\n        var timeStr = (hours < 10 ? '0' + hours : hours) + ':' + (minutes < 10 ? '0' + minutes : minutes) + ':' + (seconds  < 10 ? '0' + seconds : seconds);\n\n        var fontObj = { font: '10pt Arial', fill: '#444444' };\n        var t = new PIXI.Text(timeStr, fontObj);\n        if(isHorizontal) {\n            t.x = x + 2;\n            t.y = this.linesDown ? this.height - 15 : 2;\n        }\n        else {\n            t.rotation = -Math.PI/2;\n            t.x = x ;\n            t.y = this.linesDown ? this.height - 2 : t.width + 2;\n        }\n        this.container.addChild(t);\n    };\n\n    this.moveTo = function(diffTime){\n        var oldX = this.container.x;\n        this.container.x = Math.floor(diffTime*this.pixelsPerSecond);\n        var deltaX = Math.abs(oldX-this.container.x);\n        _.forOwn(this.noteDict, function(channelDict) {\n            _.forOwn(channelDict, function(noteDef) {\n                if(noteDef.graphics) {\n                    noteDef.graphics.width = noteDef.graphics.width + deltaX;\n                }\n            });\n        });\n    };\n\n    this.move = function() {\n        var diff = (this.startTs - Date.now())/1000;\n        this.moveTo(diff);\n    };\n\n    this.removePassedObjets = function(){\n        var childrenToRemove = [];\n        _(_this.container.children).forEach(function(child) {\n            return typeof(child) === 'undefined' ||\n                (isHidden(child) && childrenToRemove.push(child));\n        });\n        childrenToRemove.forEach(function(child) {\n            _this.container.removeChild(child);\n        });\n    };\n\n    this.start = function() {\n        if(!started) {\n            this.startTs = Date.now();\n            this.addLine();\n            started = true;\n        }\n        this.verticalLinesInterval = setInterval(function() { _this.addLine(); }, this.lineInterval);\n        this.cleanInterval = setInterval(function () { _this.removePassedObjets(); }, 1000 * this.width / this.pixelsPerSecond );\n    };\n\n    this.stop = function() {\n        //window.clearInterval(this.moveInterval);\n        clearInterval(this.verticalLinesInterval);\n        clearInterval(this.cleanInterval);\n    };\n\n\n}\n\nmodule.exports = PianoRoll;\n","/**\n* scripts/stageview.js\n*\n* This is the starting point for your application.\n* Take a look at http://browserify.org/ for more info\n*/\n\n/* global document: false */\n\n'use strict';\n\n\nvar PIXI = require('pixi');\nvar _ = require('lodash');\n\nvar defaultConfig = {\n    externalRefresh: false,\n    logger: undefined,\n    sceneWidth: 1024,\n    sceneHeight: 768,\n    framerate: 25,\n    sceneBgColor: 0xFFFFFF,\n    canvasContainer: 'canvasContainer',\n};\n\nfunction StageView(options) {\n\n    var _this = this;\n    var opts = _(options).defaults(defaultConfig).value();\n\n    var externalRefresh = opts.externalRefresh;\n\n    this.logger = opts.logger;\n    this.framerate = opts.framerate;\n    var sceneBgColor = opts.sceneBgColor;\n    var sceneWidth = opts.sceneWidth;\n    var sceneHeight = opts.sceneHeight;\n    var canvasContainer = opts.canvasContainer;\n    var timeContainer = [];\n    var components = [];\n\n    //create an new instance of a pixi stage\n    this.stage = new PIXI.Stage(sceneBgColor);\n    //create a renderer instance.\n    var renderer = PIXI.autoDetectRenderer(sceneWidth, sceneHeight);\n\n    this.init = function() {\n\n        if(typeof(canvasContainer) === 'string') {\n            canvasContainer = document.getElementById(canvasContainer);\n        }\n        if(typeof(timeContainer) === 'string') {\n            timeContainer = document.getElementById(timeContainer);\n        }\n\n        canvasContainer.appendChild(renderer.view);\n\n        components.forEach(function(c){\n    \t\tc.init();\n    \t});\n    };\n\n    this.registerTimeContainer = function(container) {\n    \ttimeContainer.push(container);\n    };\n\n    this.registerComponent = function(component) {\n    \tcomponents.push(component);\n    \tthis.stage.addChild(component.container);\n    };\n\n    this.refresh = function() {\n    \tcomponents.forEach(function(c){\n    \t\tc.refresh();\n    \t});\n        renderer.render(this.stage);\n    };\n\n    // Init page and intervals\n    var refreshInterval;\n\n    this.start = function() {\n\n        if(!externalRefresh) {\n            refreshInterval = setInterval(function() {_this.refresh();}, 1000/this.framerate);\n        }\n\n        components.forEach(function(c){\n    \t\tc.start();\n    \t});\n    };\n\n    this.stop = function() {\n        if(!externalRefresh) {\n            clearInterval(refreshInterval);\n        }\n        \n        components.forEach(function(c){\n    \t\tc.stop();\n    \t});\n    };\n\n\n    this.log = function(m) {\n        if(this.logger) {\n            this.logger.log(m);\n        }\n    };\n\n\n    return this;\n}\n\nmodule.exports = {\n    StageView: StageView\n};\n","/**\n* js/utils.js\n*\n* basic tools\n*\n*/\n/*jshint bitwise: false*/\n/*jshint camelcase: false */\n\n'use strict';\n\nvar PIXI = require('pixi');\nvar _ = require('lodash');\nvar moment = require('moment');\n\nfunction formatTime (ts) {\n    return moment(ts).format('HH:mm:ss');\n}\n\nfunction colorToHex(c) {\n    var m = /rgba?\\(\\s*(\\d+)\\s*,\\s*(\\d+)\\s*,\\s*(\\d+)/.exec(c);\n    return m ? '#' + (1 << 24 | m[1] << 16 | m[2] << 8 | m[3]).toString(16).substr(1) : c;\n}\n\nfunction getAnnotCategories(urlCategories, annotCategories) {\n\n    var jsonLoader = new PIXI.JsonLoader(urlCategories, true);\n\n    jsonLoader.on('loaded', function(res) {\n        var data = res.target.json;\n\n        while(annotCategories.length > 0) {\n            annotCategories.pop();\n        }\n\n        data.sessions.forEach(function(session) {\n            var annotCat = {\n                ts: session.start_ts === null ? new Date(0) : Date.parse(session.start_ts),\n                colors: {}\n            };\n            var categoriesJson = session.categories_json;\n            annotCat.order = categoriesJson.order;\n            if (typeof(annotCat.order['default']) === 'undefined'){\n                annotCat.order.push('default');\n            }\n            var catList = _.clone(categoriesJson.categories);\n            while(catList.length > 0) {\n                var cat = catList.pop();\n                if(cat.code) {\n                    annotCat.colors[cat.code] = colorToHex(cat.color);\n                }\n                if(cat.subcategories) {\n                    catList = catList.concat(cat.subcategories);\n                }\n            }\n            categoriesJson.categories.forEach(function(cat) {\n                if(cat.code) {\n                    annotCat.colors[cat.code] = colorToHex(cat.color);\n                }\n            });\n            annotCat.defaultColor = categoriesJson.defaultColor || '#536991';\n            annotCategories.push(annotCat);\n        });\n    });\n\n    jsonLoader.load();\n\n}\n\nmodule.exports = {\n    formatTime: formatTime,\n    getAnnotCategories: getAnnotCategories,\n    colorToHex: colorToHex\n};\n","/**\n* js/wswrapper.js\n*\n* simple webservice wrapper to register callbacks on onmessage\n*\n*/\n\n/* global WebSocket: false */\n\n'use strict';\n\nfunction WsWrapper(wsurl, logger) {\n\n    var url = wsurl;\n    var sock = new WebSocket(url);\n    var loggerObj = logger;\n\n    var log = function(msg) {\n        if(loggerObj) {\n            loggerObj.log(msg);\n        }\n    };\n\n    var handlers = [];\n\n    sock.onopen = function() {\n        log('Connected to ' + url);\n    };\n\n    sock.onclose = function(e) {\n        log('Connection closed (wasClean = ' + e.wasClean + ', code = ' + e.code + ', reason = \\'' + e.reason + '\\')');\n        sock = null;\n    };\n\n    sock.onmessage = function(e) {\n        log('received ' + e.data);\n        var data = JSON.parse(e.data);\n        handlers.forEach(function(handler) {\n            handler(data);\n        });\n    };\n\n    this.message = function(handler) {\n        if(handler) {\n            handlers.push(handler);\n        }\n    };\n\n}\n\nmodule.exports = {\n    WsWrapper: WsWrapper\n};\n"]}
--- a/annot-server/static/js/annotviz.min.js	Fri Jan 23 18:18:33 2015 +0100
+++ b/annot-server/static/js/annotviz.min.js	Fri Jan 23 23:19:45 2015 +0100
@@ -1,1 +1,1 @@
-!function(t){if("object"==typeof exports&&"undefined"!=typeof module)module.exports=t();else if("function"==typeof define&&define.amd)define([],t);else{var i;"undefined"!=typeof window?i=window:"undefined"!=typeof global?i=global:"undefined"!=typeof self&&(i=self),i.annotviz=t()}}(function(){return function t(i,e,n){function o(r,a){if(!e[r]){if(!i[r]){var h="function"==typeof require&&require;if(!a&&h)return h(r,!0);if(s)return s(r,!0);var l=new Error("Cannot find module '"+r+"'");throw l.code="MODULE_NOT_FOUND",l}var c=e[r]={exports:{}};i[r][0].call(c.exports,function(t){var e=i[r][1][t];return o(e?e:t)},c,c.exports,t,i,e,n)}return e[r].exports}for(var s="function"==typeof require&&require,r=0;r<n.length;r++)o(n[r]);return o}({1:[function(t,i){"use strict";var e=t("./doubleroll"),n=t("./annotsroll"),o=t("./annotstimeline"),s=t("./annotsvizview"),r=t("./stageview"),a=t("./wswrapper"),h=t("./logger"),l=t("./utils"),c=t("lodash");i.exports=c({}).extend(e).extend(n).extend(o).extend(s).extend(r).extend(a).extend(h).extend(l).value()},{"./annotsroll":2,"./annotstimeline":3,"./annotsvizview":4,"./doubleroll":5,"./logger":6,"./stageview":8,"./utils":9,"./wswrapper":10,lodash:"lodash"}],2:[function(t,i){"use strict";function e(t){var i=this,e=o(t).defaults(a).value();this.container=new n.DisplayObjectContainer,this.container.x=e.xInit,this.container.y=e.yInit,this.container.width=e.width,this.height=e.height,this.width=e.width,this.widthRoll=e.widthRoll,this.pixelsPerSecond=e.pixelsPerSecond,this.annotColors=e.annotColors,this.startTs=e.startTs||Date.now(),this.ignoreAnnots=e.ignoreAnnots;var h=e.yInit,l=o(e.annotStyles).defaults(r).value(),c=15;for(var d in l)l[d].wordWrap===!0&&(l[d].wordWrapWidth=this.widthRoll-c);var f,g=!1,u=e.ws,v=e.externalRefresh,p=e.stageView,w=0;p.registerComponent(this);var m=function(t){var i=t.toGlobal(new n.Point(0,0));return i.x+t.width<0||i.y+t.height<0};this.addAnnots=function(t){var i=t.content.category.label,e=t.content.text,n=t.content.user,o=Date.parse(t.ts),s=t.content.color||this.getColor(o,t.content.category.code);this.addAnnot(i,e,n,s,o)},this.getColor=function(t,i){var e;o(this.annotColors).eachRight(function(i){return i.ts<t?(e=i.colors,!1):void 0});var n;return e&&(n=e[i]),n||(n=e.defaultColor||s),n},this.addAnnot=function(t,e,s,r,a){var d=0,g=(a-this.startTs)*this.pixelsPerSecond/1e3+h,u=parseInt(r.replace(/^#/,""),16);if(0===w){var v=(new n.Graphics).beginFill(u).drawRect(d,g,10,3).endFill();this.container.addChild(v);var p=0,m=new n.Text(t,o(l.label).extend({fill:r}).value());if(m.x=d+c,m.y=g-23,this.container.addChild(m),p+=m.height-23+2,e){var x=new n.Text(e,l.text);x.x=d+c,x.y=g+p,this.container.addChild(x),p+=x.height+2}var C=new n.Text(s,l.user);C.x=d+c,C.y=g+p,this.container.addChild(C),p+=C.height+8,this.ignoreAnnots===!0&&(w=p/this.pixelsPerSecond,f=setInterval(function(){i.refreshWait()},1e3))}this.addAnnotLine(u,g)},this.addAnnotLine=function(t,i){var e=this.widthRoll,o=(new n.Graphics).beginFill(t).drawRect(e,i,this.width-e,3).endFill();this.container.addChild(o)},this.moveTo=function(t){this.container.y=Math.floor(t*this.pixelsPerSecond)},this.move=this.refresh=function(){var t=(this.startTs-Date.now())/1e3;this.moveTo(t)},this.refreshWait=function(){w-=1,0>w&&(w=0,clearInterval(f))},this.removePassedObjets=function(){var t=[];o(i.container.children).forEach(function(i){return"undefined"==typeof i||m(i)&&t.push(i)}),t.forEach(function(t){i.container.removeChild(t)})},this.init=function(){u.message(function(t){i.addAnnots(t)})},this.start=function(){g||(this.startTs=Date.now(),g=!0),this.cleanInterval=setInterval(function(){i.removePassedObjets()},1e3*this.height/this.pixelsPerSecond),v||(this.refreshInterval=setInterval(function(){i.move()},1e3/this.framerate))},this.stop=function(){clearInterval(this.cleanInterval),v||clearInterval(this.refreshInterval)}}var n=t("pixi"),o=t("lodash"),s="#bababa",r={label:{font:"16pt Arial Bold",fill:"#65A954",wordWrap:!0},text:{font:"12pt Arial Regular",fill:"#444444",wordWrap:!0},user:{font:"14pt Arial regular",fill:"#666666"}},a={externalRefresh:!1,defaultColor:s,annotStyles:r,ignoreAnnots:!1};i.exports={AnnotsRoll:e}},{lodash:"lodash",pixi:"pixi"}],3:[function(t,i){"use strict";function e(t){var i=this,e=s(t).defaults(r).value();this.container=new n.DisplayObjectContainer,this.container.x=e.xInit,this.container.y=e.yInit,this.container.width=e.width,this.container.height=e.height,this.timeBegin=e.timeBegin,this.timeEnd=e.timeEnd,this.duration=(this.timeEnd-this.timeBegin)/1e3,this.width=e.width,this.height=e.height,this.intervalHeight=e.intervalHeight,this.intervalWidth=e.intervalWidth,this.maxCellHeight=e.maxCellHeight,this.annotCategories=e.annotCategories,this.startTs=t.startTs||Date.now(),this.showClockGraphics=e.showClockGraphics,this.archive=e.archive,this.circleX=e.circleX||this.width/2,this.circleY=e.circleY||this.height/2,this.radius=e.radius;var a=2*Math.PI*this.radius;this.intervalDuration=this.intervalWidth*this.duration/a;var h=e.channel,l=e.eventCode,c=e.serverUrl,d=e.maxPages,f=Math.floor(a/this.intervalWidth);this.cells=[];for(var g=0;g<a/this.intervalWidth;g++)this.cells[g]=[],this.cells[g].i=g,this.cells[g].totalAnnots=0,this.cells[g].categories={};var u=e.ws,v=e.stageView,p=new n.Graphics;p.lineStyle(2,6579300).drawCircle(this.circleX,this.circleY,this.radius-3).endFill(),this.container.addChild(p),v.registerComponent(this);var w=function(){var t=i.timeBegin,e=i.startTs,o=c+"/p/api/v1/annotation",s=[{name:"ts",op:">",val:new Date(t).toISOString()},{name:"ts",op:"<=",val:new Date(e).toISOString()},{name:"channel",op:"==",val:h},{name:"event_code",op:"==",val:l}];o=o+"?q="+JSON.stringify({filters:s});var r=1,a=1,f=function(t){if(t){var e=t.target.json;r=Math.min(d,parseInt(e.total_pages)),e.objects.forEach(function(t){i.addAnnot(t)})}if(r>=a){var s=new n.JsonLoader(o+"&page="+a,!0);s.on("loaded",f),s.load(),a++}};f()};return this.addAnnot=function(t){var i,e=Date.parse(t.ts);if(s(this.annotCategories).eachRight(function(t){return t.ts<e?(i=t,!1):void 0}),this.timeEnd>e){var n=Math.floor((e-this.timeBegin)/(1e3*this.intervalDuration));if("undefined"==typeof this.cells[n].graphics&&this.initCell(this.cells[n],i),"undefined"!=typeof i.colors[t.content.category.code])var o=t.content.category.code;else var o="default";this.cells[n].categories[o].count+=1,this.cells[n].totalAnnots+=1,this.redrawCell(this.cells[n],i)}},this.initClockGraphics=function(){var t=new n.Text(o.formatTime(this.timeBegin),{font:"12pt Gothic Standard",fill:"#646464"});t.x=this.circleX+15,t.y=this.circleY-this.radius-this.maxCellHeight-10,this.container.addChild(t);var i=new n.Text(o.formatTime(this.timeEnd),{font:"12pt Gothic Standard",fill:"#646464"});i.x=this.circleX-15-i.width,i.y=this.circleY-this.radius-this.maxCellHeight-10,this.container.addChild(i);var e=new n.Text(o.formatTime((this.timeEnd-this.timeBegin)/4+this.timeBegin),{font:"12pt Gothic Standard",fill:"#646464"});e.x=this.circleX+this.radius+this.maxCellHeight+10,e.y=this.circleY-e.height,e.rotation=Math.PI/2,this.container.addChild(e);var s=new n.Text(o.formatTime((this.timeEnd-this.timeBegin)/2+this.timeBegin),{font:"12pt Gothic Standard",fill:"#646464"});s.x=this.circleX-s.width/2,s.y=this.circleY+this.radius+this.maxCellHeight-2,this.container.addChild(s);var r=new n.Text(o.formatTime(3*(this.timeEnd-this.timeBegin)/4+this.timeBegin),{font:"12pt Gothic Standard",fill:"#646464"});r.x=this.circleX-this.radius-this.maxCellHeight-10,r.y=this.circleY+e.height,r.rotation=-Math.PI/2,this.container.addChild(r);var a=new n.Graphics;a.lineStyle(1,6579300).moveTo(this.circleX,this.circleY-this.radius/3/2).lineTo(this.circleX,this.circleY-this.radius-this.maxCellHeight-10).endFill(),this.container.addChild(a)},this.redrawCell=function(t,i){var e,n=0;e=t.totalAnnots*this.intervalHeight>this.maxCellHeight?this.maxCellHeight/t.totalAnnots:this.intervalHeight;for(var o=0;o<i.order.length;o++){var s=i.order[o];t.graphics.beginFill(t.categories[s].color.replace("#","0x")).drawRect(0,n,this.intervalWidth-1,-t.categories[s].count*e).endFill(),n-=t.categories[s].count*e}},this.initCell=function(t,i){t.graphics=new n.Graphics,t.graphics.position.x=this.circleX+this.radius*Math.sin(t.i*(360/f)*(Math.PI/180)),t.graphics.position.y=this.circleY-this.radius*Math.cos(t.i*(360/f)*(Math.PI/180)),t.graphics.rotation=t.i*(360/f)*(Math.PI/180)+360/(2*f)*(Math.PI/180),this.container.addChild(t.graphics);for(var e in i.colors)t.categories[e]={count:0,color:i.colors[e]};"undefined"==typeof t.categories["default"]&&(t.categories["default"]={count:0,color:i.defaultColor})},this.init=function(){this.archive||u.message(function(t){i.addAnnot(t)}),this.showClockGraphics&&this.initClockGraphics()},this.start=function(){this.startTs=Date.now(),w()},this.refresh=function(){},this.stop=function(){},this}var n=t("pixi"),o=t("./utils.js"),s=t("lodash"),r={logger:void 0,intervalWidth:10,intervalHeight:5,maxCellHeight:200,radius:300,serverUrl:"http://127.0.0.1:8080",channel:"ANNOT",maxPages:1e3,showClockGraphics:!0,archive:!1};i.exports={AnnotsTimeLine:e}},{"./utils.js":9,lodash:"lodash",pixi:"pixi"}],4:[function(t,i){"use strict";function e(t){var i=this,e=o(t).defaults(l).value();this.container=new n.DisplayObjectContainer,this.container.x=e.xInit,this.container.y=e.yInit,this.width=e.width,this.height=e.height,this.timeBegin=e.timeBegin,this.timeEnd=e.timeEnd,this.annotCategories=[],h.getAnnotCategories(e.urlCategories,this.annotCategories);var c=e.wsPianoroll,d=e.wsAnnot,f=e.stageView,g=(Date.now()+36e5,e.eventCode),u=e.channel;f.registerComponent(this);var v=new r.AnnotsTimeLine({stageView:f,logger:logger,ws:new annotviz.WsWrapper(wsUriAnnotation,logger),xInit:0,yInit:0,width:624,height:568,timeBegin:this.timeBegin,timeEnd:this.timeEnd,intervalWidth:6,intervalHeight:10,maxCellHeight:70,radius:200,eventCode:g,channel:u,annotCategories:this.annotCategories}),p=(new r.AnnotsTimeLine({stageView:f,archive:!0,xInit:0,yInit:0,width:624,height:568,timeBegin:Date.parse("2015-01-22T09:30:00+01:00"),timeEnd:Date.parse("2015-01-22T18:30:00+01:00"),circleX:v.circleX,circleY:v.circleY,intervalWidth:2*v.radius/3*v.intervalWidth/v.radius,intervalHeight:v.intervalHeight*(v.radius-2*v.radius/3)/v.maxCellHeight,maxCellHeight:(v.radius-2*v.radius/3)/4,radius:2*v.radius/3,annotCategories:this.annotCategories,eventCode:g,channel:u,showClockGraphics:!1}),new n.Text("-- : -- : --",{font:"18pt Gothic Standard",fill:"#646464"}));p.x=v.circleX-p.width/2,p.y=v.circleY-p.height/2,this.container.addChild(p);var w=new n.Graphics;w.lineStyle(1,6579300).drawCircle(v.circleX,v.circleY,v.radius/3).endFill(),this.container.addChild(w);var m=(new s.DoubleRoll({stageView:f,logger:logger,ws:c,yInit:this.height-200,sceneHeight:200,pianorolls:[{height:200,timeWidth:10,lineInterval:5e3,noteHeight:10}]}),new s.DoubleRoll({stageView:f,logger:logger,ws:c,orientation:"vertical",sceneHeight:568,pianorolls:[{height:200,timeWidth:60,lineInterval:5e3,noteHeight:5}]})),x=new a.AnnotsRoll({stageView:f,logger:logger,ws:d,parentContainer:m.stage,xInit:624,yInit:568,width:400,height:568,widthRoll:200,framerate:m.framerate,pixelsPerSecond:Math.floor(1024/60),annotColors:this.annotCategories}),C=(new n.Graphics).lineStyle(1,6579300).moveTo(x.container.x,x.container.y).lineTo(x.container.x,x.container.y-x.height).moveTo(x.container.x+x.widthRoll,x.container.y).lineTo(x.container.x+x.widthRoll,x.container.y-x.height).moveTo(0,this.height-200).lineTo(this.width,this.height-200).drawRect(0,0,this.width-1,this.height-1).beginFill(15527148).drawRect(824,0,200,568).endFill();this.container.addChild(C),this.init=function(){},this.updateTime=function(){p.setText(h.formatTime(Date.now()))};var y;return this.start=function(){y=setInterval(function(){i.updateTime()},1e3)},this.refresh=function(){},this.stop=function(){},this}var n=t("pixi"),o=t("lodash"),s=t("./doubleroll.js"),r=t("./annotstimeline.js"),a=t("./annotsroll.js"),h=t("./utils.js"),l={xInit:0,yInit:0,width:1024,height:768};i.exports={AnnotsVizView:e}},{"./annotsroll.js":2,"./annotstimeline.js":3,"./doubleroll.js":5,"./utils.js":9,lodash:"lodash",pixi:"pixi"}],5:[function(t,i){"use strict";function e(t){var i=this,e=o(t).defaults(r).value(),a=e.orientation,h="vertical"!==a;this.logger=e.logger,this.lineColor=e.lineColor,this.lineFillColor=e.lineFillColor,this.framerate=e.framerate,this.offsetMusic=e.offsetMusic,this.noteColors=e.noteColors;var l=e.noteHeight,c=(e.sceneBgColor,e.sceneHeight||o(e.pianorolls).reduce(function(t,i){return t+i.height},0)),d=e.timeWidth,f=e.lineInterval,g=e.offsetMusic,u=e.sceneWidth,v=e.stageView,p=e.zeroShift,w=e.ws,m={};this.container=new n.DisplayObjectContainer,this.container.x=Math.floor(u*p),this.container.y=0,v.registerComponent(this);var x=[],C={parentContainer:this.container,orientation:a,xInit:0,width:u,noteColors:this.noteColors,colorsReg:m,lineColor:this.lineColor,lineInterval:f,offsetMusic:g},y=e.yInit||0,I=!0;o(e.pianorolls).forEach(function(t,r){var a=l||t.noteHeight||t.height/128,h=t.timeWidth||d;if(x.push(new s(o({yInit:y,height:t.height,linesDown:I,pixelsPerSecond:Math.floor(u/h),noteHeight:a,lineInterval:t.lineInterval}).defaults(C).value())),y+=t.height,I=!I,r<e.pianorolls.length-1){var c=(new n.Graphics).beginFill(i.lineFillColor).lineStyle(1,i.lineColor).moveTo(Math.floor(u*p),y).lineTo(-u-Math.floor(u*p),y).endFill();i.container.addChild(c)}}),h||(this.container.rotation=Math.PI/2,this.container.y=c,this.container.x=u),this.init=function(){w.message(function(t){i.addNotes(t)})},this.addNotes=function(t){x.forEach(function(i){i.addNoteRaw(t)})},this.refresh=function(){x.forEach(function(t){t.move()})};var T;return this.start=function(){T=Date.now(),x.forEach(function(t){t.start()})},this.stop=function(){x.forEach(function(t){t.stop()})},this.log=function(t){this.logger&&this.logger.log(t)},this}var n=t("pixi"),o=t("lodash"),s=t("./pianoroll.js"),r={orientation:"horizontal",logger:void 0,sceneWidth:1024,pianorolls:[{height:435,timeWidth:10,lineInterval:5e3,noteHeight:void 0},{height:645,timeWidth:60,lineInterval:5e3,noteHeight:void 0}],framerate:25,offsetMusic:!1,sceneBgColor:16777215,lineColor:4473924,lineFillColor:16776960,noteColors:[12124160,4971889,11506462,1845434,5466513],noteHeight:void 0,zeroShift:.9,timeWidth:60,lineInterval:5e3};i.exports={DoubleRoll:e}},{"./pianoroll.js":7,lodash:"lodash",pixi:"pixi"}],6:[function(t,i){"use strict";function e(t,i){var e=i;"string"==typeof i&&(e=document.getElementById(i)),t||(document.body.removeChild(e),e=void 0),this.log=function(i){t&&e&&(e.innerHTML+=i+"\n",e.scrollTop=e.scrollHeight)}}function n(t){this.log=function(i){t&&console.log(i)}}i.exports={HtmlLogger:e,ConsoleLogger:n}},{}],7:[function(t,i){"use strict";function e(t){var i=this;this.container=new n.DisplayObjectContainer,this.container.x=t.xInit,this.container.y=t.yInit,t.parentContainer.addChild(this.container);var e=t.orientation,a="vertical"!==e;this.linesDown=t.linesDown,this.height=t.height,this.pixelsPerSecond=t.pixelsPerSecond,this.width=t.width,this.noteColors=t.noteColors,this.colorsReg=t.colorsReg||{},this.lineColor=t.lineColor,this.lineInterval=t.lineInterval,this.offsetMusic=t.offsetMusic||!1,this.noteHeight=t.noteHeight,this.noteDict={},this.startTs=t.startTs||Date.now();var h=!1,l=function(t){var i=t.toGlobal(new n.Point(0,0));return i.x+t.width<0||i.y+t.height<0};this.getColor=function(t){var i=this.colorsReg[t];if("undefined"==typeof i){var e=Object.keys(this.colorsReg).length;i=this.colorsReg[t]=e<this.noteColors.length?this.noteColors[e]:parseInt(o({luminosity:"light",hue:"random",format:"hex"}).replace(/^#/,""),16)}return i},this.getNoteRect=function(t,i,e,o,s,r){var a=new n.Graphics;return a.beginFill(e,o),a.drawRect(0,0,s,r),a.endFill(),a.x=t,a.y=i,a.width=s,a.height=r,a},this.addNoteRaw=function(t){var i=t.content[3],e=t.content[4],n=1e3*(t.content[0]-r),o=t.content[2],s=t.content[1];this.addNote(i,n,s,e,o,0)},this.addNote=function(t,i,e,n,o,s){var r=i;this.offsetMusic&&(r=this.startTs+e);var a,h=s,l=n;if(!s)if("undefined"==typeof this.noteDict[o]&&(this.noteDict[o]={}),0===n){if("undefined"!=typeof this.noteDict[o][t]){var c=this.noteDict[o][t];delete this.noteDict[o][t],h=e-c.sessionTs,a=c.graphics,l=c.velocity,r=c.ts}}else h=Date.now()-r,this.noteDict[o][t]={ts:r,velocity:n,sessionTs:e};if(!this.offsetMusic||0===n){var d=h*this.pixelsPerSecond/1e3;if(a)a.width=d;else{var f=(r-this.startTs)*this.pixelsPerSecond/1e3;if(f+d<Math.abs(this.container.x)-this.width)return;var g=Math.floor((128-t+.5)*this.height/128-this.noteHeight/2),u=this.getColor(o),v=l/128;a=this.getNoteRect(f,g,u,v,d,this.noteHeight),this.container.addChild(a)}!s&&n&&(this.noteDict[o][t].graphics=a)}},this.addLine=function(t){"undefined"==typeof t&&(t=new Date);var i=-this.container.x,e=this.linesDown?this.height-20:0,o=(new n.Graphics).beginFill(16776960).lineStyle(1,this.lineColor).moveTo(0,0).lineTo(0,20).endFill();o.x=i,o.y=e,this.container.addChild(o);var s=t.getHours(),r=t.getMinutes(),h=t.getSeconds(),l=(10>s?"0"+s:s)+":"+(10>r?"0"+r:r)+":"+(10>h?"0"+h:h),c={font:"10pt Arial",fill:"#444444"},d=new n.Text(l,c);a?(d.x=i+2,d.y=this.linesDown?this.height-15:2):(d.rotation=-Math.PI/2,d.x=i,d.y=this.linesDown?this.height-2:d.width+2),this.container.addChild(d)},this.moveTo=function(t){var i=this.container.x;this.container.x=Math.floor(t*this.pixelsPerSecond);var e=Math.abs(i-this.container.x);s.forOwn(this.noteDict,function(t){s.forOwn(t,function(t){t.graphics&&(t.graphics.width=t.graphics.width+e)})})},this.move=function(){var t=(this.startTs-Date.now())/1e3;this.moveTo(t)},this.removePassedObjets=function(){var t=[];s(i.container.children).forEach(function(i){return"undefined"==typeof i||l(i)&&t.push(i)}),t.forEach(function(t){i.container.removeChild(t)})},this.start=function(){h||(this.startTs=Date.now(),this.addLine(),h=!0),this.verticalLinesInterval=setInterval(function(){i.addLine()},this.lineInterval),this.cleanInterval=setInterval(function(){i.removePassedObjets()},1e3*this.width/this.pixelsPerSecond)},this.stop=function(){clearInterval(this.verticalLinesInterval),clearInterval(this.cleanInterval)}}var n=t("pixi"),o=t("randomColor"),s=t("lodash"),r=2208988800;i.exports=e},{lodash:"lodash",pixi:"pixi",randomColor:"randomColor"}],8:[function(t,i){"use strict";function e(t){var i=this,e=o(t).defaults(s).value(),r=e.externalRefresh;this.logger=e.logger,this.framerate=e.framerate;var a=e.sceneBgColor,h=e.sceneWidth,l=e.sceneHeight,c=e.canvasContainer,d=[],f=[];this.stage=new n.Stage(a);var g=n.autoDetectRenderer(h,l);this.init=function(){"string"==typeof c&&(c=document.getElementById(c)),"string"==typeof d&&(d=document.getElementById(d)),c.appendChild(g.view),f.forEach(function(t){t.init()})},this.registerTimeContainer=function(t){d.push(t)},this.registerComponent=function(t){f.push(t),this.stage.addChild(t.container)},this.refresh=function(){f.forEach(function(t){t.refresh()}),g.render(this.stage)};var u;return this.start=function(){r||(u=setInterval(function(){i.refresh()},1e3/this.framerate)),f.forEach(function(t){t.start()})},this.stop=function(){r||clearInterval(u),clearInterval(refreshTimeInterval),f.forEach(function(t){t.stop()})},this.log=function(t){this.logger&&this.logger.log(t)},this}var n=t("pixi"),o=t("lodash"),s={externalRefresh:!1,logger:void 0,sceneWidth:1024,sceneHeight:768,framerate:25,sceneBgColor:16777215,canvasContainer:"canvasContainer"};i.exports={StageView:e}},{lodash:"lodash",pixi:"pixi"}],9:[function(t,i){"use strict";function e(t){return a(t).format("HH:mm:ss")}function n(t){var i=/rgba?\(\s*(\d+)\s*,\s*(\d+)\s*,\s*(\d+)/.exec(t);return i?"#"+(1<<24|i[1]<<16|i[2]<<8|i[3]).toString(16).substr(1):t}function o(t,i){var e=new s.JsonLoader(t,!0);e.on("loaded",function(t){for(var e=t.target.json;i.length>0;)i.pop();e.sessions.forEach(function(t){var e={ts:null===t.start_ts?new Date(0):Date.parse(t.start_ts),colors:{}},o=t.categories_json;e.order=o.order,"undefined"==typeof e.order["default"]&&e.order.push("default");for(var s=r.clone(o.categories);s.length>0;){var a=s.pop();a.code&&(e.colors[a.code]=n(a.color)),a.subcategories&&(s=s.concat(a.subcategories))}o.categories.forEach(function(t){t.code&&(e.colors[t.code]=n(t.color))}),e.defaultColor=o.defaultColor||"#536991",i.push(e)})}),e.load()}var s=t("pixi"),r=t("lodash"),a=t("moment");i.exports={formatTime:e,getAnnotCategories:o,colorToHex:n}},{lodash:"lodash",moment:"moment",pixi:"pixi"}],10:[function(t,i){"use strict";function e(t,i){var e=t,n=new WebSocket(e),o=i,s=function(t){o&&o.log(t)},r=[];n.onopen=function(){s("Connected to "+e)},n.onclose=function(t){s("Connection closed (wasClean = "+t.wasClean+", code = "+t.code+", reason = '"+t.reason+"')"),n=null},n.onmessage=function(t){s("received "+t.data);var i=JSON.parse(t.data);r.forEach(function(t){t(i)})},this.message=function(t){t&&r.push(t)}}i.exports={WsWrapper:e}},{}]},{},[1])(1)});
\ No newline at end of file
+!function(t){if("object"==typeof exports&&"undefined"!=typeof module)module.exports=t();else if("function"==typeof define&&define.amd)define([],t);else{var e;"undefined"!=typeof window?e=window:"undefined"!=typeof global?e=global:"undefined"!=typeof self&&(e=self),e.annotviz=t()}}(function(){return function t(e,i,n){function o(r,a){if(!i[r]){if(!e[r]){var h="function"==typeof require&&require;if(!a&&h)return h(r,!0);if(s)return s(r,!0);var l=new Error("Cannot find module '"+r+"'");throw l.code="MODULE_NOT_FOUND",l}var c=i[r]={exports:{}};e[r][0].call(c.exports,function(t){var i=e[r][1][t];return o(i?i:t)},c,c.exports,t,e,i,n)}return i[r].exports}for(var s="function"==typeof require&&require,r=0;r<n.length;r++)o(n[r]);return o}({1:[function(t,e){"use strict";var i=t("./doubleroll"),n=t("./annotsroll"),o=t("./annotstimeline"),s=t("./annotsvizview"),r=t("./stageview"),a=t("./wswrapper"),h=t("./logger"),l=t("./utils"),c=t("lodash");e.exports=c({}).extend(i).extend(n).extend(o).extend(s).extend(r).extend(a).extend(h).extend(l).value()},{"./annotsroll":2,"./annotstimeline":3,"./annotsvizview":4,"./doubleroll":5,"./logger":6,"./stageview":8,"./utils":9,"./wswrapper":10,lodash:"lodash"}],2:[function(t,e){"use strict";function i(t){var e=this,i=o(t).defaults(a).value();this.container=new n.DisplayObjectContainer,this.container.x=i.xInit,this.container.y=i.yInit,this.container.width=i.width,this.height=i.height,this.width=i.width,this.widthRoll=i.widthRoll,this.pixelsPerSecond=i.pixelsPerSecond,this.annotColors=i.annotColors,this.startTs=i.startTs||Date.now(),this.ignoreAnnots=i.ignoreAnnots;var h=i.yInit,l=o(i.annotStyles).defaults(r).value(),c=15;for(var d in l)l[d].wordWrap===!0&&(l[d].wordWrapWidth=this.widthRoll-c);var f,g=!1,u=i.ws,v=i.externalRefresh,p=i.stageView,w=0;p.registerComponent(this);var m=function(t){var e=t.toGlobal(new n.Point(0,0));return e.x+t.width<0||e.y+t.height<0};this.addAnnots=function(t){var e=t.content.category.label,i=t.content.text,n=t.content.user,o=Date.parse(t.ts),s=t.content.color||this.getColor(o,t.content.category.code);this.addAnnot(e,i,n,s,o)},this.getColor=function(t,e){var i;o(this.annotColors).eachRight(function(e){return e.ts<t?(i=e.colors,!1):void 0});var n;return i&&(n=i[e]),n||(n=i.defaultColor||s),n},this.addAnnot=function(t,i,s,r,a){var d=0,g=(a-this.startTs)*this.pixelsPerSecond/1e3+h,u=parseInt(r.replace(/^#/,""),16);if(0===w){var v=(new n.Graphics).beginFill(u).drawRect(d,g,10,3).endFill();this.container.addChild(v);var p=0,m=new n.Text(t,o(l.label).extend({fill:r}).value());if(m.x=d+c,m.y=g-23,this.container.addChild(m),p+=m.height-23+2,i){var C=new n.Text(i,l.text);C.x=d+c,C.y=g+p,this.container.addChild(C),p+=C.height+2}var x=new n.Text(s,l.user);x.x=d+c,x.y=g+p,this.container.addChild(x),p+=x.height+8,this.ignoreAnnots===!0&&(w=p/this.pixelsPerSecond,f=setInterval(function(){e.refreshWait()},1e3))}this.addAnnotLine(u,g)},this.addAnnotLine=function(t,e){var i=this.widthRoll,o=(new n.Graphics).beginFill(t).drawRect(i,e,this.width-i,3).endFill();this.container.addChild(o)},this.moveTo=function(t){this.container.y=Math.floor(t*this.pixelsPerSecond)},this.move=this.refresh=function(){var t=(this.startTs-Date.now())/1e3;this.moveTo(t)},this.refreshWait=function(){w-=1,0>w&&(w=0,clearInterval(f))},this.removePassedObjets=function(){var t=[];o(e.container.children).forEach(function(e){return"undefined"==typeof e||m(e)&&t.push(e)}),t.forEach(function(t){e.container.removeChild(t)})},this.init=function(){u.message(function(t){e.addAnnots(t)})},this.start=function(){g||(this.startTs=Date.now(),g=!0),this.cleanInterval=setInterval(function(){e.removePassedObjets()},1e3*this.height/this.pixelsPerSecond),v||(this.refreshInterval=setInterval(function(){e.move()},1e3/this.framerate))},this.stop=function(){clearInterval(this.cleanInterval),v||clearInterval(this.refreshInterval)}}var n=t("pixi"),o=t("lodash"),s="#bababa",r={label:{font:"16pt Arial Bold",fill:"#65A954",wordWrap:!0},text:{font:"12pt Arial Regular",fill:"#444444",wordWrap:!0},user:{font:"14pt Arial regular",fill:"#666666"}},a={externalRefresh:!1,defaultColor:s,annotStyles:r,ignoreAnnots:!1};e.exports={AnnotsRoll:i}},{lodash:"lodash",pixi:"pixi"}],3:[function(t,e){"use strict";function i(t){var e=this,i=s(t).defaults(r).value();this.container=new n.DisplayObjectContainer,this.container.x=i.xInit,this.container.y=i.yInit,this.container.width=i.width,this.container.height=i.height,this.timeBegin=i.timeBegin,this.timeEnd=i.timeEnd,this.duration=(this.timeEnd-this.timeBegin)/1e3,this.width=i.width,this.height=i.height,this.intervalHeight=i.intervalHeight,this.intervalWidth=i.intervalWidth,this.maxCellHeight=i.maxCellHeight,this.annotCategories=i.annotCategories,this.startTs=t.startTs||Date.now(),this.showClockGraphics=i.showClockGraphics,this.archive=i.archive,this.circleX=i.circleX||this.width/2,this.circleY=i.circleY||this.height/2,this.radius=i.radius;var a=2*Math.PI*this.radius;this.intervalDuration=this.intervalWidth*this.duration/a;var h=i.channel,l=i.eventCode,c=i.serverUrl,d=i.maxPages,f=Math.floor(a/this.intervalWidth);this.cells=[];for(var g=0;g<a/this.intervalWidth;g++)this.cells[g]=[],this.cells[g].i=g,this.cells[g].totalAnnots=0,this.cells[g].categories={};var u=i.ws,v=i.stageView,p=new n.Graphics;p.lineStyle(2,6579300).drawCircle(this.circleX,this.circleY,this.radius-3).endFill(),this.container.addChild(p),v.registerComponent(this);var w=function(){var t=e.timeBegin,i=Math.min(e.timeEnd,e.startTs);console.log("START TS",new Date(t).toISOString()),console.log("END TS",new Date(i).toISOString());var o=c+"/p/api/v1/annotation",s=[{name:"ts",op:">",val:new Date(t).toISOString()},{name:"ts",op:"<=",val:new Date(i).toISOString()},{name:"channel",op:"==",val:h},{name:"event_code",op:"==",val:l}];o=o+"?q="+JSON.stringify({filters:s});var r=1,a=1,f=function(t){if(t){var i=t.target.json;r=Math.min(d,parseInt(i.total_pages)),i.objects.forEach(function(t){e.addAnnot(t)})}if(r>=a){var s=new n.JsonLoader(o+"&page="+a,!0);s.on("loaded",f),s.load(),a++}};f()};return this.addAnnot=function(t){var e,i=Date.parse(t.ts);if(s(this.annotCategories).eachRight(function(t){return t.ts<i?(e=t,!1):void 0}),e&&this.timeEnd>i){var n=Math.floor((i-this.timeBegin)/(1e3*this.intervalDuration));"undefined"==typeof this.cells[n].graphics&&this.initCell(this.cells[n],e);var o;o="undefined"!=typeof e.colors[t.content.category.code]?t.content.category.code:"default",this.cells[n].categories[o].count+=1,this.cells[n].totalAnnots+=1,this.redrawCell(this.cells[n],e)}},this.initClockGraphics=function(){var t=new n.Text(o.formatTime(this.timeBegin),{font:"12pt Gothic Standard",fill:"#646464"});t.x=this.circleX+15,t.y=this.circleY-this.radius-this.maxCellHeight-10,this.container.addChild(t);var e=new n.Text(o.formatTime(this.timeEnd),{font:"12pt Gothic Standard",fill:"#646464"});e.x=this.circleX-15-e.width,e.y=this.circleY-this.radius-this.maxCellHeight-10,this.container.addChild(e);var i=new n.Text(o.formatTime((this.timeEnd-this.timeBegin)/4+this.timeBegin),{font:"12pt Gothic Standard",fill:"#646464"});i.x=this.circleX+this.radius+this.maxCellHeight+10,i.y=this.circleY-i.height,i.rotation=Math.PI/2,this.container.addChild(i);var s=new n.Text(o.formatTime((this.timeEnd-this.timeBegin)/2+this.timeBegin),{font:"12pt Gothic Standard",fill:"#646464"});s.x=this.circleX-s.width/2,s.y=this.circleY+this.radius+this.maxCellHeight-2,this.container.addChild(s);var r=new n.Text(o.formatTime(3*(this.timeEnd-this.timeBegin)/4+this.timeBegin),{font:"12pt Gothic Standard",fill:"#646464"});r.x=this.circleX-this.radius-this.maxCellHeight-10,r.y=this.circleY+i.height,r.rotation=-Math.PI/2,this.container.addChild(r);var a=new n.Graphics;a.lineStyle(1,6579300).moveTo(this.circleX,this.circleY-this.radius/3/2).lineTo(this.circleX,this.circleY-this.radius-this.maxCellHeight-10).endFill(),this.container.addChild(a)},this.redrawCell=function(t,e){var i,n=0;i=t.totalAnnots*this.intervalHeight>this.maxCellHeight?this.maxCellHeight/t.totalAnnots:this.intervalHeight;for(var o=0;o<e.order.length;o++){var s=e.order[o];t.graphics.beginFill(t.categories[s].color.replace("#","0x")).drawRect(0,n,this.intervalWidth-1,-t.categories[s].count*i).endFill(),n-=t.categories[s].count*i}},this.initCell=function(t,e){t.graphics=new n.Graphics,t.graphics.position.x=this.circleX+this.radius*Math.sin(t.i*(360/f)*(Math.PI/180)),t.graphics.position.y=this.circleY-this.radius*Math.cos(t.i*(360/f)*(Math.PI/180)),t.graphics.rotation=t.i*(360/f)*(Math.PI/180)+360/(2*f)*(Math.PI/180),this.container.addChild(t.graphics);for(var i in e.colors)t.categories[i]={count:0,color:e.colors[i]};"undefined"==typeof t.categories["default"]&&(t.categories["default"]={count:0,color:e.defaultColor})},this.init=function(){this.archive||u.message(function(t){e.addAnnot(t)}),this.showClockGraphics&&this.initClockGraphics()},this.start=function(){this.startTs=Date.now(),w()},this.refresh=function(){},this.stop=function(){},this}var n=t("pixi"),o=t("./utils.js"),s=t("lodash"),r={logger:void 0,intervalWidth:10,intervalHeight:5,maxCellHeight:200,radius:300,serverUrl:"http://127.0.0.1:8080",channel:"ANNOT",maxPages:1e3,showClockGraphics:!0,archive:!1};e.exports={AnnotsTimeLine:i}},{"./utils.js":9,lodash:"lodash",pixi:"pixi"}],4:[function(t,e){"use strict";function i(t){var e=this,i=o(t).defaults(l).value();this.container=new n.DisplayObjectContainer,this.container.x=i.xInit,this.container.y=i.yInit,this.width=i.width,this.height=i.height,this.timeBegin=i.timeBegin,this.timeEnd=i.timeEnd,this.annotCategories=[],this.annotCategoriesDay2=[],this.annotCategoriesDay1=[];var c=i.wsPianoroll,d=i.wsAnnot,f=i.stageView,g=(Date.now()+36e5,i.channel),u=i.serverUrl,v=i.eventCode,p=i.eventCodeSessionDay2,w=i.eventCodeSessionDay1;h.getAnnotCategories(i.urlCategories+v,this.annotCategories),h.getAnnotCategories(i.urlCategories+p,this.annotCategoriesDay2),h.getAnnotCategories(i.urlCategories+w,this.annotCategoriesDay1),f.registerComponent(this);var m=new r.AnnotsTimeLine({stageView:f,logger:logger,ws:d,xInit:0,yInit:0,width:624,height:568,timeBegin:this.timeBegin,timeEnd:this.timeEnd,intervalWidth:6,intervalHeight:10,maxCellHeight:70,radius:200,eventCode:v,channel:g,serverUrl:u,annotCategories:this.annotCategories}),C=new r.AnnotsTimeLine({stageView:f,xInit:0,yInit:0,width:624,height:568,timeBegin:Date.parse("2015-01-23T09:30:00+01:00"),timeEnd:Date.parse("2015-01-23T18:30:00+01:00"),circleX:m.circleX,circleY:m.circleY,intervalWidth:2*m.radius/3*m.intervalWidth/m.radius,intervalHeight:m.intervalHeight*(m.radius-2*m.radius/3)/m.maxCellHeight,maxCellHeight:(m.radius-2*m.radius/3)/2,radius:2*m.radius/3,annotCategories:this.annotCategoriesDay2,eventCode:p,channel:g,serverUrl:u,showClockGraphics:!1,archive:!0}),x=(new r.AnnotsTimeLine({stageView:f,xInit:0,yInit:0,width:624,height:568,timeBegin:Date.parse("2015-01-22T09:30:00+01:00"),timeEnd:Date.parse("2015-01-22T18:30:00+01:00"),circleX:m.circleX,circleY:m.circleY,intervalWidth:2*m.radius/3*m.intervalWidth/m.radius,intervalHeight:m.intervalHeight*(m.radius-m.radius/3)/m.maxCellHeight,maxCellHeight:(2*m.radius/3-m.radius/3)/4,radius:m.radius/3,annotCategories:this.annotCategoriesDay1,eventCode:w,channel:g,serverUrl:u,showClockGraphics:!1,archive:!0}),new n.Text("-- : -- : --",{font:"18pt Gothic Standard",fill:"#646464"}));x.x=C.circleX-x.width/2,x.y=C.circleY-x.height/2,this.container.addChild(x);var y=(new s.DoubleRoll({stageView:f,logger:logger,ws:c,yInit:this.height-200,sceneHeight:200,pianorolls:[{height:200,timeWidth:10,lineInterval:5e3,noteHeight:10}]}),new s.DoubleRoll({stageView:f,logger:logger,ws:c,orientation:"vertical",sceneHeight:568,pianorolls:[{height:200,timeWidth:60,lineInterval:5e3,noteHeight:5}]})),I=new a.AnnotsRoll({stageView:f,logger:logger,ws:d,parentContainer:y.stage,xInit:624,yInit:568,width:400,height:568,widthRoll:200,framerate:y.framerate,pixelsPerSecond:Math.floor(1024/60),annotColors:this.annotCategories}),T=(new n.Graphics).lineStyle(1,6579300).moveTo(I.container.x,I.container.y).lineTo(I.container.x,I.container.y-I.height).moveTo(I.container.x+I.widthRoll,I.container.y).lineTo(I.container.x+I.widthRoll,I.container.y-I.height).moveTo(0,this.height-200).lineTo(this.width,this.height-200).drawRect(0,0,this.width-1,this.height-1).beginFill(15527148).drawRect(824,0,200,568).endFill();this.container.addChild(T),this.init=function(){},this.updateTime=function(){x.setText(h.formatTime(Date.now()))};var D;return this.start=function(){D=setInterval(function(){e.updateTime()},1e3)},this.refresh=function(){},this.stop=function(){},this}var n=t("pixi"),o=t("lodash"),s=t("./doubleroll.js"),r=t("./annotstimeline.js"),a=t("./annotsroll.js"),h=t("./utils.js"),l={xInit:0,yInit:0,width:1024,height:768};e.exports={AnnotsVizView:i}},{"./annotsroll.js":2,"./annotstimeline.js":3,"./doubleroll.js":5,"./utils.js":9,lodash:"lodash",pixi:"pixi"}],5:[function(t,e){"use strict";function i(t){var e=this,i=o(t).defaults(r).value(),a=i.orientation,h="vertical"!==a;this.logger=i.logger,this.lineColor=i.lineColor,this.lineFillColor=i.lineFillColor,this.framerate=i.framerate,this.offsetMusic=i.offsetMusic,this.noteColors=i.noteColors;var l=i.noteHeight,c=i.sceneHeight||o(i.pianorolls).reduce(function(t,e){return t+e.height},0),d=i.timeWidth,f=i.lineInterval,g=i.offsetMusic,u=i.sceneWidth,v=i.stageView,p=i.zeroShift,w=i.ws,m={};this.container=new n.DisplayObjectContainer,this.container.x=Math.floor(u*p),this.container.y=0,v.registerComponent(this);var C=[],x={parentContainer:this.container,orientation:a,xInit:0,width:u,noteColors:this.noteColors,colorsReg:m,lineColor:this.lineColor,lineInterval:f,offsetMusic:g},y=i.yInit||0,I=!0;o(i.pianorolls).forEach(function(t,r){var a=l||t.noteHeight||t.height/128,h=t.timeWidth||d;if(C.push(new s(o({yInit:y,height:t.height,linesDown:I,pixelsPerSecond:Math.floor(u/h),noteHeight:a,lineInterval:t.lineInterval}).defaults(x).value())),y+=t.height,I=!I,r<i.pianorolls.length-1){var c=(new n.Graphics).beginFill(e.lineFillColor).lineStyle(1,e.lineColor).moveTo(Math.floor(u*p),y).lineTo(-u-Math.floor(u*p),y).endFill();e.container.addChild(c)}}),h||(this.container.rotation=Math.PI/2,this.container.y=c,this.container.x=u),this.init=function(){w.message(function(t){e.addNotes(t)})},this.addNotes=function(t){C.forEach(function(e){e.addNoteRaw(t)})},this.refresh=function(){C.forEach(function(t){t.move()})};var T;return this.start=function(){T=Date.now(),C.forEach(function(t){t.start()})},this.stop=function(){C.forEach(function(t){t.stop()})},this.log=function(t){this.logger&&this.logger.log(t)},this}var n=t("pixi"),o=t("lodash"),s=t("./pianoroll"),r={orientation:"horizontal",logger:void 0,sceneWidth:1024,pianorolls:[{height:435,timeWidth:10,lineInterval:5e3,noteHeight:void 0},{height:645,timeWidth:60,lineInterval:5e3,noteHeight:void 0}],framerate:25,offsetMusic:!1,sceneBgColor:16777215,lineColor:4473924,lineFillColor:16776960,noteColors:[12124160,4971889,11506462,1845434,5466513],noteHeight:void 0,zeroShift:.9,timeWidth:60,lineInterval:5e3};e.exports={DoubleRoll:i}},{"./pianoroll":7,lodash:"lodash",pixi:"pixi"}],6:[function(t,e){"use strict";function i(t,e){var i=e;"string"==typeof e&&(i=document.getElementById(e)),t||(document.body.removeChild(i),i=void 0),this.log=function(e){t&&i&&(i.innerHTML+=e+"\n",i.scrollTop=i.scrollHeight)}}function n(t){this.log=function(e){t&&console.log(e)}}e.exports={HtmlLogger:i,ConsoleLogger:n}},{}],7:[function(t,e){"use strict";function i(t){var e=this;this.container=new n.DisplayObjectContainer,this.container.x=t.xInit,this.container.y=t.yInit,t.parentContainer.addChild(this.container);var i=t.orientation,a="vertical"!==i;this.linesDown=t.linesDown,this.height=t.height,this.pixelsPerSecond=t.pixelsPerSecond,this.width=t.width,this.noteColors=t.noteColors,this.colorsReg=t.colorsReg||{},this.lineColor=t.lineColor,this.lineInterval=t.lineInterval,this.offsetMusic=t.offsetMusic||!1,this.noteHeight=t.noteHeight,this.noteDict={},this.startTs=t.startTs||Date.now();var h=!1,l=function(t){var e=t.toGlobal(new n.Point(0,0));return e.x+t.width<0||e.y+t.height<0};this.getColor=function(t){var e=this.colorsReg[t];if("undefined"==typeof e){var i=Object.keys(this.colorsReg).length;e=this.colorsReg[t]=i<this.noteColors.length?this.noteColors[i]:parseInt(o({luminosity:"light",hue:"random",format:"hex"}).replace(/^#/,""),16)}return e},this.getNoteRect=function(t,e,i,o,s,r){var a=new n.Graphics;return a.beginFill(i,o),a.drawRect(0,0,s,r),a.endFill(),a.x=t,a.y=e,a.width=s,a.height=r,a},this.addNoteRaw=function(t){var e=t.content[3],i=t.content[4],n=1e3*(t.content[0]-r),o=t.content[2],s=t.content[1];this.addNote(e,n,s,i,o,0)},this.addNote=function(t,e,i,n,o,s){var r=e;this.offsetMusic&&(r=this.startTs+i);var a,h=s,l=n;if(!s)if("undefined"==typeof this.noteDict[o]&&(this.noteDict[o]={}),0===n){if("undefined"!=typeof this.noteDict[o][t]){var c=this.noteDict[o][t];delete this.noteDict[o][t],h=i-c.sessionTs,a=c.graphics,l=c.velocity,r=c.ts}}else h=Math.abs(Date.now()-r),this.noteDict[o][t]={ts:r,velocity:n,sessionTs:i};if(!this.offsetMusic||0===n){var d=h*this.pixelsPerSecond/1e3;if(a)a.width=d;else{var f=(r-this.startTs)*this.pixelsPerSecond/1e3;if(f+d<Math.abs(this.container.x)-this.width)return;var g=Math.floor((128-t+.5)*this.height/128-this.noteHeight/2),u=this.getColor(o),v=l/128;a=this.getNoteRect(f,g,u,v,d,this.noteHeight),this.container.addChild(a)}!s&&n&&(this.noteDict[o][t].graphics=a)}},this.addLine=function(t){"undefined"==typeof t&&(t=new Date);var e=-this.container.x,i=this.linesDown?this.height-20:0,o=(new n.Graphics).beginFill(16776960).lineStyle(1,this.lineColor).moveTo(0,0).lineTo(0,20).endFill();o.x=e,o.y=i,this.container.addChild(o);var s=t.getHours(),r=t.getMinutes(),h=t.getSeconds(),l=(10>s?"0"+s:s)+":"+(10>r?"0"+r:r)+":"+(10>h?"0"+h:h),c={font:"10pt Arial",fill:"#444444"},d=new n.Text(l,c);a?(d.x=e+2,d.y=this.linesDown?this.height-15:2):(d.rotation=-Math.PI/2,d.x=e,d.y=this.linesDown?this.height-2:d.width+2),this.container.addChild(d)},this.moveTo=function(t){var e=this.container.x;this.container.x=Math.floor(t*this.pixelsPerSecond);var i=Math.abs(e-this.container.x);s.forOwn(this.noteDict,function(t){s.forOwn(t,function(t){t.graphics&&(t.graphics.width=t.graphics.width+i)})})},this.move=function(){var t=(this.startTs-Date.now())/1e3;this.moveTo(t)},this.removePassedObjets=function(){var t=[];s(e.container.children).forEach(function(e){return"undefined"==typeof e||l(e)&&t.push(e)}),t.forEach(function(t){e.container.removeChild(t)})},this.start=function(){h||(this.startTs=Date.now(),this.addLine(),h=!0),this.verticalLinesInterval=setInterval(function(){e.addLine()},this.lineInterval),this.cleanInterval=setInterval(function(){e.removePassedObjets()},1e3*this.width/this.pixelsPerSecond)},this.stop=function(){clearInterval(this.verticalLinesInterval),clearInterval(this.cleanInterval)}}var n=t("pixi"),o=t("randomColor"),s=t("lodash"),r=2208988800;e.exports=i},{lodash:"lodash",pixi:"pixi",randomColor:"randomColor"}],8:[function(t,e){"use strict";function i(t){var e=this,i=o(t).defaults(s).value(),r=i.externalRefresh;this.logger=i.logger,this.framerate=i.framerate;var a=i.sceneBgColor,h=i.sceneWidth,l=i.sceneHeight,c=i.canvasContainer,d=[],f=[];this.stage=new n.Stage(a);var g=n.autoDetectRenderer(h,l);this.init=function(){"string"==typeof c&&(c=document.getElementById(c)),"string"==typeof d&&(d=document.getElementById(d)),c.appendChild(g.view),f.forEach(function(t){t.init()})},this.registerTimeContainer=function(t){d.push(t)},this.registerComponent=function(t){f.push(t),this.stage.addChild(t.container)},this.refresh=function(){f.forEach(function(t){t.refresh()}),g.render(this.stage)};var u;return this.start=function(){r||(u=setInterval(function(){e.refresh()},1e3/this.framerate)),f.forEach(function(t){t.start()})},this.stop=function(){r||clearInterval(u),f.forEach(function(t){t.stop()})},this.log=function(t){this.logger&&this.logger.log(t)},this}var n=t("pixi"),o=t("lodash"),s={externalRefresh:!1,logger:void 0,sceneWidth:1024,sceneHeight:768,framerate:25,sceneBgColor:16777215,canvasContainer:"canvasContainer"};e.exports={StageView:i}},{lodash:"lodash",pixi:"pixi"}],9:[function(t,e){"use strict";function i(t){return a(t).format("HH:mm:ss")}function n(t){var e=/rgba?\(\s*(\d+)\s*,\s*(\d+)\s*,\s*(\d+)/.exec(t);return e?"#"+(1<<24|e[1]<<16|e[2]<<8|e[3]).toString(16).substr(1):t}function o(t,e){var i=new s.JsonLoader(t,!0);i.on("loaded",function(t){for(var i=t.target.json;e.length>0;)e.pop();i.sessions.forEach(function(t){var i={ts:null===t.start_ts?new Date(0):Date.parse(t.start_ts),colors:{}},o=t.categories_json;i.order=o.order,"undefined"==typeof i.order["default"]&&i.order.push("default");for(var s=r.clone(o.categories);s.length>0;){var a=s.pop();a.code&&(i.colors[a.code]=n(a.color)),a.subcategories&&(s=s.concat(a.subcategories))}o.categories.forEach(function(t){t.code&&(i.colors[t.code]=n(t.color))}),i.defaultColor=o.defaultColor||"#536991",e.push(i)})}),i.load()}var s=t("pixi"),r=t("lodash"),a=t("moment");e.exports={formatTime:i,getAnnotCategories:o,colorToHex:n}},{lodash:"lodash",moment:"moment",pixi:"pixi"}],10:[function(t,e){"use strict";function i(t,e){var i=t,n=new WebSocket(i),o=e,s=function(t){o&&o.log(t)},r=[];n.onopen=function(){s("Connected to "+i)},n.onclose=function(t){s("Connection closed (wasClean = "+t.wasClean+", code = "+t.code+", reason = '"+t.reason+"')"),n=null},n.onmessage=function(t){s("received "+t.data);var e=JSON.parse(t.data);r.forEach(function(t){t(e)})},this.message=function(t){t&&r.push(t)}}e.exports={WsWrapper:i}},{}]},{},[1])(1)});
\ No newline at end of file
--- a/annot-server/webapp/templates/annotviz.html	Fri Jan 23 18:18:33 2015 +0100
+++ b/annot-server/webapp/templates/annotviz.html	Fri Jan 23 23:19:45 2015 +0100
@@ -8,11 +8,6 @@
     <h1>Piano Roll {{event.label}}</h1>
     <noscript>You must enable JavaScript</noscript>
     <div id="canvasContainer"></div>
-    <p>
-        <a href="#" onclick="stop(); return false;">stop intervals</a> -
-        <a href="#" onclick="start(); return false;">start intervals</a> -
-        temps écoulé : <span id="timeStarted"></span>
-    </p>
     <pre id="log"></pre>
     <script src="{{ config['STATIC_URL'] }}/js/libs-annotviz.min.js"></script>
     <script src="{{ config['STATIC_URL'] }}/js/annotviz.min.js"></script>
@@ -21,6 +16,9 @@
     var pianorollChannel  = 'PIANOROLL';
     var annotationChannel = 'ANNOT';
     var eventCode = '{{event.code}}';
+    var eventCodeSessionDay2 = 'atelier2_jour2';
+    var eventCodeSessionDay1 = 'atelier2';
+    
     var wsUri = 'ws://' + window.location.hostname + ':8090/broadcast';
 
     wsUriPianoroll = wsUri + '?channel=' + pianorollChannel + '&event_code=' + eventCode;
@@ -31,19 +29,21 @@
     var stageView = new annotviz.StageView({
         logger: logger
     });
+    console.log("COUCOU");
 
     var serverUrl = "http://" + window.location.hostname + ":8080";
     var annotsvizview = new annotviz.AnnotsVizView({
         urlCategories: serverUrl + "/p/api/v1/event/",
-        serverUrl: serverUrl,
         logger: logger,
         stageView: stageView,
+        serverUrl: serverUrl,
         wsPianoroll: new annotviz.WsWrapper(wsUriPianoroll, logger),
         wsAnnot: new annotviz.WsWrapper(wsUriAnnotation, logger),
-        timeBegin: Date.parse("2015-01-23T09:30:00+01:00"),
-        timeEnd: Date.parse("2015-01-23T18:30:00+01:00"),
+        timeBegin: Date.parse("2015-01-24T09:30:00+01:00"),
+        timeEnd: Date.parse("2015-01-24T18:30:00+01:00"),
         eventCode: eventCode,
-        eventCodeSessionDay1: "atelier2",
+        eventCodeSessionDay2: eventCodeSessionDay2,
+        eventCodeSessionDay1: eventCodeSessionDay1,
         channel: annotationChannel
     });