|
1 !function(e){if("object"==typeof exports&&"undefined"!=typeof module)module.exports=e();else if("function"==typeof define&&define.amd)define([],e);else{var n;"undefined"!=typeof window?n=window:"undefined"!=typeof global?n=global:"undefined"!=typeof self&&(n=self),n.annotviz=e()}}(function(){var define,module,exports;return (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})({1:[function(require,module,exports){ |
|
2 /** |
|
3 * scripts/main.js |
|
4 * |
|
5 * This is the starting point for your application. |
|
6 * Take a look at http://browserify.org/ for more info |
|
7 */ |
|
8 |
|
9 'use strict'; |
|
10 |
|
11 var doubleroll = require('./doubleroll'); |
|
12 var annotsroll = require('./annotsroll'); |
|
13 var annotstimeline = require('./annotstimeline'); |
|
14 var annotsvizview = require('./annotsvizview'); |
|
15 var stageview = require('./stageview'); |
|
16 var wswrapper = require('./wswrapper'); |
|
17 var logger = require('./logger'); |
|
18 |
|
19 var _ = require('lodash'); |
|
20 |
|
21 module.exports = _({}) |
|
22 .extend(doubleroll) |
|
23 .extend(annotsroll) |
|
24 .extend(annotstimeline) |
|
25 .extend(annotsvizview) |
|
26 .extend(stageview) |
|
27 .extend(wswrapper) |
|
28 .extend(logger) |
|
29 .value(); |
|
30 },{"./annotsroll":2,"./annotstimeline":3,"./annotsvizview":4,"./doubleroll":5,"./logger":6,"./stageview":8,"./wswrapper":10,"lodash":"lodash"}],2:[function(require,module,exports){ |
|
31 /** |
|
32 * js/annotsRoll.js |
|
33 * |
|
34 * annotsRoll basic component |
|
35 * |
|
36 */ |
|
37 |
|
38 'use strict'; |
|
39 |
|
40 var PIXI = require('pixi'); |
|
41 var _ = require('lodash'); |
|
42 |
|
43 var DEFAULT_ANNOT_COLOR = '#bababa'; |
|
44 |
|
45 var defaultAnnotStyles = { |
|
46 'label': { font: '16pt Arial Bold', fill: '#65A954', wordWrap: true}, |
|
47 'text' : { font: '12pt Arial Regular', fill: '#444444', wordWrap: true}, |
|
48 'user' : { font: '14pt Arial regular', fill: '#666666' }, |
|
49 }; |
|
50 |
|
51 var defaultOptions = { |
|
52 externalRefresh: false, |
|
53 defaultColor: DEFAULT_ANNOT_COLOR, |
|
54 annotStyles: defaultAnnotStyles |
|
55 }; |
|
56 |
|
57 function AnnotsRoll(options) { |
|
58 |
|
59 //parentContainer, xInit, yInit, width, height, widthRoll, pixelsPerSecond, annotColors |
|
60 var _this = this; |
|
61 var opts = _(options).defaults(defaultOptions).value(); |
|
62 |
|
63 |
|
64 this.container = new PIXI.DisplayObjectContainer(); |
|
65 this.container.x = opts.xInit; |
|
66 this.container.y = opts.yInit; |
|
67 this.container.width = opts.width; |
|
68 |
|
69 this.height = opts.height; |
|
70 this.width = opts.width; |
|
71 this.widthRoll = opts.widthRoll; |
|
72 this.pixelsPerSecond = opts.pixelsPerSecond; |
|
73 this.annotColors = opts.annotColors; |
|
74 this.startTs = options.startTs || Date.now(); |
|
75 |
|
76 var yInit = opts.yInit; |
|
77 var annotStyles = _(opts.annotStyles).defaults(defaultAnnotStyles).value(); |
|
78 for(var style in annotStyles) { |
|
79 if (annotStyles[style].wordWrap === true){ |
|
80 annotStyles[style].wordWrapWidth = this.widthRoll; |
|
81 } |
|
82 } |
|
83 var started = false; |
|
84 var ws = opts.ws; |
|
85 var externalRefresh = opts.externalRefresh; |
|
86 var stageView = opts.stageView; |
|
87 |
|
88 stageView.registerComponent(this); |
|
89 |
|
90 var isHidden = function(child) { |
|
91 // TODO: the origin point is an approximation. Should refine this |
|
92 var globalPos = child.toGlobal(new PIXI.Point(0,0)); |
|
93 return ((globalPos.x + child.width) < 0) || ((globalPos.y + child.height) < 0) ; |
|
94 }; |
|
95 |
|
96 this.addAnnots = function(data) { |
|
97 |
|
98 //var title = data.content.category.label; |
|
99 //var user = data.content.user; |
|
100 //Test cat and color |
|
101 //var colorAnnot = 0x65A954; |
|
102 var category = data.content.category.label, |
|
103 text = data.content.text, |
|
104 user = data.content.user, |
|
105 ts = Date.parse(data.ts), |
|
106 color = data.content.color || this.getColor(ts, data.content.category.code); |
|
107 |
|
108 this.addAnnot(category, text, user, color, ts); |
|
109 }; |
|
110 |
|
111 this.getColor = function(ts, code) { |
|
112 var colorsDef; |
|
113 _(this.annotColors).eachRight(function(cdef) { |
|
114 console.log("cDef", cdef); |
|
115 console.log("cDef ts", cdef.ts, ts); |
|
116 if(cdef.ts < ts) { |
|
117 colorsDef = cdef.colors; |
|
118 return false; |
|
119 } |
|
120 }); |
|
121 var resColor; |
|
122 console.log("colorsDef", colorsDef); |
|
123 if(colorsDef) { |
|
124 resColor = colorsDef[code]; |
|
125 } |
|
126 if(!resColor) { |
|
127 resColor = DEFAULT_ANNOT_COLOR; |
|
128 } |
|
129 return resColor; |
|
130 } |
|
131 |
|
132 this.addAnnot = function(category, text, user, catColor, ts){ |
|
133 |
|
134 var color = catColor ? catColor : DEFAULT_ANNOT_COLOR; |
|
135 var x = 0; |
|
136 var y = (ts-this.startTs) * this.pixelsPerSecond / 1000 + yInit; |
|
137 |
|
138 var colorHex = parseInt(color.replace(/^#/, ''), 16); |
|
139 |
|
140 var graphics = new PIXI.Graphics() |
|
141 .beginFill(colorHex) |
|
142 .drawRect(x, y, 10, 3) |
|
143 .endFill(); |
|
144 |
|
145 this.container.addChild(graphics); |
|
146 |
|
147 var textHeight = 0; |
|
148 var catLabel = new PIXI.Text( |
|
149 category, |
|
150 _(annotStyles.label).extend({fill: color}).value() |
|
151 ); |
|
152 catLabel.x = x + 20; |
|
153 catLabel.y = y - 23; |
|
154 this.container.addChild(catLabel); |
|
155 textHeight += (catLabel.height - 23 + 2); |
|
156 |
|
157 if(text) { |
|
158 var catText = new PIXI.Text(text, annotStyles.text); |
|
159 catText.x = x + 20; |
|
160 catText.y = y + 2; |
|
161 this.container.addChild(catText); |
|
162 textHeight += (catText.height + 2); |
|
163 } |
|
164 |
|
165 var catUser = new PIXI.Text(user, annotStyles.user); |
|
166 catUser.x = x + 20; |
|
167 catUser.y = y + 2 + textHeight; |
|
168 this.container.addChild(catUser); |
|
169 |
|
170 this.addAnnotLine(colorHex, y); |
|
171 }; |
|
172 |
|
173 this.addAnnotLine = function(color, y) { |
|
174 var x = this.widthRoll; |
|
175 |
|
176 |
|
177 var graphics = new PIXI.Graphics() |
|
178 .beginFill(color) |
|
179 .drawRect(x, y, this.width - x, 3) |
|
180 .endFill(); |
|
181 |
|
182 this.container.addChild(graphics); |
|
183 }; |
|
184 |
|
185 this.moveTo = function(diffTime){ |
|
186 this.container.y = Math.floor(diffTime*this.pixelsPerSecond); |
|
187 }; |
|
188 |
|
189 this.move = this.refresh = function() { |
|
190 var diff = (this.startTs - Date.now())/1000; |
|
191 this.moveTo(diff); |
|
192 }; |
|
193 |
|
194 this.removePassedObjets = function(){ |
|
195 var childrenToRemove = []; |
|
196 _(_this.container.children).forEach(function(child) { |
|
197 return typeof(child) === 'undefined' || |
|
198 (isHidden(child) && childrenToRemove.push(child)); |
|
199 }); |
|
200 childrenToRemove.forEach(function(child) { |
|
201 _this.container.removeChild(child); |
|
202 }); |
|
203 }; |
|
204 |
|
205 this.init = function() { |
|
206 |
|
207 ws.message(function(data) { |
|
208 _this.addAnnots(data); |
|
209 }); |
|
210 |
|
211 }; |
|
212 |
|
213 this.start = function() { |
|
214 if(!started) { |
|
215 this.startTs = Date.now(); |
|
216 started = true; |
|
217 } |
|
218 this.cleanInterval = setInterval(function () { _this.removePassedObjets(); }, 1000 * this.height / this.pixelsPerSecond ); |
|
219 if(!externalRefresh) { |
|
220 this.refreshInterval = setInterval(function() {_this.move();}, 1000/this.framerate); |
|
221 } |
|
222 }; |
|
223 |
|
224 this.stop = function() { |
|
225 clearInterval(this.cleanInterval); |
|
226 if(!externalRefresh) { |
|
227 clearInterval(this.refreshInterval); |
|
228 } |
|
229 }; |
|
230 |
|
231 } |
|
232 |
|
233 module.exports = { |
|
234 AnnotsRoll: AnnotsRoll, |
|
235 }; |
|
236 |
|
237 },{"lodash":"lodash","pixi":"pixi"}],3:[function(require,module,exports){ |
|
238 /** |
|
239 * js/annotstimeline |
|
240 * |
|
241 * annotstimeline basic component |
|
242 * |
|
243 */ |
|
244 |
|
245 'use strict'; |
|
246 |
|
247 var PIXI = require('pixi'); |
|
248 var Utils = require('./utils.js'); |
|
249 var _ = require('lodash'); |
|
250 |
|
251 var defaultOptions = { |
|
252 logger: undefined, |
|
253 intervalWidth: 10, |
|
254 intervalHeight: 5, |
|
255 maxCellHeight: 200, |
|
256 radius: 300 |
|
257 }; |
|
258 |
|
259 |
|
260 function AnnotsTimeLine(options){ |
|
261 var _this = this; |
|
262 var opts = _(options).defaults(defaultOptions).value(); |
|
263 |
|
264 this.container = new PIXI.DisplayObjectContainer(); |
|
265 this.container.x = opts.xInit; |
|
266 this.container.y = opts.yInit; |
|
267 this.container.width = opts.width; |
|
268 this.container.height = opts.height; |
|
269 |
|
270 this.timeBegin = opts.timeBegin; |
|
271 this.timeEnd = opts.timeEnd; |
|
272 this.duration = (this.timeEnd - this.timeBegin)/1000; |
|
273 this.width = opts.width; |
|
274 this.height = opts.height; |
|
275 this.intervalHeight = opts.intervalHeight; |
|
276 this.intervalWidth = opts.intervalWidth; |
|
277 this.maxCellHeight = opts.maxCellHeight; |
|
278 this.annotCategories = opts.annotCategories; |
|
279 |
|
280 this.circleX = opts.circleX || (this.width/2); |
|
281 this.circleY = opts.circleY || (this.height/2); |
|
282 this.radius = opts.radius; |
|
283 this.perimeter = 2*Math.PI* this.radius; |
|
284 this.intervalDuration = (this.intervalWidth * this.duration / this.perimeter); |
|
285 |
|
286 var currentTime = this.timeBegin; |
|
287 var totalIndex = Math.floor(this.perimeter/this.intervalWidth); |
|
288 |
|
289 this.cells = [] |
|
290 for (var i=0; i<(this.perimeter/this.intervalWidth) ; i++){ |
|
291 this.cells[i] = []; |
|
292 this.cells[i].i = i; |
|
293 this.cells[i].totalAnnots = 0; |
|
294 this.cells[i].categories = {}; |
|
295 |
|
296 for (var category in this.annotCategories[0].colors){ |
|
297 this.cells[i].categories[category] = { |
|
298 "count": 0, |
|
299 "color": this.annotCategories[0].colors[category] |
|
300 }; |
|
301 } |
|
302 } |
|
303 |
|
304 var ws = opts.ws; |
|
305 var stageView = opts.stageView; |
|
306 |
|
307 //draw the base - circle and line to locate the scene |
|
308 var graphics = new PIXI.Graphics(); |
|
309 graphics.lineStyle(2, 0x646464) |
|
310 .drawCircle(this.circleX, this.circleY, this.radius - 3) |
|
311 .lineStyle(1, 0xD7D7D7) |
|
312 .drawCircle(this.circleX, this.circleY, this.radius*2/3) |
|
313 .drawCircle(this.circleX, this.circleY, this.radius/3) |
|
314 .lineStyle(1, 0x646464) |
|
315 .moveTo(this.circleX, this.circleY - (this.radius/3)/2) |
|
316 .lineTo(this.circleX, this.circleY - this.radius - this.maxCellHeight - 10) |
|
317 .endFill() |
|
318 this.container.addChild(graphics); |
|
319 |
|
320 //set time text |
|
321 //TODO : move this to annotsvizview |
|
322 var currentTimeText = new PIXI.Text("-- : -- : --", { font: '18pt Gothic Standard', fill: '#646464' }); |
|
323 currentTimeText.x = this.circleX - currentTimeText.width/2; |
|
324 currentTimeText.y = this.circleY - currentTimeText.height/2; |
|
325 this.container.addChild(currentTimeText); |
|
326 |
|
327 stageView.registerComponent(this); |
|
328 |
|
329 //Add Annotation to the TimeLine |
|
330 this.addAnnot = function(data){ |
|
331 if (typeof(this.annotCategories[0].colors[data.content.category.code]) !== 'undefined'){ |
|
332 var annotCode = data.content.category.code; |
|
333 } else { |
|
334 var annotCode = this.annotCategories[0].order[this.annotCategories[0].order.length -1]; |
|
335 } |
|
336 var annotTime = Date.parse(data.ts); |
|
337 |
|
338 if (this.timeEnd > Date.parse(data.ts)){ |
|
339 var i = Math.floor((Date.parse(data.ts)-this.timeBegin)/(1000*this.intervalDuration)); |
|
340 |
|
341 this.cells[i].categories[annotCode].count += 1; |
|
342 this.cells[i].totalAnnots +=1; |
|
343 this.redrawCell(this.cells[i], i); |
|
344 } |
|
345 }; |
|
346 |
|
347 this.initGraphics = function(cell){ |
|
348 cell.graphics = new PIXI.Graphics(); |
|
349 cell.graphics.position.x = this.circleX + this.radius * Math.sin(cell.i*(360/totalIndex)*(Math.PI/180)); |
|
350 cell.graphics.position.y = this.circleY - this.radius * Math.cos(cell.i*(360/totalIndex)*(Math.PI/180)); |
|
351 cell.graphics.rotation = (cell.i)*(360/totalIndex)*(Math.PI/180) + (360/(totalIndex*2))*(Math.PI/180); |
|
352 this.container.addChild(cell.graphics); |
|
353 } |
|
354 |
|
355 this.initTimeTexts = function() { |
|
356 var tBeg = new PIXI.Text(Utils.formatTime(this.timeBegin), { font: '12pt Gothic Standard', fill: '#646464' }); |
|
357 tBeg.x = this.circleX + 15; |
|
358 tBeg.y = this.circleY - this.radius - this.maxCellHeight - 10; |
|
359 this.container.addChild(tBeg); |
|
360 |
|
361 var tEnd = new PIXI.Text(Utils.formatTime(this.timeEnd), { font: '12pt Gothic Standard', fill: '#646464' }); |
|
362 tEnd.x = this.circleX - 15 - tEnd.width; |
|
363 tEnd.y = this.circleY - this.radius - this.maxCellHeight - 10; |
|
364 this.container.addChild(tEnd); |
|
365 |
|
366 var t15 = new PIXI.Text(Utils.formatTime(((this.timeEnd - this.timeBegin)/4) + this.timeBegin), { font: '12pt Gothic Standard', fill: '#646464' }); |
|
367 t15.x = this.circleX + this.radius + this.maxCellHeight + 10 ; |
|
368 t15.y = this.circleY - t15.height; |
|
369 t15.rotation = Math.PI /2; |
|
370 this.container.addChild(t15); |
|
371 |
|
372 var t30 = new PIXI.Text(Utils.formatTime(((this.timeEnd - this.timeBegin)/2) + this.timeBegin), { font: '12pt Gothic Standard', fill: '#646464' }); |
|
373 t30.x = this.circleX - t30.width/2; |
|
374 t30.y = this.circleY + this.radius + this.maxCellHeight - 2; |
|
375 this.container.addChild(t30); |
|
376 |
|
377 var t45 = new PIXI.Text(Utils.formatTime(((this.timeEnd - this.timeBegin)*3/4) + this.timeBegin), { font: '12pt Gothic Standard', fill: '#646464' }); |
|
378 t45.x = this.circleX - this.radius - this.maxCellHeight - 10 ; |
|
379 t45.y = this.circleY + t15.height; |
|
380 t45.rotation = -Math.PI/2; |
|
381 this.container.addChild(t45); |
|
382 } |
|
383 |
|
384 //Draw the cellule |
|
385 this.redrawCell = function(cell){ |
|
386 |
|
387 if (typeof(cell.graphics) === 'undefined'){ |
|
388 this.initGraphics(cell); |
|
389 } else { |
|
390 cell.graphics.clear(); |
|
391 } |
|
392 |
|
393 var y = 0; |
|
394 |
|
395 //Check if total height is higher than Max Cell Height |
|
396 if ((cell.totalAnnots*this.intervalHeight) > this.maxCellHeight){ |
|
397 var heightStep = this.maxCellHeight/cell.totalAnnots; |
|
398 } else { |
|
399 var heightStep = this.intervalHeight; |
|
400 } |
|
401 |
|
402 //Draw the rect depending on the height step calculated |
|
403 for (var i=0; i< this.annotCategories[0].order.length; i++){ |
|
404 var currentCode = this.annotCategories[0].order[i]; |
|
405 cell.graphics.beginFill(cell.categories[currentCode].color.replace("#", "0x")) |
|
406 .drawRect(0, y, this.intervalWidth-1, -cell.categories[currentCode].count * heightStep) |
|
407 .endFill(); |
|
408 y -= cell.categories[currentCode].count*heightStep; |
|
409 } |
|
410 } |
|
411 |
|
412 this.init = function() { |
|
413 ws.message(function(data) { |
|
414 _this.addAnnot(data); |
|
415 }); |
|
416 |
|
417 this.initTimeTexts(); |
|
418 }; |
|
419 |
|
420 this.updateTime = function(){ |
|
421 currentTime += 1000; |
|
422 |
|
423 var nbSec = currentTime / 1000; |
|
424 var hours = Math.floor( nbSec / 3600 ) % 24; |
|
425 var minutes = Math.floor( nbSec / 60 ) % 60; |
|
426 var seconds = Math.floor(nbSec % 60); |
|
427 var timeStr = (hours < 10 ? '0' + hours : hours) + ':' + (minutes < 10 ? '0' + minutes : minutes) + ':' + (seconds < 10 ? '0' + seconds : seconds); |
|
428 |
|
429 currentTimeText.setText(timeStr); |
|
430 }; |
|
431 |
|
432 var refreshTimeInterval; |
|
433 |
|
434 this.start = function() { |
|
435 refreshTimeInterval = setInterval(function() {_this.updateTime();}, 1000); |
|
436 }; |
|
437 |
|
438 this.refresh = function() { |
|
439 |
|
440 }; |
|
441 |
|
442 this.stop = function(){ |
|
443 console.log(this.cells); |
|
444 }; |
|
445 |
|
446 return this; |
|
447 } |
|
448 |
|
449 module.exports = { |
|
450 AnnotsTimeLine: AnnotsTimeLine |
|
451 }; |
|
452 |
|
453 },{"./utils.js":9,"lodash":"lodash","pixi":"pixi"}],4:[function(require,module,exports){ |
|
454 /** |
|
455 * js/annotsvizview.js |
|
456 * |
|
457 * This is the starting point for your application. |
|
458 * Take a look at http://browserify.org/ for more info |
|
459 */ |
|
460 |
|
461 'use strict'; |
|
462 |
|
463 var PIXI = require('pixi'); |
|
464 var _ = require('lodash'); |
|
465 var DoubleRoll = require('./doubleroll.js'); |
|
466 var AnnotsTimeLine = require('./annotstimeline.js'); |
|
467 var AnnotsRoll = require('./annotsroll.js'); |
|
468 |
|
469 var defaultOptions = { |
|
470 xInit: 0, |
|
471 yInit: 0, |
|
472 width: 1024, |
|
473 height: 768, |
|
474 annotCategories: [ |
|
475 { |
|
476 "ts": 1421928213000, |
|
477 "colors": { |
|
478 "transgressions": "#b90000", |
|
479 "rythmique": "#af931e", |
|
480 "narration": "#4bdd71", |
|
481 "relation": "#1c28ba" |
|
482 }, |
|
483 "order": [ |
|
484 "transgressions", |
|
485 "rythmique", |
|
486 "narration", |
|
487 "relation" |
|
488 ], |
|
489 "defaultColor": "#536991" |
|
490 }] |
|
491 }; |
|
492 |
|
493 function AnnotsVizView(options){ |
|
494 var _this = this; |
|
495 var opts = _(options).defaults(defaultOptions).value(); |
|
496 |
|
497 this.container = new PIXI.DisplayObjectContainer(); |
|
498 this.container.x = opts.xInit; |
|
499 this.container.y = opts.yInit; |
|
500 this.width = opts.width; |
|
501 this.height= opts.height; |
|
502 this.annotCategories = opts.annotCategories; |
|
503 |
|
504 var wsPianoroll = opts.wsPianoroll; |
|
505 var wsAnnot = opts.wsAnnot; |
|
506 var stageView = opts.stageView; |
|
507 |
|
508 stageView.registerComponent(this); |
|
509 |
|
510 var timeLine = new AnnotsTimeLine.AnnotsTimeLine({ |
|
511 stageView : stageView, |
|
512 logger: logger, |
|
513 ws: new annotviz.WsWrapper(wsUriAnnotation, logger), |
|
514 xInit: 0, |
|
515 yInit: 0, |
|
516 width: 1024 - 200 - 200, |
|
517 height: 768-200, |
|
518 timeBegin: Date.now(), |
|
519 timeEnd: Date.now() + 3000000, |
|
520 intervalWidth: 6, |
|
521 intervalHeight: 10, |
|
522 maxCellHeight: 70, |
|
523 radius: 200, |
|
524 annotCategories: this.annotCategories |
|
525 }); |
|
526 |
|
527 var doubleRollH = new DoubleRoll.DoubleRoll({ |
|
528 stageView : stageView, |
|
529 logger: logger, |
|
530 ws: wsPianoroll, |
|
531 yInit: (this.height - 200), |
|
532 sceneHeight: 200, |
|
533 pianorolls : [ |
|
534 { |
|
535 height: 200, |
|
536 timeWidth: 10, |
|
537 lineInterval: 5000, |
|
538 noteHeight: 10 |
|
539 }, |
|
540 ] |
|
541 }); |
|
542 |
|
543 var doubleRollV = new DoubleRoll.DoubleRoll({ |
|
544 stageView : stageView, |
|
545 logger: logger, |
|
546 ws: wsPianoroll, |
|
547 orientation: 'vertical', |
|
548 sceneHeight: 768-200, |
|
549 pianorolls : [ |
|
550 { |
|
551 height: 200, |
|
552 timeWidth: 60, |
|
553 lineInterval: 5000, |
|
554 noteHeight: 5, |
|
555 }, |
|
556 ] |
|
557 }); |
|
558 |
|
559 var annotsRoll = new AnnotsRoll.AnnotsRoll({ |
|
560 stageView : stageView, |
|
561 logger: logger, |
|
562 ws: wsAnnot, |
|
563 parentContainer: doubleRollV.stage, |
|
564 xInit: 1024 - 200 - 200, |
|
565 yInit: 768-200, |
|
566 width: 200 + 200, |
|
567 height: 768-200, |
|
568 widthRoll: 200, |
|
569 framerate: doubleRollV.framerate, |
|
570 pixelsPerSecond: Math.floor(1024 / 60), |
|
571 annotColors: this.annotCategories |
|
572 }); |
|
573 |
|
574 var limiters = new PIXI.Graphics() |
|
575 .lineStyle(1, 0x646464) |
|
576 .moveTo(annotsRoll.container.x, annotsRoll.container.y) |
|
577 .lineTo(annotsRoll.container.x, annotsRoll.container.y - annotsRoll.height) |
|
578 .moveTo(annotsRoll.container.x + annotsRoll.widthRoll, annotsRoll.container.y) |
|
579 .lineTo(annotsRoll.container.x + annotsRoll.widthRoll, annotsRoll.container.y - annotsRoll.height) |
|
580 .moveTo(0, this.height - 200) |
|
581 .lineTo(this.width, this.height - 200) |
|
582 .drawRect(0, 0, this.width -1, this.height -1) |
|
583 .beginFill(0xECECEC) |
|
584 .drawRect(1024 - 200, 0, 200, 768-200) |
|
585 .endFill(); |
|
586 this.container.addChild(limiters); |
|
587 |
|
588 // var doubleRollV = new DoubleRoll.DoubleRoll({}); |
|
589 |
|
590 this.init = function(){ |
|
591 |
|
592 } |
|
593 |
|
594 this.start = function() { |
|
595 }; |
|
596 |
|
597 this.refresh = function() { |
|
598 }; |
|
599 |
|
600 this.stop = function(){ |
|
601 }; |
|
602 |
|
603 return this; |
|
604 |
|
605 } |
|
606 |
|
607 module.exports = { |
|
608 AnnotsVizView: AnnotsVizView |
|
609 }; |
|
610 |
|
611 },{"./annotsroll.js":2,"./annotstimeline.js":3,"./doubleroll.js":5,"lodash":"lodash","pixi":"pixi"}],5:[function(require,module,exports){ |
|
612 /** |
|
613 * scripts/doubleroll.js |
|
614 * |
|
615 * This is the starting point for your application. |
|
616 * Take a look at http://browserify.org/ for more info |
|
617 */ |
|
618 |
|
619 /* global document: false */ |
|
620 |
|
621 'use strict'; |
|
622 |
|
623 |
|
624 var PIXI = require('pixi'); |
|
625 var _ = require('lodash'); |
|
626 var PianoRoll = require('./pianoroll.js'); |
|
627 |
|
628 var defaultConfig = { |
|
629 orientation: 'horizontal', |
|
630 logger: undefined, |
|
631 sceneWidth: 1024, |
|
632 pianorolls : [ |
|
633 { |
|
634 height: 435, |
|
635 timeWidth: 10, |
|
636 lineInterval: 5000, |
|
637 noteHeight: undefined |
|
638 }, |
|
639 { |
|
640 height: 645, |
|
641 timeWidth: 60, |
|
642 lineInterval: 5000, |
|
643 noteHeight: undefined |
|
644 }, |
|
645 ], |
|
646 framerate: 25, |
|
647 offsetMusic: false, |
|
648 sceneBgColor: 0xFFFFFF, |
|
649 lineColor: 0x444444, |
|
650 lineFillColor: 0xFFFF00, |
|
651 noteColors: [0xB90000, 0x4BDD71, 0xAF931E, 0x1C28BA, 0x536991], |
|
652 noteHeight: undefined, |
|
653 zeroShift: 0.9, |
|
654 timeWidth: 60, |
|
655 lineInterval: 5000, |
|
656 // wsUri: undefined, |
|
657 // eventCode: undefined |
|
658 |
|
659 }; |
|
660 |
|
661 function DoubleRoll(options) { |
|
662 |
|
663 var _this = this; |
|
664 var opts = _(options).defaults(defaultConfig).value(); |
|
665 |
|
666 var orientation = opts.orientation; |
|
667 var isHorizontal = (orientation !== 'vertical'); |
|
668 |
|
669 this.logger = opts.logger; |
|
670 this.lineColor = opts.lineColor; |
|
671 this.lineFillColor = opts.lineFillColor; |
|
672 this.framerate = opts.framerate; |
|
673 this.offsetMusic = opts.offsetMusic; |
|
674 this.noteColors = opts.noteColors; |
|
675 |
|
676 var noteHeight = opts.noteHeight; |
|
677 var sceneBgColor = opts.sceneBgColor; |
|
678 var sceneHeight = opts.sceneHeight || _(opts.pianorolls).reduce(function(s,p) { return s + p.height; }, 0); |
|
679 var timeWidth = opts.timeWidth; |
|
680 var lineInterval = opts.lineInterval; |
|
681 var offsetMusic = opts.offsetMusic; |
|
682 |
|
683 var sceneWidth = opts.sceneWidth; |
|
684 var stageView = opts.stageView; |
|
685 |
|
686 var zeroShift = opts.zeroShift; |
|
687 |
|
688 var ws = opts.ws; |
|
689 |
|
690 var colorsReg = {}; |
|
691 |
|
692 this.container = new PIXI.DisplayObjectContainer(); |
|
693 this.container.x = Math.floor(sceneWidth*zeroShift); |
|
694 this.container.y = 0; |
|
695 |
|
696 stageView.registerComponent(this); |
|
697 |
|
698 var pianorollList = []; |
|
699 |
|
700 var pianorollOptions = { |
|
701 parentContainer: this.container, |
|
702 orientation: orientation, |
|
703 xInit: 0, |
|
704 width: sceneWidth, |
|
705 noteColors: this.noteColors, |
|
706 colorsReg: colorsReg, |
|
707 lineColor: this.lineColor, |
|
708 lineInterval: lineInterval, |
|
709 offsetMusic: offsetMusic, |
|
710 }; |
|
711 |
|
712 var yInit = opts.yInit || 0; |
|
713 var linesDown = true; |
|
714 _(opts.pianorolls).forEach(function(prDef, i) { |
|
715 var prNoteHeight = noteHeight || prDef.noteHeight || prDef.height / 128; |
|
716 var prTimeWidth = prDef.timeWidth || timeWidth; |
|
717 pianorollList.push(new PianoRoll(_({ |
|
718 yInit: yInit, |
|
719 height: prDef.height, |
|
720 linesDown: linesDown, |
|
721 pixelsPerSecond: Math.floor(sceneWidth / prTimeWidth), |
|
722 noteHeight: prNoteHeight, |
|
723 lineInterval: prDef.lineInterval |
|
724 }).defaults(pianorollOptions).value())); |
|
725 yInit += prDef.height; |
|
726 linesDown = !linesDown; |
|
727 |
|
728 if(i<(opts.pianorolls.length-1)) { |
|
729 var lineGraphics = new PIXI.Graphics() |
|
730 .beginFill(_this.lineFillColor) |
|
731 .lineStyle(1, _this.lineColor) |
|
732 .moveTo(Math.floor(sceneWidth*zeroShift), yInit) |
|
733 .lineTo(-sceneWidth - Math.floor(sceneWidth*zeroShift), yInit) |
|
734 .endFill(); |
|
735 _this.container.addChild(lineGraphics); |
|
736 } |
|
737 }); |
|
738 |
|
739 if(!isHorizontal) { |
|
740 this.container.rotation = Math.PI/2; |
|
741 this.container.y = sceneHeight; |
|
742 this.container.x = sceneWidth; |
|
743 } |
|
744 |
|
745 |
|
746 this.init = function() { |
|
747 |
|
748 ws.message(function(data) { |
|
749 _this.addNotes(data); |
|
750 }); |
|
751 |
|
752 }; |
|
753 |
|
754 |
|
755 this.addNotes = function(data) { |
|
756 |
|
757 pianorollList.forEach(function(c) { |
|
758 c.addNoteRaw(data); |
|
759 }); |
|
760 }; |
|
761 |
|
762 this.refresh = function() { |
|
763 pianorollList.forEach(function(c) { |
|
764 c.move(); |
|
765 }); |
|
766 }; |
|
767 |
|
768 // Init page and intervals |
|
769 var startTs; |
|
770 |
|
771 this.start = function() { |
|
772 |
|
773 startTs = Date.now(); |
|
774 pianorollList.forEach(function(c) { |
|
775 c.start(); |
|
776 }); |
|
777 }; |
|
778 |
|
779 this.stop = function() { |
|
780 |
|
781 pianorollList.forEach(function(c) { |
|
782 c.stop(); |
|
783 }); |
|
784 }; |
|
785 |
|
786 |
|
787 this.log = function(m) { |
|
788 if(this.logger) { |
|
789 this.logger.log(m); |
|
790 } |
|
791 }; |
|
792 |
|
793 |
|
794 |
|
795 return this; |
|
796 } |
|
797 |
|
798 module.exports = { |
|
799 DoubleRoll: DoubleRoll |
|
800 }; |
|
801 |
|
802 },{"./pianoroll.js":7,"lodash":"lodash","pixi":"pixi"}],6:[function(require,module,exports){ |
|
803 /** |
|
804 * js/wswrapper.js |
|
805 * |
|
806 * simple logger service |
|
807 * |
|
808 */ |
|
809 |
|
810 /* global document: false */ |
|
811 |
|
812 'use strict'; |
|
813 |
|
814 function HtmlLogger(doLog, container) { |
|
815 |
|
816 var logContainer = container; |
|
817 if(typeof(container) === 'string') { |
|
818 logContainer = document.getElementById(container); |
|
819 } |
|
820 if(!doLog) { |
|
821 document.body.removeChild(logContainer); |
|
822 logContainer = undefined; |
|
823 } |
|
824 |
|
825 |
|
826 this.log = function(msg) { |
|
827 if(doLog && logContainer) { |
|
828 logContainer.innerHTML += msg + '\n'; |
|
829 logContainer.scrollTop = logContainer.scrollHeight; |
|
830 } |
|
831 }; |
|
832 } |
|
833 |
|
834 function ConsoleLogger(doLog) { |
|
835 |
|
836 this.log = function(msg) { |
|
837 if(doLog) { |
|
838 console.log(msg); |
|
839 } |
|
840 } |
|
841 |
|
842 } |
|
843 |
|
844 module.exports = { |
|
845 HtmlLogger: HtmlLogger, |
|
846 ConsoleLogger: ConsoleLogger |
|
847 }; |
|
848 |
|
849 },{}],7:[function(require,module,exports){ |
|
850 /** |
|
851 * js/pianoroll.js |
|
852 * |
|
853 * pianoroll basic component |
|
854 * |
|
855 */ |
|
856 |
|
857 'use strict'; |
|
858 |
|
859 |
|
860 var PIXI = require('pixi'); |
|
861 var randomColor = require('randomColor'); |
|
862 var _ = require('lodash'); |
|
863 |
|
864 var NTP_EPOCH_DELTA = 2208988800; //c.f. RFC 868 |
|
865 |
|
866 function PianoRoll(options) { |
|
867 var _this = this; |
|
868 this.container = new PIXI.DisplayObjectContainer(); |
|
869 this.container.x = options.xInit; |
|
870 this.container.y = options.yInit; |
|
871 options.parentContainer.addChild(this.container); |
|
872 |
|
873 var orientation = options.orientation; |
|
874 var isHorizontal = (orientation !== 'vertical'); |
|
875 |
|
876 this.linesDown = options.linesDown; |
|
877 this.height = options.height; |
|
878 this.pixelsPerSecond = options.pixelsPerSecond; |
|
879 this.width = options.width; |
|
880 this.noteColors = options.noteColors; |
|
881 this.colorsReg = options.colorsReg || {}; |
|
882 this.lineColor = options.lineColor; |
|
883 this.lineInterval = options.lineInterval; |
|
884 this.offsetMusic = options.offsetMusic || false; |
|
885 this.noteHeight = options.noteHeight; |
|
886 this.noteDict = {}; |
|
887 this.startTs = options.startTs || Date.now(); |
|
888 |
|
889 var started = false; |
|
890 |
|
891 var isHidden = function(child) { |
|
892 // TODO: the origin point is an approximation. Should refine this |
|
893 var globalPos = child.toGlobal(new PIXI.Point(0,0)); |
|
894 return ((globalPos.x + child.width) < 0) || ((globalPos.y + child.height) < 0) ; |
|
895 }; |
|
896 |
|
897 //TODO: I do not like the "regColor" object. This should not be global, but local |
|
898 this.getColor = function(canal) { |
|
899 var color = this.colorsReg[canal]; |
|
900 if(typeof(color) === 'undefined') { |
|
901 var colorsRegSize = Object.keys(this.colorsReg).length; |
|
902 if(colorsRegSize < this.noteColors.length) { |
|
903 color = this.colorsReg[canal] = this.noteColors[colorsRegSize]; |
|
904 } |
|
905 else { |
|
906 color = this.colorsReg[canal] = parseInt(randomColor({ luminosity: 'light', hue: 'random', format:'hex'}).replace(/^#/, ''), 16); |
|
907 } |
|
908 } |
|
909 return color; |
|
910 }; |
|
911 |
|
912 this.getNoteRect = function(x, y, color, alpha, width, height) { |
|
913 var graphics = new PIXI.Graphics(); |
|
914 graphics.beginFill(color, alpha); |
|
915 graphics.drawRect(0, 0, width, height); |
|
916 graphics.endFill(); |
|
917 graphics.x = x; |
|
918 graphics.y = y; |
|
919 graphics.width = width; |
|
920 graphics.height = height; |
|
921 return graphics; |
|
922 }; |
|
923 |
|
924 this.addNoteRaw = function(data) { |
|
925 console.log(data); |
|
926 var note = data.content[3]; |
|
927 var velocity = data.content[4]; |
|
928 var ts = (data.content[0] - NTP_EPOCH_DELTA)*1000; |
|
929 var channel = data.content[2]; |
|
930 var sessionTs = data.content[1]; |
|
931 |
|
932 this.addNote(note, ts, sessionTs, velocity, channel, 0); |
|
933 }; |
|
934 |
|
935 this.addNote = function(note, startTime, sessionTs, velocity, channel, duration) { |
|
936 |
|
937 var ts = startTime; |
|
938 if(this.offsetMusic) { |
|
939 ts = this.startTs + sessionTs; |
|
940 } |
|
941 |
|
942 var noteDuration = duration; |
|
943 var noteVelocity = velocity; |
|
944 var graphics; |
|
945 if(!duration) { |
|
946 if(typeof this.noteDict[channel]==='undefined'){ |
|
947 this.noteDict[channel] = {}; |
|
948 } |
|
949 if(velocity===0) { |
|
950 if(typeof this.noteDict[channel][note] !== 'undefined') { |
|
951 var noteDef = this.noteDict[channel][note]; |
|
952 delete this.noteDict[channel][note]; |
|
953 noteDuration = sessionTs - noteDef.sessionTs; |
|
954 graphics = noteDef.graphics; |
|
955 noteVelocity = noteDef.velocity; |
|
956 ts = noteDef.ts; |
|
957 } |
|
958 } |
|
959 else { |
|
960 noteDuration = Date.now() - ts; |
|
961 this.noteDict[channel][note] = { ts: ts, velocity: velocity, sessionTs: sessionTs}; |
|
962 } |
|
963 } |
|
964 |
|
965 |
|
966 if(!this.offsetMusic || velocity===0) { |
|
967 |
|
968 var width = noteDuration * this.pixelsPerSecond / 1000; |
|
969 if(!graphics) { |
|
970 var x = (ts-this.startTs) * this.pixelsPerSecond / 1000; |
|
971 if((x+width) < (Math.abs(this.container.x) - this.width)) { |
|
972 // not visible. do nothing |
|
973 return; |
|
974 } |
|
975 var y = Math.floor((128-note+0.5) * this.height / 128 - (this.noteHeight/2)); |
|
976 var color = this.getColor(channel); |
|
977 var alpha = (noteVelocity / 128); |
|
978 |
|
979 graphics = this.getNoteRect(x, y, color, alpha, width, this.noteHeight); |
|
980 this.container.addChild(graphics); |
|
981 } |
|
982 else { |
|
983 graphics.width = width; |
|
984 } |
|
985 |
|
986 if(!duration && velocity) { |
|
987 this.noteDict[channel][note].graphics = graphics; |
|
988 } |
|
989 } |
|
990 }; |
|
991 |
|
992 this.addLine = function(ts){ |
|
993 |
|
994 if(typeof(ts) === 'undefined') { |
|
995 ts = new Date(); |
|
996 } |
|
997 var x = -this.container.x; |
|
998 var y = this.linesDown ? this.height - 20 : 0; |
|
999 |
|
1000 var graphics = new PIXI.Graphics() |
|
1001 .beginFill(0xFFFF00) |
|
1002 .lineStyle(1, this.lineColor) |
|
1003 .moveTo(0, 0) |
|
1004 .lineTo(0, 20) |
|
1005 .endFill(); |
|
1006 graphics.x = x; |
|
1007 graphics.y = y; |
|
1008 this.container.addChild(graphics); |
|
1009 // Add text |
|
1010 //var totalSec = lineNb * this.lineInterval / 1000; |
|
1011 var hours = ts.getHours(); |
|
1012 var minutes =ts.getMinutes(); |
|
1013 var seconds = ts.getSeconds(); |
|
1014 var timeStr = (hours < 10 ? '0' + hours : hours) + ':' + (minutes < 10 ? '0' + minutes : minutes) + ':' + (seconds < 10 ? '0' + seconds : seconds); |
|
1015 |
|
1016 var fontObj = { font: '10pt Arial', fill: '#444444' }; |
|
1017 var t = new PIXI.Text(timeStr, fontObj); |
|
1018 if(isHorizontal) { |
|
1019 t.x = x + 2; |
|
1020 t.y = this.linesDown ? this.height - 15 : 2; |
|
1021 } |
|
1022 else { |
|
1023 t.rotation = -Math.PI/2; |
|
1024 t.x = x ; |
|
1025 t.y = this.linesDown ? this.height - 2 : t.width + 2; |
|
1026 } |
|
1027 this.container.addChild(t); |
|
1028 }; |
|
1029 |
|
1030 this.moveTo = function(diffTime){ |
|
1031 var oldX = this.container.x; |
|
1032 this.container.x = Math.floor(diffTime*this.pixelsPerSecond); |
|
1033 var deltaX = Math.abs(oldX-this.container.x); |
|
1034 _.forOwn(this.noteDict, function(channelDict) { |
|
1035 _.forOwn(channelDict, function(noteDef) { |
|
1036 if(noteDef.graphics) { |
|
1037 noteDef.graphics.width = noteDef.graphics.width + deltaX; |
|
1038 } |
|
1039 }); |
|
1040 }); |
|
1041 }; |
|
1042 |
|
1043 this.move = function() { |
|
1044 var diff = (this.startTs - Date.now())/1000; |
|
1045 this.moveTo(diff); |
|
1046 }; |
|
1047 |
|
1048 this.removePassedObjets = function(){ |
|
1049 var childrenToRemove = []; |
|
1050 _(_this.container.children).forEach(function(child) { |
|
1051 return typeof(child) === 'undefined' || |
|
1052 (isHidden(child) && childrenToRemove.push(child)); |
|
1053 }); |
|
1054 childrenToRemove.forEach(function(child) { |
|
1055 _this.container.removeChild(child); |
|
1056 }); |
|
1057 }; |
|
1058 |
|
1059 this.start = function() { |
|
1060 if(!started) { |
|
1061 this.startTs = Date.now(); |
|
1062 this.addLine(); |
|
1063 started = true; |
|
1064 } |
|
1065 this.verticalLinesInterval = setInterval(function() { _this.addLine(); }, this.lineInterval); |
|
1066 this.cleanInterval = setInterval(function () { _this.removePassedObjets(); }, 1000 * this.width / this.pixelsPerSecond ); |
|
1067 }; |
|
1068 |
|
1069 this.stop = function() { |
|
1070 //window.clearInterval(this.moveInterval); |
|
1071 clearInterval(this.verticalLinesInterval); |
|
1072 clearInterval(this.cleanInterval); |
|
1073 }; |
|
1074 |
|
1075 |
|
1076 } |
|
1077 |
|
1078 module.exports = PianoRoll; |
|
1079 |
|
1080 },{"lodash":"lodash","pixi":"pixi","randomColor":"randomColor"}],8:[function(require,module,exports){ |
|
1081 /** |
|
1082 * scripts/stageview.js |
|
1083 * |
|
1084 * This is the starting point for your application. |
|
1085 * Take a look at http://browserify.org/ for more info |
|
1086 */ |
|
1087 |
|
1088 /* global document: false */ |
|
1089 |
|
1090 'use strict'; |
|
1091 |
|
1092 |
|
1093 var PIXI = require('pixi'); |
|
1094 var _ = require('lodash'); |
|
1095 |
|
1096 var defaultConfig = { |
|
1097 externalRefresh: false, |
|
1098 logger: undefined, |
|
1099 sceneWidth: 1024, |
|
1100 sceneHeight: 768, |
|
1101 framerate: 25, |
|
1102 sceneBgColor: 0xFFFFFF, |
|
1103 canvasContainer: 'canvasContainer', |
|
1104 }; |
|
1105 |
|
1106 function StageView(options) { |
|
1107 |
|
1108 var _this = this; |
|
1109 var opts = _(options).defaults(defaultConfig).value(); |
|
1110 |
|
1111 var externalRefresh = opts.externalRefresh; |
|
1112 |
|
1113 this.logger = opts.logger; |
|
1114 this.framerate = opts.framerate; |
|
1115 var sceneBgColor = opts.sceneBgColor; |
|
1116 var sceneWidth = opts.sceneWidth; |
|
1117 var sceneHeight = opts.sceneHeight; |
|
1118 var canvasContainer = opts.canvasContainer; |
|
1119 var timeContainer = []; |
|
1120 var components = []; |
|
1121 |
|
1122 //create an new instance of a pixi stage |
|
1123 this.stage = new PIXI.Stage(sceneBgColor); |
|
1124 //create a renderer instance. |
|
1125 var renderer = PIXI.autoDetectRenderer(sceneWidth, sceneHeight); |
|
1126 |
|
1127 this.init = function() { |
|
1128 |
|
1129 if(typeof(canvasContainer) === 'string') { |
|
1130 canvasContainer = document.getElementById(canvasContainer); |
|
1131 } |
|
1132 if(typeof(timeContainer) === 'string') { |
|
1133 timeContainer = document.getElementById(timeContainer); |
|
1134 } |
|
1135 |
|
1136 canvasContainer.appendChild(renderer.view); |
|
1137 |
|
1138 components.forEach(function(c){ |
|
1139 c.init(); |
|
1140 }); |
|
1141 }; |
|
1142 |
|
1143 this.registerTimeContainer = function(container) { |
|
1144 timeContainer.push(container); |
|
1145 }; |
|
1146 |
|
1147 this.registerComponent = function(component) { |
|
1148 components.push(component); |
|
1149 this.stage.addChild(component.container); |
|
1150 }; |
|
1151 |
|
1152 this.refresh = function() { |
|
1153 components.forEach(function(c){ |
|
1154 c.refresh(); |
|
1155 }); |
|
1156 renderer.render(this.stage); |
|
1157 }; |
|
1158 |
|
1159 // Init page and intervals |
|
1160 var refreshInterval; |
|
1161 |
|
1162 this.start = function() { |
|
1163 |
|
1164 if(!externalRefresh) { |
|
1165 refreshInterval = setInterval(function() {_this.refresh();}, 1000/this.framerate); |
|
1166 } |
|
1167 |
|
1168 components.forEach(function(c){ |
|
1169 c.start(); |
|
1170 }); |
|
1171 }; |
|
1172 |
|
1173 this.stop = function() { |
|
1174 if(!externalRefresh) { |
|
1175 clearInterval(refreshInterval); |
|
1176 } |
|
1177 clearInterval(refreshTimeInterval); |
|
1178 |
|
1179 components.forEach(function(c){ |
|
1180 c.stop(); |
|
1181 }); |
|
1182 }; |
|
1183 |
|
1184 |
|
1185 this.log = function(m) { |
|
1186 if(this.logger) { |
|
1187 this.logger.log(m); |
|
1188 } |
|
1189 }; |
|
1190 |
|
1191 |
|
1192 return this; |
|
1193 } |
|
1194 |
|
1195 module.exports = { |
|
1196 StageView: StageView |
|
1197 }; |
|
1198 |
|
1199 },{"lodash":"lodash","pixi":"pixi"}],9:[function(require,module,exports){ |
|
1200 /** |
|
1201 * js/utils.js |
|
1202 * |
|
1203 * basic tools |
|
1204 * |
|
1205 */ |
|
1206 |
|
1207 'use strict'; |
|
1208 |
|
1209 function formatTime (ts) { |
|
1210 var hours = Math.floor( (ts/1000) / 3600 ) % 24; |
|
1211 var minutes = Math.floor( (ts/1000) / 60 ) % 60; |
|
1212 var seconds = Math.floor( (ts/1000) % 60); |
|
1213 return ((hours < 10 ? '0' + hours : hours) + ':' + (minutes < 10 ? '0' + minutes : minutes) + ':' + (seconds < 10 ? '0' + seconds : seconds)); |
|
1214 } |
|
1215 |
|
1216 |
|
1217 module.exports = { |
|
1218 formatTime: formatTime |
|
1219 }; |
|
1220 |
|
1221 },{}],10:[function(require,module,exports){ |
|
1222 /** |
|
1223 * js/wswrapper.js |
|
1224 * |
|
1225 * simple webservice wrapper to register callbacks on onmessage |
|
1226 * |
|
1227 */ |
|
1228 |
|
1229 /* global WebSocket: false */ |
|
1230 |
|
1231 'use strict'; |
|
1232 |
|
1233 function WsWrapper(wsurl, logger) { |
|
1234 |
|
1235 var url = wsurl; |
|
1236 var sock = new WebSocket(url); |
|
1237 var loggerObj = logger; |
|
1238 |
|
1239 var log = function(msg) { |
|
1240 if(loggerObj) { |
|
1241 loggerObj.log(msg); |
|
1242 } |
|
1243 }; |
|
1244 |
|
1245 var handlers = []; |
|
1246 |
|
1247 sock.onopen = function() { |
|
1248 log('Connected to ' + url); |
|
1249 }; |
|
1250 |
|
1251 sock.onclose = function(e) { |
|
1252 log('Connection closed (wasClean = ' + e.wasClean + ', code = ' + e.code + ', reason = \'' + e.reason + '\')'); |
|
1253 sock = null; |
|
1254 }; |
|
1255 |
|
1256 sock.onmessage = function(e) { |
|
1257 log('received ' + e.data); |
|
1258 var data = JSON.parse(e.data); |
|
1259 handlers.forEach(function(handler) { |
|
1260 handler(data); |
|
1261 }); |
|
1262 }; |
|
1263 |
|
1264 this.message = function(handler) { |
|
1265 if(handler) { |
|
1266 handlers.push(handler); |
|
1267 } |
|
1268 }; |
|
1269 |
|
1270 } |
|
1271 |
|
1272 module.exports = { |
|
1273 WsWrapper: WsWrapper |
|
1274 }; |
|
1275 |
|
1276 },{}]},{},[1])(1) |
|
1277 }); |
|
1278 //# 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;;AC3BA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC7MA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;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;;AC5JA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;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;AACA;;ACrOA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;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;;ACpBA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;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');\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    .value();","/**\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};\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 = options.startTs || Date.now();\n\n    var yInit = opts.yInit;\n    var annotStyles = _(opts.annotStyles).defaults(defaultAnnotStyles).value();\n    for(var style in annotStyles) {\n    \tif (annotStyles[style].wordWrap === true){\n    \t\tannotStyles[style].wordWrapWidth = this.widthRoll;\n    \t}\n    }\n    var started = false;\n    var ws = opts.ws;\n    var externalRefresh = opts.externalRefresh;\n    var stageView = opts.stageView;\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            console.log(\"cDef\", cdef);\n            console.log(\"cDef ts\", cdef.ts, ts);\n            if(cdef.ts < ts) {\n                colorsDef = cdef.colors;\n                return false;\n            }\n        });\n        var resColor;\n        console.log(\"colorsDef\", colorsDef);\n        if(colorsDef) {\n            resColor = colorsDef[code];\n        }\n        if(!resColor) {\n            resColor = DEFAULT_ANNOT_COLOR;\n        }\n        return resColor;\n    }\n\n    this.addAnnot = function(category, text, user, catColor, ts){\n\n        var color = catColor ? catColor : DEFAULT_ANNOT_COLOR;\n        var x = 0;\n        var y = (ts-this.startTs) * this.pixelsPerSecond / 1000 + yInit;\n\n        var colorHex = parseInt(color.replace(/^#/, ''), 16);\n\n        var graphics = new PIXI.Graphics()\n            .beginFill(colorHex)\n            .drawRect(x, y, 10, 3)\n            .endFill();\n\n        this.container.addChild(graphics);\n\n        var textHeight = 0;\n        var catLabel = new PIXI.Text(\n            category,\n            _(annotStyles.label).extend({fill: color}).value()\n        );\n        catLabel.x = x + 20;\n        catLabel.y = y - 23;\n        this.container.addChild(catLabel);\n        textHeight += (catLabel.height - 23 + 2);\n\n        if(text) {\n            var catText = new PIXI.Text(text, annotStyles.text);\n            catText.x = x + 20;\n            catText.y = y + 2;\n            this.container.addChild(catText);\n            textHeight += (catText.height + 2);\n        }\n\n        var catUser = new PIXI.Text(user, annotStyles.user);\n        catUser.x = x + 20;\n        catUser.y = y + 2 + textHeight;\n        this.container.addChild(catUser);\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.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    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};\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\n    this.circleX = opts.circleX || (this.width/2);\n    this.circleY = opts.circleY || (this.height/2);\n    this.radius = opts.radius;\n    this.perimeter = 2*Math.PI* this.radius;\n    this.intervalDuration = (this.intervalWidth * this.duration / this.perimeter);\n\n    var currentTime = this.timeBegin;\n    var totalIndex = Math.floor(this.perimeter/this.intervalWidth);\n\n    this.cells = []\n    for (var i=0; i<(this.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    \tfor (var category in this.annotCategories[0].colors){\n    \t\tthis.cells[i].categories[category] = {\n\t\t\t\t\"count\": 0,\n\t\t\t\t\"color\": this.annotCategories[0].colors[category]\n    \t\t};\n    \t}\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.lineStyle(1, 0xD7D7D7)\n    \t.drawCircle(this.circleX, this.circleY, this.radius*2/3)\n    \t.drawCircle(this.circleX, this.circleY, this.radius/3)\n    \t.lineStyle(1, 0x646464)\n    \t.moveTo(this.circleX, this.circleY - (this.radius/3)/2)\n    \t.lineTo(this.circleX, this.circleY - this.radius - this.maxCellHeight - 10)\n    \t.endFill()\n    this.container.addChild(graphics);\n\n    //set time text\n    //TODO : move this to annotsvizview\n    var currentTimeText = new PIXI.Text(\"-- : -- : --\", { font: '18pt Gothic Standard', fill: '#646464' });\n    currentTimeText.x = this.circleX - currentTimeText.width/2;\n    currentTimeText.y = this.circleY - currentTimeText.height/2;\n    this.container.addChild(currentTimeText);\n\n    stageView.registerComponent(this);\n\n    //Add Annotation to the TimeLine\n    this.addAnnot = function(data){\n    \tif (typeof(this.annotCategories[0].colors[data.content.category.code]) !== 'undefined'){\n    \t\tvar annotCode = data.content.category.code;\n    \t} else {\n    \t\tvar annotCode = this.annotCategories[0].order[this.annotCategories[0].order.length -1];\n    \t}\n    \tvar annotTime = Date.parse(data.ts);\n\n    \tif (this.timeEnd > Date.parse(data.ts)){\n\t    \tvar i = Math.floor((Date.parse(data.ts)-this.timeBegin)/(1000*this.intervalDuration));\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], i);\n    \t}\n    };\n\n    this.initGraphics = function(cell){\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\n    this.initTimeTexts = 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    }\n\n    //Draw the cellule\n    this.redrawCell = function(cell){\n\n    \tif (typeof(cell.graphics) === 'undefined'){\n    \t\tthis.initGraphics(cell);\n    \t} else {\n    \t\tcell.graphics.clear();\n    \t}\n\n    \tvar y = 0;\n\n    \t//Check if total height is higher than Max Cell Height\n    \tif ((cell.totalAnnots*this.intervalHeight) > this.maxCellHeight){\n    \t\tvar heightStep = this.maxCellHeight/cell.totalAnnots;\n    \t} else {\n    \t\tvar heightStep = this.intervalHeight;\n    \t}\n\n    \t//Draw the rect depending on the height step calculated\n    \tfor (var i=0; i< this.annotCategories[0].order.length; i++){\n    \t\tvar currentCode = this.annotCategories[0].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.init = function() {\n    \tws.message(function(data) {\n            _this.addAnnot(data);\n        });\n\n    \tthis.initTimeTexts();\n    };\n\n    this.updateTime = function(){\n    \tcurrentTime += 1000;\n\n        var nbSec = currentTime / 1000;\n        var hours = Math.floor( nbSec / 3600 ) % 24;\n        var minutes = Math.floor( nbSec / 60 ) % 60;\n        var seconds = Math.floor(nbSec % 60);\n        var timeStr = (hours < 10 ? '0' + hours : hours) + ':' + (minutes < 10 ? '0' + minutes : minutes) + ':' + (seconds  < 10 ? '0' + seconds : seconds);\n\n        currentTimeText.setText(timeStr);\n    };\n\n    var refreshTimeInterval;\n\n    this.start = function() {\n    \trefreshTimeInterval = setInterval(function() {_this.updateTime();}, 1000);\n    };\n\n    this.refresh = function() {\n\n    };\n\n    this.stop = function(){\n    \tconsole.log(this.cells);\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');\n\nvar defaultOptions = {\n    xInit: 0,\n    yInit: 0,\n    width: 1024,\n    height: 768,\n    annotCategories: [\n    {\n        \"ts\": 1421928213000,\n        \"colors\": {\n            \"transgressions\": \"#b90000\",\n            \"rythmique\": \"#af931e\",\n            \"narration\": \"#4bdd71\",\n            \"relation\": \"#1c28ba\"\n        },\n        \"order\": [\n            \"transgressions\",\n            \"rythmique\",\n            \"narration\",\n            \"relation\"\n        ],\n        \"defaultColor\": \"#536991\"\n    }]\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.annotCategories = opts.annotCategories;\n\n\tvar wsPianoroll = opts.wsPianoroll;\n\tvar wsAnnot = opts.wsAnnot;\n\tvar stageView = opts.stageView;\n\n\tstageView.registerComponent(this);\n\n\tvar timeLine = 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: Date.now(),\n        timeEnd: Date.now() + 3000000,\n        intervalWidth: 6,\n        intervalHeight: 10,\n        maxCellHeight: 70,\n        radius: 200,\n        annotCategories: this.annotCategories\n    });\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//\tvar doubleRollV = new DoubleRoll.DoubleRoll({});\n\n\tthis.init = function(){\n\n\t}\n\n\tthis.start = function() {\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    \tconsole.log(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\nfunction formatTime (ts) {\n\tvar hours = Math.floor( (ts/1000) / 3600 ) % 24;\n\tvar minutes = Math.floor( (ts/1000) / 60 ) % 60;\n\tvar seconds = Math.floor( (ts/1000) % 60);\n\treturn ((hours < 10 ? '0' + hours : hours) + ':' + (minutes < 10 ? '0' + minutes : minutes) + ':' + (seconds  < 10 ? '0' + seconds : seconds));\n}\n\n\nmodule.exports = {\n\tformatTime: formatTime\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"]} |