# HG changeset patch # User rougeronj # Date 1421915389 -3600 # Node ID 3e075a48e19e81cef606849d0e14aa6798a57879 # Parent a7b72620d227bd926d5161293c9aafbb993d3837# Parent e4f0c105090d38f4b60699716dd20166e38e40da Merge with e4f0c105090d38f4b60699716dd20166e38e40da diff -r a7b72620d227 -r 3e075a48e19e annot-server/static/css/annotviz.css --- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/annot-server/static/css/annotviz.css Thu Jan 22 09:29:49 2015 +0100 @@ -0,0 +1,22 @@ +/* ========================================================================== + App Stylesheet + app.less + ========================================================================== */ +/* Mixins (User defined mixins) + ======================================================================== */ +/* Components + ======================================================================== */ +/* ========================================================================== + Base Stylesheet + base.less + ========================================================================== */ +body { + margin: 0; + padding: 0; + background-color: #FFF; +} +#log { + height: 20em; + overflow-y: scroll; + background-color: #faa; +} diff -r a7b72620d227 -r 3e075a48e19e annot-server/static/css/annotviz.min.css --- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/annot-server/static/css/annotviz.min.css Thu Jan 22 09:29:49 2015 +0100 @@ -0,0 +1,1 @@ +body{margin:0;padding:0;background-color:#FFF}#log{height:20em;overflow-y:scroll;background-color:#faa} \ No newline at end of file diff -r a7b72620d227 -r 3e075a48e19e annot-server/static/css/app.css --- a/annot-server/static/css/app.css Thu Jan 22 09:26:43 2015 +0100 +++ b/annot-server/static/css/app.css Thu Jan 22 09:29:49 2015 +0100 @@ -63,7 +63,7 @@ font-size: 2vw; } .send{ - background-color: #4cae4c; + background-color: #536991; } .return{ background-color: #e6e6e6; diff -r a7b72620d227 -r 3e075a48e19e annot-server/static/css/app.min.css --- a/annot-server/static/css/app.min.css Thu Jan 22 09:26:43 2015 +0100 +++ b/annot-server/static/css/app.min.css Thu Jan 22 09:29:49 2015 +0100 @@ -10,7 +10,7 @@ .mons-button .content .table .table-cell{display:table-cell;text-align:center;vertical-align:middle;text-shadow:0 0 5px #fff} .large-cat{font-size:300%;font-size:4vw} .normal-cat{font-size:100%;font-size:2vw} -.send{background-color:#4cae4c} +.send{background-color:#536991} .return{background-color:#e6e6e6} .return,.send{padding-bottom:15%} .row input{font-size:1.2em} diff -r a7b72620d227 -r 3e075a48e19e annot-server/static/css/lib.min.css --- a/annot-server/static/css/lib.min.css Thu Jan 22 09:26:43 2015 +0100 +++ b/annot-server/static/css/lib.min.css Thu Jan 22 09:29:49 2015 +0100 @@ -83,7 +83,7 @@ * Copyright 2011-2014 Twitter, Inc. * Licensed under MIT (https://github.com/twbs/bootstrap/blob/master/LICENSE) */ - /*! normalize.css v3.0.0 | MIT License | git.io/normalize */ +/*! normalize.css v3.0.0 | MIT License | git.io/normalize */ html{font-family:sans-serif;-webkit-text-size-adjust:100%;-ms-text-size-adjust:100%} body{margin:0} article,aside,details,figcaption,figure,footer,header,hgroup,main,nav,section,summary{display:block} @@ -141,7 +141,7 @@ button,input,select,textarea{font-family:inherit;font-size:inherit;line-height:inherit} a{color:#428bca;text-decoration:none} a:focus,a:hover{color:#2a6496;text-decoration:underline} -a:focus{outline:-webkit-focus-ring-color auto 5px;outline-offset:-2px} +a:focus{outline:dotted thin;outline:-webkit-focus-ring-color auto 5px;outline-offset:-2px} figure{margin:0} img{vertical-align:middle} .carousel-inner>.item>a>img,.carousel-inner>.item>img,.img-responsive,.thumbnail a>img,.thumbnail>img{display:block;max-width:100%;height:auto} @@ -475,7 +475,7 @@ input[type=file]{display:block} input[type=range]{display:block;width:100%} select[multiple],select[size]{height:auto} -input[type=checkbox]:focus,input[type=file]:focus,input[type=radio]:focus{outline:dotted thin;outline:-webkit-focus-ring-color auto 5px;outline-offset:-2px} +input[type=checkbox]:focus,input[type=radio]:focus,input[type=file]:focus{outline:dotted thin;outline:-webkit-focus-ring-color auto 5px;outline-offset:-2px} output{display:block;padding-top:7px;font-size:14px;line-height:1.42857143;color:#555} .form-control{display:block;width:100%;height:34px;padding:6px 12px;font-size:14px;line-height:1.42857143;color:#555;background-color:#fff;background-image:none;border:1px solid #ccc;border-radius:4px;-webkit-box-shadow:inset 0 1px 1px rgba(0,0,0,.075);box-shadow:inset 0 1px 1px rgba(0,0,0,.075);-webkit-transition:border-color ease-in-out .15s,box-shadow ease-in-out .15s;transition:border-color ease-in-out .15s,box-shadow ease-in-out .15s} .form-control:focus{border-color:#66afe9;outline:0;-webkit-box-shadow:inset 0 1px 1px rgba(0,0,0,.075),0 0 8px rgba(102,175,233,.6);box-shadow:inset 0 1px 1px rgba(0,0,0,.075),0 0 8px rgba(102,175,233,.6)} @@ -1039,7 +1039,7 @@ .label:empty{display:none} .btn .label{position:relative;top:-1px} .label-default{background-color:#999} -.label-default[href]:focus,.label-default[href]:hover{background-color:gray} +.label-default[href]:focus,.label-default[href]:hover{background-color:grey} .label-primary{background-color:#428bca} .label-primary[href]:focus,.label-primary[href]:hover{background-color:#3071a9} .label-success{background-color:#5cb85c} diff -r a7b72620d227 -r 3e075a48e19e annot-server/static/js/annotviz.js --- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/annot-server/static/js/annotviz.js Thu Jan 22 09:29:49 2015 +0100 @@ -0,0 +1,1278 @@ +!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 Date.parse(data.ts)){ + var i = Math.floor((Date.parse(data.ts)-this.timeBegin)/(1000*this.intervalDuration)); + + this.cells[i].categories[annotCode].count += 1; + this.cells[i].totalAnnots +=1; + this.redrawCell(this.cells[i], i); + } + }; + + this.initGraphics = function(cell){ + 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); + } + + this.initTimeTexts = 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 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 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); + } + + //Draw the cellule + this.redrawCell = function(cell){ + + if (typeof(cell.graphics) === 'undefined'){ + this.initGraphics(cell); + } else { + cell.graphics.clear(); + } + + var y = 0; + + //Check if total height is higher than Max Cell Height + if ((cell.totalAnnots*this.intervalHeight) > this.maxCellHeight){ + var heightStep = this.maxCellHeight/cell.totalAnnots; + } else { + var heightStep = this.intervalHeight; + } + + //Draw the rect depending on the height step calculated + for (var i=0; i< this.annotCategories[0].order.length; i++){ + var currentCode = this.annotCategories[0].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.init = function() { + ws.message(function(data) { + _this.addAnnot(data); + }); + + this.initTimeTexts(); + }; + + this.updateTime = function(){ + currentTime += 1000; + + var nbSec = currentTime / 1000; + var hours = Math.floor( nbSec / 3600 ) % 24; + var minutes = Math.floor( nbSec / 60 ) % 60; + var seconds = Math.floor(nbSec % 60); + var timeStr = (hours < 10 ? '0' + hours : hours) + ':' + (minutes < 10 ? '0' + minutes : minutes) + ':' + (seconds < 10 ? '0' + seconds : seconds); + + currentTimeText.setText(timeStr); + }; + + var refreshTimeInterval; + + this.start = function() { + refreshTimeInterval = setInterval(function() {_this.updateTime();}, 1000); + }; + + this.refresh = function() { + + }; + + this.stop = function(){ + console.log(this.cells); + }; + + return this; +} + +module.exports = { + AnnotsTimeLine: AnnotsTimeLine +}; + +},{"./utils.js":9,"lodash":"lodash","pixi":"pixi"}],4:[function(require,module,exports){ +/** +* js/annotsvizview.js +* +* This is the starting point for your application. +* Take a look at http://browserify.org/ for more info +*/ + +'use strict'; + +var PIXI = require('pixi'); +var _ = require('lodash'); +var DoubleRoll = require('./doubleroll.js'); +var AnnotsTimeLine = require('./annotstimeline.js'); +var AnnotsRoll = require('./annotsroll.js'); + +var defaultOptions = { + xInit: 0, + yInit: 0, + width: 1024, + height: 768, + annotCategories: [ + { + "ts": 1421928213000, + "colors": { + "transgressions": "#b90000", + "rythmique": "#af931e", + "narration": "#4bdd71", + "relation": "#1c28ba" + }, + "order": [ + "transgressions", + "rythmique", + "narration", + "relation" + ], + "defaultColor": "#536991" + }] +}; + +function AnnotsVizView(options){ + 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.annotCategories = opts.annotCategories; + + var wsPianoroll = opts.wsPianoroll; + var wsAnnot = opts.wsAnnot; + var stageView = opts.stageView; + + stageView.registerComponent(this); + + var timeLine = new AnnotsTimeLine.AnnotsTimeLine({ + stageView : stageView, + logger: logger, + ws: new annotviz.WsWrapper(wsUriAnnotation, logger), + xInit: 0, + yInit: 0, + width: 1024 - 200 - 200, + height: 768-200, + timeBegin: Date.now(), + timeEnd: Date.now() + 3000000, + intervalWidth: 6, + intervalHeight: 10, + maxCellHeight: 70, + radius: 200, + annotCategories: this.annotCategories + }); + + var doubleRollH = new DoubleRoll.DoubleRoll({ + stageView : stageView, + logger: logger, + ws: wsPianoroll, + yInit: (this.height - 200), + sceneHeight: 200, + pianorolls : [ + { + height: 200, + timeWidth: 10, + lineInterval: 5000, + noteHeight: 10 + }, + ] + }); + + var doubleRollV = new DoubleRoll.DoubleRoll({ + stageView : stageView, + logger: logger, + ws: wsPianoroll, + orientation: 'vertical', + sceneHeight: 768-200, + pianorolls : [ + { + height: 200, + timeWidth: 60, + lineInterval: 5000, + noteHeight: 5, + }, + ] + }); + + var annotsRoll = new AnnotsRoll.AnnotsRoll({ + stageView : stageView, + logger: logger, + ws: wsAnnot, + parentContainer: doubleRollV.stage, + xInit: 1024 - 200 - 200, + yInit: 768-200, + width: 200 + 200, + height: 768-200, + widthRoll: 200, + framerate: doubleRollV.framerate, + pixelsPerSecond: Math.floor(1024 / 60), + 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 doubleRollV = new DoubleRoll.DoubleRoll({}); + + this.init = function(){ + + } + + this.start = function() { + }; + + this.refresh = function() { + }; + + this.stop = function(){ + }; + + return this; + +} + +module.exports = { + AnnotsVizView: AnnotsVizView +}; + +},{"./annotsroll.js":2,"./annotstimeline.js":3,"./doubleroll.js":5,"lodash":"lodash","pixi":"pixi"}],5:[function(require,module,exports){ +/** +* scripts/doubleroll.js +* +* This is the starting point for your application. +* 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 defaultConfig = { + orientation: 'horizontal', + logger: undefined, + sceneWidth: 1024, + pianorolls : [ + { + height: 435, + timeWidth: 10, + lineInterval: 5000, + noteHeight: undefined + }, + { + height: 645, + timeWidth: 60, + lineInterval: 5000, + noteHeight: undefined + }, + ], + framerate: 25, + offsetMusic: false, + sceneBgColor: 0xFFFFFF, + lineColor: 0x444444, + lineFillColor: 0xFFFF00, + noteColors: [0xB90000, 0x4BDD71, 0xAF931E, 0x1C28BA, 0x536991], + noteHeight: undefined, + zeroShift: 0.9, + timeWidth: 60, + lineInterval: 5000, +// wsUri: undefined, +// eventCode: undefined + +}; + +function DoubleRoll(options) { + + var _this = this; + var opts = _(options).defaults(defaultConfig).value(); + + var orientation = opts.orientation; + var isHorizontal = (orientation !== 'vertical'); + + this.logger = opts.logger; + this.lineColor = opts.lineColor; + this.lineFillColor = opts.lineFillColor; + this.framerate = opts.framerate; + this.offsetMusic = opts.offsetMusic; + 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; + var offsetMusic = opts.offsetMusic; + + var sceneWidth = opts.sceneWidth; + var stageView = opts.stageView; + + var zeroShift = opts.zeroShift; + + var ws = opts.ws; + + var colorsReg = {}; + + this.container = new PIXI.DisplayObjectContainer(); + this.container.x = Math.floor(sceneWidth*zeroShift); + this.container.y = 0; + + stageView.registerComponent(this); + + var pianorollList = []; + + var pianorollOptions = { + parentContainer: this.container, + orientation: orientation, + xInit: 0, + width: sceneWidth, + noteColors: this.noteColors, + colorsReg: colorsReg, + lineColor: this.lineColor, + lineInterval: lineInterval, + offsetMusic: offsetMusic, + }; + + var yInit = opts.yInit || 0; + var linesDown = true; + _(opts.pianorolls).forEach(function(prDef, i) { + var prNoteHeight = noteHeight || prDef.noteHeight || prDef.height / 128; + var prTimeWidth = prDef.timeWidth || timeWidth; + pianorollList.push(new PianoRoll(_({ + yInit: yInit, + height: prDef.height, + linesDown: linesDown, + pixelsPerSecond: Math.floor(sceneWidth / prTimeWidth), + noteHeight: prNoteHeight, + lineInterval: prDef.lineInterval + }).defaults(pianorollOptions).value())); + yInit += prDef.height; + linesDown = !linesDown; + + if(i<(opts.pianorolls.length-1)) { + var lineGraphics = new PIXI.Graphics() + .beginFill(_this.lineFillColor) + .lineStyle(1, _this.lineColor) + .moveTo(Math.floor(sceneWidth*zeroShift), yInit) + .lineTo(-sceneWidth - Math.floor(sceneWidth*zeroShift), yInit) + .endFill(); + _this.container.addChild(lineGraphics); + } + }); + + if(!isHorizontal) { + this.container.rotation = Math.PI/2; + this.container.y = sceneHeight; + this.container.x = sceneWidth; + } + + + this.init = function() { + + ws.message(function(data) { + _this.addNotes(data); + }); + + }; + + + this.addNotes = function(data) { + + pianorollList.forEach(function(c) { + c.addNoteRaw(data); + }); + }; + + this.refresh = function() { + pianorollList.forEach(function(c) { + c.move(); + }); + }; + + // Init page and intervals + var startTs; + + this.start = function() { + + startTs = Date.now(); + pianorollList.forEach(function(c) { + c.start(); + }); + }; + + this.stop = function() { + + pianorollList.forEach(function(c) { + c.stop(); + }); + }; + + + this.log = function(m) { + if(this.logger) { + this.logger.log(m); + } + }; + + + + return this; +} + +module.exports = { + DoubleRoll: DoubleRoll +}; + +},{"./pianoroll.js":7,"lodash":"lodash","pixi":"pixi"}],6:[function(require,module,exports){ +/** +* js/wswrapper.js +* +* simple logger service +* +*/ + +/* global document: false */ + +'use strict'; + +function HtmlLogger(doLog, container) { + + var logContainer = container; + if(typeof(container) === 'string') { + logContainer = document.getElementById(container); + } + if(!doLog) { + document.body.removeChild(logContainer); + logContainer = undefined; + } + + + this.log = function(msg) { + if(doLog && logContainer) { + logContainer.innerHTML += msg + '\n'; + logContainer.scrollTop = logContainer.scrollHeight; + } + }; +} + +function ConsoleLogger(doLog) { + + this.log = function(msg) { + if(doLog) { + console.log(msg); + } + } + +} + +module.exports = { + HtmlLogger: HtmlLogger, + ConsoleLogger: ConsoleLogger +}; + +},{}],7:[function(require,module,exports){ +/** +* js/pianoroll.js +* +* pianoroll basic component +* +*/ + +'use strict'; + + +var PIXI = require('pixi'); +var randomColor = require('randomColor'); +var _ = require('lodash'); + +var NTP_EPOCH_DELTA = 2208988800; //c.f. RFC 868 + +function PianoRoll(options) { + var _this = this; + this.container = new PIXI.DisplayObjectContainer(); + this.container.x = options.xInit; + this.container.y = options.yInit; + options.parentContainer.addChild(this.container); + + var orientation = options.orientation; + var isHorizontal = (orientation !== 'vertical'); + + this.linesDown = options.linesDown; + this.height = options.height; + this.pixelsPerSecond = options.pixelsPerSecond; + this.width = options.width; + this.noteColors = options.noteColors; + this.colorsReg = options.colorsReg || {}; + this.lineColor = options.lineColor; + this.lineInterval = options.lineInterval; + this.offsetMusic = options.offsetMusic || false; + this.noteHeight = options.noteHeight; + this.noteDict = {}; + this.startTs = options.startTs || Date.now(); + + var started = false; + + var isHidden = function(child) { + // TODO: the origin point is an approximation. Should refine this + var globalPos = child.toGlobal(new PIXI.Point(0,0)); + return ((globalPos.x + child.width) < 0) || ((globalPos.y + child.height) < 0) ; + }; + + //TODO: I do not like the "regColor" object. This should not be global, but local + this.getColor = function(canal) { + var color = this.colorsReg[canal]; + if(typeof(color) === 'undefined') { + var colorsRegSize = Object.keys(this.colorsReg).length; + if(colorsRegSize < this.noteColors.length) { + color = this.colorsReg[canal] = this.noteColors[colorsRegSize]; + } + else { + color = this.colorsReg[canal] = parseInt(randomColor({ luminosity: 'light', hue: 'random', format:'hex'}).replace(/^#/, ''), 16); + } + } + return color; + }; + + this.getNoteRect = function(x, y, color, alpha, width, height) { + var graphics = new PIXI.Graphics(); + graphics.beginFill(color, alpha); + graphics.drawRect(0, 0, width, height); + graphics.endFill(); + graphics.x = x; + graphics.y = y; + graphics.width = width; + graphics.height = height; + return graphics; + }; + + this.addNoteRaw = function(data) { + console.log(data); + var note = data.content[3]; + var velocity = data.content[4]; + var ts = (data.content[0] - NTP_EPOCH_DELTA)*1000; + var channel = data.content[2]; + var sessionTs = data.content[1]; + + this.addNote(note, ts, sessionTs, velocity, channel, 0); + }; + + this.addNote = function(note, startTime, sessionTs, velocity, channel, duration) { + + var ts = startTime; + if(this.offsetMusic) { + ts = this.startTs + sessionTs; + } + + var noteDuration = duration; + var noteVelocity = velocity; + var graphics; + if(!duration) { + if(typeof this.noteDict[channel]==='undefined'){ + this.noteDict[channel] = {}; + } + if(velocity===0) { + if(typeof this.noteDict[channel][note] !== 'undefined') { + var noteDef = this.noteDict[channel][note]; + delete this.noteDict[channel][note]; + noteDuration = sessionTs - noteDef.sessionTs; + graphics = noteDef.graphics; + noteVelocity = noteDef.velocity; + ts = noteDef.ts; + } + } + else { + noteDuration = Date.now() - ts; + this.noteDict[channel][note] = { ts: ts, velocity: velocity, sessionTs: sessionTs}; + } + } + + + if(!this.offsetMusic || velocity===0) { + + var width = noteDuration * this.pixelsPerSecond / 1000; + if(!graphics) { + var x = (ts-this.startTs) * this.pixelsPerSecond / 1000; + if((x+width) < (Math.abs(this.container.x) - this.width)) { + // not visible. do nothing + return; + } + var y = Math.floor((128-note+0.5) * this.height / 128 - (this.noteHeight/2)); + var color = this.getColor(channel); + var alpha = (noteVelocity / 128); + + graphics = this.getNoteRect(x, y, color, alpha, width, this.noteHeight); + this.container.addChild(graphics); + } + else { + graphics.width = width; + } + + if(!duration && velocity) { + this.noteDict[channel][note].graphics = graphics; + } + } + }; + + this.addLine = function(ts){ + + if(typeof(ts) === 'undefined') { + ts = new Date(); + } + var x = -this.container.x; + var y = this.linesDown ? this.height - 20 : 0; + + var graphics = new PIXI.Graphics() + .beginFill(0xFFFF00) + .lineStyle(1, this.lineColor) + .moveTo(0, 0) + .lineTo(0, 20) + .endFill(); + graphics.x = x; + graphics.y = y; + this.container.addChild(graphics); + // Add text + //var totalSec = lineNb * this.lineInterval / 1000; + var hours = ts.getHours(); + var minutes =ts.getMinutes(); + var seconds = ts.getSeconds(); + var timeStr = (hours < 10 ? '0' + hours : hours) + ':' + (minutes < 10 ? '0' + minutes : minutes) + ':' + (seconds < 10 ? '0' + seconds : seconds); + + var fontObj = { font: '10pt Arial', fill: '#444444' }; + var t = new PIXI.Text(timeStr, fontObj); + if(isHorizontal) { + t.x = x + 2; + t.y = this.linesDown ? this.height - 15 : 2; + } + else { + t.rotation = -Math.PI/2; + t.x = x ; + t.y = this.linesDown ? this.height - 2 : t.width + 2; + } + this.container.addChild(t); + }; + + this.moveTo = function(diffTime){ + var oldX = this.container.x; + this.container.x = Math.floor(diffTime*this.pixelsPerSecond); + var deltaX = Math.abs(oldX-this.container.x); + _.forOwn(this.noteDict, function(channelDict) { + _.forOwn(channelDict, function(noteDef) { + if(noteDef.graphics) { + noteDef.graphics.width = noteDef.graphics.width + deltaX; + } + }); + }); + }; + + this.move = function() { + var diff = (this.startTs - Date.now())/1000; + this.moveTo(diff); + }; + + this.removePassedObjets = function(){ + var childrenToRemove = []; + _(_this.container.children).forEach(function(child) { + return typeof(child) === 'undefined' || + (isHidden(child) && childrenToRemove.push(child)); + }); + childrenToRemove.forEach(function(child) { + _this.container.removeChild(child); + }); + }; + + this.start = function() { + if(!started) { + this.startTs = Date.now(); + this.addLine(); + started = true; + } + this.verticalLinesInterval = setInterval(function() { _this.addLine(); }, this.lineInterval); + this.cleanInterval = setInterval(function () { _this.removePassedObjets(); }, 1000 * this.width / this.pixelsPerSecond ); + }; + + this.stop = function() { + //window.clearInterval(this.moveInterval); + clearInterval(this.verticalLinesInterval); + clearInterval(this.cleanInterval); + }; + + +} + +module.exports = PianoRoll; + +},{"lodash":"lodash","pixi":"pixi","randomColor":"randomColor"}],8:[function(require,module,exports){ +/** +* scripts/stageview.js +* +* This is the starting point for your application. +* Take a look at http://browserify.org/ for more info +*/ + +/* global document: false */ + +'use strict'; + + +var PIXI = require('pixi'); +var _ = require('lodash'); + +var defaultConfig = { + externalRefresh: false, + logger: undefined, + sceneWidth: 1024, + sceneHeight: 768, + framerate: 25, + sceneBgColor: 0xFFFFFF, + canvasContainer: 'canvasContainer', +}; + +function StageView(options) { + + var _this = this; + var opts = _(options).defaults(defaultConfig).value(); + + var externalRefresh = opts.externalRefresh; + + this.logger = opts.logger; + this.framerate = opts.framerate; + var sceneBgColor = opts.sceneBgColor; + var sceneWidth = opts.sceneWidth; + var sceneHeight = opts.sceneHeight; + var canvasContainer = opts.canvasContainer; + var timeContainer = []; + 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') { + canvasContainer = document.getElementById(canvasContainer); + } + if(typeof(timeContainer) === 'string') { + timeContainer = document.getElementById(timeContainer); + } + + 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); + }; + + this.refresh = function() { + components.forEach(function(c){ + c.refresh(); + }); + renderer.render(this.stage); + }; + + // Init page and intervals + var refreshInterval; + + this.start = function() { + + if(!externalRefresh) { + refreshInterval = setInterval(function() {_this.refresh();}, 1000/this.framerate); + } + + components.forEach(function(c){ + c.start(); + }); + }; + + this.stop = function() { + if(!externalRefresh) { + clearInterval(refreshInterval); + } + clearInterval(refreshTimeInterval); + + components.forEach(function(c){ + c.stop(); + }); + }; + + + this.log = function(m) { + if(this.logger) { + this.logger.log(m); + } + }; + + + return this; +} + +module.exports = { + StageView: StageView +}; + +},{"lodash":"lodash","pixi":"pixi"}],9:[function(require,module,exports){ +/** +* js/utils.js +* +* basic tools +* +*/ + +'use strict'; + +function formatTime (ts) { + var hours = Math.floor( (ts/1000) / 3600 ) % 24; + var minutes = Math.floor( (ts/1000) / 60 ) % 60; + var seconds = Math.floor( (ts/1000) % 60); + return ((hours < 10 ? '0' + hours : hours) + ':' + (minutes < 10 ? '0' + minutes : minutes) + ':' + (seconds < 10 ? '0' + seconds : seconds)); +} + + +module.exports = { + formatTime: formatTime +}; + +},{}],10:[function(require,module,exports){ +/** +* js/wswrapper.js +* +* simple webservice wrapper to register callbacks on onmessage +* +*/ + +/* global WebSocket: false */ + +'use strict'; + +function WsWrapper(wsurl, logger) { + + var url = wsurl; + var sock = new WebSocket(url); + var loggerObj = logger; + + var log = function(msg) { + if(loggerObj) { + loggerObj.log(msg); + } + }; + + var handlers = []; + + sock.onopen = function() { + log('Connected to ' + url); + }; + + sock.onclose = function(e) { + log('Connection closed (wasClean = ' + e.wasClean + ', code = ' + e.code + ', reason = \'' + e.reason + '\')'); + sock = null; + }; + + sock.onmessage = function(e) { + log('received ' + e.data); + var data = JSON.parse(e.data); + handlers.forEach(function(handler) { + handler(data); + }); + }; + + this.message = function(handler) { + if(handler) { + handlers.push(handler); + } + }; + +} + +module.exports = { + WsWrapper: WsWrapper +}; + +},{}]},{},[1])(1) +}); +//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIm5vZGVfbW9kdWxlcy9icm93c2VyaWZ5L25vZGVfbW9kdWxlcy9icm93c2VyLXBhY2svX3ByZWx1ZGUuanMiLCIuL2FwcC9qcy9tYWluLmpzIiwiL1VzZXJzL3ltaC9kZXYvcHJvamVjdHMvbW9ucy9kZXYvY2xpZW50L2Fubm90dml6L2FwcC9qcy9hbm5vdHNyb2xsLmpzIiwiL1VzZXJzL3ltaC9kZXYvcHJvamVjdHMvbW9ucy9kZXYvY2xpZW50L2Fubm90dml6L2FwcC9qcy9hbm5vdHN0aW1lbGluZS5qcyIsIi9Vc2Vycy95bWgvZGV2L3Byb2plY3RzL21vbnMvZGV2L2NsaWVudC9hbm5vdHZpei9hcHAvanMvYW5ub3Rzdml6dmlldy5qcyIsIi9Vc2Vycy95bWgvZGV2L3Byb2plY3RzL21vbnMvZGV2L2NsaWVudC9hbm5vdHZpei9hcHAvanMvZG91Ymxlcm9sbC5qcyIsIi9Vc2Vycy95bWgvZGV2L3Byb2plY3RzL21vbnMvZGV2L2NsaWVudC9hbm5vdHZpei9hcHAvanMvbG9nZ2VyLmpzIiwiL1VzZXJzL3ltaC9kZXYvcHJvamVjdHMvbW9ucy9kZXYvY2xpZW50L2Fubm90dml6L2FwcC9qcy9waWFub3JvbGwuanMiLCIvVXNlcnMveW1oL2Rldi9wcm9qZWN0cy9tb25zL2Rldi9jbGllbnQvYW5ub3R2aXovYXBwL2pzL3N0YWdldmlldy5qcyIsIi9Vc2Vycy95bWgvZGV2L3Byb2plY3RzL21vbnMvZGV2L2NsaWVudC9hbm5vdHZpei9hcHAvanMvdXRpbHMuanMiLCIvVXNlcnMveW1oL2Rldi9wcm9qZWN0cy9tb25zL2Rldi9jbGllbnQvYW5ub3R2aXovYXBwL2pzL3dzd3JhcHBlci5qcyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQTtBQ0FBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBOztBQzNCQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBOztBQzdNQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBOztBQ3ROQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTs7QUM1SkE7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7O0FDN0xBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBOztBQzdDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBOztBQ3JPQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTs7QUNySEE7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBOztBQ3BCQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0E7QUFDQTtBQUNBO0FBQ0EiLCJmaWxlIjoiZ2VuZXJhdGVkLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXNDb250ZW50IjpbIihmdW5jdGlvbiBlKHQsbixyKXtmdW5jdGlvbiBzKG8sdSl7aWYoIW5bb10pe2lmKCF0W29dKXt2YXIgYT10eXBlb2YgcmVxdWlyZT09XCJmdW5jdGlvblwiJiZyZXF1aXJlO2lmKCF1JiZhKXJldHVybiBhKG8sITApO2lmKGkpcmV0dXJuIGkobywhMCk7dmFyIGY9bmV3IEVycm9yKFwiQ2Fubm90IGZpbmQgbW9kdWxlICdcIitvK1wiJ1wiKTt0aHJvdyBmLmNvZGU9XCJNT0RVTEVfTk9UX0ZPVU5EXCIsZn12YXIgbD1uW29dPXtleHBvcnRzOnt9fTt0W29dWzBdLmNhbGwobC5leHBvcnRzLGZ1bmN0aW9uKGUpe3ZhciBuPXRbb11bMV1bZV07cmV0dXJuIHMobj9uOmUpfSxsLGwuZXhwb3J0cyxlLHQsbixyKX1yZXR1cm4gbltvXS5leHBvcnRzfXZhciBpPXR5cGVvZiByZXF1aXJlPT1cImZ1bmN0aW9uXCImJnJlcXVpcmU7Zm9yKHZhciBvPTA7bzxyLmxlbmd0aDtvKyspcyhyW29dKTtyZXR1cm4gc30pIiwiLyoqXG4gKiBzY3JpcHRzL21haW4uanNcbiAqXG4gKiBUaGlzIGlzIHRoZSBzdGFydGluZyBwb2ludCBmb3IgeW91ciBhcHBsaWNhdGlvbi5cbiAqIFRha2UgYSBsb29rIGF0IGh0dHA6Ly9icm93c2VyaWZ5Lm9yZy8gZm9yIG1vcmUgaW5mb1xuICovXG5cbid1c2Ugc3RyaWN0JztcblxudmFyIGRvdWJsZXJvbGwgPSByZXF1aXJlKCcuL2RvdWJsZXJvbGwnKTtcbnZhciBhbm5vdHNyb2xsID0gcmVxdWlyZSgnLi9hbm5vdHNyb2xsJyk7XG52YXIgYW5ub3RzdGltZWxpbmUgPSByZXF1aXJlKCcuL2Fubm90c3RpbWVsaW5lJyk7XG52YXIgYW5ub3Rzdml6dmlldyA9IHJlcXVpcmUoJy4vYW5ub3Rzdml6dmlldycpO1xudmFyIHN0YWdldmlldyA9IHJlcXVpcmUoJy4vc3RhZ2V2aWV3Jyk7XG52YXIgd3N3cmFwcGVyID0gcmVxdWlyZSgnLi93c3dyYXBwZXInKTtcbnZhciBsb2dnZXIgPSByZXF1aXJlKCcuL2xvZ2dlcicpO1xuXG52YXIgXyA9IHJlcXVpcmUoJ2xvZGFzaCcpO1xuXG5tb2R1bGUuZXhwb3J0cyA9IF8oe30pXG4gICAgLmV4dGVuZChkb3VibGVyb2xsKVxuICAgIC5leHRlbmQoYW5ub3Rzcm9sbClcbiAgICAuZXh0ZW5kKGFubm90c3RpbWVsaW5lKVxuICAgIC5leHRlbmQoYW5ub3Rzdml6dmlldylcbiAgICAuZXh0ZW5kKHN0YWdldmlldylcbiAgICAuZXh0ZW5kKHdzd3JhcHBlcilcbiAgICAuZXh0ZW5kKGxvZ2dlcilcbiAgICAudmFsdWUoKTsiLCIvKipcbioganMvYW5ub3RzUm9sbC5qc1xuKlxuKiBhbm5vdHNSb2xsIGJhc2ljIGNvbXBvbmVudFxuKlxuKi9cblxuJ3VzZSBzdHJpY3QnO1xuXG52YXIgUElYSSA9IHJlcXVpcmUoJ3BpeGknKTtcbnZhciBfID0gcmVxdWlyZSgnbG9kYXNoJyk7XG5cbnZhciBERUZBVUxUX0FOTk9UX0NPTE9SID0gJyNiYWJhYmEnO1xuXG52YXIgZGVmYXVsdEFubm90U3R5bGVzID0ge1xuICAgICdsYWJlbCc6IHsgZm9udDogJzE2cHQgQXJpYWwgQm9sZCcsIGZpbGw6ICcjNjVBOTU0Jywgd29yZFdyYXA6IHRydWV9LFxuICAgICd0ZXh0JyA6IHsgZm9udDogJzEycHQgQXJpYWwgUmVndWxhcicsIGZpbGw6ICcjNDQ0NDQ0Jywgd29yZFdyYXA6IHRydWV9LFxuICAgICd1c2VyJyA6IHsgZm9udDogJzE0cHQgQXJpYWwgcmVndWxhcicsIGZpbGw6ICcjNjY2NjY2JyB9LFxufTtcblxudmFyIGRlZmF1bHRPcHRpb25zID0ge1xuICAgIGV4dGVybmFsUmVmcmVzaDogZmFsc2UsXG4gICAgZGVmYXVsdENvbG9yOiBERUZBVUxUX0FOTk9UX0NPTE9SLFxuICAgIGFubm90U3R5bGVzOiBkZWZhdWx0QW5ub3RTdHlsZXNcbn07XG5cbmZ1bmN0aW9uIEFubm90c1JvbGwob3B0aW9ucykge1xuXG4vL3BhcmVudENvbnRhaW5lciwgeEluaXQsIHlJbml0LCB3aWR0aCwgaGVpZ2h0LCB3aWR0aFJvbGwsIHBpeGVsc1BlclNlY29uZCwgYW5ub3RDb2xvcnNcbiAgICB2YXIgX3RoaXMgPSB0aGlzO1xuICAgIHZhciBvcHRzID0gXyhvcHRpb25zKS5kZWZhdWx0cyhkZWZhdWx0T3B0aW9ucykudmFsdWUoKTtcblxuXG4gICAgdGhpcy5jb250YWluZXIgPSBuZXcgUElYSS5EaXNwbGF5T2JqZWN0Q29udGFpbmVyKCk7XG4gICAgdGhpcy5jb250YWluZXIueCA9IG9wdHMueEluaXQ7XG4gICAgdGhpcy5jb250YWluZXIueSA9IG9wdHMueUluaXQ7XG4gICAgdGhpcy5jb250YWluZXIud2lkdGggPSBvcHRzLndpZHRoO1xuXG4gICAgdGhpcy5oZWlnaHQgPSBvcHRzLmhlaWdodDtcbiAgICB0aGlzLndpZHRoID0gb3B0cy53aWR0aDtcbiAgICB0aGlzLndpZHRoUm9sbCA9IG9wdHMud2lkdGhSb2xsO1xuICAgIHRoaXMucGl4ZWxzUGVyU2Vjb25kID0gb3B0cy5waXhlbHNQZXJTZWNvbmQ7XG4gICAgdGhpcy5hbm5vdENvbG9ycyA9IG9wdHMuYW5ub3RDb2xvcnM7XG4gICAgdGhpcy5zdGFydFRzID0gb3B0aW9ucy5zdGFydFRzIHx8IERhdGUubm93KCk7XG5cbiAgICB2YXIgeUluaXQgPSBvcHRzLnlJbml0O1xuICAgIHZhciBhbm5vdFN0eWxlcyA9IF8ob3B0cy5hbm5vdFN0eWxlcykuZGVmYXVsdHMoZGVmYXVsdEFubm90U3R5bGVzKS52YWx1ZSgpO1xuICAgIGZvcih2YXIgc3R5bGUgaW4gYW5ub3RTdHlsZXMpIHtcbiAgICBcdGlmIChhbm5vdFN0eWxlc1tzdHlsZV0ud29yZFdyYXAgPT09IHRydWUpe1xuICAgIFx0XHRhbm5vdFN0eWxlc1tzdHlsZV0ud29yZFdyYXBXaWR0aCA9IHRoaXMud2lkdGhSb2xsO1xuICAgIFx0fVxuICAgIH1cbiAgICB2YXIgc3RhcnRlZCA9IGZhbHNlO1xuICAgIHZhciB3cyA9IG9wdHMud3M7XG4gICAgdmFyIGV4dGVybmFsUmVmcmVzaCA9IG9wdHMuZXh0ZXJuYWxSZWZyZXNoO1xuICAgIHZhciBzdGFnZVZpZXcgPSBvcHRzLnN0YWdlVmlldztcblxuICAgIHN0YWdlVmlldy5yZWdpc3RlckNvbXBvbmVudCh0aGlzKTtcblxuICAgIHZhciBpc0hpZGRlbiA9IGZ1bmN0aW9uKGNoaWxkKSB7XG4gICAgICAgIC8vIFRPRE86IHRoZSBvcmlnaW4gcG9pbnQgaXMgYW4gYXBwcm94aW1hdGlvbi4gU2hvdWxkIHJlZmluZSB0aGlzXG4gICAgICAgIHZhciBnbG9iYWxQb3MgPSBjaGlsZC50b0dsb2JhbChuZXcgUElYSS5Qb2ludCgwLDApKTtcbiAgICAgICAgcmV0dXJuICgoZ2xvYmFsUG9zLnggKyBjaGlsZC53aWR0aCkgPCAwKSB8fCAoKGdsb2JhbFBvcy55ICsgY2hpbGQuaGVpZ2h0KSA8IDApIDtcbiAgICB9O1xuXG4gICAgdGhpcy5hZGRBbm5vdHMgPSBmdW5jdGlvbihkYXRhKSB7XG5cbiAgICAgICAgLy92YXIgdGl0bGUgPSBkYXRhLmNvbnRlbnQuY2F0ZWdvcnkubGFiZWw7XG4gICAgICAgIC8vdmFyIHVzZXIgPSBkYXRhLmNvbnRlbnQudXNlcjtcbiAgICAgICAgLy9UZXN0IGNhdCBhbmQgY29sb3JcbiAgICAgICAgLy92YXIgY29sb3JBbm5vdCA9IDB4NjVBOTU0O1xuICAgICAgICB2YXIgY2F0ZWdvcnkgPSBkYXRhLmNvbnRlbnQuY2F0ZWdvcnkubGFiZWwsXG4gICAgICAgICAgICB0ZXh0ICAgICA9IGRhdGEuY29udGVudC50ZXh0LFxuICAgICAgICAgICAgdXNlciAgICAgPSBkYXRhLmNvbnRlbnQudXNlcixcbiAgICAgICAgICAgIHRzICAgICAgID0gRGF0ZS5wYXJzZShkYXRhLnRzKSxcbiAgICAgICAgICAgIGNvbG9yICAgID0gZGF0YS5jb250ZW50LmNvbG9yIHx8IHRoaXMuZ2V0Q29sb3IodHMsIGRhdGEuY29udGVudC5jYXRlZ29yeS5jb2RlKTtcblxuICAgICAgICB0aGlzLmFkZEFubm90KGNhdGVnb3J5LCB0ZXh0LCB1c2VyLCBjb2xvciwgdHMpO1xuICAgIH07XG5cbiAgICB0aGlzLmdldENvbG9yID0gZnVuY3Rpb24odHMsIGNvZGUpIHtcbiAgICAgICAgdmFyIGNvbG9yc0RlZjtcbiAgICAgICAgXyh0aGlzLmFubm90Q29sb3JzKS5lYWNoUmlnaHQoZnVuY3Rpb24oY2RlZikge1xuICAgICAgICAgICAgY29uc29sZS5sb2coXCJjRGVmXCIsIGNkZWYpO1xuICAgICAgICAgICAgY29uc29sZS5sb2coXCJjRGVmIHRzXCIsIGNkZWYudHMsIHRzKTtcbiAgICAgICAgICAgIGlmKGNkZWYudHMgPCB0cykge1xuICAgICAgICAgICAgICAgIGNvbG9yc0RlZiA9IGNkZWYuY29sb3JzO1xuICAgICAgICAgICAgICAgIHJldHVybiBmYWxzZTtcbiAgICAgICAgICAgIH1cbiAgICAgICAgfSk7XG4gICAgICAgIHZhciByZXNDb2xvcjtcbiAgICAgICAgY29uc29sZS5sb2coXCJjb2xvcnNEZWZcIiwgY29sb3JzRGVmKTtcbiAgICAgICAgaWYoY29sb3JzRGVmKSB7XG4gICAgICAgICAgICByZXNDb2xvciA9IGNvbG9yc0RlZltjb2RlXTtcbiAgICAgICAgfVxuICAgICAgICBpZighcmVzQ29sb3IpIHtcbiAgICAgICAgICAgIHJlc0NvbG9yID0gREVGQVVMVF9BTk5PVF9DT0xPUjtcbiAgICAgICAgfVxuICAgICAgICByZXR1cm4gcmVzQ29sb3I7XG4gICAgfVxuXG4gICAgdGhpcy5hZGRBbm5vdCA9IGZ1bmN0aW9uKGNhdGVnb3J5LCB0ZXh0LCB1c2VyLCBjYXRDb2xvciwgdHMpe1xuXG4gICAgICAgIHZhciBjb2xvciA9IGNhdENvbG9yID8gY2F0Q29sb3IgOiBERUZBVUxUX0FOTk9UX0NPTE9SO1xuICAgICAgICB2YXIgeCA9IDA7XG4gICAgICAgIHZhciB5ID0gKHRzLXRoaXMuc3RhcnRUcykgKiB0aGlzLnBpeGVsc1BlclNlY29uZCAvIDEwMDAgKyB5SW5pdDtcblxuICAgICAgICB2YXIgY29sb3JIZXggPSBwYXJzZUludChjb2xvci5yZXBsYWNlKC9eIy8sICcnKSwgMTYpO1xuXG4gICAgICAgIHZhciBncmFwaGljcyA9IG5ldyBQSVhJLkdyYXBoaWNzKClcbiAgICAgICAgICAgIC5iZWdpbkZpbGwoY29sb3JIZXgpXG4gICAgICAgICAgICAuZHJhd1JlY3QoeCwgeSwgMTAsIDMpXG4gICAgICAgICAgICAuZW5kRmlsbCgpO1xuXG4gICAgICAgIHRoaXMuY29udGFpbmVyLmFkZENoaWxkKGdyYXBoaWNzKTtcblxuICAgICAgICB2YXIgdGV4dEhlaWdodCA9IDA7XG4gICAgICAgIHZhciBjYXRMYWJlbCA9IG5ldyBQSVhJLlRleHQoXG4gICAgICAgICAgICBjYXRlZ29yeSxcbiAgICAgICAgICAgIF8oYW5ub3RTdHlsZXMubGFiZWwpLmV4dGVuZCh7ZmlsbDogY29sb3J9KS52YWx1ZSgpXG4gICAgICAgICk7XG4gICAgICAgIGNhdExhYmVsLnggPSB4ICsgMjA7XG4gICAgICAgIGNhdExhYmVsLnkgPSB5IC0gMjM7XG4gICAgICAgIHRoaXMuY29udGFpbmVyLmFkZENoaWxkKGNhdExhYmVsKTtcbiAgICAgICAgdGV4dEhlaWdodCArPSAoY2F0TGFiZWwuaGVpZ2h0IC0gMjMgKyAyKTtcblxuICAgICAgICBpZih0ZXh0KSB7XG4gICAgICAgICAgICB2YXIgY2F0VGV4dCA9IG5ldyBQSVhJLlRleHQodGV4dCwgYW5ub3RTdHlsZXMudGV4dCk7XG4gICAgICAgICAgICBjYXRUZXh0LnggPSB4ICsgMjA7XG4gICAgICAgICAgICBjYXRUZXh0LnkgPSB5ICsgMjtcbiAgICAgICAgICAgIHRoaXMuY29udGFpbmVyLmFkZENoaWxkKGNhdFRleHQpO1xuICAgICAgICAgICAgdGV4dEhlaWdodCArPSAoY2F0VGV4dC5oZWlnaHQgKyAyKTtcbiAgICAgICAgfVxuXG4gICAgICAgIHZhciBjYXRVc2VyID0gbmV3IFBJWEkuVGV4dCh1c2VyLCBhbm5vdFN0eWxlcy51c2VyKTtcbiAgICAgICAgY2F0VXNlci54ID0geCArIDIwO1xuICAgICAgICBjYXRVc2VyLnkgPSB5ICsgMiArIHRleHRIZWlnaHQ7XG4gICAgICAgIHRoaXMuY29udGFpbmVyLmFkZENoaWxkKGNhdFVzZXIpO1xuXG4gICAgICAgIHRoaXMuYWRkQW5ub3RMaW5lKGNvbG9ySGV4LCB5KTtcbiAgICB9O1xuXG4gICAgdGhpcy5hZGRBbm5vdExpbmUgPSBmdW5jdGlvbihjb2xvciwgeSkge1xuICAgICAgICB2YXIgeCA9IHRoaXMud2lkdGhSb2xsO1xuXG5cbiAgICAgICAgdmFyIGdyYXBoaWNzID0gbmV3IFBJWEkuR3JhcGhpY3MoKVxuICAgICAgICAgICAgLmJlZ2luRmlsbChjb2xvcilcbiAgICAgICAgICAgIC5kcmF3UmVjdCh4LCB5LCB0aGlzLndpZHRoIC0geCwgMylcbiAgICAgICAgICAgIC5lbmRGaWxsKCk7XG5cbiAgICAgICAgdGhpcy5jb250YWluZXIuYWRkQ2hpbGQoZ3JhcGhpY3MpO1xuICAgIH07XG5cbiAgICB0aGlzLm1vdmVUbyA9IGZ1bmN0aW9uKGRpZmZUaW1lKXtcbiAgICBcdHRoaXMuY29udGFpbmVyLnkgPSBNYXRoLmZsb29yKGRpZmZUaW1lKnRoaXMucGl4ZWxzUGVyU2Vjb25kKTtcbiAgICB9O1xuXG4gICAgdGhpcy5tb3ZlID0gdGhpcy5yZWZyZXNoID0gZnVuY3Rpb24oKSB7XG4gICAgICAgIHZhciBkaWZmID0gKHRoaXMuc3RhcnRUcyAtIERhdGUubm93KCkpLzEwMDA7XG4gICAgICAgIHRoaXMubW92ZVRvKGRpZmYpO1xuICAgIH07XG5cbiAgICB0aGlzLnJlbW92ZVBhc3NlZE9iamV0cyA9IGZ1bmN0aW9uKCl7XG4gICAgICAgIHZhciBjaGlsZHJlblRvUmVtb3ZlID0gW107XG4gICAgICAgIF8oX3RoaXMuY29udGFpbmVyLmNoaWxkcmVuKS5mb3JFYWNoKGZ1bmN0aW9uKGNoaWxkKSB7XG4gICAgICAgICAgICByZXR1cm4gdHlwZW9mKGNoaWxkKSA9PT0gJ3VuZGVmaW5lZCcgfHxcbiAgICAgICAgICAgICAgICAoaXNIaWRkZW4oY2hpbGQpICYmIGNoaWxkcmVuVG9SZW1vdmUucHVzaChjaGlsZCkpO1xuICAgICAgICB9KTtcbiAgICAgICAgY2hpbGRyZW5Ub1JlbW92ZS5mb3JFYWNoKGZ1bmN0aW9uKGNoaWxkKSB7XG4gICAgICAgICAgICBfdGhpcy5jb250YWluZXIucmVtb3ZlQ2hpbGQoY2hpbGQpO1xuICAgICAgICB9KTtcbiAgICB9O1xuXG4gICAgdGhpcy5pbml0ID0gZnVuY3Rpb24oKSB7XG5cbiAgICAgICAgd3MubWVzc2FnZShmdW5jdGlvbihkYXRhKSB7XG4gICAgICAgICAgICBfdGhpcy5hZGRBbm5vdHMoZGF0YSk7XG4gICAgICAgIH0pO1xuXG4gICAgfTtcblxuICAgIHRoaXMuc3RhcnQgPSBmdW5jdGlvbigpIHtcbiAgICAgICAgaWYoIXN0YXJ0ZWQpIHtcbiAgICAgICAgICAgIHRoaXMuc3RhcnRUcyA9IERhdGUubm93KCk7XG4gICAgICAgICAgICBzdGFydGVkID0gdHJ1ZTtcbiAgICAgICAgfVxuICAgICAgICB0aGlzLmNsZWFuSW50ZXJ2YWwgPSBzZXRJbnRlcnZhbChmdW5jdGlvbiAoKSB7IF90aGlzLnJlbW92ZVBhc3NlZE9iamV0cygpOyB9LCAxMDAwICogdGhpcy5oZWlnaHQgLyB0aGlzLnBpeGVsc1BlclNlY29uZCApO1xuICAgICAgICBpZighZXh0ZXJuYWxSZWZyZXNoKSB7XG4gICAgICAgICAgICB0aGlzLnJlZnJlc2hJbnRlcnZhbCA9IHNldEludGVydmFsKGZ1bmN0aW9uKCkge190aGlzLm1vdmUoKTt9LCAxMDAwL3RoaXMuZnJhbWVyYXRlKTtcbiAgICAgICAgfVxuICAgIH07XG5cbiAgICB0aGlzLnN0b3AgPSBmdW5jdGlvbigpIHtcbiAgICAgICAgY2xlYXJJbnRlcnZhbCh0aGlzLmNsZWFuSW50ZXJ2YWwpO1xuICAgICAgICBpZighZXh0ZXJuYWxSZWZyZXNoKSB7XG4gICAgICAgICAgICBjbGVhckludGVydmFsKHRoaXMucmVmcmVzaEludGVydmFsKTtcbiAgICAgICAgfVxuICAgIH07XG5cbn1cblxubW9kdWxlLmV4cG9ydHMgPSB7XG4gICAgQW5ub3RzUm9sbDogQW5ub3RzUm9sbCxcbn07XG4iLCIvKipcbioganMvYW5ub3RzdGltZWxpbmVcbipcbiogYW5ub3RzdGltZWxpbmUgYmFzaWMgY29tcG9uZW50XG4qXG4qL1xuXG4ndXNlIHN0cmljdCc7XG5cbnZhciBQSVhJID0gcmVxdWlyZSgncGl4aScpO1xudmFyIFV0aWxzID0gcmVxdWlyZSgnLi91dGlscy5qcycpO1xudmFyIF8gPSByZXF1aXJlKCdsb2Rhc2gnKTtcblxudmFyIGRlZmF1bHRPcHRpb25zID0ge1xuICAgIGxvZ2dlcjogdW5kZWZpbmVkLFxuICAgIGludGVydmFsV2lkdGg6IDEwLFxuICAgIGludGVydmFsSGVpZ2h0OiA1LFxuICAgIG1heENlbGxIZWlnaHQ6IDIwMCxcbiAgICByYWRpdXM6IDMwMFxufTtcblxuXG5mdW5jdGlvbiBBbm5vdHNUaW1lTGluZShvcHRpb25zKXtcbiAgICB2YXIgX3RoaXMgPSB0aGlzO1xuICAgIHZhciBvcHRzID0gXyhvcHRpb25zKS5kZWZhdWx0cyhkZWZhdWx0T3B0aW9ucykudmFsdWUoKTtcblxuICAgIHRoaXMuY29udGFpbmVyID0gbmV3IFBJWEkuRGlzcGxheU9iamVjdENvbnRhaW5lcigpO1xuICAgIHRoaXMuY29udGFpbmVyLnggPSBvcHRzLnhJbml0O1xuICAgIHRoaXMuY29udGFpbmVyLnkgPSBvcHRzLnlJbml0O1xuICAgIHRoaXMuY29udGFpbmVyLndpZHRoID0gb3B0cy53aWR0aDtcbiAgICB0aGlzLmNvbnRhaW5lci5oZWlnaHQgPSBvcHRzLmhlaWdodDtcblxuICAgIHRoaXMudGltZUJlZ2luID0gb3B0cy50aW1lQmVnaW47XG4gICAgdGhpcy50aW1lRW5kID0gb3B0cy50aW1lRW5kO1xuICAgIHRoaXMuZHVyYXRpb24gPSAodGhpcy50aW1lRW5kIC0gdGhpcy50aW1lQmVnaW4pLzEwMDA7XG4gICAgdGhpcy53aWR0aCA9IG9wdHMud2lkdGg7XG4gICAgdGhpcy5oZWlnaHQgPSBvcHRzLmhlaWdodDtcbiAgICB0aGlzLmludGVydmFsSGVpZ2h0ID0gb3B0cy5pbnRlcnZhbEhlaWdodDtcbiAgICB0aGlzLmludGVydmFsV2lkdGggPSBvcHRzLmludGVydmFsV2lkdGg7XG4gICAgdGhpcy5tYXhDZWxsSGVpZ2h0ID0gb3B0cy5tYXhDZWxsSGVpZ2h0O1xuICAgIHRoaXMuYW5ub3RDYXRlZ29yaWVzID0gb3B0cy5hbm5vdENhdGVnb3JpZXM7XG5cbiAgICB0aGlzLmNpcmNsZVggPSBvcHRzLmNpcmNsZVggfHwgKHRoaXMud2lkdGgvMik7XG4gICAgdGhpcy5jaXJjbGVZID0gb3B0cy5jaXJjbGVZIHx8ICh0aGlzLmhlaWdodC8yKTtcbiAgICB0aGlzLnJhZGl1cyA9IG9wdHMucmFkaXVzO1xuICAgIHRoaXMucGVyaW1ldGVyID0gMipNYXRoLlBJKiB0aGlzLnJhZGl1cztcbiAgICB0aGlzLmludGVydmFsRHVyYXRpb24gPSAodGhpcy5pbnRlcnZhbFdpZHRoICogdGhpcy5kdXJhdGlvbiAvIHRoaXMucGVyaW1ldGVyKTtcblxuICAgIHZhciBjdXJyZW50VGltZSA9IHRoaXMudGltZUJlZ2luO1xuICAgIHZhciB0b3RhbEluZGV4ID0gTWF0aC5mbG9vcih0aGlzLnBlcmltZXRlci90aGlzLmludGVydmFsV2lkdGgpO1xuXG4gICAgdGhpcy5jZWxscyA9IFtdXG4gICAgZm9yICh2YXIgaT0wOyBpPCh0aGlzLnBlcmltZXRlci90aGlzLmludGVydmFsV2lkdGgpIDsgaSsrKXtcbiAgICBcdHRoaXMuY2VsbHNbaV0gPSBbXTtcbiAgICBcdHRoaXMuY2VsbHNbaV0uaSA9IGk7XG4gICAgXHR0aGlzLmNlbGxzW2ldLnRvdGFsQW5ub3RzID0gMDtcbiAgICBcdHRoaXMuY2VsbHNbaV0uY2F0ZWdvcmllcyA9IHt9O1xuXG4gICAgXHRmb3IgKHZhciBjYXRlZ29yeSBpbiB0aGlzLmFubm90Q2F0ZWdvcmllc1swXS5jb2xvcnMpe1xuICAgIFx0XHR0aGlzLmNlbGxzW2ldLmNhdGVnb3JpZXNbY2F0ZWdvcnldID0ge1xuXHRcdFx0XHRcImNvdW50XCI6IDAsXG5cdFx0XHRcdFwiY29sb3JcIjogdGhpcy5hbm5vdENhdGVnb3JpZXNbMF0uY29sb3JzW2NhdGVnb3J5XVxuICAgIFx0XHR9O1xuICAgIFx0fVxuICAgIH1cblxuICAgIHZhciB3cyA9IG9wdHMud3M7XG4gICAgdmFyIHN0YWdlVmlldyA9IG9wdHMuc3RhZ2VWaWV3O1xuXG4gICAgLy9kcmF3IHRoZSBiYXNlIC0gY2lyY2xlIGFuZCBsaW5lIHRvIGxvY2F0ZSB0aGUgc2NlbmVcbiAgICB2YXIgZ3JhcGhpY3MgPSBuZXcgUElYSS5HcmFwaGljcygpO1xuICAgIGdyYXBoaWNzLmxpbmVTdHlsZSgyLCAweDY0NjQ2NClcbiAgICBcdC5kcmF3Q2lyY2xlKHRoaXMuY2lyY2xlWCwgdGhpcy5jaXJjbGVZLCB0aGlzLnJhZGl1cyAtIDMpXG4gICAgXHQubGluZVN0eWxlKDEsIDB4RDdEN0Q3KVxuICAgIFx0LmRyYXdDaXJjbGUodGhpcy5jaXJjbGVYLCB0aGlzLmNpcmNsZVksIHRoaXMucmFkaXVzKjIvMylcbiAgICBcdC5kcmF3Q2lyY2xlKHRoaXMuY2lyY2xlWCwgdGhpcy5jaXJjbGVZLCB0aGlzLnJhZGl1cy8zKVxuICAgIFx0LmxpbmVTdHlsZSgxLCAweDY0NjQ2NClcbiAgICBcdC5tb3ZlVG8odGhpcy5jaXJjbGVYLCB0aGlzLmNpcmNsZVkgLSAodGhpcy5yYWRpdXMvMykvMilcbiAgICBcdC5saW5lVG8odGhpcy5jaXJjbGVYLCB0aGlzLmNpcmNsZVkgLSB0aGlzLnJhZGl1cyAtIHRoaXMubWF4Q2VsbEhlaWdodCAtIDEwKVxuICAgIFx0LmVuZEZpbGwoKVxuICAgIHRoaXMuY29udGFpbmVyLmFkZENoaWxkKGdyYXBoaWNzKTtcblxuICAgIC8vc2V0IHRpbWUgdGV4dFxuICAgIC8vVE9ETyA6IG1vdmUgdGhpcyB0byBhbm5vdHN2aXp2aWV3XG4gICAgdmFyIGN1cnJlbnRUaW1lVGV4dCA9IG5ldyBQSVhJLlRleHQoXCItLSA6IC0tIDogLS1cIiwgeyBmb250OiAnMThwdCBHb3RoaWMgU3RhbmRhcmQnLCBmaWxsOiAnIzY0NjQ2NCcgfSk7XG4gICAgY3VycmVudFRpbWVUZXh0LnggPSB0aGlzLmNpcmNsZVggLSBjdXJyZW50VGltZVRleHQud2lkdGgvMjtcbiAgICBjdXJyZW50VGltZVRleHQueSA9IHRoaXMuY2lyY2xlWSAtIGN1cnJlbnRUaW1lVGV4dC5oZWlnaHQvMjtcbiAgICB0aGlzLmNvbnRhaW5lci5hZGRDaGlsZChjdXJyZW50VGltZVRleHQpO1xuXG4gICAgc3RhZ2VWaWV3LnJlZ2lzdGVyQ29tcG9uZW50KHRoaXMpO1xuXG4gICAgLy9BZGQgQW5ub3RhdGlvbiB0byB0aGUgVGltZUxpbmVcbiAgICB0aGlzLmFkZEFubm90ID0gZnVuY3Rpb24oZGF0YSl7XG4gICAgXHRpZiAodHlwZW9mKHRoaXMuYW5ub3RDYXRlZ29yaWVzWzBdLmNvbG9yc1tkYXRhLmNvbnRlbnQuY2F0ZWdvcnkuY29kZV0pICE9PSAndW5kZWZpbmVkJyl7XG4gICAgXHRcdHZhciBhbm5vdENvZGUgPSBkYXRhLmNvbnRlbnQuY2F0ZWdvcnkuY29kZTtcbiAgICBcdH0gZWxzZSB7XG4gICAgXHRcdHZhciBhbm5vdENvZGUgPSB0aGlzLmFubm90Q2F0ZWdvcmllc1swXS5vcmRlclt0aGlzLmFubm90Q2F0ZWdvcmllc1swXS5vcmRlci5sZW5ndGggLTFdO1xuICAgIFx0fVxuICAgIFx0dmFyIGFubm90VGltZSA9IERhdGUucGFyc2UoZGF0YS50cyk7XG5cbiAgICBcdGlmICh0aGlzLnRpbWVFbmQgPiBEYXRlLnBhcnNlKGRhdGEudHMpKXtcblx0ICAgIFx0dmFyIGkgPSBNYXRoLmZsb29yKChEYXRlLnBhcnNlKGRhdGEudHMpLXRoaXMudGltZUJlZ2luKS8oMTAwMCp0aGlzLmludGVydmFsRHVyYXRpb24pKTtcblxuXHRcdFx0dGhpcy5jZWxsc1tpXS5jYXRlZ29yaWVzW2Fubm90Q29kZV0uY291bnQgKz0gMTtcblx0XHRcdHRoaXMuY2VsbHNbaV0udG90YWxBbm5vdHMgKz0xO1xuXHRcdFx0dGhpcy5yZWRyYXdDZWxsKHRoaXMuY2VsbHNbaV0sIGkpO1xuICAgIFx0fVxuICAgIH07XG5cbiAgICB0aGlzLmluaXRHcmFwaGljcyA9IGZ1bmN0aW9uKGNlbGwpe1xuICAgIFx0Y2VsbC5ncmFwaGljcyA9IG5ldyBQSVhJLkdyYXBoaWNzKCk7XG4gICAgXHRjZWxsLmdyYXBoaWNzLnBvc2l0aW9uLnggPSB0aGlzLmNpcmNsZVggKyB0aGlzLnJhZGl1cyAqIE1hdGguc2luKGNlbGwuaSooMzYwL3RvdGFsSW5kZXgpKihNYXRoLlBJLzE4MCkpO1xuICAgIFx0Y2VsbC5ncmFwaGljcy5wb3NpdGlvbi55ID0gdGhpcy5jaXJjbGVZIC0gdGhpcy5yYWRpdXMgKiBNYXRoLmNvcyhjZWxsLmkqKDM2MC90b3RhbEluZGV4KSooTWF0aC5QSS8xODApKTtcbiAgICBcdGNlbGwuZ3JhcGhpY3Mucm90YXRpb24gPSAoY2VsbC5pKSooMzYwL3RvdGFsSW5kZXgpKihNYXRoLlBJLzE4MCkgKyAoMzYwLyh0b3RhbEluZGV4KjIpKSooTWF0aC5QSS8xODApO1xuICAgIFx0dGhpcy5jb250YWluZXIuYWRkQ2hpbGQoY2VsbC5ncmFwaGljcyk7XG4gICAgfVxuXG4gICAgdGhpcy5pbml0VGltZVRleHRzID0gZnVuY3Rpb24oKSB7XG5cdCAgICB2YXIgdEJlZyA9IG5ldyBQSVhJLlRleHQoVXRpbHMuZm9ybWF0VGltZSh0aGlzLnRpbWVCZWdpbiksIHsgZm9udDogJzEycHQgR290aGljIFN0YW5kYXJkJywgZmlsbDogJyM2NDY0NjQnIH0pO1xuXHQgICAgdEJlZy54ID0gdGhpcy5jaXJjbGVYICsgMTU7XG5cdCAgICB0QmVnLnkgPSB0aGlzLmNpcmNsZVkgLSB0aGlzLnJhZGl1cyAtIHRoaXMubWF4Q2VsbEhlaWdodCAtIDEwO1xuXHQgICAgdGhpcy5jb250YWluZXIuYWRkQ2hpbGQodEJlZyk7XG5cblx0ICAgIHZhciB0RW5kID0gbmV3IFBJWEkuVGV4dChVdGlscy5mb3JtYXRUaW1lKHRoaXMudGltZUVuZCksIHsgZm9udDogJzEycHQgR290aGljIFN0YW5kYXJkJywgZmlsbDogJyM2NDY0NjQnIH0pO1xuXHQgICAgdEVuZC54ID0gdGhpcy5jaXJjbGVYIC0gMTUgLSB0RW5kLndpZHRoO1xuXHQgICAgdEVuZC55ID0gdGhpcy5jaXJjbGVZIC0gdGhpcy5yYWRpdXMgLSB0aGlzLm1heENlbGxIZWlnaHQgLSAxMDtcblx0ICAgIHRoaXMuY29udGFpbmVyLmFkZENoaWxkKHRFbmQpO1xuXG5cdCAgICB2YXIgdDE1ID0gbmV3IFBJWEkuVGV4dChVdGlscy5mb3JtYXRUaW1lKCgodGhpcy50aW1lRW5kIC0gdGhpcy50aW1lQmVnaW4pLzQpICsgdGhpcy50aW1lQmVnaW4pLCB7IGZvbnQ6ICcxMnB0IEdvdGhpYyBTdGFuZGFyZCcsIGZpbGw6ICcjNjQ2NDY0JyB9KTtcblx0ICAgIHQxNS54ID0gdGhpcy5jaXJjbGVYICsgdGhpcy5yYWRpdXMgKyB0aGlzLm1heENlbGxIZWlnaHQgKyAxMCA7XG5cdCAgICB0MTUueSA9IHRoaXMuY2lyY2xlWSAtIHQxNS5oZWlnaHQ7XG5cdCAgICB0MTUucm90YXRpb24gPSBNYXRoLlBJIC8yO1xuXHQgICAgdGhpcy5jb250YWluZXIuYWRkQ2hpbGQodDE1KTtcblxuXHQgICAgdmFyIHQzMCA9IG5ldyBQSVhJLlRleHQoVXRpbHMuZm9ybWF0VGltZSgoKHRoaXMudGltZUVuZCAtIHRoaXMudGltZUJlZ2luKS8yKSArIHRoaXMudGltZUJlZ2luKSwgeyBmb250OiAnMTJwdCBHb3RoaWMgU3RhbmRhcmQnLCBmaWxsOiAnIzY0NjQ2NCcgfSk7XG5cdCAgICB0MzAueCA9IHRoaXMuY2lyY2xlWCAtIHQzMC53aWR0aC8yO1xuXHQgICAgdDMwLnkgPSB0aGlzLmNpcmNsZVkgKyB0aGlzLnJhZGl1cyArIHRoaXMubWF4Q2VsbEhlaWdodCAtIDI7XG5cdCAgICB0aGlzLmNvbnRhaW5lci5hZGRDaGlsZCh0MzApO1xuXG5cdCAgICB2YXIgdDQ1ID0gbmV3IFBJWEkuVGV4dChVdGlscy5mb3JtYXRUaW1lKCgodGhpcy50aW1lRW5kIC0gdGhpcy50aW1lQmVnaW4pKjMvNCkgKyB0aGlzLnRpbWVCZWdpbiksIHsgZm9udDogJzEycHQgR290aGljIFN0YW5kYXJkJywgZmlsbDogJyM2NDY0NjQnIH0pO1xuXHQgICAgdDQ1LnggPSB0aGlzLmNpcmNsZVggLSB0aGlzLnJhZGl1cyAtIHRoaXMubWF4Q2VsbEhlaWdodCAtIDEwIDtcblx0ICAgIHQ0NS55ID0gdGhpcy5jaXJjbGVZICsgdDE1LmhlaWdodDtcblx0ICAgIHQ0NS5yb3RhdGlvbiA9IC1NYXRoLlBJLzI7XG5cdCAgICB0aGlzLmNvbnRhaW5lci5hZGRDaGlsZCh0NDUpO1xuICAgIH1cblxuICAgIC8vRHJhdyB0aGUgY2VsbHVsZVxuICAgIHRoaXMucmVkcmF3Q2VsbCA9IGZ1bmN0aW9uKGNlbGwpe1xuXG4gICAgXHRpZiAodHlwZW9mKGNlbGwuZ3JhcGhpY3MpID09PSAndW5kZWZpbmVkJyl7XG4gICAgXHRcdHRoaXMuaW5pdEdyYXBoaWNzKGNlbGwpO1xuICAgIFx0fSBlbHNlIHtcbiAgICBcdFx0Y2VsbC5ncmFwaGljcy5jbGVhcigpO1xuICAgIFx0fVxuXG4gICAgXHR2YXIgeSA9IDA7XG5cbiAgICBcdC8vQ2hlY2sgaWYgdG90YWwgaGVpZ2h0IGlzIGhpZ2hlciB0aGFuIE1heCBDZWxsIEhlaWdodFxuICAgIFx0aWYgKChjZWxsLnRvdGFsQW5ub3RzKnRoaXMuaW50ZXJ2YWxIZWlnaHQpID4gdGhpcy5tYXhDZWxsSGVpZ2h0KXtcbiAgICBcdFx0dmFyIGhlaWdodFN0ZXAgPSB0aGlzLm1heENlbGxIZWlnaHQvY2VsbC50b3RhbEFubm90cztcbiAgICBcdH0gZWxzZSB7XG4gICAgXHRcdHZhciBoZWlnaHRTdGVwID0gdGhpcy5pbnRlcnZhbEhlaWdodDtcbiAgICBcdH1cblxuICAgIFx0Ly9EcmF3IHRoZSByZWN0IGRlcGVuZGluZyBvbiB0aGUgaGVpZ2h0IHN0ZXAgY2FsY3VsYXRlZFxuICAgIFx0Zm9yICh2YXIgaT0wOyBpPCB0aGlzLmFubm90Q2F0ZWdvcmllc1swXS5vcmRlci5sZW5ndGg7IGkrKyl7XG4gICAgXHRcdHZhciBjdXJyZW50Q29kZSA9IHRoaXMuYW5ub3RDYXRlZ29yaWVzWzBdLm9yZGVyW2ldO1xuXHRcdFx0Y2VsbC5ncmFwaGljcy5iZWdpbkZpbGwoY2VsbC5jYXRlZ29yaWVzW2N1cnJlbnRDb2RlXS5jb2xvci5yZXBsYWNlKFwiI1wiLCBcIjB4XCIpKVxuICAgIFx0XHRcdC5kcmF3UmVjdCgwLCB5LCB0aGlzLmludGVydmFsV2lkdGgtMSwgLWNlbGwuY2F0ZWdvcmllc1tjdXJyZW50Q29kZV0uY291bnQgKiBoZWlnaHRTdGVwKVxuICAgIFx0XHRcdC5lbmRGaWxsKCk7XG4gICAgXHRcdHkgLT0gY2VsbC5jYXRlZ29yaWVzW2N1cnJlbnRDb2RlXS5jb3VudCpoZWlnaHRTdGVwO1xuICAgIFx0fVxuICAgIH1cblxuICAgIHRoaXMuaW5pdCA9IGZ1bmN0aW9uKCkge1xuICAgIFx0d3MubWVzc2FnZShmdW5jdGlvbihkYXRhKSB7XG4gICAgICAgICAgICBfdGhpcy5hZGRBbm5vdChkYXRhKTtcbiAgICAgICAgfSk7XG5cbiAgICBcdHRoaXMuaW5pdFRpbWVUZXh0cygpO1xuICAgIH07XG5cbiAgICB0aGlzLnVwZGF0ZVRpbWUgPSBmdW5jdGlvbigpe1xuICAgIFx0Y3VycmVudFRpbWUgKz0gMTAwMDtcblxuICAgICAgICB2YXIgbmJTZWMgPSBjdXJyZW50VGltZSAvIDEwMDA7XG4gICAgICAgIHZhciBob3VycyA9IE1hdGguZmxvb3IoIG5iU2VjIC8gMzYwMCApICUgMjQ7XG4gICAgICAgIHZhciBtaW51dGVzID0gTWF0aC5mbG9vciggbmJTZWMgLyA2MCApICUgNjA7XG4gICAgICAgIHZhciBzZWNvbmRzID0gTWF0aC5mbG9vcihuYlNlYyAlIDYwKTtcbiAgICAgICAgdmFyIHRpbWVTdHIgPSAoaG91cnMgPCAxMCA/ICcwJyArIGhvdXJzIDogaG91cnMpICsgJzonICsgKG1pbnV0ZXMgPCAxMCA/ICcwJyArIG1pbnV0ZXMgOiBtaW51dGVzKSArICc6JyArIChzZWNvbmRzICA8IDEwID8gJzAnICsgc2Vjb25kcyA6IHNlY29uZHMpO1xuXG4gICAgICAgIGN1cnJlbnRUaW1lVGV4dC5zZXRUZXh0KHRpbWVTdHIpO1xuICAgIH07XG5cbiAgICB2YXIgcmVmcmVzaFRpbWVJbnRlcnZhbDtcblxuICAgIHRoaXMuc3RhcnQgPSBmdW5jdGlvbigpIHtcbiAgICBcdHJlZnJlc2hUaW1lSW50ZXJ2YWwgPSBzZXRJbnRlcnZhbChmdW5jdGlvbigpIHtfdGhpcy51cGRhdGVUaW1lKCk7fSwgMTAwMCk7XG4gICAgfTtcblxuICAgIHRoaXMucmVmcmVzaCA9IGZ1bmN0aW9uKCkge1xuXG4gICAgfTtcblxuICAgIHRoaXMuc3RvcCA9IGZ1bmN0aW9uKCl7XG4gICAgXHRjb25zb2xlLmxvZyh0aGlzLmNlbGxzKTtcbiAgICB9O1xuXG4gICAgcmV0dXJuIHRoaXM7XG59XG5cbm1vZHVsZS5leHBvcnRzID0ge1xuXHRBbm5vdHNUaW1lTGluZTogQW5ub3RzVGltZUxpbmVcbn07XG4iLCIvKipcbioganMvYW5ub3Rzdml6dmlldy5qc1xuKlxuKiBUaGlzIGlzIHRoZSBzdGFydGluZyBwb2ludCBmb3IgeW91ciBhcHBsaWNhdGlvbi5cbiogVGFrZSBhIGxvb2sgYXQgaHR0cDovL2Jyb3dzZXJpZnkub3JnLyBmb3IgbW9yZSBpbmZvXG4qL1xuXG4ndXNlIHN0cmljdCc7XG5cbnZhciBQSVhJID0gcmVxdWlyZSgncGl4aScpO1xudmFyIF8gPSByZXF1aXJlKCdsb2Rhc2gnKTtcbnZhciBEb3VibGVSb2xsID0gcmVxdWlyZSgnLi9kb3VibGVyb2xsLmpzJyk7XG52YXIgQW5ub3RzVGltZUxpbmUgPSByZXF1aXJlKCcuL2Fubm90c3RpbWVsaW5lLmpzJyk7XG52YXIgQW5ub3RzUm9sbCA9IHJlcXVpcmUoJy4vYW5ub3Rzcm9sbC5qcycpO1xuXG52YXIgZGVmYXVsdE9wdGlvbnMgPSB7XG4gICAgeEluaXQ6IDAsXG4gICAgeUluaXQ6IDAsXG4gICAgd2lkdGg6IDEwMjQsXG4gICAgaGVpZ2h0OiA3NjgsXG4gICAgYW5ub3RDYXRlZ29yaWVzOiBbXG4gICAge1xuICAgICAgICBcInRzXCI6IDE0MjE5MjgyMTMwMDAsXG4gICAgICAgIFwiY29sb3JzXCI6IHtcbiAgICAgICAgICAgIFwidHJhbnNncmVzc2lvbnNcIjogXCIjYjkwMDAwXCIsXG4gICAgICAgICAgICBcInJ5dGhtaXF1ZVwiOiBcIiNhZjkzMWVcIixcbiAgICAgICAgICAgIFwibmFycmF0aW9uXCI6IFwiIzRiZGQ3MVwiLFxuICAgICAgICAgICAgXCJyZWxhdGlvblwiOiBcIiMxYzI4YmFcIlxuICAgICAgICB9LFxuICAgICAgICBcIm9yZGVyXCI6IFtcbiAgICAgICAgICAgIFwidHJhbnNncmVzc2lvbnNcIixcbiAgICAgICAgICAgIFwicnl0aG1pcXVlXCIsXG4gICAgICAgICAgICBcIm5hcnJhdGlvblwiLFxuICAgICAgICAgICAgXCJyZWxhdGlvblwiXG4gICAgICAgIF0sXG4gICAgICAgIFwiZGVmYXVsdENvbG9yXCI6IFwiIzUzNjk5MVwiXG4gICAgfV1cbn07XG5cbmZ1bmN0aW9uIEFubm90c1ZpelZpZXcob3B0aW9ucyl7XG5cdHZhciBfdGhpcyA9IHRoaXM7XG4gICAgdmFyIG9wdHMgPSBfKG9wdGlvbnMpLmRlZmF1bHRzKGRlZmF1bHRPcHRpb25zKS52YWx1ZSgpO1xuXG4gICAgdGhpcy5jb250YWluZXIgPSBuZXcgUElYSS5EaXNwbGF5T2JqZWN0Q29udGFpbmVyKCk7XG4gICAgdGhpcy5jb250YWluZXIueCA9IG9wdHMueEluaXQ7XG4gICAgdGhpcy5jb250YWluZXIueSA9IG9wdHMueUluaXQ7XG5cdHRoaXMud2lkdGggPSBvcHRzLndpZHRoO1xuXHR0aGlzLmhlaWdodD0gb3B0cy5oZWlnaHQ7XG4gICAgdGhpcy5hbm5vdENhdGVnb3JpZXMgPSBvcHRzLmFubm90Q2F0ZWdvcmllcztcblxuXHR2YXIgd3NQaWFub3JvbGwgPSBvcHRzLndzUGlhbm9yb2xsO1xuXHR2YXIgd3NBbm5vdCA9IG9wdHMud3NBbm5vdDtcblx0dmFyIHN0YWdlVmlldyA9IG9wdHMuc3RhZ2VWaWV3O1xuXG5cdHN0YWdlVmlldy5yZWdpc3RlckNvbXBvbmVudCh0aGlzKTtcblxuXHR2YXIgdGltZUxpbmUgPSBuZXcgQW5ub3RzVGltZUxpbmUuQW5ub3RzVGltZUxpbmUoe1xuICAgIFx0c3RhZ2VWaWV3IDogc3RhZ2VWaWV3LFxuICAgICAgICBsb2dnZXI6IGxvZ2dlcixcbiAgICAgICAgd3M6IG5ldyBhbm5vdHZpei5Xc1dyYXBwZXIod3NVcmlBbm5vdGF0aW9uLCBsb2dnZXIpLFxuICAgICAgICB4SW5pdDogMCxcbiAgICAgICAgeUluaXQ6IDAsXG4gICAgICAgIHdpZHRoOiAxMDI0IC0gMjAwIC0gMjAwLFxuICAgICAgICBoZWlnaHQ6IDc2OC0yMDAsXG4gICAgICAgIHRpbWVCZWdpbjogRGF0ZS5ub3coKSxcbiAgICAgICAgdGltZUVuZDogRGF0ZS5ub3coKSArIDMwMDAwMDAsXG4gICAgICAgIGludGVydmFsV2lkdGg6IDYsXG4gICAgICAgIGludGVydmFsSGVpZ2h0OiAxMCxcbiAgICAgICAgbWF4Q2VsbEhlaWdodDogNzAsXG4gICAgICAgIHJhZGl1czogMjAwLFxuICAgICAgICBhbm5vdENhdGVnb3JpZXM6IHRoaXMuYW5ub3RDYXRlZ29yaWVzXG4gICAgfSk7XG5cblx0dmFyIGRvdWJsZVJvbGxIID0gbmV3IERvdWJsZVJvbGwuRG91YmxlUm9sbCh7XG4gICAgICAgIHN0YWdlVmlldyA6IHN0YWdlVmlldyxcbiAgICBcdGxvZ2dlcjogbG9nZ2VyLFxuICAgICAgICB3czogd3NQaWFub3JvbGwsXG4gICAgICAgIHlJbml0OiAodGhpcy5oZWlnaHQgLSAyMDApLFxuICAgICAgICBzY2VuZUhlaWdodDogMjAwLFxuICAgICAgICBwaWFub3JvbGxzIDogW1xuICAgICAgICAgICAge1xuICAgICAgICAgICAgICAgIGhlaWdodDogMjAwLFxuICAgICAgICAgICAgICAgIHRpbWVXaWR0aDogMTAsXG4gICAgICAgICAgICAgICAgbGluZUludGVydmFsOiA1MDAwLFxuICAgICAgICAgICAgICAgIG5vdGVIZWlnaHQ6IDEwXG4gICAgICAgICAgICB9LFxuICAgICAgICBdXG4gICAgfSk7XG5cblx0dmFyIGRvdWJsZVJvbGxWID0gbmV3IERvdWJsZVJvbGwuRG91YmxlUm9sbCh7XG4gICAgICAgIHN0YWdlVmlldyA6IHN0YWdlVmlldyxcbiAgICBcdGxvZ2dlcjogbG9nZ2VyLFxuICAgICAgICB3czogd3NQaWFub3JvbGwsXG4gICAgICAgIG9yaWVudGF0aW9uOiAndmVydGljYWwnLFxuICAgICAgICBzY2VuZUhlaWdodDogNzY4LTIwMCxcbiAgICAgICAgcGlhbm9yb2xscyA6IFtcbiAgICAgICAgICAgIHtcbiAgICAgICAgICAgICAgICBoZWlnaHQ6IDIwMCxcbiAgICAgICAgICAgICAgICB0aW1lV2lkdGg6IDYwLFxuICAgICAgICAgICAgICAgIGxpbmVJbnRlcnZhbDogNTAwMCxcbiAgICAgICAgICAgICAgICBub3RlSGVpZ2h0OiA1LFxuICAgICAgICAgICAgfSxcbiAgICAgICAgXVxuICAgIH0pO1xuXG5cdHZhciBhbm5vdHNSb2xsID0gbmV3IEFubm90c1JvbGwuQW5ub3RzUm9sbCh7XG4gICAgXHRzdGFnZVZpZXcgOiBzdGFnZVZpZXcsXG4gICAgICAgIGxvZ2dlcjogbG9nZ2VyLFxuICAgICAgICB3czogd3NBbm5vdCxcbiAgICAgICAgcGFyZW50Q29udGFpbmVyOiBkb3VibGVSb2xsVi5zdGFnZSxcbiAgICAgICAgeEluaXQ6IDEwMjQgLSAyMDAgLSAyMDAsXG4gICAgICAgIHlJbml0OiA3NjgtMjAwLFxuICAgICAgICB3aWR0aDogMjAwICsgMjAwLFxuICAgICAgICBoZWlnaHQ6IDc2OC0yMDAsXG4gICAgICAgIHdpZHRoUm9sbDogMjAwLFxuICAgICAgICBmcmFtZXJhdGU6IGRvdWJsZVJvbGxWLmZyYW1lcmF0ZSxcbiAgICAgICAgcGl4ZWxzUGVyU2Vjb25kOiBNYXRoLmZsb29yKDEwMjQgLyA2MCksXG4gICAgICAgIGFubm90Q29sb3JzOiB0aGlzLmFubm90Q2F0ZWdvcmllc1xuICAgIH0pO1xuXG5cdHZhciBsaW1pdGVycyA9IG5ldyBQSVhJLkdyYXBoaWNzKClcblx0XHQubGluZVN0eWxlKDEsIDB4NjQ2NDY0KVxuXHRcdC5tb3ZlVG8oYW5ub3RzUm9sbC5jb250YWluZXIueCwgYW5ub3RzUm9sbC5jb250YWluZXIueSlcblx0XHQubGluZVRvKGFubm90c1JvbGwuY29udGFpbmVyLngsIGFubm90c1JvbGwuY29udGFpbmVyLnkgLSBhbm5vdHNSb2xsLmhlaWdodClcblx0XHQubW92ZVRvKGFubm90c1JvbGwuY29udGFpbmVyLnggKyBhbm5vdHNSb2xsLndpZHRoUm9sbCwgYW5ub3RzUm9sbC5jb250YWluZXIueSlcblx0XHQubGluZVRvKGFubm90c1JvbGwuY29udGFpbmVyLnggKyBhbm5vdHNSb2xsLndpZHRoUm9sbCwgYW5ub3RzUm9sbC5jb250YWluZXIueSAtIGFubm90c1JvbGwuaGVpZ2h0KVxuXHRcdC5tb3ZlVG8oMCwgdGhpcy5oZWlnaHQgLSAyMDApXG5cdFx0LmxpbmVUbyh0aGlzLndpZHRoLCB0aGlzLmhlaWdodCAtIDIwMClcblx0XHQuZHJhd1JlY3QoMCwgMCwgdGhpcy53aWR0aCAtMSwgdGhpcy5oZWlnaHQgLTEpXG5cdFx0LmJlZ2luRmlsbCgweEVDRUNFQylcblx0XHQuZHJhd1JlY3QoMTAyNCAtIDIwMCwgMCwgMjAwLCA3NjgtMjAwKVxuXHRcdC5lbmRGaWxsKCk7XG5cdHRoaXMuY29udGFpbmVyLmFkZENoaWxkKGxpbWl0ZXJzKTtcblxuLy9cdHZhciBkb3VibGVSb2xsViA9IG5ldyBEb3VibGVSb2xsLkRvdWJsZVJvbGwoe30pO1xuXG5cdHRoaXMuaW5pdCA9IGZ1bmN0aW9uKCl7XG5cblx0fVxuXG5cdHRoaXMuc3RhcnQgPSBmdW5jdGlvbigpIHtcbiAgICB9O1xuXG4gICAgdGhpcy5yZWZyZXNoID0gZnVuY3Rpb24oKSB7XG4gICAgfTtcblxuICAgIHRoaXMuc3RvcCA9IGZ1bmN0aW9uKCl7XG4gICAgfTtcblxuICAgIHJldHVybiB0aGlzO1xuXG59XG5cbm1vZHVsZS5leHBvcnRzID0ge1xuXHRBbm5vdHNWaXpWaWV3OiBBbm5vdHNWaXpWaWV3XG59O1xuIiwiLyoqXG4qIHNjcmlwdHMvZG91Ymxlcm9sbC5qc1xuKlxuKiBUaGlzIGlzIHRoZSBzdGFydGluZyBwb2ludCBmb3IgeW91ciBhcHBsaWNhdGlvbi5cbiogVGFrZSBhIGxvb2sgYXQgaHR0cDovL2Jyb3dzZXJpZnkub3JnLyBmb3IgbW9yZSBpbmZvXG4qL1xuXG4vKiBnbG9iYWwgZG9jdW1lbnQ6IGZhbHNlICovXG5cbid1c2Ugc3RyaWN0JztcblxuXG52YXIgUElYSSA9IHJlcXVpcmUoJ3BpeGknKTtcbnZhciBfID0gcmVxdWlyZSgnbG9kYXNoJyk7XG52YXIgUGlhbm9Sb2xsID0gcmVxdWlyZSgnLi9waWFub3JvbGwuanMnKTtcblxudmFyIGRlZmF1bHRDb25maWcgPSB7XG4gICAgb3JpZW50YXRpb246ICdob3Jpem9udGFsJyxcbiAgICBsb2dnZXI6IHVuZGVmaW5lZCxcbiAgICBzY2VuZVdpZHRoOiAxMDI0LFxuICAgIHBpYW5vcm9sbHMgOiBbXG4gICAgICB7XG4gICAgICAgIGhlaWdodDogNDM1LFxuICAgICAgICB0aW1lV2lkdGg6IDEwLFxuICAgICAgICBsaW5lSW50ZXJ2YWw6IDUwMDAsXG4gICAgICAgIG5vdGVIZWlnaHQ6IHVuZGVmaW5lZFxuICAgICAgfSxcbiAgICAgIHtcbiAgICAgICAgaGVpZ2h0OiA2NDUsXG4gICAgICAgIHRpbWVXaWR0aDogNjAsXG4gICAgICAgIGxpbmVJbnRlcnZhbDogNTAwMCxcbiAgICAgICAgbm90ZUhlaWdodDogdW5kZWZpbmVkXG4gICAgICB9LFxuICAgIF0sXG4gICAgZnJhbWVyYXRlOiAyNSxcbiAgICBvZmZzZXRNdXNpYzogZmFsc2UsXG4gICAgc2NlbmVCZ0NvbG9yOiAweEZGRkZGRixcbiAgICBsaW5lQ29sb3I6IDB4NDQ0NDQ0LFxuICAgIGxpbmVGaWxsQ29sb3I6IDB4RkZGRjAwLFxuICAgIG5vdGVDb2xvcnM6IFsweEI5MDAwMCwgMHg0QkRENzEsIDB4QUY5MzFFLCAweDFDMjhCQSwgMHg1MzY5OTFdLFxuICAgIG5vdGVIZWlnaHQ6IHVuZGVmaW5lZCxcbiAgICB6ZXJvU2hpZnQ6IDAuOSxcbiAgICB0aW1lV2lkdGg6IDYwLFxuICAgIGxpbmVJbnRlcnZhbDogNTAwMCxcbi8vICAgIHdzVXJpOiB1bmRlZmluZWQsXG4vLyAgICBldmVudENvZGU6IHVuZGVmaW5lZFxuXG59O1xuXG5mdW5jdGlvbiBEb3VibGVSb2xsKG9wdGlvbnMpIHtcblxuICAgIHZhciBfdGhpcyA9IHRoaXM7XG4gICAgdmFyIG9wdHMgPSBfKG9wdGlvbnMpLmRlZmF1bHRzKGRlZmF1bHRDb25maWcpLnZhbHVlKCk7XG5cbiAgICB2YXIgb3JpZW50YXRpb24gPSBvcHRzLm9yaWVudGF0aW9uO1xuICAgIHZhciBpc0hvcml6b250YWwgPSAob3JpZW50YXRpb24gIT09ICd2ZXJ0aWNhbCcpO1xuXG4gICAgdGhpcy5sb2dnZXIgPSBvcHRzLmxvZ2dlcjtcbiAgICB0aGlzLmxpbmVDb2xvciA9IG9wdHMubGluZUNvbG9yO1xuICAgIHRoaXMubGluZUZpbGxDb2xvciA9IG9wdHMubGluZUZpbGxDb2xvcjtcbiAgICB0aGlzLmZyYW1lcmF0ZSA9IG9wdHMuZnJhbWVyYXRlO1xuICAgIHRoaXMub2Zmc2V0TXVzaWMgPSBvcHRzLm9mZnNldE11c2ljO1xuICAgIHRoaXMubm90ZUNvbG9ycyA9IG9wdHMubm90ZUNvbG9ycztcblxuICAgIHZhciBub3RlSGVpZ2h0ID0gb3B0cy5ub3RlSGVpZ2h0O1xuICAgIHZhciBzY2VuZUJnQ29sb3IgPSBvcHRzLnNjZW5lQmdDb2xvcjtcbiAgICB2YXIgc2NlbmVIZWlnaHQgPSBvcHRzLnNjZW5lSGVpZ2h0IHx8IF8ob3B0cy5waWFub3JvbGxzKS5yZWR1Y2UoZnVuY3Rpb24ocyxwKSB7IHJldHVybiBzICsgcC5oZWlnaHQ7IH0sIDApO1xuICAgIHZhciB0aW1lV2lkdGggPSBvcHRzLnRpbWVXaWR0aDtcbiAgICB2YXIgbGluZUludGVydmFsID0gb3B0cy5saW5lSW50ZXJ2YWw7XG4gICAgdmFyIG9mZnNldE11c2ljID0gb3B0cy5vZmZzZXRNdXNpYztcblxuICAgIHZhciBzY2VuZVdpZHRoID0gb3B0cy5zY2VuZVdpZHRoO1xuICAgIHZhciBzdGFnZVZpZXcgPSBvcHRzLnN0YWdlVmlldztcblxuICAgIHZhciB6ZXJvU2hpZnQgPSBvcHRzLnplcm9TaGlmdDtcblxuICAgIHZhciB3cyA9IG9wdHMud3M7XG5cbiAgICB2YXIgY29sb3JzUmVnID0ge307XG5cbiAgICB0aGlzLmNvbnRhaW5lciA9IG5ldyBQSVhJLkRpc3BsYXlPYmplY3RDb250YWluZXIoKTtcbiAgICB0aGlzLmNvbnRhaW5lci54ID0gTWF0aC5mbG9vcihzY2VuZVdpZHRoKnplcm9TaGlmdCk7XG4gICAgdGhpcy5jb250YWluZXIueSA9IDA7XG4gICAgXG4gICAgc3RhZ2VWaWV3LnJlZ2lzdGVyQ29tcG9uZW50KHRoaXMpO1xuXG4gICAgdmFyIHBpYW5vcm9sbExpc3QgPSBbXTtcblxuICAgIHZhciBwaWFub3JvbGxPcHRpb25zID0ge1xuICAgICAgICBwYXJlbnRDb250YWluZXI6IHRoaXMuY29udGFpbmVyLFxuICAgICAgICBvcmllbnRhdGlvbjogb3JpZW50YXRpb24sXG4gICAgICAgIHhJbml0OiAwLFxuICAgICAgICB3aWR0aDogc2NlbmVXaWR0aCxcbiAgICAgICAgbm90ZUNvbG9yczogdGhpcy5ub3RlQ29sb3JzLFxuICAgICAgICBjb2xvcnNSZWc6IGNvbG9yc1JlZyxcbiAgICAgICAgbGluZUNvbG9yOiB0aGlzLmxpbmVDb2xvcixcbiAgICAgICAgbGluZUludGVydmFsOiBsaW5lSW50ZXJ2YWwsXG4gICAgICAgIG9mZnNldE11c2ljOiBvZmZzZXRNdXNpYyxcbiAgICB9O1xuXG4gICAgdmFyIHlJbml0ID0gb3B0cy55SW5pdCB8fCAwO1xuICAgIHZhciBsaW5lc0Rvd24gPSB0cnVlO1xuICAgIF8ob3B0cy5waWFub3JvbGxzKS5mb3JFYWNoKGZ1bmN0aW9uKHByRGVmLCBpKSB7XG4gICAgICAgIHZhciBwck5vdGVIZWlnaHQgPSBub3RlSGVpZ2h0IHx8IHByRGVmLm5vdGVIZWlnaHQgfHwgcHJEZWYuaGVpZ2h0IC8gMTI4O1xuICAgICAgICB2YXIgcHJUaW1lV2lkdGggPSBwckRlZi50aW1lV2lkdGggfHwgdGltZVdpZHRoO1xuICAgICAgICBwaWFub3JvbGxMaXN0LnB1c2gobmV3IFBpYW5vUm9sbChfKHtcbiAgICAgICAgICAgIHlJbml0OiB5SW5pdCxcbiAgICAgICAgICAgIGhlaWdodDogcHJEZWYuaGVpZ2h0LFxuICAgICAgICAgICAgbGluZXNEb3duOiBsaW5lc0Rvd24sXG4gICAgICAgICAgICBwaXhlbHNQZXJTZWNvbmQ6IE1hdGguZmxvb3Ioc2NlbmVXaWR0aCAvIHByVGltZVdpZHRoKSxcbiAgICAgICAgICAgIG5vdGVIZWlnaHQ6IHByTm90ZUhlaWdodCxcbiAgICAgICAgICAgIGxpbmVJbnRlcnZhbDogcHJEZWYubGluZUludGVydmFsXG4gICAgICAgIH0pLmRlZmF1bHRzKHBpYW5vcm9sbE9wdGlvbnMpLnZhbHVlKCkpKTtcbiAgICAgICAgeUluaXQgKz0gcHJEZWYuaGVpZ2h0O1xuICAgICAgICBsaW5lc0Rvd24gPSAhbGluZXNEb3duO1xuXG4gICAgICAgIGlmKGk8KG9wdHMucGlhbm9yb2xscy5sZW5ndGgtMSkpIHtcbiAgICAgICAgICAgIHZhciBsaW5lR3JhcGhpY3MgPSBuZXcgUElYSS5HcmFwaGljcygpXG4gICAgICAgICAgICAgICAgLmJlZ2luRmlsbChfdGhpcy5saW5lRmlsbENvbG9yKVxuICAgICAgICAgICAgICAgIC5saW5lU3R5bGUoMSwgX3RoaXMubGluZUNvbG9yKVxuICAgICAgICAgICAgICAgIC5tb3ZlVG8oTWF0aC5mbG9vcihzY2VuZVdpZHRoKnplcm9TaGlmdCksIHlJbml0KVxuICAgICAgICAgICAgICAgIC5saW5lVG8oLXNjZW5lV2lkdGggLSBNYXRoLmZsb29yKHNjZW5lV2lkdGgqemVyb1NoaWZ0KSwgeUluaXQpXG4gICAgICAgICAgICAgICAgLmVuZEZpbGwoKTtcbiAgICAgICAgICAgIF90aGlzLmNvbnRhaW5lci5hZGRDaGlsZChsaW5lR3JhcGhpY3MpO1xuICAgICAgICB9XG4gICAgfSk7XG5cbiAgICBpZighaXNIb3Jpem9udGFsKSB7XG4gICAgICAgIHRoaXMuY29udGFpbmVyLnJvdGF0aW9uID0gTWF0aC5QSS8yO1xuICAgICAgICB0aGlzLmNvbnRhaW5lci55ID0gc2NlbmVIZWlnaHQ7XG4gICAgICAgIHRoaXMuY29udGFpbmVyLnggPSBzY2VuZVdpZHRoO1xuICAgIH1cblxuXG4gICAgdGhpcy5pbml0ID0gZnVuY3Rpb24oKSB7XG5cbiAgICBcdHdzLm1lc3NhZ2UoZnVuY3Rpb24oZGF0YSkge1xuICAgICAgICAgICAgX3RoaXMuYWRkTm90ZXMoZGF0YSk7XG4gICAgICAgIH0pO1xuXG4gICAgfTtcblxuXG4gICAgdGhpcy5hZGROb3RlcyA9IGZ1bmN0aW9uKGRhdGEpIHtcblxuICAgICAgICBwaWFub3JvbGxMaXN0LmZvckVhY2goZnVuY3Rpb24oYykge1xuICAgICAgICAgICAgYy5hZGROb3RlUmF3KGRhdGEpO1xuICAgICAgICB9KTtcbiAgICB9O1xuXG4gICAgdGhpcy5yZWZyZXNoID0gZnVuY3Rpb24oKSB7XG4gICAgICAgIHBpYW5vcm9sbExpc3QuZm9yRWFjaChmdW5jdGlvbihjKSB7XG4gICAgICAgICAgICBjLm1vdmUoKTtcbiAgICAgICAgfSk7XG4gICAgfTtcblxuICAgIC8vIEluaXQgcGFnZSBhbmQgaW50ZXJ2YWxzXG4gICAgdmFyIHN0YXJ0VHM7XG5cbiAgICB0aGlzLnN0YXJ0ID0gZnVuY3Rpb24oKSB7XG5cbiAgICAgICAgc3RhcnRUcyA9IERhdGUubm93KCk7XG4gICAgICAgIHBpYW5vcm9sbExpc3QuZm9yRWFjaChmdW5jdGlvbihjKSB7XG4gICAgICAgICAgICBjLnN0YXJ0KCk7XG4gICAgICAgIH0pO1xuICAgIH07XG5cbiAgICB0aGlzLnN0b3AgPSBmdW5jdGlvbigpIHtcbiAgICBcdFxuICAgICAgICBwaWFub3JvbGxMaXN0LmZvckVhY2goZnVuY3Rpb24oYykge1xuICAgICAgICAgICAgYy5zdG9wKCk7XG4gICAgICAgIH0pO1xuICAgIH07XG5cblxuICAgIHRoaXMubG9nID0gZnVuY3Rpb24obSkge1xuICAgICAgICBpZih0aGlzLmxvZ2dlcikge1xuICAgICAgICAgICAgdGhpcy5sb2dnZXIubG9nKG0pO1xuICAgICAgICB9XG4gICAgfTtcblxuXG5cbiAgICByZXR1cm4gdGhpcztcbn1cblxubW9kdWxlLmV4cG9ydHMgPSB7XG4gICAgRG91YmxlUm9sbDogRG91YmxlUm9sbFxufTtcbiIsIi8qKlxuKiBqcy93c3dyYXBwZXIuanNcbipcbiogc2ltcGxlIGxvZ2dlciBzZXJ2aWNlXG4qXG4qL1xuXG4vKiBnbG9iYWwgZG9jdW1lbnQ6IGZhbHNlICovXG5cbid1c2Ugc3RyaWN0JztcblxuZnVuY3Rpb24gSHRtbExvZ2dlcihkb0xvZywgY29udGFpbmVyKSB7XG5cbiAgICB2YXIgbG9nQ29udGFpbmVyID0gY29udGFpbmVyO1xuICAgIGlmKHR5cGVvZihjb250YWluZXIpID09PSAnc3RyaW5nJykge1xuICAgICAgICBsb2dDb250YWluZXIgPSBkb2N1bWVudC5nZXRFbGVtZW50QnlJZChjb250YWluZXIpO1xuICAgIH1cbiAgICBpZighZG9Mb2cpIHtcbiAgICAgICAgZG9jdW1lbnQuYm9keS5yZW1vdmVDaGlsZChsb2dDb250YWluZXIpO1xuICAgICAgICBsb2dDb250YWluZXIgPSB1bmRlZmluZWQ7XG4gICAgfVxuXG5cbiAgICB0aGlzLmxvZyA9IGZ1bmN0aW9uKG1zZykge1xuICAgICAgICBpZihkb0xvZyAmJiBsb2dDb250YWluZXIpIHtcbiAgICAgICAgICAgIGxvZ0NvbnRhaW5lci5pbm5lckhUTUwgKz0gbXNnICsgJ1xcbic7XG4gICAgICAgICAgICBsb2dDb250YWluZXIuc2Nyb2xsVG9wID0gbG9nQ29udGFpbmVyLnNjcm9sbEhlaWdodDtcbiAgICAgICAgfVxuICAgIH07XG59XG5cbmZ1bmN0aW9uIENvbnNvbGVMb2dnZXIoZG9Mb2cpIHtcblxuICAgIHRoaXMubG9nID0gZnVuY3Rpb24obXNnKSB7XG4gICAgICAgIGlmKGRvTG9nKSB7XG4gICAgICAgICAgICBjb25zb2xlLmxvZyhtc2cpO1xuICAgICAgICB9XG4gICAgfVxuXG59XG5cbm1vZHVsZS5leHBvcnRzID0ge1xuICAgIEh0bWxMb2dnZXI6IEh0bWxMb2dnZXIsXG4gICAgQ29uc29sZUxvZ2dlcjogQ29uc29sZUxvZ2dlclxufTtcbiIsIi8qKlxuKiBqcy9waWFub3JvbGwuanNcbipcbiogcGlhbm9yb2xsIGJhc2ljIGNvbXBvbmVudFxuKlxuKi9cblxuJ3VzZSBzdHJpY3QnO1xuXG5cbnZhciBQSVhJID0gcmVxdWlyZSgncGl4aScpO1xudmFyIHJhbmRvbUNvbG9yID0gcmVxdWlyZSgncmFuZG9tQ29sb3InKTtcbnZhciBfID0gcmVxdWlyZSgnbG9kYXNoJyk7XG5cbnZhciBOVFBfRVBPQ0hfREVMVEEgPSAyMjA4OTg4ODAwOyAvL2MuZi4gUkZDIDg2OFxuXG5mdW5jdGlvbiBQaWFub1JvbGwob3B0aW9ucykge1xuICAgIHZhciBfdGhpcyA9IHRoaXM7XG4gICAgdGhpcy5jb250YWluZXIgPSBuZXcgUElYSS5EaXNwbGF5T2JqZWN0Q29udGFpbmVyKCk7XG4gICAgdGhpcy5jb250YWluZXIueCA9IG9wdGlvbnMueEluaXQ7XG4gICAgdGhpcy5jb250YWluZXIueSA9IG9wdGlvbnMueUluaXQ7XG4gICAgb3B0aW9ucy5wYXJlbnRDb250YWluZXIuYWRkQ2hpbGQodGhpcy5jb250YWluZXIpO1xuXG4gICAgdmFyIG9yaWVudGF0aW9uID0gb3B0aW9ucy5vcmllbnRhdGlvbjtcbiAgICB2YXIgaXNIb3Jpem9udGFsID0gKG9yaWVudGF0aW9uICE9PSAndmVydGljYWwnKTtcblxuICAgIHRoaXMubGluZXNEb3duID0gb3B0aW9ucy5saW5lc0Rvd247XG4gICAgdGhpcy5oZWlnaHQgPSBvcHRpb25zLmhlaWdodDtcbiAgICB0aGlzLnBpeGVsc1BlclNlY29uZCA9IG9wdGlvbnMucGl4ZWxzUGVyU2Vjb25kO1xuICAgIHRoaXMud2lkdGggPSBvcHRpb25zLndpZHRoO1xuICAgIHRoaXMubm90ZUNvbG9ycyA9IG9wdGlvbnMubm90ZUNvbG9ycztcbiAgICB0aGlzLmNvbG9yc1JlZyA9IG9wdGlvbnMuY29sb3JzUmVnIHx8IHt9O1xuICAgIHRoaXMubGluZUNvbG9yID0gb3B0aW9ucy5saW5lQ29sb3I7XG4gICAgdGhpcy5saW5lSW50ZXJ2YWwgPSBvcHRpb25zLmxpbmVJbnRlcnZhbDtcbiAgICB0aGlzLm9mZnNldE11c2ljID0gb3B0aW9ucy5vZmZzZXRNdXNpYyB8fCBmYWxzZTtcbiAgICB0aGlzLm5vdGVIZWlnaHQgPSBvcHRpb25zLm5vdGVIZWlnaHQ7XG4gICAgdGhpcy5ub3RlRGljdCA9IHt9O1xuICAgIHRoaXMuc3RhcnRUcyA9IG9wdGlvbnMuc3RhcnRUcyB8fCBEYXRlLm5vdygpO1xuXG4gICAgdmFyIHN0YXJ0ZWQgPSBmYWxzZTtcblxuICAgIHZhciBpc0hpZGRlbiA9IGZ1bmN0aW9uKGNoaWxkKSB7XG4gICAgICAgIC8vIFRPRE86IHRoZSBvcmlnaW4gcG9pbnQgaXMgYW4gYXBwcm94aW1hdGlvbi4gU2hvdWxkIHJlZmluZSB0aGlzXG4gICAgICAgIHZhciBnbG9iYWxQb3MgPSBjaGlsZC50b0dsb2JhbChuZXcgUElYSS5Qb2ludCgwLDApKTtcbiAgICAgICAgcmV0dXJuICgoZ2xvYmFsUG9zLnggKyBjaGlsZC53aWR0aCkgPCAwKSB8fCAoKGdsb2JhbFBvcy55ICsgY2hpbGQuaGVpZ2h0KSA8IDApIDtcbiAgICB9O1xuXG4gICAgLy9UT0RPOiBJIGRvIG5vdCBsaWtlIHRoZSBcInJlZ0NvbG9yXCIgb2JqZWN0LiBUaGlzIHNob3VsZCBub3QgYmUgZ2xvYmFsLCBidXQgbG9jYWxcbiAgICB0aGlzLmdldENvbG9yID0gZnVuY3Rpb24oY2FuYWwpIHtcbiAgICAgICAgdmFyIGNvbG9yID0gdGhpcy5jb2xvcnNSZWdbY2FuYWxdO1xuICAgICAgICBpZih0eXBlb2YoY29sb3IpID09PSAndW5kZWZpbmVkJykge1xuICAgICAgICAgICAgdmFyIGNvbG9yc1JlZ1NpemUgPSBPYmplY3Qua2V5cyh0aGlzLmNvbG9yc1JlZykubGVuZ3RoO1xuICAgICAgICAgICAgaWYoY29sb3JzUmVnU2l6ZSA8IHRoaXMubm90ZUNvbG9ycy5sZW5ndGgpIHtcbiAgICAgICAgICAgICAgICBjb2xvciA9IHRoaXMuY29sb3JzUmVnW2NhbmFsXSA9IHRoaXMubm90ZUNvbG9yc1tjb2xvcnNSZWdTaXplXTtcbiAgICAgICAgICAgIH1cbiAgICAgICAgICAgIGVsc2Uge1xuICAgICAgICAgICAgICAgIGNvbG9yID0gdGhpcy5jb2xvcnNSZWdbY2FuYWxdID0gcGFyc2VJbnQocmFuZG9tQ29sb3IoeyBsdW1pbm9zaXR5OiAnbGlnaHQnLCBodWU6ICdyYW5kb20nLCBmb3JtYXQ6J2hleCd9KS5yZXBsYWNlKC9eIy8sICcnKSwgMTYpO1xuICAgICAgICAgICAgfVxuICAgICAgICB9XG4gICAgICAgIHJldHVybiBjb2xvcjtcbiAgICB9O1xuXG4gICAgdGhpcy5nZXROb3RlUmVjdCA9IGZ1bmN0aW9uKHgsIHksIGNvbG9yLCBhbHBoYSwgd2lkdGgsIGhlaWdodCkge1xuICAgICAgICB2YXIgZ3JhcGhpY3MgPSBuZXcgUElYSS5HcmFwaGljcygpO1xuICAgICAgICBncmFwaGljcy5iZWdpbkZpbGwoY29sb3IsIGFscGhhKTtcbiAgICAgICAgZ3JhcGhpY3MuZHJhd1JlY3QoMCwgMCwgd2lkdGgsIGhlaWdodCk7XG4gICAgICAgIGdyYXBoaWNzLmVuZEZpbGwoKTtcbiAgICAgICAgZ3JhcGhpY3MueCA9IHg7XG4gICAgICAgIGdyYXBoaWNzLnkgPSB5O1xuICAgICAgICBncmFwaGljcy53aWR0aCA9IHdpZHRoO1xuICAgICAgICBncmFwaGljcy5oZWlnaHQgPSBoZWlnaHQ7XG4gICAgICAgIHJldHVybiBncmFwaGljcztcbiAgICB9O1xuXG4gICAgdGhpcy5hZGROb3RlUmF3ID0gZnVuY3Rpb24oZGF0YSkge1xuICAgIFx0Y29uc29sZS5sb2coZGF0YSk7XG4gICAgICAgIHZhciBub3RlID0gZGF0YS5jb250ZW50WzNdO1xuICAgICAgICB2YXIgdmVsb2NpdHkgPSBkYXRhLmNvbnRlbnRbNF07XG4gICAgICAgIHZhciB0cyA9IChkYXRhLmNvbnRlbnRbMF0gLSBOVFBfRVBPQ0hfREVMVEEpKjEwMDA7XG4gICAgICAgIHZhciBjaGFubmVsID0gZGF0YS5jb250ZW50WzJdO1xuICAgICAgICB2YXIgc2Vzc2lvblRzID0gZGF0YS5jb250ZW50WzFdO1xuXG4gICAgICAgIHRoaXMuYWRkTm90ZShub3RlLCB0cywgc2Vzc2lvblRzLCB2ZWxvY2l0eSwgY2hhbm5lbCwgMCk7XG4gICAgfTtcblxuICAgIHRoaXMuYWRkTm90ZSA9IGZ1bmN0aW9uKG5vdGUsIHN0YXJ0VGltZSwgc2Vzc2lvblRzLCB2ZWxvY2l0eSwgY2hhbm5lbCwgZHVyYXRpb24pIHtcblxuICAgICAgICB2YXIgdHMgPSBzdGFydFRpbWU7XG4gICAgICAgIGlmKHRoaXMub2Zmc2V0TXVzaWMpIHtcbiAgICAgICAgICAgIHRzID0gdGhpcy5zdGFydFRzICsgc2Vzc2lvblRzO1xuICAgICAgICB9XG5cbiAgICAgICAgdmFyIG5vdGVEdXJhdGlvbiA9IGR1cmF0aW9uO1xuICAgICAgICB2YXIgbm90ZVZlbG9jaXR5ID0gdmVsb2NpdHk7XG4gICAgICAgIHZhciBncmFwaGljcztcbiAgICAgICAgaWYoIWR1cmF0aW9uKSB7XG4gICAgICAgICAgICBpZih0eXBlb2YgdGhpcy5ub3RlRGljdFtjaGFubmVsXT09PSd1bmRlZmluZWQnKXtcbiAgICAgICAgICAgICAgICB0aGlzLm5vdGVEaWN0W2NoYW5uZWxdID0ge307XG4gICAgICAgICAgICB9XG4gICAgICAgICAgICBpZih2ZWxvY2l0eT09PTApIHtcbiAgICAgICAgICAgICAgICBpZih0eXBlb2YgdGhpcy5ub3RlRGljdFtjaGFubmVsXVtub3RlXSAhPT0gJ3VuZGVmaW5lZCcpIHtcbiAgICAgICAgICAgICAgICAgICAgdmFyIG5vdGVEZWYgPSB0aGlzLm5vdGVEaWN0W2NoYW5uZWxdW25vdGVdO1xuICAgICAgICAgICAgICAgICAgICBkZWxldGUgdGhpcy5ub3RlRGljdFtjaGFubmVsXVtub3RlXTtcbiAgICAgICAgICAgICAgICAgICAgbm90ZUR1cmF0aW9uID0gc2Vzc2lvblRzIC0gbm90ZURlZi5zZXNzaW9uVHM7XG4gICAgICAgICAgICAgICAgICAgIGdyYXBoaWNzID0gbm90ZURlZi5ncmFwaGljcztcbiAgICAgICAgICAgICAgICAgICAgbm90ZVZlbG9jaXR5ID0gbm90ZURlZi52ZWxvY2l0eTtcbiAgICAgICAgICAgICAgICAgICAgdHMgPSBub3RlRGVmLnRzO1xuICAgICAgICAgICAgICAgIH1cbiAgICAgICAgICAgIH1cbiAgICAgICAgICAgIGVsc2Uge1xuICAgICAgICAgICAgICAgIG5vdGVEdXJhdGlvbiA9IERhdGUubm93KCkgLSB0cztcbiAgICAgICAgICAgICAgICB0aGlzLm5vdGVEaWN0W2NoYW5uZWxdW25vdGVdID0geyB0czogdHMsIHZlbG9jaXR5OiB2ZWxvY2l0eSwgc2Vzc2lvblRzOiBzZXNzaW9uVHN9O1xuICAgICAgICAgICAgfVxuICAgICAgICB9XG5cblxuICAgICAgICBpZighdGhpcy5vZmZzZXRNdXNpYyB8fCB2ZWxvY2l0eT09PTApIHtcblxuICAgICAgICAgICAgdmFyIHdpZHRoID0gbm90ZUR1cmF0aW9uICogdGhpcy5waXhlbHNQZXJTZWNvbmQgLyAxMDAwO1xuICAgICAgICAgICAgaWYoIWdyYXBoaWNzKSB7XG4gICAgICAgICAgICAgICAgdmFyIHggPSAodHMtdGhpcy5zdGFydFRzKSAqIHRoaXMucGl4ZWxzUGVyU2Vjb25kIC8gMTAwMDtcbiAgICAgICAgICAgICAgICBpZigoeCt3aWR0aCkgPCAgKE1hdGguYWJzKHRoaXMuY29udGFpbmVyLngpIC0gdGhpcy53aWR0aCkpIHtcbiAgICAgICAgICAgICAgICAgICAgLy8gbm90IHZpc2libGUuIGRvIG5vdGhpbmdcbiAgICAgICAgICAgICAgICAgICAgcmV0dXJuO1xuICAgICAgICAgICAgICAgIH1cbiAgICAgICAgICAgICAgICB2YXIgeSA9IE1hdGguZmxvb3IoKDEyOC1ub3RlKzAuNSkgKiB0aGlzLmhlaWdodCAvIDEyOCAtICh0aGlzLm5vdGVIZWlnaHQvMikpO1xuICAgICAgICAgICAgICAgIHZhciBjb2xvciA9IHRoaXMuZ2V0Q29sb3IoY2hhbm5lbCk7XG4gICAgICAgICAgICAgICAgdmFyIGFscGhhID0gKG5vdGVWZWxvY2l0eSAvIDEyOCk7XG5cbiAgICAgICAgICAgICAgICBncmFwaGljcyA9IHRoaXMuZ2V0Tm90ZVJlY3QoeCwgeSwgY29sb3IsIGFscGhhLCB3aWR0aCwgdGhpcy5ub3RlSGVpZ2h0KTtcbiAgICAgICAgICAgICAgICB0aGlzLmNvbnRhaW5lci5hZGRDaGlsZChncmFwaGljcyk7XG4gICAgICAgICAgICB9XG4gICAgICAgICAgICBlbHNlIHtcbiAgICAgICAgICAgICAgICBncmFwaGljcy53aWR0aCA9IHdpZHRoO1xuICAgICAgICAgICAgfVxuXG4gICAgICAgICAgICBpZighZHVyYXRpb24gJiYgdmVsb2NpdHkpIHtcbiAgICAgICAgICAgICAgICB0aGlzLm5vdGVEaWN0W2NoYW5uZWxdW25vdGVdLmdyYXBoaWNzID0gZ3JhcGhpY3M7XG4gICAgICAgICAgICB9XG4gICAgICAgIH1cbiAgICB9O1xuXG4gICAgdGhpcy5hZGRMaW5lID0gZnVuY3Rpb24odHMpe1xuXG4gICAgICAgIGlmKHR5cGVvZih0cykgPT09ICd1bmRlZmluZWQnKSB7XG4gICAgICAgICAgICB0cyA9IG5ldyBEYXRlKCk7XG4gICAgICAgIH1cbiAgICAgICAgdmFyIHggPSAtdGhpcy5jb250YWluZXIueDtcbiAgICAgICAgdmFyIHkgPSB0aGlzLmxpbmVzRG93biA/IHRoaXMuaGVpZ2h0IC0gMjAgOiAwO1xuXG4gICAgICAgIHZhciBncmFwaGljcyA9IG5ldyBQSVhJLkdyYXBoaWNzKClcbiAgICAgICAgICAgIC5iZWdpbkZpbGwoMHhGRkZGMDApXG4gICAgICAgICAgICAubGluZVN0eWxlKDEsIHRoaXMubGluZUNvbG9yKVxuICAgICAgICAgICAgLm1vdmVUbygwLCAwKVxuICAgICAgICAgICAgLmxpbmVUbygwLCAyMClcbiAgICAgICAgICAgIC5lbmRGaWxsKCk7XG4gICAgICAgIGdyYXBoaWNzLnggPSB4O1xuICAgICAgICBncmFwaGljcy55ID0geTtcbiAgICAgICAgdGhpcy5jb250YWluZXIuYWRkQ2hpbGQoZ3JhcGhpY3MpO1xuICAgICAgICAvLyBBZGQgdGV4dFxuICAgICAgICAvL3ZhciB0b3RhbFNlYyA9IGxpbmVOYiAqIHRoaXMubGluZUludGVydmFsIC8gMTAwMDtcbiAgICAgICAgdmFyIGhvdXJzID0gdHMuZ2V0SG91cnMoKTtcbiAgICAgICAgdmFyIG1pbnV0ZXMgPXRzLmdldE1pbnV0ZXMoKTtcbiAgICAgICAgdmFyIHNlY29uZHMgPSB0cy5nZXRTZWNvbmRzKCk7XG4gICAgICAgIHZhciB0aW1lU3RyID0gKGhvdXJzIDwgMTAgPyAnMCcgKyBob3VycyA6IGhvdXJzKSArICc6JyArIChtaW51dGVzIDwgMTAgPyAnMCcgKyBtaW51dGVzIDogbWludXRlcykgKyAnOicgKyAoc2Vjb25kcyAgPCAxMCA/ICcwJyArIHNlY29uZHMgOiBzZWNvbmRzKTtcblxuICAgICAgICB2YXIgZm9udE9iaiA9IHsgZm9udDogJzEwcHQgQXJpYWwnLCBmaWxsOiAnIzQ0NDQ0NCcgfTtcbiAgICAgICAgdmFyIHQgPSBuZXcgUElYSS5UZXh0KHRpbWVTdHIsIGZvbnRPYmopO1xuICAgICAgICBpZihpc0hvcml6b250YWwpIHtcbiAgICAgICAgICAgIHQueCA9IHggKyAyO1xuICAgICAgICAgICAgdC55ID0gdGhpcy5saW5lc0Rvd24gPyB0aGlzLmhlaWdodCAtIDE1IDogMjtcbiAgICAgICAgfVxuICAgICAgICBlbHNlIHtcbiAgICAgICAgICAgIHQucm90YXRpb24gPSAtTWF0aC5QSS8yO1xuICAgICAgICAgICAgdC54ID0geCA7XG4gICAgICAgICAgICB0LnkgPSB0aGlzLmxpbmVzRG93biA/IHRoaXMuaGVpZ2h0IC0gMiA6IHQud2lkdGggKyAyO1xuICAgICAgICB9XG4gICAgICAgIHRoaXMuY29udGFpbmVyLmFkZENoaWxkKHQpO1xuICAgIH07XG5cbiAgICB0aGlzLm1vdmVUbyA9IGZ1bmN0aW9uKGRpZmZUaW1lKXtcbiAgICAgICAgdmFyIG9sZFggPSB0aGlzLmNvbnRhaW5lci54O1xuICAgICAgICB0aGlzLmNvbnRhaW5lci54ID0gTWF0aC5mbG9vcihkaWZmVGltZSp0aGlzLnBpeGVsc1BlclNlY29uZCk7XG4gICAgICAgIHZhciBkZWx0YVggPSBNYXRoLmFicyhvbGRYLXRoaXMuY29udGFpbmVyLngpO1xuICAgICAgICBfLmZvck93bih0aGlzLm5vdGVEaWN0LCBmdW5jdGlvbihjaGFubmVsRGljdCkge1xuICAgICAgICAgICAgXy5mb3JPd24oY2hhbm5lbERpY3QsIGZ1bmN0aW9uKG5vdGVEZWYpIHtcbiAgICAgICAgICAgICAgICBpZihub3RlRGVmLmdyYXBoaWNzKSB7XG4gICAgICAgICAgICAgICAgICAgIG5vdGVEZWYuZ3JhcGhpY3Mud2lkdGggPSBub3RlRGVmLmdyYXBoaWNzLndpZHRoICsgZGVsdGFYO1xuICAgICAgICAgICAgICAgIH1cbiAgICAgICAgICAgIH0pO1xuICAgICAgICB9KTtcbiAgICB9O1xuXG4gICAgdGhpcy5tb3ZlID0gZnVuY3Rpb24oKSB7XG4gICAgICAgIHZhciBkaWZmID0gKHRoaXMuc3RhcnRUcyAtIERhdGUubm93KCkpLzEwMDA7XG4gICAgICAgIHRoaXMubW92ZVRvKGRpZmYpO1xuICAgIH07XG5cbiAgICB0aGlzLnJlbW92ZVBhc3NlZE9iamV0cyA9IGZ1bmN0aW9uKCl7XG4gICAgICAgIHZhciBjaGlsZHJlblRvUmVtb3ZlID0gW107XG4gICAgICAgIF8oX3RoaXMuY29udGFpbmVyLmNoaWxkcmVuKS5mb3JFYWNoKGZ1bmN0aW9uKGNoaWxkKSB7XG4gICAgICAgICAgICByZXR1cm4gdHlwZW9mKGNoaWxkKSA9PT0gJ3VuZGVmaW5lZCcgfHxcbiAgICAgICAgICAgICAgICAoaXNIaWRkZW4oY2hpbGQpICYmIGNoaWxkcmVuVG9SZW1vdmUucHVzaChjaGlsZCkpO1xuICAgICAgICB9KTtcbiAgICAgICAgY2hpbGRyZW5Ub1JlbW92ZS5mb3JFYWNoKGZ1bmN0aW9uKGNoaWxkKSB7XG4gICAgICAgICAgICBfdGhpcy5jb250YWluZXIucmVtb3ZlQ2hpbGQoY2hpbGQpO1xuICAgICAgICB9KTtcbiAgICB9O1xuXG4gICAgdGhpcy5zdGFydCA9IGZ1bmN0aW9uKCkge1xuICAgICAgICBpZighc3RhcnRlZCkge1xuICAgICAgICAgICAgdGhpcy5zdGFydFRzID0gRGF0ZS5ub3coKTtcbiAgICAgICAgICAgIHRoaXMuYWRkTGluZSgpO1xuICAgICAgICAgICAgc3RhcnRlZCA9IHRydWU7XG4gICAgICAgIH1cbiAgICAgICAgdGhpcy52ZXJ0aWNhbExpbmVzSW50ZXJ2YWwgPSBzZXRJbnRlcnZhbChmdW5jdGlvbigpIHsgX3RoaXMuYWRkTGluZSgpOyB9LCB0aGlzLmxpbmVJbnRlcnZhbCk7XG4gICAgICAgIHRoaXMuY2xlYW5JbnRlcnZhbCA9IHNldEludGVydmFsKGZ1bmN0aW9uICgpIHsgX3RoaXMucmVtb3ZlUGFzc2VkT2JqZXRzKCk7IH0sIDEwMDAgKiB0aGlzLndpZHRoIC8gdGhpcy5waXhlbHNQZXJTZWNvbmQgKTtcbiAgICB9O1xuXG4gICAgdGhpcy5zdG9wID0gZnVuY3Rpb24oKSB7XG4gICAgICAgIC8vd2luZG93LmNsZWFySW50ZXJ2YWwodGhpcy5tb3ZlSW50ZXJ2YWwpO1xuICAgICAgICBjbGVhckludGVydmFsKHRoaXMudmVydGljYWxMaW5lc0ludGVydmFsKTtcbiAgICAgICAgY2xlYXJJbnRlcnZhbCh0aGlzLmNsZWFuSW50ZXJ2YWwpO1xuICAgIH07XG5cblxufVxuXG5tb2R1bGUuZXhwb3J0cyA9IFBpYW5vUm9sbDtcbiIsIi8qKlxuKiBzY3JpcHRzL3N0YWdldmlldy5qc1xuKlxuKiBUaGlzIGlzIHRoZSBzdGFydGluZyBwb2ludCBmb3IgeW91ciBhcHBsaWNhdGlvbi5cbiogVGFrZSBhIGxvb2sgYXQgaHR0cDovL2Jyb3dzZXJpZnkub3JnLyBmb3IgbW9yZSBpbmZvXG4qL1xuXG4vKiBnbG9iYWwgZG9jdW1lbnQ6IGZhbHNlICovXG5cbid1c2Ugc3RyaWN0JztcblxuXG52YXIgUElYSSA9IHJlcXVpcmUoJ3BpeGknKTtcbnZhciBfID0gcmVxdWlyZSgnbG9kYXNoJyk7XG5cbnZhciBkZWZhdWx0Q29uZmlnID0ge1xuICAgIGV4dGVybmFsUmVmcmVzaDogZmFsc2UsXG4gICAgbG9nZ2VyOiB1bmRlZmluZWQsXG4gICAgc2NlbmVXaWR0aDogMTAyNCxcbiAgICBzY2VuZUhlaWdodDogNzY4LFxuICAgIGZyYW1lcmF0ZTogMjUsXG4gICAgc2NlbmVCZ0NvbG9yOiAweEZGRkZGRixcbiAgICBjYW52YXNDb250YWluZXI6ICdjYW52YXNDb250YWluZXInLFxufTtcblxuZnVuY3Rpb24gU3RhZ2VWaWV3KG9wdGlvbnMpIHtcblxuICAgIHZhciBfdGhpcyA9IHRoaXM7XG4gICAgdmFyIG9wdHMgPSBfKG9wdGlvbnMpLmRlZmF1bHRzKGRlZmF1bHRDb25maWcpLnZhbHVlKCk7XG5cbiAgICB2YXIgZXh0ZXJuYWxSZWZyZXNoID0gb3B0cy5leHRlcm5hbFJlZnJlc2g7XG5cbiAgICB0aGlzLmxvZ2dlciA9IG9wdHMubG9nZ2VyO1xuICAgIHRoaXMuZnJhbWVyYXRlID0gb3B0cy5mcmFtZXJhdGU7XG4gICAgdmFyIHNjZW5lQmdDb2xvciA9IG9wdHMuc2NlbmVCZ0NvbG9yO1xuICAgIHZhciBzY2VuZVdpZHRoID0gb3B0cy5zY2VuZVdpZHRoO1xuICAgIHZhciBzY2VuZUhlaWdodCA9IG9wdHMuc2NlbmVIZWlnaHQ7XG4gICAgdmFyIGNhbnZhc0NvbnRhaW5lciA9IG9wdHMuY2FudmFzQ29udGFpbmVyO1xuICAgIHZhciB0aW1lQ29udGFpbmVyID0gW107XG4gICAgdmFyIGNvbXBvbmVudHMgPSBbXTsgXG4gICAgXG4gICAgLy9jcmVhdGUgYW4gbmV3IGluc3RhbmNlIG9mIGEgcGl4aSBzdGFnZVxuICAgIHRoaXMuc3RhZ2UgPSBuZXcgUElYSS5TdGFnZShzY2VuZUJnQ29sb3IpO1xuICAgIC8vY3JlYXRlIGEgcmVuZGVyZXIgaW5zdGFuY2UuXG4gICAgdmFyIHJlbmRlcmVyID0gUElYSS5hdXRvRGV0ZWN0UmVuZGVyZXIoc2NlbmVXaWR0aCwgc2NlbmVIZWlnaHQpO1xuICAgIFx0XG4gICAgdGhpcy5pbml0ID0gZnVuY3Rpb24oKSB7XG5cbiAgICAgICAgaWYodHlwZW9mKGNhbnZhc0NvbnRhaW5lcikgPT09ICdzdHJpbmcnKSB7XG4gICAgICAgICAgICBjYW52YXNDb250YWluZXIgPSBkb2N1bWVudC5nZXRFbGVtZW50QnlJZChjYW52YXNDb250YWluZXIpO1xuICAgICAgICB9XG4gICAgICAgIGlmKHR5cGVvZih0aW1lQ29udGFpbmVyKSA9PT0gJ3N0cmluZycpIHtcbiAgICAgICAgICAgIHRpbWVDb250YWluZXIgPSBkb2N1bWVudC5nZXRFbGVtZW50QnlJZCh0aW1lQ29udGFpbmVyKTtcbiAgICAgICAgfVxuXG4gICAgICAgIGNhbnZhc0NvbnRhaW5lci5hcHBlbmRDaGlsZChyZW5kZXJlci52aWV3KTtcbiAgICAgICAgXG4gICAgICAgIGNvbXBvbmVudHMuZm9yRWFjaChmdW5jdGlvbihjKXtcbiAgICBcdFx0Yy5pbml0KCk7XG4gICAgXHR9KTtcbiAgICB9O1xuICAgIFxuICAgIHRoaXMucmVnaXN0ZXJUaW1lQ29udGFpbmVyID0gZnVuY3Rpb24oY29udGFpbmVyKSB7XG4gICAgXHR0aW1lQ29udGFpbmVyLnB1c2goY29udGFpbmVyKTtcbiAgICB9O1xuICAgIFxuICAgIHRoaXMucmVnaXN0ZXJDb21wb25lbnQgPSBmdW5jdGlvbihjb21wb25lbnQpIHtcbiAgICBcdGNvbXBvbmVudHMucHVzaChjb21wb25lbnQpO1xuICAgIFx0dGhpcy5zdGFnZS5hZGRDaGlsZChjb21wb25lbnQuY29udGFpbmVyKTtcbiAgICB9O1xuXG4gICAgdGhpcy5yZWZyZXNoID0gZnVuY3Rpb24oKSB7XG4gICAgXHRjb21wb25lbnRzLmZvckVhY2goZnVuY3Rpb24oYyl7XG4gICAgXHRcdGMucmVmcmVzaCgpO1xuICAgIFx0fSk7XG4gICAgICAgIHJlbmRlcmVyLnJlbmRlcih0aGlzLnN0YWdlKTtcbiAgICB9O1xuXG4gICAgLy8gSW5pdCBwYWdlIGFuZCBpbnRlcnZhbHNcbiAgICB2YXIgcmVmcmVzaEludGVydmFsO1xuXG4gICAgdGhpcy5zdGFydCA9IGZ1bmN0aW9uKCkge1xuXG4gICAgICAgIGlmKCFleHRlcm5hbFJlZnJlc2gpIHtcbiAgICAgICAgICAgIHJlZnJlc2hJbnRlcnZhbCA9IHNldEludGVydmFsKGZ1bmN0aW9uKCkge190aGlzLnJlZnJlc2goKTt9LCAxMDAwL3RoaXMuZnJhbWVyYXRlKTtcbiAgICAgICAgfVxuICAgICAgICBcbiAgICAgICAgY29tcG9uZW50cy5mb3JFYWNoKGZ1bmN0aW9uKGMpe1xuICAgIFx0XHRjLnN0YXJ0KCk7XG4gICAgXHR9KTtcbiAgICB9O1xuXG4gICAgdGhpcy5zdG9wID0gZnVuY3Rpb24oKSB7XG4gICAgICAgIGlmKCFleHRlcm5hbFJlZnJlc2gpIHtcbiAgICAgICAgICAgIGNsZWFySW50ZXJ2YWwocmVmcmVzaEludGVydmFsKTtcbiAgICAgICAgfVxuICAgICAgICBjbGVhckludGVydmFsKHJlZnJlc2hUaW1lSW50ZXJ2YWwpO1xuICAgICAgICBcbiAgICAgICAgY29tcG9uZW50cy5mb3JFYWNoKGZ1bmN0aW9uKGMpe1xuICAgIFx0XHRjLnN0b3AoKTtcbiAgICBcdH0pO1xuICAgIH07XG5cblxuICAgIHRoaXMubG9nID0gZnVuY3Rpb24obSkge1xuICAgICAgICBpZih0aGlzLmxvZ2dlcikge1xuICAgICAgICAgICAgdGhpcy5sb2dnZXIubG9nKG0pO1xuICAgICAgICB9XG4gICAgfTtcblxuXG4gICAgcmV0dXJuIHRoaXM7XG59XG5cbm1vZHVsZS5leHBvcnRzID0ge1xuICAgIFN0YWdlVmlldzogU3RhZ2VWaWV3XG59O1xuIiwiLyoqXG4qIGpzL3V0aWxzLmpzXG4qXG4qIGJhc2ljIHRvb2xzXG4qXG4qL1xuXG4ndXNlIHN0cmljdCc7XG5cbmZ1bmN0aW9uIGZvcm1hdFRpbWUgKHRzKSB7XG5cdHZhciBob3VycyA9IE1hdGguZmxvb3IoICh0cy8xMDAwKSAvIDM2MDAgKSAlIDI0O1xuXHR2YXIgbWludXRlcyA9IE1hdGguZmxvb3IoICh0cy8xMDAwKSAvIDYwICkgJSA2MDtcblx0dmFyIHNlY29uZHMgPSBNYXRoLmZsb29yKCAodHMvMTAwMCkgJSA2MCk7XG5cdHJldHVybiAoKGhvdXJzIDwgMTAgPyAnMCcgKyBob3VycyA6IGhvdXJzKSArICc6JyArIChtaW51dGVzIDwgMTAgPyAnMCcgKyBtaW51dGVzIDogbWludXRlcykgKyAnOicgKyAoc2Vjb25kcyAgPCAxMCA/ICcwJyArIHNlY29uZHMgOiBzZWNvbmRzKSk7XG59XG5cblxubW9kdWxlLmV4cG9ydHMgPSB7XG5cdGZvcm1hdFRpbWU6IGZvcm1hdFRpbWVcbn07XG4iLCIvKipcbioganMvd3N3cmFwcGVyLmpzXG4qXG4qIHNpbXBsZSB3ZWJzZXJ2aWNlIHdyYXBwZXIgdG8gcmVnaXN0ZXIgY2FsbGJhY2tzIG9uIG9ubWVzc2FnZVxuKlxuKi9cblxuLyogZ2xvYmFsIFdlYlNvY2tldDogZmFsc2UgKi9cblxuJ3VzZSBzdHJpY3QnO1xuXG5mdW5jdGlvbiBXc1dyYXBwZXIod3N1cmwsIGxvZ2dlcikge1xuXG4gICAgdmFyIHVybCA9IHdzdXJsO1xuICAgIHZhciBzb2NrID0gbmV3IFdlYlNvY2tldCh1cmwpO1xuICAgIHZhciBsb2dnZXJPYmogPSBsb2dnZXI7XG5cbiAgICB2YXIgbG9nID0gZnVuY3Rpb24obXNnKSB7XG4gICAgICAgIGlmKGxvZ2dlck9iaikge1xuICAgICAgICAgICAgbG9nZ2VyT2JqLmxvZyhtc2cpO1xuICAgICAgICB9XG4gICAgfTtcblxuICAgIHZhciBoYW5kbGVycyA9IFtdO1xuXG4gICAgc29jay5vbm9wZW4gPSBmdW5jdGlvbigpIHtcbiAgICAgICAgbG9nKCdDb25uZWN0ZWQgdG8gJyArIHVybCk7XG4gICAgfTtcblxuICAgIHNvY2sub25jbG9zZSA9IGZ1bmN0aW9uKGUpIHtcbiAgICAgICAgbG9nKCdDb25uZWN0aW9uIGNsb3NlZCAod2FzQ2xlYW4gPSAnICsgZS53YXNDbGVhbiArICcsIGNvZGUgPSAnICsgZS5jb2RlICsgJywgcmVhc29uID0gXFwnJyArIGUucmVhc29uICsgJ1xcJyknKTtcbiAgICAgICAgc29jayA9IG51bGw7XG4gICAgfTtcblxuICAgIHNvY2sub25tZXNzYWdlID0gZnVuY3Rpb24oZSkge1xuICAgICAgICBsb2coJ3JlY2VpdmVkICcgKyBlLmRhdGEpO1xuICAgICAgICB2YXIgZGF0YSA9IEpTT04ucGFyc2UoZS5kYXRhKTtcbiAgICAgICAgaGFuZGxlcnMuZm9yRWFjaChmdW5jdGlvbihoYW5kbGVyKSB7XG4gICAgICAgICAgICBoYW5kbGVyKGRhdGEpO1xuICAgICAgICB9KTtcbiAgICB9O1xuXG4gICAgdGhpcy5tZXNzYWdlID0gZnVuY3Rpb24oaGFuZGxlcikge1xuICAgICAgICBpZihoYW5kbGVyKSB7XG4gICAgICAgICAgICBoYW5kbGVycy5wdXNoKGhhbmRsZXIpO1xuICAgICAgICB9XG4gICAgfTtcblxufVxuXG5tb2R1bGUuZXhwb3J0cyA9IHtcbiAgICBXc1dyYXBwZXI6IFdzV3JhcHBlclxufTtcbiJdfQ== diff -r a7b72620d227 -r 3e075a48e19e annot-server/static/js/annotviz.min.js --- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/annot-server/static/js/annotviz.min.js Thu Jan 22 09:29:49 2015 +0100 @@ -0,0 +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;rDate.parse(t.ts)){var e=Math.floor((Date.parse(t.ts)-this.timeBegin)/(1e3*this.intervalDuration));this.cells[e].categories[i].count+=1,this.cells[e].totalAnnots+=1,this.redrawCell(this.cells[e],e)}},this.initGraphics=function(t){t.graphics=new n.Graphics,t.graphics.position.x=this.circleX+this.radius*Math.sin(t.i*(360/h)*(Math.PI/180)),t.graphics.position.y=this.circleY-this.radius*Math.cos(t.i*(360/h)*(Math.PI/180)),t.graphics.rotation=t.i*(360/h)*(Math.PI/180)+360/(2*h)*(Math.PI/180),this.container.addChild(t.graphics)},this.initTimeTexts=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)},this.redrawCell=function(t){"undefined"==typeof t.graphics?this.initGraphics(t):t.graphics.clear();var i=0;if(t.totalAnnots*this.intervalHeight>this.maxCellHeight)var e=this.maxCellHeight/t.totalAnnots;else var e=this.intervalHeight;for(var n=0;ni?"0"+i:i)+":"+(10>e?"0"+e:e)+":"+(10>n?"0"+n:n);u.setText(o)};var v;return this.start=function(){v=setInterval(function(){i.updateTime()},1e3)},this.refresh=function(){},this.stop=function(){console.log(this.cells)},this}var n=t("pixi"),o=t("./utils.js"),s=t("lodash"),r={logger:void 0,intervalWidth:10,intervalHeight:5,maxCellHeight:200,radius:300};i.exports={AnnotsTimeLine:e}},{"./utils.js":9,lodash:"lodash",pixi:"pixi"}],4:[function(t,i){"use strict";function e(t){var i=o(t).defaults(h).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.annotCategories=i.annotCategories;var e=i.wsPianoroll,l=i.wsAnnot,c=i.stageView;c.registerComponent(this);var d=(new r.AnnotsTimeLine({stageView:c,logger:logger,ws:new annotviz.WsWrapper(wsUriAnnotation,logger),xInit:0,yInit:0,width:624,height:568,timeBegin:Date.now(),timeEnd:Date.now()+3e6,intervalWidth:6,intervalHeight:10,maxCellHeight:70,radius:200,annotCategories:this.annotCategories}),new s.DoubleRoll({stageView:c,logger:logger,ws:e,yInit:this.height-200,sceneHeight:200,pianorolls:[{height:200,timeWidth:10,lineInterval:5e3,noteHeight:10}]}),new s.DoubleRoll({stageView:c,logger:logger,ws:e,orientation:"vertical",sceneHeight:568,pianorolls:[{height:200,timeWidth:60,lineInterval:5e3,noteHeight:5}]})),f=new a.AnnotsRoll({stageView:c,logger:logger,ws:l,parentContainer:d.stage,xInit:624,yInit:568,width:400,height:568,widthRoll:200,framerate:d.framerate,pixelsPerSecond:Math.floor(1024/60),annotColors:this.annotCategories}),g=(new n.Graphics).lineStyle(1,6579300).moveTo(f.container.x,f.container.y).lineTo(f.container.x,f.container.y-f.height).moveTo(f.container.x+f.widthRoll,f.container.y).lineTo(f.container.x+f.widthRoll,f.container.y-f.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();return this.container.addChild(g),this.init=function(){},this.start=function(){},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={xInit:0,yInit:0,width:1024,height:768,annotCategories:[{ts:1421928213e3,colors:{transgressions:"#b90000",rythmique:"#af931e",narration:"#4bdd71",relation:"#1c28ba"},order:["transgressions","rythmique","narration","relation"],defaultColor:"#536991"}]};i.exports={AnnotsVizView:e}},{"./annotsroll.js":2,"./annotstimeline.js":3,"./doubleroll.js":5,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,rs?"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){var i=Math.floor(t/1e3/3600)%24,e=Math.floor(t/1e3/60)%60,n=Math.floor(t/1e3%60);return(10>i?"0"+i:i)+":"+(10>e?"0"+e:e)+":"+(10>n?"0"+n:n)}i.exports={formatTime:e}},{}],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 diff -r a7b72620d227 -r 3e075a48e19e annot-server/static/js/app.js --- a/annot-server/static/js/app.js Thu Jan 22 09:26:43 2015 +0100 +++ b/annot-server/static/js/app.js Thu Jan 22 09:29:49 2015 +0100 @@ -35,6 +35,11 @@ ); } + 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; + } + $scope.data = dataModel.data; var process_categories = function(data) { @@ -50,7 +55,12 @@ } } } - $scope.allCatLabels = cats; + if(typeof data.autocomplete === 'undefined' || data.autocomplete.length === 0) { + $scope.allCatLabels = cats; + } + else { + $scope.allCatLabels = data.autocomplete; + } } }; @@ -101,7 +111,7 @@ } else { wsuri = 'ws://' + window.location.hostname + ':8090/annot'; } - + var eventCode = context.event_code; if(typeof eventCode==='undefined' || eventCode===''){ eventCode = $location.search().event; @@ -173,7 +183,12 @@ }; } - $scope.sendAnnotation = function(label, code, c){ + $scope.sendFreeAnnotation = function(label, text) { + $scope.sendAnnotation(label, window.S(label).slugify().s, label, text, $scope.data.defaultColor || "#536991"); + }; + + $scope.sendAnnotation = function(label, code, freeLabel, freetext, color, c){ + if($scope.username==='' || typeof $scope.username==='undefined'){ showAlert('Vous devez indiquer un nom d\'utilisateur.', false); return; @@ -187,12 +202,24 @@ if(typeof code==='undefined' || code===''){ code = window.S(label).slugify().s; } + + var hexc; + if(color.substring(0, 4) === 'rgb(') { + hexc = colorToHex(color); + } + else { + hexc = color; + } + + var new_annot = { category: {code: code, label: label}, + text: freetext, + color: hexc, user : $scope.username }; sock.send(JSON.stringify(new_annot)); - if(context.logging===true){ + if(context.logging===true){ log('Sent: ' + JSON.stringify(new_annot)); } if(typeof c==='undefined'){ @@ -201,6 +228,8 @@ else{ $scope.annotPile.push(c); } + $scope.catText = ""; + $scope.catLabel = ""; } else { showAlert('La socket ne fonctionne pas.', false); if(context.logging===true){ @@ -210,7 +239,7 @@ }; // Interface management - $scope.selectLevel = function(label, code, c){ + $scope.selectLevel = function(label, code, freelabel, freetext, color, c){ if(typeof c==='undefined'){ $scope.returnVisStyle = {visibility:'hidden'}; $scope.selectedlevel = false; @@ -223,7 +252,7 @@ else{ // Send query //console.log('send ntm', c); - $scope.sendAnnotation(label, code, c); + $scope.sendAnnotation(label, code, freelabel, freetext, color, c); } }; diff -r a7b72620d227 -r 3e075a48e19e annot-server/static/js/app.min.js --- a/annot-server/static/js/app.min.js Thu Jan 22 09:26:43 2015 +0100 +++ b/annot-server/static/js/app.min.js Thu Jan 22 09:29:49 2015 +0100 @@ -1,1 +1,1 @@ -!function(){"use strict";angular.module("mons",["ngResource","ngRoute","autocomplete"]).config(function(e){e.when("/",{controller:"homeCtrl"}).otherwise({redirectTo:"/"})}).config(function(e){e.debugEnabled(!0)}).service("dataApi",function(e,n){this.dataResource=e(n.urls.dataUrl)}).service("dataModel",function(e,n){this.data="undefined"!=typeof n.categories_json&&n.categories_json?JSON.parse(n.categories_json):e.dataResource.get()}).controller("homeCtrl",function(e,n,t,o,i){function r(e){return decodeURI((new RegExp(e+"=(.+?)(&|$)").exec(location.search)||[,null])[1])}function s(n,t){e.alertMessage=n,e.showSuccessAlert=t,e.showAlertDiv=!0,e.$$phase||e.$apply(),e.currentInterval&&(i.cancel(e.currentInterval),e.currentInterval=!1),e.currentInterval=i(function(){i.cancel(e.currentInterval),e.showAlertDiv=!1},2e3,1)}function a(e){d&&(d.innerHTML+=e+"\n",d.scrollTop=d.scrollHeight)}e.data=t.data;var c=function(n){if("undefined"!=typeof n.categories&&n.categories.length>0){for(var t=[],o=n.categories.length,i=0;o>i;i++)if(t.push(n.categories[i].label),"undefined"!=typeof n.categories[i].subcategories&&n.categories[i].subcategories.length>0)for(var r=n.categories[i].subcategories.length,s=0;r>s;s++)t.push(n.categories[i].subcategories[s].label);e.allCatLabels=t}};"undefined"!=typeof t.data.$promise?t.data.$promise.then(c):c(t.data),e.selectedlevel=!1,e.currentInterval=!1,e.showSuccessAlert=!1,e.showAlertDiv=!1,e.annotPile=[];var l=null,d=null;d=document.getElementById("log");var u;u="file:"===window.location.protocol?"ws://127.0.0.1:8090/annot":"ws://"+window.location.hostname+":8090/annot";var f=o.event_code;return"undefined"!=typeof f&&""!==f||(f=n.search().event,"undefined"!=typeof f&&""!==f||(f=r("event"),"undefined"!=typeof f&&""!==f))?(u=u+"?event="+f,"WebSocket"in window?l=new WebSocket(u):"MozWebSocket"in window?l=new window.MozWebSocket(u):(o.logging===!0&&a("Browser does not support WebSocket!"),window.location="http://autobahn.ws/unsupportedbrowser"),l&&(l.onopen=function(){o.logging===!0&&a("Connected to "+u)},l.onclose=function(e){o.logging===!0&&a("Connection closed (wasClean = "+e.wasClean+", code = "+e.code+", reason = "+e.reason+")"),s("Communication interrompue : la socket vient de se fermer.",!1),l=null},l.onmessage=function(n){var t=JSON.parse(n.data);if(o.logging&&a("Got message: "+n.data),e.annotPile.length>0){var r,s=e.annotPile.shift(),c="OK"===t.status;s===!1?(e.sendBtnSuccess=c,e.sendBtnError=!c,r=i(function(){i.cancel(r),e.sendBtnSuccess=!1,e.sendBtnError=!1},2e3,1)):(s.sendSuccess=c,s.sendError=!c,r=i(function(){i.cancel(r),s.sendSuccess=!1,s.sendError=!1},2e3,1)),e.$$phase||e.$apply()}}),e.sendAnnotation=function(n,t,i){if(""===e.username||"undefined"==typeof e.username)return void s("Vous devez indiquer un nom d'utilisateur.",!1);if(""===n||"undefined"==typeof n)return void s("Vous devez indiquer un nom de catégorie.",!1);if(l){("undefined"==typeof t||""===t)&&(t=window.S(n).slugify().s);var r={category:{code:t,label:n},user:e.username};l.send(JSON.stringify(r)),o.logging===!0&&a("Sent: "+JSON.stringify(r)),e.annotPile.push("undefined"==typeof i?!1:i)}else s("La socket ne fonctionne pas.",!1),o.logging===!0&&a("Not connected.")},void(e.selectLevel=function(n,t,o){return"undefined"==typeof o?(e.returnVisStyle={visibility:"hidden"},void(e.selectedlevel=!1)):void("undefined"!=typeof o.subcategories&&o.subcategories.length>0?(e.selectedlevel=o.subcategories,e.returnVisStyle={visibility:"show"}):e.sendAnnotation(n,t,o))})):void alert("le code de l'événement doit être indiqué dans un paramètre de template u dans l'url selon ?event=CODE_EVENEMENT.")})}(); \ No newline at end of file +!function(){"use strict";angular.module("mons",["ngResource","ngRoute","autocomplete"]).config(function(e){e.when("/",{controller:"homeCtrl"}).otherwise({redirectTo:"/"})}).config(function(e){e.debugEnabled(!0)}).service("dataApi",function(e,n){this.dataResource=e(n.urls.dataUrl)}).service("dataModel",function(e,n){this.data="undefined"!=typeof n.categories_json&&n.categories_json?JSON.parse(n.categories_json):e.dataResource.get()}).controller("homeCtrl",function(e,n,t,o,r){function s(e){return decodeURI((new RegExp(e+"=(.+?)(&|$)").exec(location.search)||[,null])[1])}function i(e){var n=/rgba?\(\s*(\d+)\s*,\s*(\d+)\s*,\s*(\d+)/.exec(e);return n?"#"+(1<<24|n[1]<<16|n[2]<<8|n[3]).toString(16).substr(1):e}function a(n,t){e.alertMessage=n,e.showSuccessAlert=t,e.showAlertDiv=!0,e.$$phase||e.$apply(),e.currentInterval&&(r.cancel(e.currentInterval),e.currentInterval=!1),e.currentInterval=r(function(){r.cancel(e.currentInterval),e.showAlertDiv=!1},2e3,1)}function c(e){u&&(u.innerHTML+=e+"\n",u.scrollTop=u.scrollHeight)}e.data=t.data;var l=function(n){if("undefined"!=typeof n.categories&&n.categories.length>0){for(var t=[],o=n.categories.length,r=0;o>r;r++)if(t.push(n.categories[r].label),"undefined"!=typeof n.categories[r].subcategories&&n.categories[r].subcategories.length>0)for(var s=n.categories[r].subcategories.length,i=0;s>i;i++)t.push(n.categories[r].subcategories[i].label);e.allCatLabels="undefined"==typeof n.autocomplete||0===n.autocomplete.length?t:n.autocomplete}};"undefined"!=typeof t.data.$promise?t.data.$promise.then(l):l(t.data),e.selectedlevel=!1,e.currentInterval=!1,e.showSuccessAlert=!1,e.showAlertDiv=!1,e.annotPile=[];var d=null,u=null;u=document.getElementById("log");var f;f="file:"===window.location.protocol?"ws://127.0.0.1:8090/annot":"ws://"+window.location.hostname+":8090/annot";var g=o.event_code;return"undefined"!=typeof g&&""!==g||(g=n.search().event,"undefined"!=typeof g&&""!==g||(g=s("event"),"undefined"!=typeof g&&""!==g))?(f=f+"?event="+g,"WebSocket"in window?d=new WebSocket(f):"MozWebSocket"in window?d=new window.MozWebSocket(f):(o.logging===!0&&c("Browser does not support WebSocket!"),window.location="http://autobahn.ws/unsupportedbrowser"),d&&(d.onopen=function(){o.logging===!0&&c("Connected to "+f)},d.onclose=function(e){o.logging===!0&&c("Connection closed (wasClean = "+e.wasClean+", code = "+e.code+", reason = "+e.reason+")"),a("Communication interrompue : la socket vient de se fermer.",!1),d=null},d.onmessage=function(n){var t=JSON.parse(n.data);if(o.logging&&c("Got message: "+n.data),e.annotPile.length>0){var s,i=e.annotPile.shift(),a="OK"===t.status;i===!1?(e.sendBtnSuccess=a,e.sendBtnError=!a,s=r(function(){r.cancel(s),e.sendBtnSuccess=!1,e.sendBtnError=!1},2e3,1)):(i.sendSuccess=a,i.sendError=!a,s=r(function(){r.cancel(s),i.sendSuccess=!1,i.sendError=!1},2e3,1)),e.$$phase||e.$apply()}}),e.sendFreeAnnotation=function(n,t){e.sendAnnotation(n,window.S(n).slugify().s,n,t,e.data.defaultColor||"#536991")},e.sendAnnotation=function(n,t,r,s,l,u){if(""===e.username||"undefined"==typeof e.username)return void a("Vous devez indiquer un nom d'utilisateur.",!1);if(""===n||"undefined"==typeof n)return void a("Vous devez indiquer un nom de catégorie.",!1);if(d){("undefined"==typeof t||""===t)&&(t=window.S(n).slugify().s);var f;f="rgb("===l.substring(0,4)?i(l):l;var g={category:{code:t,label:n},text:s,color:f,user:e.username};d.send(JSON.stringify(g)),o.logging===!0&&c("Sent: "+JSON.stringify(g)),e.annotPile.push("undefined"==typeof u?!1:u),e.catText="",e.catLabel=""}else a("La socket ne fonctionne pas.",!1),o.logging===!0&&c("Not connected.")},void(e.selectLevel=function(n,t,o,r,s,i){return"undefined"==typeof i?(e.returnVisStyle={visibility:"hidden"},void(e.selectedlevel=!1)):void("undefined"!=typeof i.subcategories&&i.subcategories.length>0?(e.selectedlevel=i.subcategories,e.returnVisStyle={visibility:"show"}):e.sendAnnotation(n,t,o,r,s,i))})):void alert("le code de l'événement doit être indiqué dans un paramètre de template u dans l'url selon ?event=CODE_EVENEMENT.")})}(); \ No newline at end of file diff -r a7b72620d227 -r 3e075a48e19e annot-server/static/js/libs-annotviz.js --- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/annot-server/static/js/libs-annotviz.js Thu Jan 22 09:29:49 2015 +0100 @@ -0,0 +1,7173 @@ +require=(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 + * Build: `lodash modern -o ./dist/lodash.js` + * Copyright 2012-2013 The Dojo Foundation + * Based on Underscore.js 1.5.2 + * Copyright 2009-2013 Jeremy Ashkenas, DocumentCloud and Investigative Reporters & Editors + * Available under MIT license + */ +;(function() { + + /** Used as a safe reference for `undefined` in pre ES5 environments */ + var undefined; + + /** Used to pool arrays and objects used internally */ + var arrayPool = [], + objectPool = []; + + /** Used to generate unique IDs */ + var idCounter = 0; + + /** Used to prefix keys to avoid issues with `__proto__` and properties on `Object.prototype` */ + var keyPrefix = +new Date + ''; + + /** Used as the size when optimizations are enabled for large arrays */ + var largeArraySize = 75; + + /** Used as the max size of the `arrayPool` and `objectPool` */ + var maxPoolSize = 40; + + /** Used to detect and test whitespace */ + var whitespace = ( + // whitespace + ' \t\x0B\f\xA0\ufeff' + + + // line terminators + '\n\r\u2028\u2029' + + + // unicode category "Zs" space separators + '\u1680\u180e\u2000\u2001\u2002\u2003\u2004\u2005\u2006\u2007\u2008\u2009\u200a\u202f\u205f\u3000' + ); + + /** Used to match empty string literals in compiled template source */ + var reEmptyStringLeading = /\b__p \+= '';/g, + reEmptyStringMiddle = /\b(__p \+=) '' \+/g, + reEmptyStringTrailing = /(__e\(.*?\)|\b__t\)) \+\n'';/g; + + /** + * Used to match ES6 template delimiters + * http://people.mozilla.org/~jorendorff/es6-draft.html#sec-literals-string-literals + */ + var reEsTemplate = /\$\{([^\\}]*(?:\\.[^\\}]*)*)\}/g; + + /** Used to match regexp flags from their coerced string values */ + var reFlags = /\w*$/; + + /** Used to detected named functions */ + var reFuncName = /^\s*function[ \n\r\t]+\w/; + + /** Used to match "interpolate" template delimiters */ + var reInterpolate = /<%=([\s\S]+?)%>/g; + + /** Used to match leading whitespace and zeros to be removed */ + var reLeadingSpacesAndZeros = RegExp('^[' + whitespace + ']*0+(?=.$)'); + + /** Used to ensure capturing order of template delimiters */ + var reNoMatch = /($^)/; + + /** Used to detect functions containing a `this` reference */ + var reThis = /\bthis\b/; + + /** Used to match unescaped characters in compiled string literals */ + var reUnescapedString = /['\n\r\t\u2028\u2029\\]/g; + + /** Used to assign default `context` object properties */ + var contextProps = [ + 'Array', 'Boolean', 'Date', 'Function', 'Math', 'Number', 'Object', + 'RegExp', 'String', '_', 'attachEvent', 'clearTimeout', 'isFinite', 'isNaN', + 'parseInt', 'setTimeout' + ]; + + /** Used to make template sourceURLs easier to identify */ + var templateCounter = 0; + + /** `Object#toString` result shortcuts */ + var argsClass = '[object Arguments]', + arrayClass = '[object Array]', + boolClass = '[object Boolean]', + dateClass = '[object Date]', + funcClass = '[object Function]', + numberClass = '[object Number]', + objectClass = '[object Object]', + regexpClass = '[object RegExp]', + stringClass = '[object String]'; + + /** Used to identify object classifications that `_.clone` supports */ + var cloneableClasses = {}; + cloneableClasses[funcClass] = false; + cloneableClasses[argsClass] = cloneableClasses[arrayClass] = + cloneableClasses[boolClass] = cloneableClasses[dateClass] = + cloneableClasses[numberClass] = cloneableClasses[objectClass] = + cloneableClasses[regexpClass] = cloneableClasses[stringClass] = true; + + /** Used as an internal `_.debounce` options object */ + var debounceOptions = { + 'leading': false, + 'maxWait': 0, + 'trailing': false + }; + + /** Used as the property descriptor for `__bindData__` */ + var descriptor = { + 'configurable': false, + 'enumerable': false, + 'value': null, + 'writable': false + }; + + /** Used to determine if values are of the language type Object */ + var objectTypes = { + 'boolean': false, + 'function': true, + 'object': true, + 'number': false, + 'string': false, + 'undefined': false + }; + + /** Used to escape characters for inclusion in compiled string literals */ + var stringEscapes = { + '\\': '\\', + "'": "'", + '\n': 'n', + '\r': 'r', + '\t': 't', + '\u2028': 'u2028', + '\u2029': 'u2029' + }; + + /** Used as a reference to the global object */ + var root = (objectTypes[typeof window] && window) || this; + + /** Detect free variable `exports` */ + var freeExports = objectTypes[typeof exports] && exports && !exports.nodeType && exports; + + /** Detect free variable `module` */ + var freeModule = objectTypes[typeof module] && module && !module.nodeType && module; + + /** Detect the popular CommonJS extension `module.exports` */ + var moduleExports = freeModule && freeModule.exports === freeExports && freeExports; + + /** Detect free variable `global` from Node.js or Browserified code and use it as `root` */ + var freeGlobal = objectTypes[typeof global] && global; + if (freeGlobal && (freeGlobal.global === freeGlobal || freeGlobal.window === freeGlobal)) { + root = freeGlobal; + } + + /*--------------------------------------------------------------------------*/ + + /** + * The base implementation of `_.indexOf` without support for binary searches + * or `fromIndex` constraints. + * + * @private + * @param {Array} array The array to search. + * @param {*} value The value to search for. + * @param {number} [fromIndex=0] The index to search from. + * @returns {number} Returns the index of the matched value or `-1`. + */ + function baseIndexOf(array, value, fromIndex) { + var index = (fromIndex || 0) - 1, + length = array ? array.length : 0; + + while (++index < length) { + if (array[index] === value) { + return index; + } + } + return -1; + } + + /** + * An implementation of `_.contains` for cache objects that mimics the return + * signature of `_.indexOf` by returning `0` if the value is found, else `-1`. + * + * @private + * @param {Object} cache The cache object to inspect. + * @param {*} value The value to search for. + * @returns {number} Returns `0` if `value` is found, else `-1`. + */ + function cacheIndexOf(cache, value) { + var type = typeof value; + cache = cache.cache; + + if (type == 'boolean' || value == null) { + return cache[value] ? 0 : -1; + } + if (type != 'number' && type != 'string') { + type = 'object'; + } + var key = type == 'number' ? value : keyPrefix + value; + cache = (cache = cache[type]) && cache[key]; + + return type == 'object' + ? (cache && baseIndexOf(cache, value) > -1 ? 0 : -1) + : (cache ? 0 : -1); + } + + /** + * Adds a given value to the corresponding cache object. + * + * @private + * @param {*} value The value to add to the cache. + */ + function cachePush(value) { + var cache = this.cache, + type = typeof value; + + if (type == 'boolean' || value == null) { + cache[value] = true; + } else { + if (type != 'number' && type != 'string') { + type = 'object'; + } + var key = type == 'number' ? value : keyPrefix + value, + typeCache = cache[type] || (cache[type] = {}); + + if (type == 'object') { + (typeCache[key] || (typeCache[key] = [])).push(value); + } else { + typeCache[key] = true; + } + } + } + + /** + * Used by `_.max` and `_.min` as the default callback when a given + * collection is a string value. + * + * @private + * @param {string} value The character to inspect. + * @returns {number} Returns the code unit of given character. + */ + function charAtCallback(value) { + return value.charCodeAt(0); + } + + /** + * Used by `sortBy` to compare transformed `collection` elements, stable sorting + * them in ascending order. + * + * @private + * @param {Object} a The object to compare to `b`. + * @param {Object} b The object to compare to `a`. + * @returns {number} Returns the sort order indicator of `1` or `-1`. + */ + function compareAscending(a, b) { + var ac = a.criteria, + bc = b.criteria, + index = -1, + length = ac.length; + + while (++index < length) { + var value = ac[index], + other = bc[index]; + + if (value !== other) { + if (value > other || typeof value == 'undefined') { + return 1; + } + if (value < other || typeof other == 'undefined') { + return -1; + } + } + } + // Fixes an `Array#sort` bug in the JS engine embedded in Adobe applications + // that causes it, under certain circumstances, to return the same value for + // `a` and `b`. See https://github.com/jashkenas/underscore/pull/1247 + // + // This also ensures a stable sort in V8 and other engines. + // See http://code.google.com/p/v8/issues/detail?id=90 + return a.index - b.index; + } + + /** + * Creates a cache object to optimize linear searches of large arrays. + * + * @private + * @param {Array} [array=[]] The array to search. + * @returns {null|Object} Returns the cache object or `null` if caching should not be used. + */ + function createCache(array) { + var index = -1, + length = array.length, + first = array[0], + mid = array[(length / 2) | 0], + last = array[length - 1]; + + if (first && typeof first == 'object' && + mid && typeof mid == 'object' && last && typeof last == 'object') { + return false; + } + var cache = getObject(); + cache['false'] = cache['null'] = cache['true'] = cache['undefined'] = false; + + var result = getObject(); + result.array = array; + result.cache = cache; + result.push = cachePush; + + while (++index < length) { + result.push(array[index]); + } + return result; + } + + /** + * Used by `template` to escape characters for inclusion in compiled + * string literals. + * + * @private + * @param {string} match The matched character to escape. + * @returns {string} Returns the escaped character. + */ + function escapeStringChar(match) { + return '\\' + stringEscapes[match]; + } + + /** + * Gets an array from the array pool or creates a new one if the pool is empty. + * + * @private + * @returns {Array} The array from the pool. + */ + function getArray() { + return arrayPool.pop() || []; + } + + /** + * Gets an object from the object pool or creates a new one if the pool is empty. + * + * @private + * @returns {Object} The object from the pool. + */ + function getObject() { + return objectPool.pop() || { + 'array': null, + 'cache': null, + 'criteria': null, + 'false': false, + 'index': 0, + 'null': false, + 'number': null, + 'object': null, + 'push': null, + 'string': null, + 'true': false, + 'undefined': false, + 'value': null + }; + } + + /** + * Releases the given array back to the array pool. + * + * @private + * @param {Array} [array] The array to release. + */ + function releaseArray(array) { + array.length = 0; + if (arrayPool.length < maxPoolSize) { + arrayPool.push(array); + } + } + + /** + * Releases the given object back to the object pool. + * + * @private + * @param {Object} [object] The object to release. + */ + function releaseObject(object) { + var cache = object.cache; + if (cache) { + releaseObject(cache); + } + object.array = object.cache = object.criteria = object.object = object.number = object.string = object.value = null; + if (objectPool.length < maxPoolSize) { + objectPool.push(object); + } + } + + /** + * Slices the `collection` from the `start` index up to, but not including, + * the `end` index. + * + * Note: This function is used instead of `Array#slice` to support node lists + * in IE < 9 and to ensure dense arrays are returned. + * + * @private + * @param {Array|Object|string} collection The collection to slice. + * @param {number} start The start index. + * @param {number} end The end index. + * @returns {Array} Returns the new array. + */ + function slice(array, start, end) { + start || (start = 0); + if (typeof end == 'undefined') { + end = array ? array.length : 0; + } + var index = -1, + length = end - start || 0, + result = Array(length < 0 ? 0 : length); + + while (++index < length) { + result[index] = array[start + index]; + } + return result; + } + + /*--------------------------------------------------------------------------*/ + + /** + * Create a new `lodash` function using the given context object. + * + * @static + * @memberOf _ + * @category Utilities + * @param {Object} [context=root] The context object. + * @returns {Function} Returns the `lodash` function. + */ + function runInContext(context) { + // Avoid issues with some ES3 environments that attempt to use values, named + // after built-in constructors like `Object`, for the creation of literals. + // ES5 clears this up by stating that literals must use built-in constructors. + // See http://es5.github.io/#x11.1.5. + context = context ? _.defaults(root.Object(), context, _.pick(root, contextProps)) : root; + + /** Native constructor references */ + var Array = context.Array, + Boolean = context.Boolean, + Date = context.Date, + Function = context.Function, + Math = context.Math, + Number = context.Number, + Object = context.Object, + RegExp = context.RegExp, + String = context.String, + TypeError = context.TypeError; + + /** + * Used for `Array` method references. + * + * Normally `Array.prototype` would suffice, however, using an array literal + * avoids issues in Narwhal. + */ + var arrayRef = []; + + /** Used for native method references */ + var objectProto = Object.prototype; + + /** Used to restore the original `_` reference in `noConflict` */ + var oldDash = context._; + + /** Used to resolve the internal [[Class]] of values */ + var toString = objectProto.toString; + + /** Used to detect if a method is native */ + var reNative = RegExp('^' + + String(toString) + .replace(/[.*+?^${}()|[\]\\]/g, '\\$&') + .replace(/toString| for [^\]]+/g, '.*?') + '$' + ); + + /** Native method shortcuts */ + var ceil = Math.ceil, + clearTimeout = context.clearTimeout, + floor = Math.floor, + fnToString = Function.prototype.toString, + getPrototypeOf = isNative(getPrototypeOf = Object.getPrototypeOf) && getPrototypeOf, + hasOwnProperty = objectProto.hasOwnProperty, + push = arrayRef.push, + setTimeout = context.setTimeout, + splice = arrayRef.splice, + unshift = arrayRef.unshift; + + /** Used to set meta data on functions */ + var defineProperty = (function() { + // IE 8 only accepts DOM elements + try { + var o = {}, + func = isNative(func = Object.defineProperty) && func, + result = func(o, o, o) && func; + } catch(e) { } + return result; + }()); + + /* Native method shortcuts for methods with the same name as other `lodash` methods */ + var nativeCreate = isNative(nativeCreate = Object.create) && nativeCreate, + nativeIsArray = isNative(nativeIsArray = Array.isArray) && nativeIsArray, + nativeIsFinite = context.isFinite, + nativeIsNaN = context.isNaN, + nativeKeys = isNative(nativeKeys = Object.keys) && nativeKeys, + nativeMax = Math.max, + nativeMin = Math.min, + nativeParseInt = context.parseInt, + nativeRandom = Math.random; + + /** Used to lookup a built-in constructor by [[Class]] */ + var ctorByClass = {}; + ctorByClass[arrayClass] = Array; + ctorByClass[boolClass] = Boolean; + ctorByClass[dateClass] = Date; + ctorByClass[funcClass] = Function; + ctorByClass[objectClass] = Object; + ctorByClass[numberClass] = Number; + ctorByClass[regexpClass] = RegExp; + ctorByClass[stringClass] = String; + + /*--------------------------------------------------------------------------*/ + + /** + * Creates a `lodash` object which wraps the given value to enable intuitive + * method chaining. + * + * In addition to Lo-Dash methods, wrappers also have the following `Array` methods: + * `concat`, `join`, `pop`, `push`, `reverse`, `shift`, `slice`, `sort`, `splice`, + * and `unshift` + * + * Chaining is supported in custom builds as long as the `value` method is + * implicitly or explicitly included in the build. + * + * The chainable wrapper functions are: + * `after`, `assign`, `bind`, `bindAll`, `bindKey`, `chain`, `compact`, + * `compose`, `concat`, `countBy`, `create`, `createCallback`, `curry`, + * `debounce`, `defaults`, `defer`, `delay`, `difference`, `filter`, `flatten`, + * `forEach`, `forEachRight`, `forIn`, `forInRight`, `forOwn`, `forOwnRight`, + * `functions`, `groupBy`, `indexBy`, `initial`, `intersection`, `invert`, + * `invoke`, `keys`, `map`, `max`, `memoize`, `merge`, `min`, `object`, `omit`, + * `once`, `pairs`, `partial`, `partialRight`, `pick`, `pluck`, `pull`, `push`, + * `range`, `reject`, `remove`, `rest`, `reverse`, `shuffle`, `slice`, `sort`, + * `sortBy`, `splice`, `tap`, `throttle`, `times`, `toArray`, `transform`, + * `union`, `uniq`, `unshift`, `unzip`, `values`, `where`, `without`, `wrap`, + * and `zip` + * + * The non-chainable wrapper functions are: + * `clone`, `cloneDeep`, `contains`, `escape`, `every`, `find`, `findIndex`, + * `findKey`, `findLast`, `findLastIndex`, `findLastKey`, `has`, `identity`, + * `indexOf`, `isArguments`, `isArray`, `isBoolean`, `isDate`, `isElement`, + * `isEmpty`, `isEqual`, `isFinite`, `isFunction`, `isNaN`, `isNull`, `isNumber`, + * `isObject`, `isPlainObject`, `isRegExp`, `isString`, `isUndefined`, `join`, + * `lastIndexOf`, `mixin`, `noConflict`, `parseInt`, `pop`, `random`, `reduce`, + * `reduceRight`, `result`, `shift`, `size`, `some`, `sortedIndex`, `runInContext`, + * `template`, `unescape`, `uniqueId`, and `value` + * + * The wrapper functions `first` and `last` return wrapped values when `n` is + * provided, otherwise they return unwrapped values. + * + * Explicit chaining can be enabled by using the `_.chain` method. + * + * @name _ + * @constructor + * @category Chaining + * @param {*} value The value to wrap in a `lodash` instance. + * @returns {Object} Returns a `lodash` instance. + * @example + * + * var wrapped = _([1, 2, 3]); + * + * // returns an unwrapped value + * wrapped.reduce(function(sum, num) { + * return sum + num; + * }); + * // => 6 + * + * // returns a wrapped value + * var squares = wrapped.map(function(num) { + * return num * num; + * }); + * + * _.isArray(squares); + * // => false + * + * _.isArray(squares.value()); + * // => true + */ + function lodash(value) { + // don't wrap if already wrapped, even if wrapped by a different `lodash` constructor + return (value && typeof value == 'object' && !isArray(value) && hasOwnProperty.call(value, '__wrapped__')) + ? value + : new lodashWrapper(value); + } + + /** + * A fast path for creating `lodash` wrapper objects. + * + * @private + * @param {*} value The value to wrap in a `lodash` instance. + * @param {boolean} chainAll A flag to enable chaining for all methods + * @returns {Object} Returns a `lodash` instance. + */ + function lodashWrapper(value, chainAll) { + this.__chain__ = !!chainAll; + this.__wrapped__ = value; + } + // ensure `new lodashWrapper` is an instance of `lodash` + lodashWrapper.prototype = lodash.prototype; + + /** + * An object used to flag environments features. + * + * @static + * @memberOf _ + * @type Object + */ + var support = lodash.support = {}; + + /** + * Detect if functions can be decompiled by `Function#toString` + * (all but PS3 and older Opera mobile browsers & avoided in Windows 8 apps). + * + * @memberOf _.support + * @type boolean + */ + support.funcDecomp = !isNative(context.WinRTError) && reThis.test(runInContext); + + /** + * Detect if `Function#name` is supported (all but IE). + * + * @memberOf _.support + * @type boolean + */ + support.funcNames = typeof Function.name == 'string'; + + /** + * By default, the template delimiters used by Lo-Dash are similar to those in + * embedded Ruby (ERB). Change the following template settings to use alternative + * delimiters. + * + * @static + * @memberOf _ + * @type Object + */ + lodash.templateSettings = { + + /** + * Used to detect `data` property values to be HTML-escaped. + * + * @memberOf _.templateSettings + * @type RegExp + */ + 'escape': /<%-([\s\S]+?)%>/g, + + /** + * Used to detect code to be evaluated. + * + * @memberOf _.templateSettings + * @type RegExp + */ + 'evaluate': /<%([\s\S]+?)%>/g, + + /** + * Used to detect `data` property values to inject. + * + * @memberOf _.templateSettings + * @type RegExp + */ + 'interpolate': reInterpolate, + + /** + * Used to reference the data object in the template text. + * + * @memberOf _.templateSettings + * @type string + */ + 'variable': '', + + /** + * Used to import variables into the compiled template. + * + * @memberOf _.templateSettings + * @type Object + */ + 'imports': { + + /** + * A reference to the `lodash` function. + * + * @memberOf _.templateSettings.imports + * @type Function + */ + '_': lodash + } + }; + + /*--------------------------------------------------------------------------*/ + + /** + * The base implementation of `_.bind` that creates the bound function and + * sets its meta data. + * + * @private + * @param {Array} bindData The bind data array. + * @returns {Function} Returns the new bound function. + */ + function baseBind(bindData) { + var func = bindData[0], + partialArgs = bindData[2], + thisArg = bindData[4]; + + function bound() { + // `Function#bind` spec + // http://es5.github.io/#x15.3.4.5 + if (partialArgs) { + // avoid `arguments` object deoptimizations by using `slice` instead + // of `Array.prototype.slice.call` and not assigning `arguments` to a + // variable as a ternary expression + var args = slice(partialArgs); + push.apply(args, arguments); + } + // mimic the constructor's `return` behavior + // http://es5.github.io/#x13.2.2 + if (this instanceof bound) { + // ensure `new bound` is an instance of `func` + var thisBinding = baseCreate(func.prototype), + result = func.apply(thisBinding, args || arguments); + return isObject(result) ? result : thisBinding; + } + return func.apply(thisArg, args || arguments); + } + setBindData(bound, bindData); + return bound; + } + + /** + * The base implementation of `_.clone` without argument juggling or support + * for `thisArg` binding. + * + * @private + * @param {*} value The value to clone. + * @param {boolean} [isDeep=false] Specify a deep clone. + * @param {Function} [callback] The function to customize cloning values. + * @param {Array} [stackA=[]] Tracks traversed source objects. + * @param {Array} [stackB=[]] Associates clones with source counterparts. + * @returns {*} Returns the cloned value. + */ + function baseClone(value, isDeep, callback, stackA, stackB) { + if (callback) { + var result = callback(value); + if (typeof result != 'undefined') { + return result; + } + } + // inspect [[Class]] + var isObj = isObject(value); + if (isObj) { + var className = toString.call(value); + if (!cloneableClasses[className]) { + return value; + } + var ctor = ctorByClass[className]; + switch (className) { + case boolClass: + case dateClass: + return new ctor(+value); + + case numberClass: + case stringClass: + return new ctor(value); + + case regexpClass: + result = ctor(value.source, reFlags.exec(value)); + result.lastIndex = value.lastIndex; + return result; + } + } else { + return value; + } + var isArr = isArray(value); + if (isDeep) { + // check for circular references and return corresponding clone + var initedStack = !stackA; + stackA || (stackA = getArray()); + stackB || (stackB = getArray()); + + var length = stackA.length; + while (length--) { + if (stackA[length] == value) { + return stackB[length]; + } + } + result = isArr ? ctor(value.length) : {}; + } + else { + result = isArr ? slice(value) : assign({}, value); + } + // add array properties assigned by `RegExp#exec` + if (isArr) { + if (hasOwnProperty.call(value, 'index')) { + result.index = value.index; + } + if (hasOwnProperty.call(value, 'input')) { + result.input = value.input; + } + } + // exit for shallow clone + if (!isDeep) { + return result; + } + // add the source value to the stack of traversed objects + // and associate it with its clone + stackA.push(value); + stackB.push(result); + + // recursively populate clone (susceptible to call stack limits) + (isArr ? forEach : forOwn)(value, function(objValue, key) { + result[key] = baseClone(objValue, isDeep, callback, stackA, stackB); + }); + + if (initedStack) { + releaseArray(stackA); + releaseArray(stackB); + } + return result; + } + + /** + * The base implementation of `_.create` without support for assigning + * properties to the created object. + * + * @private + * @param {Object} prototype The object to inherit from. + * @returns {Object} Returns the new object. + */ + function baseCreate(prototype, properties) { + return isObject(prototype) ? nativeCreate(prototype) : {}; + } + // fallback for browsers without `Object.create` + if (!nativeCreate) { + baseCreate = (function() { + function Object() {} + return function(prototype) { + if (isObject(prototype)) { + Object.prototype = prototype; + var result = new Object; + Object.prototype = null; + } + return result || context.Object(); + }; + }()); + } + + /** + * The base implementation of `_.createCallback` without support for creating + * "_.pluck" or "_.where" style callbacks. + * + * @private + * @param {*} [func=identity] The value to convert to a callback. + * @param {*} [thisArg] The `this` binding of the created callback. + * @param {number} [argCount] The number of arguments the callback accepts. + * @returns {Function} Returns a callback function. + */ + function baseCreateCallback(func, thisArg, argCount) { + if (typeof func != 'function') { + return identity; + } + // exit early for no `thisArg` or already bound by `Function#bind` + if (typeof thisArg == 'undefined' || !('prototype' in func)) { + return func; + } + var bindData = func.__bindData__; + if (typeof bindData == 'undefined') { + if (support.funcNames) { + bindData = !func.name; + } + bindData = bindData || !support.funcDecomp; + if (!bindData) { + var source = fnToString.call(func); + if (!support.funcNames) { + bindData = !reFuncName.test(source); + } + if (!bindData) { + // checks if `func` references the `this` keyword and stores the result + bindData = reThis.test(source); + setBindData(func, bindData); + } + } + } + // exit early if there are no `this` references or `func` is bound + if (bindData === false || (bindData !== true && bindData[1] & 1)) { + return func; + } + switch (argCount) { + case 1: return function(value) { + return func.call(thisArg, value); + }; + case 2: return function(a, b) { + return func.call(thisArg, a, b); + }; + case 3: return function(value, index, collection) { + return func.call(thisArg, value, index, collection); + }; + case 4: return function(accumulator, value, index, collection) { + return func.call(thisArg, accumulator, value, index, collection); + }; + } + return bind(func, thisArg); + } + + /** + * The base implementation of `createWrapper` that creates the wrapper and + * sets its meta data. + * + * @private + * @param {Array} bindData The bind data array. + * @returns {Function} Returns the new function. + */ + function baseCreateWrapper(bindData) { + var func = bindData[0], + bitmask = bindData[1], + partialArgs = bindData[2], + partialRightArgs = bindData[3], + thisArg = bindData[4], + arity = bindData[5]; + + var isBind = bitmask & 1, + isBindKey = bitmask & 2, + isCurry = bitmask & 4, + isCurryBound = bitmask & 8, + key = func; + + function bound() { + var thisBinding = isBind ? thisArg : this; + if (partialArgs) { + var args = slice(partialArgs); + push.apply(args, arguments); + } + if (partialRightArgs || isCurry) { + args || (args = slice(arguments)); + if (partialRightArgs) { + push.apply(args, partialRightArgs); + } + if (isCurry && args.length < arity) { + bitmask |= 16 & ~32; + return baseCreateWrapper([func, (isCurryBound ? bitmask : bitmask & ~3), args, null, thisArg, arity]); + } + } + args || (args = arguments); + if (isBindKey) { + func = thisBinding[key]; + } + if (this instanceof bound) { + thisBinding = baseCreate(func.prototype); + var result = func.apply(thisBinding, args); + return isObject(result) ? result : thisBinding; + } + return func.apply(thisBinding, args); + } + setBindData(bound, bindData); + return bound; + } + + /** + * The base implementation of `_.difference` that accepts a single array + * of values to exclude. + * + * @private + * @param {Array} array The array to process. + * @param {Array} [values] The array of values to exclude. + * @returns {Array} Returns a new array of filtered values. + */ + function baseDifference(array, values) { + var index = -1, + indexOf = getIndexOf(), + length = array ? array.length : 0, + isLarge = length >= largeArraySize && indexOf === baseIndexOf, + result = []; + + if (isLarge) { + var cache = createCache(values); + if (cache) { + indexOf = cacheIndexOf; + values = cache; + } else { + isLarge = false; + } + } + while (++index < length) { + var value = array[index]; + if (indexOf(values, value) < 0) { + result.push(value); + } + } + if (isLarge) { + releaseObject(values); + } + return result; + } + + /** + * The base implementation of `_.flatten` without support for callback + * shorthands or `thisArg` binding. + * + * @private + * @param {Array} array The array to flatten. + * @param {boolean} [isShallow=false] A flag to restrict flattening to a single level. + * @param {boolean} [isStrict=false] A flag to restrict flattening to arrays and `arguments` objects. + * @param {number} [fromIndex=0] The index to start from. + * @returns {Array} Returns a new flattened array. + */ + function baseFlatten(array, isShallow, isStrict, fromIndex) { + var index = (fromIndex || 0) - 1, + length = array ? array.length : 0, + result = []; + + while (++index < length) { + var value = array[index]; + + if (value && typeof value == 'object' && typeof value.length == 'number' + && (isArray(value) || isArguments(value))) { + // recursively flatten arrays (susceptible to call stack limits) + if (!isShallow) { + value = baseFlatten(value, isShallow, isStrict); + } + var valIndex = -1, + valLength = value.length, + resIndex = result.length; + + result.length += valLength; + while (++valIndex < valLength) { + result[resIndex++] = value[valIndex]; + } + } else if (!isStrict) { + result.push(value); + } + } + return result; + } + + /** + * The base implementation of `_.isEqual`, without support for `thisArg` binding, + * that allows partial "_.where" style comparisons. + * + * @private + * @param {*} a The value to compare. + * @param {*} b The other value to compare. + * @param {Function} [callback] The function to customize comparing values. + * @param {Function} [isWhere=false] A flag to indicate performing partial comparisons. + * @param {Array} [stackA=[]] Tracks traversed `a` objects. + * @param {Array} [stackB=[]] Tracks traversed `b` objects. + * @returns {boolean} Returns `true` if the values are equivalent, else `false`. + */ + function baseIsEqual(a, b, callback, isWhere, stackA, stackB) { + // used to indicate that when comparing objects, `a` has at least the properties of `b` + if (callback) { + var result = callback(a, b); + if (typeof result != 'undefined') { + return !!result; + } + } + // exit early for identical values + if (a === b) { + // treat `+0` vs. `-0` as not equal + return a !== 0 || (1 / a == 1 / b); + } + var type = typeof a, + otherType = typeof b; + + // exit early for unlike primitive values + if (a === a && + !(a && objectTypes[type]) && + !(b && objectTypes[otherType])) { + return false; + } + // exit early for `null` and `undefined` avoiding ES3's Function#call behavior + // http://es5.github.io/#x15.3.4.4 + if (a == null || b == null) { + return a === b; + } + // compare [[Class]] names + var className = toString.call(a), + otherClass = toString.call(b); + + if (className == argsClass) { + className = objectClass; + } + if (otherClass == argsClass) { + otherClass = objectClass; + } + if (className != otherClass) { + return false; + } + switch (className) { + case boolClass: + case dateClass: + // coerce dates and booleans to numbers, dates to milliseconds and booleans + // to `1` or `0` treating invalid dates coerced to `NaN` as not equal + return +a == +b; + + case numberClass: + // treat `NaN` vs. `NaN` as equal + return (a != +a) + ? b != +b + // but treat `+0` vs. `-0` as not equal + : (a == 0 ? (1 / a == 1 / b) : a == +b); + + case regexpClass: + case stringClass: + // coerce regexes to strings (http://es5.github.io/#x15.10.6.4) + // treat string primitives and their corresponding object instances as equal + return a == String(b); + } + var isArr = className == arrayClass; + if (!isArr) { + // unwrap any `lodash` wrapped values + var aWrapped = hasOwnProperty.call(a, '__wrapped__'), + bWrapped = hasOwnProperty.call(b, '__wrapped__'); + + if (aWrapped || bWrapped) { + return baseIsEqual(aWrapped ? a.__wrapped__ : a, bWrapped ? b.__wrapped__ : b, callback, isWhere, stackA, stackB); + } + // exit for functions and DOM nodes + if (className != objectClass) { + return false; + } + // in older versions of Opera, `arguments` objects have `Array` constructors + var ctorA = a.constructor, + ctorB = b.constructor; + + // non `Object` object instances with different constructors are not equal + if (ctorA != ctorB && + !(isFunction(ctorA) && ctorA instanceof ctorA && isFunction(ctorB) && ctorB instanceof ctorB) && + ('constructor' in a && 'constructor' in b) + ) { + return false; + } + } + // assume cyclic structures are equal + // the algorithm for detecting cyclic structures is adapted from ES 5.1 + // section 15.12.3, abstract operation `JO` (http://es5.github.io/#x15.12.3) + var initedStack = !stackA; + stackA || (stackA = getArray()); + stackB || (stackB = getArray()); + + var length = stackA.length; + while (length--) { + if (stackA[length] == a) { + return stackB[length] == b; + } + } + var size = 0; + result = true; + + // add `a` and `b` to the stack of traversed objects + stackA.push(a); + stackB.push(b); + + // recursively compare objects and arrays (susceptible to call stack limits) + if (isArr) { + // compare lengths to determine if a deep comparison is necessary + length = a.length; + size = b.length; + result = size == length; + + if (result || isWhere) { + // deep compare the contents, ignoring non-numeric properties + while (size--) { + var index = length, + value = b[size]; + + if (isWhere) { + while (index--) { + if ((result = baseIsEqual(a[index], value, callback, isWhere, stackA, stackB))) { + break; + } + } + } else if (!(result = baseIsEqual(a[size], value, callback, isWhere, stackA, stackB))) { + break; + } + } + } + } + else { + // deep compare objects using `forIn`, instead of `forOwn`, to avoid `Object.keys` + // which, in this case, is more costly + forIn(b, function(value, key, b) { + if (hasOwnProperty.call(b, key)) { + // count the number of properties. + size++; + // deep compare each property value. + return (result = hasOwnProperty.call(a, key) && baseIsEqual(a[key], value, callback, isWhere, stackA, stackB)); + } + }); + + if (result && !isWhere) { + // ensure both objects have the same number of properties + forIn(a, function(value, key, a) { + if (hasOwnProperty.call(a, key)) { + // `size` will be `-1` if `a` has more properties than `b` + return (result = --size > -1); + } + }); + } + } + stackA.pop(); + stackB.pop(); + + if (initedStack) { + releaseArray(stackA); + releaseArray(stackB); + } + return result; + } + + /** + * The base implementation of `_.merge` without argument juggling or support + * for `thisArg` binding. + * + * @private + * @param {Object} object The destination object. + * @param {Object} source The source object. + * @param {Function} [callback] The function to customize merging properties. + * @param {Array} [stackA=[]] Tracks traversed source objects. + * @param {Array} [stackB=[]] Associates values with source counterparts. + */ + function baseMerge(object, source, callback, stackA, stackB) { + (isArray(source) ? forEach : forOwn)(source, function(source, key) { + var found, + isArr, + result = source, + value = object[key]; + + if (source && ((isArr = isArray(source)) || isPlainObject(source))) { + // avoid merging previously merged cyclic sources + var stackLength = stackA.length; + while (stackLength--) { + if ((found = stackA[stackLength] == source)) { + value = stackB[stackLength]; + break; + } + } + if (!found) { + var isShallow; + if (callback) { + result = callback(value, source); + if ((isShallow = typeof result != 'undefined')) { + value = result; + } + } + if (!isShallow) { + value = isArr + ? (isArray(value) ? value : []) + : (isPlainObject(value) ? value : {}); + } + // add `source` and associated `value` to the stack of traversed objects + stackA.push(source); + stackB.push(value); + + // recursively merge objects and arrays (susceptible to call stack limits) + if (!isShallow) { + baseMerge(value, source, callback, stackA, stackB); + } + } + } + else { + if (callback) { + result = callback(value, source); + if (typeof result == 'undefined') { + result = source; + } + } + if (typeof result != 'undefined') { + value = result; + } + } + object[key] = value; + }); + } + + /** + * The base implementation of `_.random` without argument juggling or support + * for returning floating-point numbers. + * + * @private + * @param {number} min The minimum possible value. + * @param {number} max The maximum possible value. + * @returns {number} Returns a random number. + */ + function baseRandom(min, max) { + return min + floor(nativeRandom() * (max - min + 1)); + } + + /** + * The base implementation of `_.uniq` without support for callback shorthands + * or `thisArg` binding. + * + * @private + * @param {Array} array The array to process. + * @param {boolean} [isSorted=false] A flag to indicate that `array` is sorted. + * @param {Function} [callback] The function called per iteration. + * @returns {Array} Returns a duplicate-value-free array. + */ + function baseUniq(array, isSorted, callback) { + var index = -1, + indexOf = getIndexOf(), + length = array ? array.length : 0, + result = []; + + var isLarge = !isSorted && length >= largeArraySize && indexOf === baseIndexOf, + seen = (callback || isLarge) ? getArray() : result; + + if (isLarge) { + var cache = createCache(seen); + indexOf = cacheIndexOf; + seen = cache; + } + while (++index < length) { + var value = array[index], + computed = callback ? callback(value, index, array) : value; + + if (isSorted + ? !index || seen[seen.length - 1] !== computed + : indexOf(seen, computed) < 0 + ) { + if (callback || isLarge) { + seen.push(computed); + } + result.push(value); + } + } + if (isLarge) { + releaseArray(seen.array); + releaseObject(seen); + } else if (callback) { + releaseArray(seen); + } + return result; + } + + /** + * Creates a function that aggregates a collection, creating an object composed + * of keys generated from the results of running each element of the collection + * through a callback. The given `setter` function sets the keys and values + * of the composed object. + * + * @private + * @param {Function} setter The setter function. + * @returns {Function} Returns the new aggregator function. + */ + function createAggregator(setter) { + return function(collection, callback, thisArg) { + var result = {}; + callback = lodash.createCallback(callback, thisArg, 3); + + var index = -1, + length = collection ? collection.length : 0; + + if (typeof length == 'number') { + while (++index < length) { + var value = collection[index]; + setter(result, value, callback(value, index, collection), collection); + } + } else { + forOwn(collection, function(value, key, collection) { + setter(result, value, callback(value, key, collection), collection); + }); + } + return result; + }; + } + + /** + * Creates a function that, when called, either curries or invokes `func` + * with an optional `this` binding and partially applied arguments. + * + * @private + * @param {Function|string} func The function or method name to reference. + * @param {number} bitmask The bitmask of method flags to compose. + * The bitmask may be composed of the following flags: + * 1 - `_.bind` + * 2 - `_.bindKey` + * 4 - `_.curry` + * 8 - `_.curry` (bound) + * 16 - `_.partial` + * 32 - `_.partialRight` + * @param {Array} [partialArgs] An array of arguments to prepend to those + * provided to the new function. + * @param {Array} [partialRightArgs] An array of arguments to append to those + * provided to the new function. + * @param {*} [thisArg] The `this` binding of `func`. + * @param {number} [arity] The arity of `func`. + * @returns {Function} Returns the new function. + */ + function createWrapper(func, bitmask, partialArgs, partialRightArgs, thisArg, arity) { + var isBind = bitmask & 1, + isBindKey = bitmask & 2, + isCurry = bitmask & 4, + isCurryBound = bitmask & 8, + isPartial = bitmask & 16, + isPartialRight = bitmask & 32; + + if (!isBindKey && !isFunction(func)) { + throw new TypeError; + } + if (isPartial && !partialArgs.length) { + bitmask &= ~16; + isPartial = partialArgs = false; + } + if (isPartialRight && !partialRightArgs.length) { + bitmask &= ~32; + isPartialRight = partialRightArgs = false; + } + var bindData = func && func.__bindData__; + if (bindData && bindData !== true) { + // clone `bindData` + bindData = slice(bindData); + if (bindData[2]) { + bindData[2] = slice(bindData[2]); + } + if (bindData[3]) { + bindData[3] = slice(bindData[3]); + } + // set `thisBinding` is not previously bound + if (isBind && !(bindData[1] & 1)) { + bindData[4] = thisArg; + } + // set if previously bound but not currently (subsequent curried functions) + if (!isBind && bindData[1] & 1) { + bitmask |= 8; + } + // set curried arity if not yet set + if (isCurry && !(bindData[1] & 4)) { + bindData[5] = arity; + } + // append partial left arguments + if (isPartial) { + push.apply(bindData[2] || (bindData[2] = []), partialArgs); + } + // append partial right arguments + if (isPartialRight) { + unshift.apply(bindData[3] || (bindData[3] = []), partialRightArgs); + } + // merge flags + bindData[1] |= bitmask; + return createWrapper.apply(null, bindData); + } + // fast path for `_.bind` + var creater = (bitmask == 1 || bitmask === 17) ? baseBind : baseCreateWrapper; + return creater([func, bitmask, partialArgs, partialRightArgs, thisArg, arity]); + } + + /** + * Used by `escape` to convert characters to HTML entities. + * + * @private + * @param {string} match The matched character to escape. + * @returns {string} Returns the escaped character. + */ + function escapeHtmlChar(match) { + return htmlEscapes[match]; + } + + /** + * Gets the appropriate "indexOf" function. If the `_.indexOf` method is + * customized, this method returns the custom method, otherwise it returns + * the `baseIndexOf` function. + * + * @private + * @returns {Function} Returns the "indexOf" function. + */ + function getIndexOf() { + var result = (result = lodash.indexOf) === indexOf ? baseIndexOf : result; + return result; + } + + /** + * Checks if `value` is a native function. + * + * @private + * @param {*} value The value to check. + * @returns {boolean} Returns `true` if the `value` is a native function, else `false`. + */ + function isNative(value) { + return typeof value == 'function' && reNative.test(value); + } + + /** + * Sets `this` binding data on a given function. + * + * @private + * @param {Function} func The function to set data on. + * @param {Array} value The data array to set. + */ + var setBindData = !defineProperty ? noop : function(func, value) { + descriptor.value = value; + defineProperty(func, '__bindData__', descriptor); + }; + + /** + * A fallback implementation of `isPlainObject` which checks if a given value + * is an object created by the `Object` constructor, assuming objects created + * by the `Object` constructor have no inherited enumerable properties and that + * there are no `Object.prototype` extensions. + * + * @private + * @param {*} value The value to check. + * @returns {boolean} Returns `true` if `value` is a plain object, else `false`. + */ + function shimIsPlainObject(value) { + var ctor, + result; + + // avoid non Object objects, `arguments` objects, and DOM elements + if (!(value && toString.call(value) == objectClass) || + (ctor = value.constructor, isFunction(ctor) && !(ctor instanceof ctor))) { + return false; + } + // In most environments an object's own properties are iterated before + // its inherited properties. If the last iterated property is an object's + // own property then there are no inherited enumerable properties. + forIn(value, function(value, key) { + result = key; + }); + return typeof result == 'undefined' || hasOwnProperty.call(value, result); + } + + /** + * Used by `unescape` to convert HTML entities to characters. + * + * @private + * @param {string} match The matched character to unescape. + * @returns {string} Returns the unescaped character. + */ + function unescapeHtmlChar(match) { + return htmlUnescapes[match]; + } + + /*--------------------------------------------------------------------------*/ + + /** + * Checks if `value` is an `arguments` object. + * + * @static + * @memberOf _ + * @category Objects + * @param {*} value The value to check. + * @returns {boolean} Returns `true` if the `value` is an `arguments` object, else `false`. + * @example + * + * (function() { return _.isArguments(arguments); })(1, 2, 3); + * // => true + * + * _.isArguments([1, 2, 3]); + * // => false + */ + function isArguments(value) { + return value && typeof value == 'object' && typeof value.length == 'number' && + toString.call(value) == argsClass || false; + } + + /** + * Checks if `value` is an array. + * + * @static + * @memberOf _ + * @type Function + * @category Objects + * @param {*} value The value to check. + * @returns {boolean} Returns `true` if the `value` is an array, else `false`. + * @example + * + * (function() { return _.isArray(arguments); })(); + * // => false + * + * _.isArray([1, 2, 3]); + * // => true + */ + var isArray = nativeIsArray || function(value) { + return value && typeof value == 'object' && typeof value.length == 'number' && + toString.call(value) == arrayClass || false; + }; + + /** + * A fallback implementation of `Object.keys` which produces an array of the + * given object's own enumerable property names. + * + * @private + * @type Function + * @param {Object} object The object to inspect. + * @returns {Array} Returns an array of property names. + */ + var shimKeys = function(object) { + var index, iterable = object, result = []; + if (!iterable) return result; + if (!(objectTypes[typeof object])) return result; + for (index in iterable) { + if (hasOwnProperty.call(iterable, index)) { + result.push(index); + } + } + return result + }; + + /** + * Creates an array composed of the own enumerable property names of an object. + * + * @static + * @memberOf _ + * @category Objects + * @param {Object} object The object to inspect. + * @returns {Array} Returns an array of property names. + * @example + * + * _.keys({ 'one': 1, 'two': 2, 'three': 3 }); + * // => ['one', 'two', 'three'] (property order is not guaranteed across environments) + */ + var keys = !nativeKeys ? shimKeys : function(object) { + if (!isObject(object)) { + return []; + } + return nativeKeys(object); + }; + + /** + * Used to convert characters to HTML entities: + * + * Though the `>` character is escaped for symmetry, characters like `>` and `/` + * don't require escaping in HTML and have no special meaning unless they're part + * of a tag or an unquoted attribute value. + * http://mathiasbynens.be/notes/ambiguous-ampersands (under "semi-related fun fact") + */ + var htmlEscapes = { + '&': '&', + '<': '<', + '>': '>', + '"': '"', + "'": ''' + }; + + /** Used to convert HTML entities to characters */ + var htmlUnescapes = invert(htmlEscapes); + + /** Used to match HTML entities and HTML characters */ + var reEscapedHtml = RegExp('(' + keys(htmlUnescapes).join('|') + ')', 'g'), + reUnescapedHtml = RegExp('[' + keys(htmlEscapes).join('') + ']', 'g'); + + /*--------------------------------------------------------------------------*/ + + /** + * Assigns own enumerable properties of source object(s) to the destination + * object. Subsequent sources will overwrite property assignments of previous + * sources. If a callback is provided it will be executed to produce the + * assigned values. The callback is bound to `thisArg` and invoked with two + * arguments; (objectValue, sourceValue). + * + * @static + * @memberOf _ + * @type Function + * @alias extend + * @category Objects + * @param {Object} object The destination object. + * @param {...Object} [source] The source objects. + * @param {Function} [callback] The function to customize assigning values. + * @param {*} [thisArg] The `this` binding of `callback`. + * @returns {Object} Returns the destination object. + * @example + * + * _.assign({ 'name': 'fred' }, { 'employer': 'slate' }); + * // => { 'name': 'fred', 'employer': 'slate' } + * + * var defaults = _.partialRight(_.assign, function(a, b) { + * return typeof a == 'undefined' ? b : a; + * }); + * + * var object = { 'name': 'barney' }; + * defaults(object, { 'name': 'fred', 'employer': 'slate' }); + * // => { 'name': 'barney', 'employer': 'slate' } + */ + var assign = function(object, source, guard) { + var index, iterable = object, result = iterable; + if (!iterable) return result; + var args = arguments, + argsIndex = 0, + argsLength = typeof guard == 'number' ? 2 : args.length; + if (argsLength > 3 && typeof args[argsLength - 2] == 'function') { + var callback = baseCreateCallback(args[--argsLength - 1], args[argsLength--], 2); + } else if (argsLength > 2 && typeof args[argsLength - 1] == 'function') { + callback = args[--argsLength]; + } + while (++argsIndex < argsLength) { + iterable = args[argsIndex]; + if (iterable && objectTypes[typeof iterable]) { + var ownIndex = -1, + ownProps = objectTypes[typeof iterable] && keys(iterable), + length = ownProps ? ownProps.length : 0; + + while (++ownIndex < length) { + index = ownProps[ownIndex]; + result[index] = callback ? callback(result[index], iterable[index]) : iterable[index]; + } + } + } + return result + }; + + /** + * Creates a clone of `value`. If `isDeep` is `true` nested objects will also + * be cloned, otherwise they will be assigned by reference. If a callback + * is provided it will be executed to produce the cloned values. If the + * callback returns `undefined` cloning will be handled by the method instead. + * The callback is bound to `thisArg` and invoked with one argument; (value). + * + * @static + * @memberOf _ + * @category Objects + * @param {*} value The value to clone. + * @param {boolean} [isDeep=false] Specify a deep clone. + * @param {Function} [callback] The function to customize cloning values. + * @param {*} [thisArg] The `this` binding of `callback`. + * @returns {*} Returns the cloned value. + * @example + * + * var characters = [ + * { 'name': 'barney', 'age': 36 }, + * { 'name': 'fred', 'age': 40 } + * ]; + * + * var shallow = _.clone(characters); + * shallow[0] === characters[0]; + * // => true + * + * var deep = _.clone(characters, true); + * deep[0] === characters[0]; + * // => false + * + * _.mixin({ + * 'clone': _.partialRight(_.clone, function(value) { + * return _.isElement(value) ? value.cloneNode(false) : undefined; + * }) + * }); + * + * var clone = _.clone(document.body); + * clone.childNodes.length; + * // => 0 + */ + function clone(value, isDeep, callback, thisArg) { + // allows working with "Collections" methods without using their `index` + // and `collection` arguments for `isDeep` and `callback` + if (typeof isDeep != 'boolean' && isDeep != null) { + thisArg = callback; + callback = isDeep; + isDeep = false; + } + return baseClone(value, isDeep, typeof callback == 'function' && baseCreateCallback(callback, thisArg, 1)); + } + + /** + * Creates a deep clone of `value`. If a callback is provided it will be + * executed to produce the cloned values. If the callback returns `undefined` + * cloning will be handled by the method instead. The callback is bound to + * `thisArg` and invoked with one argument; (value). + * + * Note: This method is loosely based on the structured clone algorithm. Functions + * and DOM nodes are **not** cloned. The enumerable properties of `arguments` objects and + * objects created by constructors other than `Object` are cloned to plain `Object` objects. + * See http://www.w3.org/TR/html5/infrastructure.html#internal-structured-cloning-algorithm. + * + * @static + * @memberOf _ + * @category Objects + * @param {*} value The value to deep clone. + * @param {Function} [callback] The function to customize cloning values. + * @param {*} [thisArg] The `this` binding of `callback`. + * @returns {*} Returns the deep cloned value. + * @example + * + * var characters = [ + * { 'name': 'barney', 'age': 36 }, + * { 'name': 'fred', 'age': 40 } + * ]; + * + * var deep = _.cloneDeep(characters); + * deep[0] === characters[0]; + * // => false + * + * var view = { + * 'label': 'docs', + * 'node': element + * }; + * + * var clone = _.cloneDeep(view, function(value) { + * return _.isElement(value) ? value.cloneNode(true) : undefined; + * }); + * + * clone.node == view.node; + * // => false + */ + function cloneDeep(value, callback, thisArg) { + return baseClone(value, true, typeof callback == 'function' && baseCreateCallback(callback, thisArg, 1)); + } + + /** + * Creates an object that inherits from the given `prototype` object. If a + * `properties` object is provided its own enumerable properties are assigned + * to the created object. + * + * @static + * @memberOf _ + * @category Objects + * @param {Object} prototype The object to inherit from. + * @param {Object} [properties] The properties to assign to the object. + * @returns {Object} Returns the new object. + * @example + * + * function Shape() { + * this.x = 0; + * this.y = 0; + * } + * + * function Circle() { + * Shape.call(this); + * } + * + * Circle.prototype = _.create(Shape.prototype, { 'constructor': Circle }); + * + * var circle = new Circle; + * circle instanceof Circle; + * // => true + * + * circle instanceof Shape; + * // => true + */ + function create(prototype, properties) { + var result = baseCreate(prototype); + return properties ? assign(result, properties) : result; + } + + /** + * Assigns own enumerable properties of source object(s) to the destination + * object for all destination properties that resolve to `undefined`. Once a + * property is set, additional defaults of the same property will be ignored. + * + * @static + * @memberOf _ + * @type Function + * @category Objects + * @param {Object} object The destination object. + * @param {...Object} [source] The source objects. + * @param- {Object} [guard] Allows working with `_.reduce` without using its + * `key` and `object` arguments as sources. + * @returns {Object} Returns the destination object. + * @example + * + * var object = { 'name': 'barney' }; + * _.defaults(object, { 'name': 'fred', 'employer': 'slate' }); + * // => { 'name': 'barney', 'employer': 'slate' } + */ + var defaults = function(object, source, guard) { + var index, iterable = object, result = iterable; + if (!iterable) return result; + var args = arguments, + argsIndex = 0, + argsLength = typeof guard == 'number' ? 2 : args.length; + while (++argsIndex < argsLength) { + iterable = args[argsIndex]; + if (iterable && objectTypes[typeof iterable]) { + var ownIndex = -1, + ownProps = objectTypes[typeof iterable] && keys(iterable), + length = ownProps ? ownProps.length : 0; + + while (++ownIndex < length) { + index = ownProps[ownIndex]; + if (typeof result[index] == 'undefined') result[index] = iterable[index]; + } + } + } + return result + }; + + /** + * This method is like `_.findIndex` except that it returns the key of the + * first element that passes the callback check, instead of the element itself. + * + * If a property name is provided for `callback` the created "_.pluck" style + * callback will return the property value of the given element. + * + * If an object is provided for `callback` the created "_.where" style callback + * will return `true` for elements that have the properties of the given object, + * else `false`. + * + * @static + * @memberOf _ + * @category Objects + * @param {Object} object The object to search. + * @param {Function|Object|string} [callback=identity] The function called per + * iteration. If a property name or object is provided it will be used to + * create a "_.pluck" or "_.where" style callback, respectively. + * @param {*} [thisArg] The `this` binding of `callback`. + * @returns {string|undefined} Returns the key of the found element, else `undefined`. + * @example + * + * var characters = { + * 'barney': { 'age': 36, 'blocked': false }, + * 'fred': { 'age': 40, 'blocked': true }, + * 'pebbles': { 'age': 1, 'blocked': false } + * }; + * + * _.findKey(characters, function(chr) { + * return chr.age < 40; + * }); + * // => 'barney' (property order is not guaranteed across environments) + * + * // using "_.where" callback shorthand + * _.findKey(characters, { 'age': 1 }); + * // => 'pebbles' + * + * // using "_.pluck" callback shorthand + * _.findKey(characters, 'blocked'); + * // => 'fred' + */ + function findKey(object, callback, thisArg) { + var result; + callback = lodash.createCallback(callback, thisArg, 3); + forOwn(object, function(value, key, object) { + if (callback(value, key, object)) { + result = key; + return false; + } + }); + return result; + } + + /** + * This method is like `_.findKey` except that it iterates over elements + * of a `collection` in the opposite order. + * + * If a property name is provided for `callback` the created "_.pluck" style + * callback will return the property value of the given element. + * + * If an object is provided for `callback` the created "_.where" style callback + * will return `true` for elements that have the properties of the given object, + * else `false`. + * + * @static + * @memberOf _ + * @category Objects + * @param {Object} object The object to search. + * @param {Function|Object|string} [callback=identity] The function called per + * iteration. If a property name or object is provided it will be used to + * create a "_.pluck" or "_.where" style callback, respectively. + * @param {*} [thisArg] The `this` binding of `callback`. + * @returns {string|undefined} Returns the key of the found element, else `undefined`. + * @example + * + * var characters = { + * 'barney': { 'age': 36, 'blocked': true }, + * 'fred': { 'age': 40, 'blocked': false }, + * 'pebbles': { 'age': 1, 'blocked': true } + * }; + * + * _.findLastKey(characters, function(chr) { + * return chr.age < 40; + * }); + * // => returns `pebbles`, assuming `_.findKey` returns `barney` + * + * // using "_.where" callback shorthand + * _.findLastKey(characters, { 'age': 40 }); + * // => 'fred' + * + * // using "_.pluck" callback shorthand + * _.findLastKey(characters, 'blocked'); + * // => 'pebbles' + */ + function findLastKey(object, callback, thisArg) { + var result; + callback = lodash.createCallback(callback, thisArg, 3); + forOwnRight(object, function(value, key, object) { + if (callback(value, key, object)) { + result = key; + return false; + } + }); + return result; + } + + /** + * Iterates over own and inherited enumerable properties of an object, + * executing the callback for each property. The callback is bound to `thisArg` + * and invoked with three arguments; (value, key, object). Callbacks may exit + * iteration early by explicitly returning `false`. + * + * @static + * @memberOf _ + * @type Function + * @category Objects + * @param {Object} object The object to iterate over. + * @param {Function} [callback=identity] The function called per iteration. + * @param {*} [thisArg] The `this` binding of `callback`. + * @returns {Object} Returns `object`. + * @example + * + * function Shape() { + * this.x = 0; + * this.y = 0; + * } + * + * Shape.prototype.move = function(x, y) { + * this.x += x; + * this.y += y; + * }; + * + * _.forIn(new Shape, function(value, key) { + * console.log(key); + * }); + * // => logs 'x', 'y', and 'move' (property order is not guaranteed across environments) + */ + var forIn = function(collection, callback, thisArg) { + var index, iterable = collection, result = iterable; + if (!iterable) return result; + if (!objectTypes[typeof iterable]) return result; + callback = callback && typeof thisArg == 'undefined' ? callback : baseCreateCallback(callback, thisArg, 3); + for (index in iterable) { + if (callback(iterable[index], index, collection) === false) return result; + } + return result + }; + + /** + * This method is like `_.forIn` except that it iterates over elements + * of a `collection` in the opposite order. + * + * @static + * @memberOf _ + * @category Objects + * @param {Object} object The object to iterate over. + * @param {Function} [callback=identity] The function called per iteration. + * @param {*} [thisArg] The `this` binding of `callback`. + * @returns {Object} Returns `object`. + * @example + * + * function Shape() { + * this.x = 0; + * this.y = 0; + * } + * + * Shape.prototype.move = function(x, y) { + * this.x += x; + * this.y += y; + * }; + * + * _.forInRight(new Shape, function(value, key) { + * console.log(key); + * }); + * // => logs 'move', 'y', and 'x' assuming `_.forIn ` logs 'x', 'y', and 'move' + */ + function forInRight(object, callback, thisArg) { + var pairs = []; + + forIn(object, function(value, key) { + pairs.push(key, value); + }); + + var length = pairs.length; + callback = baseCreateCallback(callback, thisArg, 3); + while (length--) { + if (callback(pairs[length--], pairs[length], object) === false) { + break; + } + } + return object; + } + + /** + * Iterates over own enumerable properties of an object, executing the callback + * for each property. The callback is bound to `thisArg` and invoked with three + * arguments; (value, key, object). Callbacks may exit iteration early by + * explicitly returning `false`. + * + * @static + * @memberOf _ + * @type Function + * @category Objects + * @param {Object} object The object to iterate over. + * @param {Function} [callback=identity] The function called per iteration. + * @param {*} [thisArg] The `this` binding of `callback`. + * @returns {Object} Returns `object`. + * @example + * + * _.forOwn({ '0': 'zero', '1': 'one', 'length': 2 }, function(num, key) { + * console.log(key); + * }); + * // => logs '0', '1', and 'length' (property order is not guaranteed across environments) + */ + var forOwn = function(collection, callback, thisArg) { + var index, iterable = collection, result = iterable; + if (!iterable) return result; + if (!objectTypes[typeof iterable]) return result; + callback = callback && typeof thisArg == 'undefined' ? callback : baseCreateCallback(callback, thisArg, 3); + var ownIndex = -1, + ownProps = objectTypes[typeof iterable] && keys(iterable), + length = ownProps ? ownProps.length : 0; + + while (++ownIndex < length) { + index = ownProps[ownIndex]; + if (callback(iterable[index], index, collection) === false) return result; + } + return result + }; + + /** + * This method is like `_.forOwn` except that it iterates over elements + * of a `collection` in the opposite order. + * + * @static + * @memberOf _ + * @category Objects + * @param {Object} object The object to iterate over. + * @param {Function} [callback=identity] The function called per iteration. + * @param {*} [thisArg] The `this` binding of `callback`. + * @returns {Object} Returns `object`. + * @example + * + * _.forOwnRight({ '0': 'zero', '1': 'one', 'length': 2 }, function(num, key) { + * console.log(key); + * }); + * // => logs 'length', '1', and '0' assuming `_.forOwn` logs '0', '1', and 'length' + */ + function forOwnRight(object, callback, thisArg) { + var props = keys(object), + length = props.length; + + callback = baseCreateCallback(callback, thisArg, 3); + while (length--) { + var key = props[length]; + if (callback(object[key], key, object) === false) { + break; + } + } + return object; + } + + /** + * Creates a sorted array of property names of all enumerable properties, + * own and inherited, of `object` that have function values. + * + * @static + * @memberOf _ + * @alias methods + * @category Objects + * @param {Object} object The object to inspect. + * @returns {Array} Returns an array of property names that have function values. + * @example + * + * _.functions(_); + * // => ['all', 'any', 'bind', 'bindAll', 'clone', 'compact', 'compose', ...] + */ + function functions(object) { + var result = []; + forIn(object, function(value, key) { + if (isFunction(value)) { + result.push(key); + } + }); + return result.sort(); + } + + /** + * Checks if the specified property name exists as a direct property of `object`, + * instead of an inherited property. + * + * @static + * @memberOf _ + * @category Objects + * @param {Object} object The object to inspect. + * @param {string} key The name of the property to check. + * @returns {boolean} Returns `true` if key is a direct property, else `false`. + * @example + * + * _.has({ 'a': 1, 'b': 2, 'c': 3 }, 'b'); + * // => true + */ + function has(object, key) { + return object ? hasOwnProperty.call(object, key) : false; + } + + /** + * Creates an object composed of the inverted keys and values of the given object. + * + * @static + * @memberOf _ + * @category Objects + * @param {Object} object The object to invert. + * @returns {Object} Returns the created inverted object. + * @example + * + * _.invert({ 'first': 'fred', 'second': 'barney' }); + * // => { 'fred': 'first', 'barney': 'second' } + */ + function invert(object) { + var index = -1, + props = keys(object), + length = props.length, + result = {}; + + while (++index < length) { + var key = props[index]; + result[object[key]] = key; + } + return result; + } + + /** + * Checks if `value` is a boolean value. + * + * @static + * @memberOf _ + * @category Objects + * @param {*} value The value to check. + * @returns {boolean} Returns `true` if the `value` is a boolean value, else `false`. + * @example + * + * _.isBoolean(null); + * // => false + */ + function isBoolean(value) { + return value === true || value === false || + value && typeof value == 'object' && toString.call(value) == boolClass || false; + } + + /** + * Checks if `value` is a date. + * + * @static + * @memberOf _ + * @category Objects + * @param {*} value The value to check. + * @returns {boolean} Returns `true` if the `value` is a date, else `false`. + * @example + * + * _.isDate(new Date); + * // => true + */ + function isDate(value) { + return value && typeof value == 'object' && toString.call(value) == dateClass || false; + } + + /** + * Checks if `value` is a DOM element. + * + * @static + * @memberOf _ + * @category Objects + * @param {*} value The value to check. + * @returns {boolean} Returns `true` if the `value` is a DOM element, else `false`. + * @example + * + * _.isElement(document.body); + * // => true + */ + function isElement(value) { + return value && value.nodeType === 1 || false; + } + + /** + * Checks if `value` is empty. Arrays, strings, or `arguments` objects with a + * length of `0` and objects with no own enumerable properties are considered + * "empty". + * + * @static + * @memberOf _ + * @category Objects + * @param {Array|Object|string} value The value to inspect. + * @returns {boolean} Returns `true` if the `value` is empty, else `false`. + * @example + * + * _.isEmpty([1, 2, 3]); + * // => false + * + * _.isEmpty({}); + * // => true + * + * _.isEmpty(''); + * // => true + */ + function isEmpty(value) { + var result = true; + if (!value) { + return result; + } + var className = toString.call(value), + length = value.length; + + if ((className == arrayClass || className == stringClass || className == argsClass ) || + (className == objectClass && typeof length == 'number' && isFunction(value.splice))) { + return !length; + } + forOwn(value, function() { + return (result = false); + }); + return result; + } + + /** + * Performs a deep comparison between two values to determine if they are + * equivalent to each other. If a callback is provided it will be executed + * to compare values. If the callback returns `undefined` comparisons will + * be handled by the method instead. The callback is bound to `thisArg` and + * invoked with two arguments; (a, b). + * + * @static + * @memberOf _ + * @category Objects + * @param {*} a The value to compare. + * @param {*} b The other value to compare. + * @param {Function} [callback] The function to customize comparing values. + * @param {*} [thisArg] The `this` binding of `callback`. + * @returns {boolean} Returns `true` if the values are equivalent, else `false`. + * @example + * + * var object = { 'name': 'fred' }; + * var copy = { 'name': 'fred' }; + * + * object == copy; + * // => false + * + * _.isEqual(object, copy); + * // => true + * + * var words = ['hello', 'goodbye']; + * var otherWords = ['hi', 'goodbye']; + * + * _.isEqual(words, otherWords, function(a, b) { + * var reGreet = /^(?:hello|hi)$/i, + * aGreet = _.isString(a) && reGreet.test(a), + * bGreet = _.isString(b) && reGreet.test(b); + * + * return (aGreet || bGreet) ? (aGreet == bGreet) : undefined; + * }); + * // => true + */ + function isEqual(a, b, callback, thisArg) { + return baseIsEqual(a, b, typeof callback == 'function' && baseCreateCallback(callback, thisArg, 2)); + } + + /** + * Checks if `value` is, or can be coerced to, a finite number. + * + * Note: This is not the same as native `isFinite` which will return true for + * booleans and empty strings. See http://es5.github.io/#x15.1.2.5. + * + * @static + * @memberOf _ + * @category Objects + * @param {*} value The value to check. + * @returns {boolean} Returns `true` if the `value` is finite, else `false`. + * @example + * + * _.isFinite(-101); + * // => true + * + * _.isFinite('10'); + * // => true + * + * _.isFinite(true); + * // => false + * + * _.isFinite(''); + * // => false + * + * _.isFinite(Infinity); + * // => false + */ + function isFinite(value) { + return nativeIsFinite(value) && !nativeIsNaN(parseFloat(value)); + } + + /** + * Checks if `value` is a function. + * + * @static + * @memberOf _ + * @category Objects + * @param {*} value The value to check. + * @returns {boolean} Returns `true` if the `value` is a function, else `false`. + * @example + * + * _.isFunction(_); + * // => true + */ + function isFunction(value) { + return typeof value == 'function'; + } + + /** + * Checks if `value` is the language type of Object. + * (e.g. arrays, functions, objects, regexes, `new Number(0)`, and `new String('')`) + * + * @static + * @memberOf _ + * @category Objects + * @param {*} value The value to check. + * @returns {boolean} Returns `true` if the `value` is an object, else `false`. + * @example + * + * _.isObject({}); + * // => true + * + * _.isObject([1, 2, 3]); + * // => true + * + * _.isObject(1); + * // => false + */ + function isObject(value) { + // check if the value is the ECMAScript language type of Object + // http://es5.github.io/#x8 + // and avoid a V8 bug + // http://code.google.com/p/v8/issues/detail?id=2291 + return !!(value && objectTypes[typeof value]); + } + + /** + * Checks if `value` is `NaN`. + * + * Note: This is not the same as native `isNaN` which will return `true` for + * `undefined` and other non-numeric values. See http://es5.github.io/#x15.1.2.4. + * + * @static + * @memberOf _ + * @category Objects + * @param {*} value The value to check. + * @returns {boolean} Returns `true` if the `value` is `NaN`, else `false`. + * @example + * + * _.isNaN(NaN); + * // => true + * + * _.isNaN(new Number(NaN)); + * // => true + * + * isNaN(undefined); + * // => true + * + * _.isNaN(undefined); + * // => false + */ + function isNaN(value) { + // `NaN` as a primitive is the only value that is not equal to itself + // (perform the [[Class]] check first to avoid errors with some host objects in IE) + return isNumber(value) && value != +value; + } + + /** + * Checks if `value` is `null`. + * + * @static + * @memberOf _ + * @category Objects + * @param {*} value The value to check. + * @returns {boolean} Returns `true` if the `value` is `null`, else `false`. + * @example + * + * _.isNull(null); + * // => true + * + * _.isNull(undefined); + * // => false + */ + function isNull(value) { + return value === null; + } + + /** + * Checks if `value` is a number. + * + * Note: `NaN` is considered a number. See http://es5.github.io/#x8.5. + * + * @static + * @memberOf _ + * @category Objects + * @param {*} value The value to check. + * @returns {boolean} Returns `true` if the `value` is a number, else `false`. + * @example + * + * _.isNumber(8.4 * 5); + * // => true + */ + function isNumber(value) { + return typeof value == 'number' || + value && typeof value == 'object' && toString.call(value) == numberClass || false; + } + + /** + * Checks if `value` is an object created by the `Object` constructor. + * + * @static + * @memberOf _ + * @category Objects + * @param {*} value The value to check. + * @returns {boolean} Returns `true` if `value` is a plain object, else `false`. + * @example + * + * function Shape() { + * this.x = 0; + * this.y = 0; + * } + * + * _.isPlainObject(new Shape); + * // => false + * + * _.isPlainObject([1, 2, 3]); + * // => false + * + * _.isPlainObject({ 'x': 0, 'y': 0 }); + * // => true + */ + var isPlainObject = !getPrototypeOf ? shimIsPlainObject : function(value) { + if (!(value && toString.call(value) == objectClass)) { + return false; + } + var valueOf = value.valueOf, + objProto = isNative(valueOf) && (objProto = getPrototypeOf(valueOf)) && getPrototypeOf(objProto); + + return objProto + ? (value == objProto || getPrototypeOf(value) == objProto) + : shimIsPlainObject(value); + }; + + /** + * Checks if `value` is a regular expression. + * + * @static + * @memberOf _ + * @category Objects + * @param {*} value The value to check. + * @returns {boolean} Returns `true` if the `value` is a regular expression, else `false`. + * @example + * + * _.isRegExp(/fred/); + * // => true + */ + function isRegExp(value) { + return value && typeof value == 'object' && toString.call(value) == regexpClass || false; + } + + /** + * Checks if `value` is a string. + * + * @static + * @memberOf _ + * @category Objects + * @param {*} value The value to check. + * @returns {boolean} Returns `true` if the `value` is a string, else `false`. + * @example + * + * _.isString('fred'); + * // => true + */ + function isString(value) { + return typeof value == 'string' || + value && typeof value == 'object' && toString.call(value) == stringClass || false; + } + + /** + * Checks if `value` is `undefined`. + * + * @static + * @memberOf _ + * @category Objects + * @param {*} value The value to check. + * @returns {boolean} Returns `true` if the `value` is `undefined`, else `false`. + * @example + * + * _.isUndefined(void 0); + * // => true + */ + function isUndefined(value) { + return typeof value == 'undefined'; + } + + /** + * Creates an object with the same keys as `object` and values generated by + * running each own enumerable property of `object` through the callback. + * The callback is bound to `thisArg` and invoked with three arguments; + * (value, key, object). + * + * If a property name is provided for `callback` the created "_.pluck" style + * callback will return the property value of the given element. + * + * If an object is provided for `callback` the created "_.where" style callback + * will return `true` for elements that have the properties of the given object, + * else `false`. + * + * @static + * @memberOf _ + * @category Objects + * @param {Object} object The object to iterate over. + * @param {Function|Object|string} [callback=identity] The function called + * per iteration. If a property name or object is provided it will be used + * to create a "_.pluck" or "_.where" style callback, respectively. + * @param {*} [thisArg] The `this` binding of `callback`. + * @returns {Array} Returns a new object with values of the results of each `callback` execution. + * @example + * + * _.mapValues({ 'a': 1, 'b': 2, 'c': 3} , function(num) { return num * 3; }); + * // => { 'a': 3, 'b': 6, 'c': 9 } + * + * var characters = { + * 'fred': { 'name': 'fred', 'age': 40 }, + * 'pebbles': { 'name': 'pebbles', 'age': 1 } + * }; + * + * // using "_.pluck" callback shorthand + * _.mapValues(characters, 'age'); + * // => { 'fred': 40, 'pebbles': 1 } + */ + function mapValues(object, callback, thisArg) { + var result = {}; + callback = lodash.createCallback(callback, thisArg, 3); + + forOwn(object, function(value, key, object) { + result[key] = callback(value, key, object); + }); + return result; + } + + /** + * Recursively merges own enumerable properties of the source object(s), that + * don't resolve to `undefined` into the destination object. Subsequent sources + * will overwrite property assignments of previous sources. If a callback is + * provided it will be executed to produce the merged values of the destination + * and source properties. If the callback returns `undefined` merging will + * be handled by the method instead. The callback is bound to `thisArg` and + * invoked with two arguments; (objectValue, sourceValue). + * + * @static + * @memberOf _ + * @category Objects + * @param {Object} object The destination object. + * @param {...Object} [source] The source objects. + * @param {Function} [callback] The function to customize merging properties. + * @param {*} [thisArg] The `this` binding of `callback`. + * @returns {Object} Returns the destination object. + * @example + * + * var names = { + * 'characters': [ + * { 'name': 'barney' }, + * { 'name': 'fred' } + * ] + * }; + * + * var ages = { + * 'characters': [ + * { 'age': 36 }, + * { 'age': 40 } + * ] + * }; + * + * _.merge(names, ages); + * // => { 'characters': [{ 'name': 'barney', 'age': 36 }, { 'name': 'fred', 'age': 40 }] } + * + * var food = { + * 'fruits': ['apple'], + * 'vegetables': ['beet'] + * }; + * + * var otherFood = { + * 'fruits': ['banana'], + * 'vegetables': ['carrot'] + * }; + * + * _.merge(food, otherFood, function(a, b) { + * return _.isArray(a) ? a.concat(b) : undefined; + * }); + * // => { 'fruits': ['apple', 'banana'], 'vegetables': ['beet', 'carrot] } + */ + function merge(object) { + var args = arguments, + length = 2; + + if (!isObject(object)) { + return object; + } + // allows working with `_.reduce` and `_.reduceRight` without using + // their `index` and `collection` arguments + if (typeof args[2] != 'number') { + length = args.length; + } + if (length > 3 && typeof args[length - 2] == 'function') { + var callback = baseCreateCallback(args[--length - 1], args[length--], 2); + } else if (length > 2 && typeof args[length - 1] == 'function') { + callback = args[--length]; + } + var sources = slice(arguments, 1, length), + index = -1, + stackA = getArray(), + stackB = getArray(); + + while (++index < length) { + baseMerge(object, sources[index], callback, stackA, stackB); + } + releaseArray(stackA); + releaseArray(stackB); + return object; + } + + /** + * Creates a shallow clone of `object` excluding the specified properties. + * Property names may be specified as individual arguments or as arrays of + * property names. If a callback is provided it will be executed for each + * property of `object` omitting the properties the callback returns truey + * for. The callback is bound to `thisArg` and invoked with three arguments; + * (value, key, object). + * + * @static + * @memberOf _ + * @category Objects + * @param {Object} object The source object. + * @param {Function|...string|string[]} [callback] The properties to omit or the + * function called per iteration. + * @param {*} [thisArg] The `this` binding of `callback`. + * @returns {Object} Returns an object without the omitted properties. + * @example + * + * _.omit({ 'name': 'fred', 'age': 40 }, 'age'); + * // => { 'name': 'fred' } + * + * _.omit({ 'name': 'fred', 'age': 40 }, function(value) { + * return typeof value == 'number'; + * }); + * // => { 'name': 'fred' } + */ + function omit(object, callback, thisArg) { + var result = {}; + if (typeof callback != 'function') { + var props = []; + forIn(object, function(value, key) { + props.push(key); + }); + props = baseDifference(props, baseFlatten(arguments, true, false, 1)); + + var index = -1, + length = props.length; + + while (++index < length) { + var key = props[index]; + result[key] = object[key]; + } + } else { + callback = lodash.createCallback(callback, thisArg, 3); + forIn(object, function(value, key, object) { + if (!callback(value, key, object)) { + result[key] = value; + } + }); + } + return result; + } + + /** + * Creates a two dimensional array of an object's key-value pairs, + * i.e. `[[key1, value1], [key2, value2]]`. + * + * @static + * @memberOf _ + * @category Objects + * @param {Object} object The object to inspect. + * @returns {Array} Returns new array of key-value pairs. + * @example + * + * _.pairs({ 'barney': 36, 'fred': 40 }); + * // => [['barney', 36], ['fred', 40]] (property order is not guaranteed across environments) + */ + function pairs(object) { + var index = -1, + props = keys(object), + length = props.length, + result = Array(length); + + while (++index < length) { + var key = props[index]; + result[index] = [key, object[key]]; + } + return result; + } + + /** + * Creates a shallow clone of `object` composed of the specified properties. + * Property names may be specified as individual arguments or as arrays of + * property names. If a callback is provided it will be executed for each + * property of `object` picking the properties the callback returns truey + * for. The callback is bound to `thisArg` and invoked with three arguments; + * (value, key, object). + * + * @static + * @memberOf _ + * @category Objects + * @param {Object} object The source object. + * @param {Function|...string|string[]} [callback] The function called per + * iteration or property names to pick, specified as individual property + * names or arrays of property names. + * @param {*} [thisArg] The `this` binding of `callback`. + * @returns {Object} Returns an object composed of the picked properties. + * @example + * + * _.pick({ 'name': 'fred', '_userid': 'fred1' }, 'name'); + * // => { 'name': 'fred' } + * + * _.pick({ 'name': 'fred', '_userid': 'fred1' }, function(value, key) { + * return key.charAt(0) != '_'; + * }); + * // => { 'name': 'fred' } + */ + function pick(object, callback, thisArg) { + var result = {}; + if (typeof callback != 'function') { + var index = -1, + props = baseFlatten(arguments, true, false, 1), + length = isObject(object) ? props.length : 0; + + while (++index < length) { + var key = props[index]; + if (key in object) { + result[key] = object[key]; + } + } + } else { + callback = lodash.createCallback(callback, thisArg, 3); + forIn(object, function(value, key, object) { + if (callback(value, key, object)) { + result[key] = value; + } + }); + } + return result; + } + + /** + * An alternative to `_.reduce` this method transforms `object` to a new + * `accumulator` object which is the result of running each of its own + * enumerable properties through a callback, with each callback execution + * potentially mutating the `accumulator` object. The callback is bound to + * `thisArg` and invoked with four arguments; (accumulator, value, key, object). + * Callbacks may exit iteration early by explicitly returning `false`. + * + * @static + * @memberOf _ + * @category Objects + * @param {Array|Object} object The object to iterate over. + * @param {Function} [callback=identity] The function called per iteration. + * @param {*} [accumulator] The custom accumulator value. + * @param {*} [thisArg] The `this` binding of `callback`. + * @returns {*} Returns the accumulated value. + * @example + * + * var squares = _.transform([1, 2, 3, 4, 5, 6, 7, 8, 9, 10], function(result, num) { + * num *= num; + * if (num % 2) { + * return result.push(num) < 3; + * } + * }); + * // => [1, 9, 25] + * + * var mapped = _.transform({ 'a': 1, 'b': 2, 'c': 3 }, function(result, num, key) { + * result[key] = num * 3; + * }); + * // => { 'a': 3, 'b': 6, 'c': 9 } + */ + function transform(object, callback, accumulator, thisArg) { + var isArr = isArray(object); + if (accumulator == null) { + if (isArr) { + accumulator = []; + } else { + var ctor = object && object.constructor, + proto = ctor && ctor.prototype; + + accumulator = baseCreate(proto); + } + } + if (callback) { + callback = lodash.createCallback(callback, thisArg, 4); + (isArr ? forEach : forOwn)(object, function(value, index, object) { + return callback(accumulator, value, index, object); + }); + } + return accumulator; + } + + /** + * Creates an array composed of the own enumerable property values of `object`. + * + * @static + * @memberOf _ + * @category Objects + * @param {Object} object The object to inspect. + * @returns {Array} Returns an array of property values. + * @example + * + * _.values({ 'one': 1, 'two': 2, 'three': 3 }); + * // => [1, 2, 3] (property order is not guaranteed across environments) + */ + function values(object) { + var index = -1, + props = keys(object), + length = props.length, + result = Array(length); + + while (++index < length) { + result[index] = object[props[index]]; + } + return result; + } + + /*--------------------------------------------------------------------------*/ + + /** + * Creates an array of elements from the specified indexes, or keys, of the + * `collection`. Indexes may be specified as individual arguments or as arrays + * of indexes. + * + * @static + * @memberOf _ + * @category Collections + * @param {Array|Object|string} collection The collection to iterate over. + * @param {...(number|number[]|string|string[])} [index] The indexes of `collection` + * to retrieve, specified as individual indexes or arrays of indexes. + * @returns {Array} Returns a new array of elements corresponding to the + * provided indexes. + * @example + * + * _.at(['a', 'b', 'c', 'd', 'e'], [0, 2, 4]); + * // => ['a', 'c', 'e'] + * + * _.at(['fred', 'barney', 'pebbles'], 0, 2); + * // => ['fred', 'pebbles'] + */ + function at(collection) { + var args = arguments, + index = -1, + props = baseFlatten(args, true, false, 1), + length = (args[2] && args[2][args[1]] === collection) ? 1 : props.length, + result = Array(length); + + while(++index < length) { + result[index] = collection[props[index]]; + } + return result; + } + + /** + * Checks if a given value is present in a collection using strict equality + * for comparisons, i.e. `===`. If `fromIndex` is negative, it is used as the + * offset from the end of the collection. + * + * @static + * @memberOf _ + * @alias include + * @category Collections + * @param {Array|Object|string} collection The collection to iterate over. + * @param {*} target The value to check for. + * @param {number} [fromIndex=0] The index to search from. + * @returns {boolean} Returns `true` if the `target` element is found, else `false`. + * @example + * + * _.contains([1, 2, 3], 1); + * // => true + * + * _.contains([1, 2, 3], 1, 2); + * // => false + * + * _.contains({ 'name': 'fred', 'age': 40 }, 'fred'); + * // => true + * + * _.contains('pebbles', 'eb'); + * // => true + */ + function contains(collection, target, fromIndex) { + var index = -1, + indexOf = getIndexOf(), + length = collection ? collection.length : 0, + result = false; + + fromIndex = (fromIndex < 0 ? nativeMax(0, length + fromIndex) : fromIndex) || 0; + if (isArray(collection)) { + result = indexOf(collection, target, fromIndex) > -1; + } else if (typeof length == 'number') { + result = (isString(collection) ? collection.indexOf(target, fromIndex) : indexOf(collection, target, fromIndex)) > -1; + } else { + forOwn(collection, function(value) { + if (++index >= fromIndex) { + return !(result = value === target); + } + }); + } + return result; + } + + /** + * Creates an object composed of keys generated from the results of running + * each element of `collection` through the callback. The corresponding value + * of each key is the number of times the key was returned by the callback. + * The callback is bound to `thisArg` and invoked with three arguments; + * (value, index|key, collection). + * + * If a property name is provided for `callback` the created "_.pluck" style + * callback will return the property value of the given element. + * + * If an object is provided for `callback` the created "_.where" style callback + * will return `true` for elements that have the properties of the given object, + * else `false`. + * + * @static + * @memberOf _ + * @category Collections + * @param {Array|Object|string} collection The collection to iterate over. + * @param {Function|Object|string} [callback=identity] The function called + * per iteration. If a property name or object is provided it will be used + * to create a "_.pluck" or "_.where" style callback, respectively. + * @param {*} [thisArg] The `this` binding of `callback`. + * @returns {Object} Returns the composed aggregate object. + * @example + * + * _.countBy([4.3, 6.1, 6.4], function(num) { return Math.floor(num); }); + * // => { '4': 1, '6': 2 } + * + * _.countBy([4.3, 6.1, 6.4], function(num) { return this.floor(num); }, Math); + * // => { '4': 1, '6': 2 } + * + * _.countBy(['one', 'two', 'three'], 'length'); + * // => { '3': 2, '5': 1 } + */ + var countBy = createAggregator(function(result, value, key) { + (hasOwnProperty.call(result, key) ? result[key]++ : result[key] = 1); + }); + + /** + * Checks if the given callback returns truey value for **all** elements of + * a collection. The callback is bound to `thisArg` and invoked with three + * arguments; (value, index|key, collection). + * + * If a property name is provided for `callback` the created "_.pluck" style + * callback will return the property value of the given element. + * + * If an object is provided for `callback` the created "_.where" style callback + * will return `true` for elements that have the properties of the given object, + * else `false`. + * + * @static + * @memberOf _ + * @alias all + * @category Collections + * @param {Array|Object|string} collection The collection to iterate over. + * @param {Function|Object|string} [callback=identity] The function called + * per iteration. If a property name or object is provided it will be used + * to create a "_.pluck" or "_.where" style callback, respectively. + * @param {*} [thisArg] The `this` binding of `callback`. + * @returns {boolean} Returns `true` if all elements passed the callback check, + * else `false`. + * @example + * + * _.every([true, 1, null, 'yes']); + * // => false + * + * var characters = [ + * { 'name': 'barney', 'age': 36 }, + * { 'name': 'fred', 'age': 40 } + * ]; + * + * // using "_.pluck" callback shorthand + * _.every(characters, 'age'); + * // => true + * + * // using "_.where" callback shorthand + * _.every(characters, { 'age': 36 }); + * // => false + */ + function every(collection, callback, thisArg) { + var result = true; + callback = lodash.createCallback(callback, thisArg, 3); + + var index = -1, + length = collection ? collection.length : 0; + + if (typeof length == 'number') { + while (++index < length) { + if (!(result = !!callback(collection[index], index, collection))) { + break; + } + } + } else { + forOwn(collection, function(value, index, collection) { + return (result = !!callback(value, index, collection)); + }); + } + return result; + } + + /** + * Iterates over elements of a collection, returning an array of all elements + * the callback returns truey for. The callback is bound to `thisArg` and + * invoked with three arguments; (value, index|key, collection). + * + * If a property name is provided for `callback` the created "_.pluck" style + * callback will return the property value of the given element. + * + * If an object is provided for `callback` the created "_.where" style callback + * will return `true` for elements that have the properties of the given object, + * else `false`. + * + * @static + * @memberOf _ + * @alias select + * @category Collections + * @param {Array|Object|string} collection The collection to iterate over. + * @param {Function|Object|string} [callback=identity] The function called + * per iteration. If a property name or object is provided it will be used + * to create a "_.pluck" or "_.where" style callback, respectively. + * @param {*} [thisArg] The `this` binding of `callback`. + * @returns {Array} Returns a new array of elements that passed the callback check. + * @example + * + * var evens = _.filter([1, 2, 3, 4, 5, 6], function(num) { return num % 2 == 0; }); + * // => [2, 4, 6] + * + * var characters = [ + * { 'name': 'barney', 'age': 36, 'blocked': false }, + * { 'name': 'fred', 'age': 40, 'blocked': true } + * ]; + * + * // using "_.pluck" callback shorthand + * _.filter(characters, 'blocked'); + * // => [{ 'name': 'fred', 'age': 40, 'blocked': true }] + * + * // using "_.where" callback shorthand + * _.filter(characters, { 'age': 36 }); + * // => [{ 'name': 'barney', 'age': 36, 'blocked': false }] + */ + function filter(collection, callback, thisArg) { + var result = []; + callback = lodash.createCallback(callback, thisArg, 3); + + var index = -1, + length = collection ? collection.length : 0; + + if (typeof length == 'number') { + while (++index < length) { + var value = collection[index]; + if (callback(value, index, collection)) { + result.push(value); + } + } + } else { + forOwn(collection, function(value, index, collection) { + if (callback(value, index, collection)) { + result.push(value); + } + }); + } + return result; + } + + /** + * Iterates over elements of a collection, returning the first element that + * the callback returns truey for. The callback is bound to `thisArg` and + * invoked with three arguments; (value, index|key, collection). + * + * If a property name is provided for `callback` the created "_.pluck" style + * callback will return the property value of the given element. + * + * If an object is provided for `callback` the created "_.where" style callback + * will return `true` for elements that have the properties of the given object, + * else `false`. + * + * @static + * @memberOf _ + * @alias detect, findWhere + * @category Collections + * @param {Array|Object|string} collection The collection to iterate over. + * @param {Function|Object|string} [callback=identity] The function called + * per iteration. If a property name or object is provided it will be used + * to create a "_.pluck" or "_.where" style callback, respectively. + * @param {*} [thisArg] The `this` binding of `callback`. + * @returns {*} Returns the found element, else `undefined`. + * @example + * + * var characters = [ + * { 'name': 'barney', 'age': 36, 'blocked': false }, + * { 'name': 'fred', 'age': 40, 'blocked': true }, + * { 'name': 'pebbles', 'age': 1, 'blocked': false } + * ]; + * + * _.find(characters, function(chr) { + * return chr.age < 40; + * }); + * // => { 'name': 'barney', 'age': 36, 'blocked': false } + * + * // using "_.where" callback shorthand + * _.find(characters, { 'age': 1 }); + * // => { 'name': 'pebbles', 'age': 1, 'blocked': false } + * + * // using "_.pluck" callback shorthand + * _.find(characters, 'blocked'); + * // => { 'name': 'fred', 'age': 40, 'blocked': true } + */ + function find(collection, callback, thisArg) { + callback = lodash.createCallback(callback, thisArg, 3); + + var index = -1, + length = collection ? collection.length : 0; + + if (typeof length == 'number') { + while (++index < length) { + var value = collection[index]; + if (callback(value, index, collection)) { + return value; + } + } + } else { + var result; + forOwn(collection, function(value, index, collection) { + if (callback(value, index, collection)) { + result = value; + return false; + } + }); + return result; + } + } + + /** + * This method is like `_.find` except that it iterates over elements + * of a `collection` from right to left. + * + * @static + * @memberOf _ + * @category Collections + * @param {Array|Object|string} collection The collection to iterate over. + * @param {Function|Object|string} [callback=identity] The function called + * per iteration. If a property name or object is provided it will be used + * to create a "_.pluck" or "_.where" style callback, respectively. + * @param {*} [thisArg] The `this` binding of `callback`. + * @returns {*} Returns the found element, else `undefined`. + * @example + * + * _.findLast([1, 2, 3, 4], function(num) { + * return num % 2 == 1; + * }); + * // => 3 + */ + function findLast(collection, callback, thisArg) { + var result; + callback = lodash.createCallback(callback, thisArg, 3); + forEachRight(collection, function(value, index, collection) { + if (callback(value, index, collection)) { + result = value; + return false; + } + }); + return result; + } + + /** + * Iterates over elements of a collection, executing the callback for each + * element. The callback is bound to `thisArg` and invoked with three arguments; + * (value, index|key, collection). Callbacks may exit iteration early by + * explicitly returning `false`. + * + * Note: As with other "Collections" methods, objects with a `length` property + * are iterated like arrays. To avoid this behavior `_.forIn` or `_.forOwn` + * may be used for object iteration. + * + * @static + * @memberOf _ + * @alias each + * @category Collections + * @param {Array|Object|string} collection The collection to iterate over. + * @param {Function} [callback=identity] The function called per iteration. + * @param {*} [thisArg] The `this` binding of `callback`. + * @returns {Array|Object|string} Returns `collection`. + * @example + * + * _([1, 2, 3]).forEach(function(num) { console.log(num); }).join(','); + * // => logs each number and returns '1,2,3' + * + * _.forEach({ 'one': 1, 'two': 2, 'three': 3 }, function(num) { console.log(num); }); + * // => logs each number and returns the object (property order is not guaranteed across environments) + */ + function forEach(collection, callback, thisArg) { + var index = -1, + length = collection ? collection.length : 0; + + callback = callback && typeof thisArg == 'undefined' ? callback : baseCreateCallback(callback, thisArg, 3); + if (typeof length == 'number') { + while (++index < length) { + if (callback(collection[index], index, collection) === false) { + break; + } + } + } else { + forOwn(collection, callback); + } + return collection; + } + + /** + * This method is like `_.forEach` except that it iterates over elements + * of a `collection` from right to left. + * + * @static + * @memberOf _ + * @alias eachRight + * @category Collections + * @param {Array|Object|string} collection The collection to iterate over. + * @param {Function} [callback=identity] The function called per iteration. + * @param {*} [thisArg] The `this` binding of `callback`. + * @returns {Array|Object|string} Returns `collection`. + * @example + * + * _([1, 2, 3]).forEachRight(function(num) { console.log(num); }).join(','); + * // => logs each number from right to left and returns '3,2,1' + */ + function forEachRight(collection, callback, thisArg) { + var length = collection ? collection.length : 0; + callback = callback && typeof thisArg == 'undefined' ? callback : baseCreateCallback(callback, thisArg, 3); + if (typeof length == 'number') { + while (length--) { + if (callback(collection[length], length, collection) === false) { + break; + } + } + } else { + var props = keys(collection); + length = props.length; + forOwn(collection, function(value, key, collection) { + key = props ? props[--length] : --length; + return callback(collection[key], key, collection); + }); + } + return collection; + } + + /** + * Creates an object composed of keys generated from the results of running + * each element of a collection through the callback. The corresponding value + * of each key is an array of the elements responsible for generating the key. + * The callback is bound to `thisArg` and invoked with three arguments; + * (value, index|key, collection). + * + * If a property name is provided for `callback` the created "_.pluck" style + * callback will return the property value of the given element. + * + * If an object is provided for `callback` the created "_.where" style callback + * will return `true` for elements that have the properties of the given object, + * else `false` + * + * @static + * @memberOf _ + * @category Collections + * @param {Array|Object|string} collection The collection to iterate over. + * @param {Function|Object|string} [callback=identity] The function called + * per iteration. If a property name or object is provided it will be used + * to create a "_.pluck" or "_.where" style callback, respectively. + * @param {*} [thisArg] The `this` binding of `callback`. + * @returns {Object} Returns the composed aggregate object. + * @example + * + * _.groupBy([4.2, 6.1, 6.4], function(num) { return Math.floor(num); }); + * // => { '4': [4.2], '6': [6.1, 6.4] } + * + * _.groupBy([4.2, 6.1, 6.4], function(num) { return this.floor(num); }, Math); + * // => { '4': [4.2], '6': [6.1, 6.4] } + * + * // using "_.pluck" callback shorthand + * _.groupBy(['one', 'two', 'three'], 'length'); + * // => { '3': ['one', 'two'], '5': ['three'] } + */ + var groupBy = createAggregator(function(result, value, key) { + (hasOwnProperty.call(result, key) ? result[key] : result[key] = []).push(value); + }); + + /** + * Creates an object composed of keys generated from the results of running + * each element of the collection through the given callback. The corresponding + * value of each key is the last element responsible for generating the key. + * The callback is bound to `thisArg` and invoked with three arguments; + * (value, index|key, collection). + * + * If a property name is provided for `callback` the created "_.pluck" style + * callback will return the property value of the given element. + * + * If an object is provided for `callback` the created "_.where" style callback + * will return `true` for elements that have the properties of the given object, + * else `false`. + * + * @static + * @memberOf _ + * @category Collections + * @param {Array|Object|string} collection The collection to iterate over. + * @param {Function|Object|string} [callback=identity] The function called + * per iteration. If a property name or object is provided it will be used + * to create a "_.pluck" or "_.where" style callback, respectively. + * @param {*} [thisArg] The `this` binding of `callback`. + * @returns {Object} Returns the composed aggregate object. + * @example + * + * var keys = [ + * { 'dir': 'left', 'code': 97 }, + * { 'dir': 'right', 'code': 100 } + * ]; + * + * _.indexBy(keys, 'dir'); + * // => { 'left': { 'dir': 'left', 'code': 97 }, 'right': { 'dir': 'right', 'code': 100 } } + * + * _.indexBy(keys, function(key) { return String.fromCharCode(key.code); }); + * // => { 'a': { 'dir': 'left', 'code': 97 }, 'd': { 'dir': 'right', 'code': 100 } } + * + * _.indexBy(characters, function(key) { this.fromCharCode(key.code); }, String); + * // => { 'a': { 'dir': 'left', 'code': 97 }, 'd': { 'dir': 'right', 'code': 100 } } + */ + var indexBy = createAggregator(function(result, value, key) { + result[key] = value; + }); + + /** + * Invokes the method named by `methodName` on each element in the `collection` + * returning an array of the results of each invoked method. Additional arguments + * will be provided to each invoked method. If `methodName` is a function it + * will be invoked for, and `this` bound to, each element in the `collection`. + * + * @static + * @memberOf _ + * @category Collections + * @param {Array|Object|string} collection The collection to iterate over. + * @param {Function|string} methodName The name of the method to invoke or + * the function invoked per iteration. + * @param {...*} [arg] Arguments to invoke the method with. + * @returns {Array} Returns a new array of the results of each invoked method. + * @example + * + * _.invoke([[5, 1, 7], [3, 2, 1]], 'sort'); + * // => [[1, 5, 7], [1, 2, 3]] + * + * _.invoke([123, 456], String.prototype.split, ''); + * // => [['1', '2', '3'], ['4', '5', '6']] + */ + function invoke(collection, methodName) { + var args = slice(arguments, 2), + index = -1, + isFunc = typeof methodName == 'function', + length = collection ? collection.length : 0, + result = Array(typeof length == 'number' ? length : 0); + + forEach(collection, function(value) { + result[++index] = (isFunc ? methodName : value[methodName]).apply(value, args); + }); + return result; + } + + /** + * Creates an array of values by running each element in the collection + * through the callback. The callback is bound to `thisArg` and invoked with + * three arguments; (value, index|key, collection). + * + * If a property name is provided for `callback` the created "_.pluck" style + * callback will return the property value of the given element. + * + * If an object is provided for `callback` the created "_.where" style callback + * will return `true` for elements that have the properties of the given object, + * else `false`. + * + * @static + * @memberOf _ + * @alias collect + * @category Collections + * @param {Array|Object|string} collection The collection to iterate over. + * @param {Function|Object|string} [callback=identity] The function called + * per iteration. If a property name or object is provided it will be used + * to create a "_.pluck" or "_.where" style callback, respectively. + * @param {*} [thisArg] The `this` binding of `callback`. + * @returns {Array} Returns a new array of the results of each `callback` execution. + * @example + * + * _.map([1, 2, 3], function(num) { return num * 3; }); + * // => [3, 6, 9] + * + * _.map({ 'one': 1, 'two': 2, 'three': 3 }, function(num) { return num * 3; }); + * // => [3, 6, 9] (property order is not guaranteed across environments) + * + * var characters = [ + * { 'name': 'barney', 'age': 36 }, + * { 'name': 'fred', 'age': 40 } + * ]; + * + * // using "_.pluck" callback shorthand + * _.map(characters, 'name'); + * // => ['barney', 'fred'] + */ + function map(collection, callback, thisArg) { + var index = -1, + length = collection ? collection.length : 0; + + callback = lodash.createCallback(callback, thisArg, 3); + if (typeof length == 'number') { + var result = Array(length); + while (++index < length) { + result[index] = callback(collection[index], index, collection); + } + } else { + result = []; + forOwn(collection, function(value, key, collection) { + result[++index] = callback(value, key, collection); + }); + } + return result; + } + + /** + * Retrieves the maximum value of a collection. If the collection is empty or + * falsey `-Infinity` is returned. If a callback is provided it will be executed + * for each value in the collection to generate the criterion by which the value + * is ranked. The callback is bound to `thisArg` and invoked with three + * arguments; (value, index, collection). + * + * If a property name is provided for `callback` the created "_.pluck" style + * callback will return the property value of the given element. + * + * If an object is provided for `callback` the created "_.where" style callback + * will return `true` for elements that have the properties of the given object, + * else `false`. + * + * @static + * @memberOf _ + * @category Collections + * @param {Array|Object|string} collection The collection to iterate over. + * @param {Function|Object|string} [callback=identity] The function called + * per iteration. If a property name or object is provided it will be used + * to create a "_.pluck" or "_.where" style callback, respectively. + * @param {*} [thisArg] The `this` binding of `callback`. + * @returns {*} Returns the maximum value. + * @example + * + * _.max([4, 2, 8, 6]); + * // => 8 + * + * var characters = [ + * { 'name': 'barney', 'age': 36 }, + * { 'name': 'fred', 'age': 40 } + * ]; + * + * _.max(characters, function(chr) { return chr.age; }); + * // => { 'name': 'fred', 'age': 40 }; + * + * // using "_.pluck" callback shorthand + * _.max(characters, 'age'); + * // => { 'name': 'fred', 'age': 40 }; + */ + function max(collection, callback, thisArg) { + var computed = -Infinity, + result = computed; + + // allows working with functions like `_.map` without using + // their `index` argument as a callback + if (typeof callback != 'function' && thisArg && thisArg[callback] === collection) { + callback = null; + } + if (callback == null && isArray(collection)) { + var index = -1, + length = collection.length; + + while (++index < length) { + var value = collection[index]; + if (value > result) { + result = value; + } + } + } else { + callback = (callback == null && isString(collection)) + ? charAtCallback + : lodash.createCallback(callback, thisArg, 3); + + forEach(collection, function(value, index, collection) { + var current = callback(value, index, collection); + if (current > computed) { + computed = current; + result = value; + } + }); + } + return result; + } + + /** + * Retrieves the minimum value of a collection. If the collection is empty or + * falsey `Infinity` is returned. If a callback is provided it will be executed + * for each value in the collection to generate the criterion by which the value + * is ranked. The callback is bound to `thisArg` and invoked with three + * arguments; (value, index, collection). + * + * If a property name is provided for `callback` the created "_.pluck" style + * callback will return the property value of the given element. + * + * If an object is provided for `callback` the created "_.where" style callback + * will return `true` for elements that have the properties of the given object, + * else `false`. + * + * @static + * @memberOf _ + * @category Collections + * @param {Array|Object|string} collection The collection to iterate over. + * @param {Function|Object|string} [callback=identity] The function called + * per iteration. If a property name or object is provided it will be used + * to create a "_.pluck" or "_.where" style callback, respectively. + * @param {*} [thisArg] The `this` binding of `callback`. + * @returns {*} Returns the minimum value. + * @example + * + * _.min([4, 2, 8, 6]); + * // => 2 + * + * var characters = [ + * { 'name': 'barney', 'age': 36 }, + * { 'name': 'fred', 'age': 40 } + * ]; + * + * _.min(characters, function(chr) { return chr.age; }); + * // => { 'name': 'barney', 'age': 36 }; + * + * // using "_.pluck" callback shorthand + * _.min(characters, 'age'); + * // => { 'name': 'barney', 'age': 36 }; + */ + function min(collection, callback, thisArg) { + var computed = Infinity, + result = computed; + + // allows working with functions like `_.map` without using + // their `index` argument as a callback + if (typeof callback != 'function' && thisArg && thisArg[callback] === collection) { + callback = null; + } + if (callback == null && isArray(collection)) { + var index = -1, + length = collection.length; + + while (++index < length) { + var value = collection[index]; + if (value < result) { + result = value; + } + } + } else { + callback = (callback == null && isString(collection)) + ? charAtCallback + : lodash.createCallback(callback, thisArg, 3); + + forEach(collection, function(value, index, collection) { + var current = callback(value, index, collection); + if (current < computed) { + computed = current; + result = value; + } + }); + } + return result; + } + + /** + * Retrieves the value of a specified property from all elements in the collection. + * + * @static + * @memberOf _ + * @type Function + * @category Collections + * @param {Array|Object|string} collection The collection to iterate over. + * @param {string} property The name of the property to pluck. + * @returns {Array} Returns a new array of property values. + * @example + * + * var characters = [ + * { 'name': 'barney', 'age': 36 }, + * { 'name': 'fred', 'age': 40 } + * ]; + * + * _.pluck(characters, 'name'); + * // => ['barney', 'fred'] + */ + var pluck = map; + + /** + * Reduces a collection to a value which is the accumulated result of running + * each element in the collection through the callback, where each successive + * callback execution consumes the return value of the previous execution. If + * `accumulator` is not provided the first element of the collection will be + * used as the initial `accumulator` value. The callback is bound to `thisArg` + * and invoked with four arguments; (accumulator, value, index|key, collection). + * + * @static + * @memberOf _ + * @alias foldl, inject + * @category Collections + * @param {Array|Object|string} collection The collection to iterate over. + * @param {Function} [callback=identity] The function called per iteration. + * @param {*} [accumulator] Initial value of the accumulator. + * @param {*} [thisArg] The `this` binding of `callback`. + * @returns {*} Returns the accumulated value. + * @example + * + * var sum = _.reduce([1, 2, 3], function(sum, num) { + * return sum + num; + * }); + * // => 6 + * + * var mapped = _.reduce({ 'a': 1, 'b': 2, 'c': 3 }, function(result, num, key) { + * result[key] = num * 3; + * return result; + * }, {}); + * // => { 'a': 3, 'b': 6, 'c': 9 } + */ + function reduce(collection, callback, accumulator, thisArg) { + if (!collection) return accumulator; + var noaccum = arguments.length < 3; + callback = lodash.createCallback(callback, thisArg, 4); + + var index = -1, + length = collection.length; + + if (typeof length == 'number') { + if (noaccum) { + accumulator = collection[++index]; + } + while (++index < length) { + accumulator = callback(accumulator, collection[index], index, collection); + } + } else { + forOwn(collection, function(value, index, collection) { + accumulator = noaccum + ? (noaccum = false, value) + : callback(accumulator, value, index, collection) + }); + } + return accumulator; + } + + /** + * This method is like `_.reduce` except that it iterates over elements + * of a `collection` from right to left. + * + * @static + * @memberOf _ + * @alias foldr + * @category Collections + * @param {Array|Object|string} collection The collection to iterate over. + * @param {Function} [callback=identity] The function called per iteration. + * @param {*} [accumulator] Initial value of the accumulator. + * @param {*} [thisArg] The `this` binding of `callback`. + * @returns {*} Returns the accumulated value. + * @example + * + * var list = [[0, 1], [2, 3], [4, 5]]; + * var flat = _.reduceRight(list, function(a, b) { return a.concat(b); }, []); + * // => [4, 5, 2, 3, 0, 1] + */ + function reduceRight(collection, callback, accumulator, thisArg) { + var noaccum = arguments.length < 3; + callback = lodash.createCallback(callback, thisArg, 4); + forEachRight(collection, function(value, index, collection) { + accumulator = noaccum + ? (noaccum = false, value) + : callback(accumulator, value, index, collection); + }); + return accumulator; + } + + /** + * The opposite of `_.filter` this method returns the elements of a + * collection that the callback does **not** return truey for. + * + * If a property name is provided for `callback` the created "_.pluck" style + * callback will return the property value of the given element. + * + * If an object is provided for `callback` the created "_.where" style callback + * will return `true` for elements that have the properties of the given object, + * else `false`. + * + * @static + * @memberOf _ + * @category Collections + * @param {Array|Object|string} collection The collection to iterate over. + * @param {Function|Object|string} [callback=identity] The function called + * per iteration. If a property name or object is provided it will be used + * to create a "_.pluck" or "_.where" style callback, respectively. + * @param {*} [thisArg] The `this` binding of `callback`. + * @returns {Array} Returns a new array of elements that failed the callback check. + * @example + * + * var odds = _.reject([1, 2, 3, 4, 5, 6], function(num) { return num % 2 == 0; }); + * // => [1, 3, 5] + * + * var characters = [ + * { 'name': 'barney', 'age': 36, 'blocked': false }, + * { 'name': 'fred', 'age': 40, 'blocked': true } + * ]; + * + * // using "_.pluck" callback shorthand + * _.reject(characters, 'blocked'); + * // => [{ 'name': 'barney', 'age': 36, 'blocked': false }] + * + * // using "_.where" callback shorthand + * _.reject(characters, { 'age': 36 }); + * // => [{ 'name': 'fred', 'age': 40, 'blocked': true }] + */ + function reject(collection, callback, thisArg) { + callback = lodash.createCallback(callback, thisArg, 3); + return filter(collection, function(value, index, collection) { + return !callback(value, index, collection); + }); + } + + /** + * Retrieves a random element or `n` random elements from a collection. + * + * @static + * @memberOf _ + * @category Collections + * @param {Array|Object|string} collection The collection to sample. + * @param {number} [n] The number of elements to sample. + * @param- {Object} [guard] Allows working with functions like `_.map` + * without using their `index` arguments as `n`. + * @returns {Array} Returns the random sample(s) of `collection`. + * @example + * + * _.sample([1, 2, 3, 4]); + * // => 2 + * + * _.sample([1, 2, 3, 4], 2); + * // => [3, 1] + */ + function sample(collection, n, guard) { + if (collection && typeof collection.length != 'number') { + collection = values(collection); + } + if (n == null || guard) { + return collection ? collection[baseRandom(0, collection.length - 1)] : undefined; + } + var result = shuffle(collection); + result.length = nativeMin(nativeMax(0, n), result.length); + return result; + } + + /** + * Creates an array of shuffled values, using a version of the Fisher-Yates + * shuffle. See http://en.wikipedia.org/wiki/Fisher-Yates_shuffle. + * + * @static + * @memberOf _ + * @category Collections + * @param {Array|Object|string} collection The collection to shuffle. + * @returns {Array} Returns a new shuffled collection. + * @example + * + * _.shuffle([1, 2, 3, 4, 5, 6]); + * // => [4, 1, 6, 3, 5, 2] + */ + function shuffle(collection) { + var index = -1, + length = collection ? collection.length : 0, + result = Array(typeof length == 'number' ? length : 0); + + forEach(collection, function(value) { + var rand = baseRandom(0, ++index); + result[index] = result[rand]; + result[rand] = value; + }); + return result; + } + + /** + * Gets the size of the `collection` by returning `collection.length` for arrays + * and array-like objects or the number of own enumerable properties for objects. + * + * @static + * @memberOf _ + * @category Collections + * @param {Array|Object|string} collection The collection to inspect. + * @returns {number} Returns `collection.length` or number of own enumerable properties. + * @example + * + * _.size([1, 2]); + * // => 2 + * + * _.size({ 'one': 1, 'two': 2, 'three': 3 }); + * // => 3 + * + * _.size('pebbles'); + * // => 7 + */ + function size(collection) { + var length = collection ? collection.length : 0; + return typeof length == 'number' ? length : keys(collection).length; + } + + /** + * Checks if the callback returns a truey value for **any** element of a + * collection. The function returns as soon as it finds a passing value and + * does not iterate over the entire collection. The callback is bound to + * `thisArg` and invoked with three arguments; (value, index|key, collection). + * + * If a property name is provided for `callback` the created "_.pluck" style + * callback will return the property value of the given element. + * + * If an object is provided for `callback` the created "_.where" style callback + * will return `true` for elements that have the properties of the given object, + * else `false`. + * + * @static + * @memberOf _ + * @alias any + * @category Collections + * @param {Array|Object|string} collection The collection to iterate over. + * @param {Function|Object|string} [callback=identity] The function called + * per iteration. If a property name or object is provided it will be used + * to create a "_.pluck" or "_.where" style callback, respectively. + * @param {*} [thisArg] The `this` binding of `callback`. + * @returns {boolean} Returns `true` if any element passed the callback check, + * else `false`. + * @example + * + * _.some([null, 0, 'yes', false], Boolean); + * // => true + * + * var characters = [ + * { 'name': 'barney', 'age': 36, 'blocked': false }, + * { 'name': 'fred', 'age': 40, 'blocked': true } + * ]; + * + * // using "_.pluck" callback shorthand + * _.some(characters, 'blocked'); + * // => true + * + * // using "_.where" callback shorthand + * _.some(characters, { 'age': 1 }); + * // => false + */ + function some(collection, callback, thisArg) { + var result; + callback = lodash.createCallback(callback, thisArg, 3); + + var index = -1, + length = collection ? collection.length : 0; + + if (typeof length == 'number') { + while (++index < length) { + if ((result = callback(collection[index], index, collection))) { + break; + } + } + } else { + forOwn(collection, function(value, index, collection) { + return !(result = callback(value, index, collection)); + }); + } + return !!result; + } + + /** + * Creates an array of elements, sorted in ascending order by the results of + * running each element in a collection through the callback. This method + * performs a stable sort, that is, it will preserve the original sort order + * of equal elements. The callback is bound to `thisArg` and invoked with + * three arguments; (value, index|key, collection). + * + * If a property name is provided for `callback` the created "_.pluck" style + * callback will return the property value of the given element. + * + * If an array of property names is provided for `callback` the collection + * will be sorted by each property value. + * + * If an object is provided for `callback` the created "_.where" style callback + * will return `true` for elements that have the properties of the given object, + * else `false`. + * + * @static + * @memberOf _ + * @category Collections + * @param {Array|Object|string} collection The collection to iterate over. + * @param {Array|Function|Object|string} [callback=identity] The function called + * per iteration. If a property name or object is provided it will be used + * to create a "_.pluck" or "_.where" style callback, respectively. + * @param {*} [thisArg] The `this` binding of `callback`. + * @returns {Array} Returns a new array of sorted elements. + * @example + * + * _.sortBy([1, 2, 3], function(num) { return Math.sin(num); }); + * // => [3, 1, 2] + * + * _.sortBy([1, 2, 3], function(num) { return this.sin(num); }, Math); + * // => [3, 1, 2] + * + * var characters = [ + * { 'name': 'barney', 'age': 36 }, + * { 'name': 'fred', 'age': 40 }, + * { 'name': 'barney', 'age': 26 }, + * { 'name': 'fred', 'age': 30 } + * ]; + * + * // using "_.pluck" callback shorthand + * _.map(_.sortBy(characters, 'age'), _.values); + * // => [['barney', 26], ['fred', 30], ['barney', 36], ['fred', 40]] + * + * // sorting by multiple properties + * _.map(_.sortBy(characters, ['name', 'age']), _.values); + * // = > [['barney', 26], ['barney', 36], ['fred', 30], ['fred', 40]] + */ + function sortBy(collection, callback, thisArg) { + var index = -1, + isArr = isArray(callback), + length = collection ? collection.length : 0, + result = Array(typeof length == 'number' ? length : 0); + + if (!isArr) { + callback = lodash.createCallback(callback, thisArg, 3); + } + forEach(collection, function(value, key, collection) { + var object = result[++index] = getObject(); + if (isArr) { + object.criteria = map(callback, function(key) { return value[key]; }); + } else { + (object.criteria = getArray())[0] = callback(value, key, collection); + } + object.index = index; + object.value = value; + }); + + length = result.length; + result.sort(compareAscending); + while (length--) { + var object = result[length]; + result[length] = object.value; + if (!isArr) { + releaseArray(object.criteria); + } + releaseObject(object); + } + return result; + } + + /** + * Converts the `collection` to an array. + * + * @static + * @memberOf _ + * @category Collections + * @param {Array|Object|string} collection The collection to convert. + * @returns {Array} Returns the new converted array. + * @example + * + * (function() { return _.toArray(arguments).slice(1); })(1, 2, 3, 4); + * // => [2, 3, 4] + */ + function toArray(collection) { + if (collection && typeof collection.length == 'number') { + return slice(collection); + } + return values(collection); + } + + /** + * Performs a deep comparison of each element in a `collection` to the given + * `properties` object, returning an array of all elements that have equivalent + * property values. + * + * @static + * @memberOf _ + * @type Function + * @category Collections + * @param {Array|Object|string} collection The collection to iterate over. + * @param {Object} props The object of property values to filter by. + * @returns {Array} Returns a new array of elements that have the given properties. + * @example + * + * var characters = [ + * { 'name': 'barney', 'age': 36, 'pets': ['hoppy'] }, + * { 'name': 'fred', 'age': 40, 'pets': ['baby puss', 'dino'] } + * ]; + * + * _.where(characters, { 'age': 36 }); + * // => [{ 'name': 'barney', 'age': 36, 'pets': ['hoppy'] }] + * + * _.where(characters, { 'pets': ['dino'] }); + * // => [{ 'name': 'fred', 'age': 40, 'pets': ['baby puss', 'dino'] }] + */ + var where = filter; + + /*--------------------------------------------------------------------------*/ + + /** + * Creates an array with all falsey values removed. The values `false`, `null`, + * `0`, `""`, `undefined`, and `NaN` are all falsey. + * + * @static + * @memberOf _ + * @category Arrays + * @param {Array} array The array to compact. + * @returns {Array} Returns a new array of filtered values. + * @example + * + * _.compact([0, 1, false, 2, '', 3]); + * // => [1, 2, 3] + */ + function compact(array) { + var index = -1, + length = array ? array.length : 0, + result = []; + + while (++index < length) { + var value = array[index]; + if (value) { + result.push(value); + } + } + return result; + } + + /** + * Creates an array excluding all values of the provided arrays using strict + * equality for comparisons, i.e. `===`. + * + * @static + * @memberOf _ + * @category Arrays + * @param {Array} array The array to process. + * @param {...Array} [values] The arrays of values to exclude. + * @returns {Array} Returns a new array of filtered values. + * @example + * + * _.difference([1, 2, 3, 4, 5], [5, 2, 10]); + * // => [1, 3, 4] + */ + function difference(array) { + return baseDifference(array, baseFlatten(arguments, true, true, 1)); + } + + /** + * This method is like `_.find` except that it returns the index of the first + * element that passes the callback check, instead of the element itself. + * + * If a property name is provided for `callback` the created "_.pluck" style + * callback will return the property value of the given element. + * + * If an object is provided for `callback` the created "_.where" style callback + * will return `true` for elements that have the properties of the given object, + * else `false`. + * + * @static + * @memberOf _ + * @category Arrays + * @param {Array} array The array to search. + * @param {Function|Object|string} [callback=identity] The function called + * per iteration. If a property name or object is provided it will be used + * to create a "_.pluck" or "_.where" style callback, respectively. + * @param {*} [thisArg] The `this` binding of `callback`. + * @returns {number} Returns the index of the found element, else `-1`. + * @example + * + * var characters = [ + * { 'name': 'barney', 'age': 36, 'blocked': false }, + * { 'name': 'fred', 'age': 40, 'blocked': true }, + * { 'name': 'pebbles', 'age': 1, 'blocked': false } + * ]; + * + * _.findIndex(characters, function(chr) { + * return chr.age < 20; + * }); + * // => 2 + * + * // using "_.where" callback shorthand + * _.findIndex(characters, { 'age': 36 }); + * // => 0 + * + * // using "_.pluck" callback shorthand + * _.findIndex(characters, 'blocked'); + * // => 1 + */ + function findIndex(array, callback, thisArg) { + var index = -1, + length = array ? array.length : 0; + + callback = lodash.createCallback(callback, thisArg, 3); + while (++index < length) { + if (callback(array[index], index, array)) { + return index; + } + } + return -1; + } + + /** + * This method is like `_.findIndex` except that it iterates over elements + * of a `collection` from right to left. + * + * If a property name is provided for `callback` the created "_.pluck" style + * callback will return the property value of the given element. + * + * If an object is provided for `callback` the created "_.where" style callback + * will return `true` for elements that have the properties of the given object, + * else `false`. + * + * @static + * @memberOf _ + * @category Arrays + * @param {Array} array The array to search. + * @param {Function|Object|string} [callback=identity] The function called + * per iteration. If a property name or object is provided it will be used + * to create a "_.pluck" or "_.where" style callback, respectively. + * @param {*} [thisArg] The `this` binding of `callback`. + * @returns {number} Returns the index of the found element, else `-1`. + * @example + * + * var characters = [ + * { 'name': 'barney', 'age': 36, 'blocked': true }, + * { 'name': 'fred', 'age': 40, 'blocked': false }, + * { 'name': 'pebbles', 'age': 1, 'blocked': true } + * ]; + * + * _.findLastIndex(characters, function(chr) { + * return chr.age > 30; + * }); + * // => 1 + * + * // using "_.where" callback shorthand + * _.findLastIndex(characters, { 'age': 36 }); + * // => 0 + * + * // using "_.pluck" callback shorthand + * _.findLastIndex(characters, 'blocked'); + * // => 2 + */ + function findLastIndex(array, callback, thisArg) { + var length = array ? array.length : 0; + callback = lodash.createCallback(callback, thisArg, 3); + while (length--) { + if (callback(array[length], length, array)) { + return length; + } + } + return -1; + } + + /** + * Gets the first element or first `n` elements of an array. If a callback + * is provided elements at the beginning of the array are returned as long + * as the callback returns truey. The callback is bound to `thisArg` and + * invoked with three arguments; (value, index, array). + * + * If a property name is provided for `callback` the created "_.pluck" style + * callback will return the property value of the given element. + * + * If an object is provided for `callback` the created "_.where" style callback + * will return `true` for elements that have the properties of the given object, + * else `false`. + * + * @static + * @memberOf _ + * @alias head, take + * @category Arrays + * @param {Array} array The array to query. + * @param {Function|Object|number|string} [callback] The function called + * per element or the number of elements to return. If a property name or + * object is provided it will be used to create a "_.pluck" or "_.where" + * style callback, respectively. + * @param {*} [thisArg] The `this` binding of `callback`. + * @returns {*} Returns the first element(s) of `array`. + * @example + * + * _.first([1, 2, 3]); + * // => 1 + * + * _.first([1, 2, 3], 2); + * // => [1, 2] + * + * _.first([1, 2, 3], function(num) { + * return num < 3; + * }); + * // => [1, 2] + * + * var characters = [ + * { 'name': 'barney', 'blocked': true, 'employer': 'slate' }, + * { 'name': 'fred', 'blocked': false, 'employer': 'slate' }, + * { 'name': 'pebbles', 'blocked': true, 'employer': 'na' } + * ]; + * + * // using "_.pluck" callback shorthand + * _.first(characters, 'blocked'); + * // => [{ 'name': 'barney', 'blocked': true, 'employer': 'slate' }] + * + * // using "_.where" callback shorthand + * _.pluck(_.first(characters, { 'employer': 'slate' }), 'name'); + * // => ['barney', 'fred'] + */ + function first(array, callback, thisArg) { + var n = 0, + length = array ? array.length : 0; + + if (typeof callback != 'number' && callback != null) { + var index = -1; + callback = lodash.createCallback(callback, thisArg, 3); + while (++index < length && callback(array[index], index, array)) { + n++; + } + } else { + n = callback; + if (n == null || thisArg) { + return array ? array[0] : undefined; + } + } + return slice(array, 0, nativeMin(nativeMax(0, n), length)); + } + + /** + * Flattens a nested array (the nesting can be to any depth). If `isShallow` + * is truey, the array will only be flattened a single level. If a callback + * is provided each element of the array is passed through the callback before + * flattening. The callback is bound to `thisArg` and invoked with three + * arguments; (value, index, array). + * + * If a property name is provided for `callback` the created "_.pluck" style + * callback will return the property value of the given element. + * + * If an object is provided for `callback` the created "_.where" style callback + * will return `true` for elements that have the properties of the given object, + * else `false`. + * + * @static + * @memberOf _ + * @category Arrays + * @param {Array} array The array to flatten. + * @param {boolean} [isShallow=false] A flag to restrict flattening to a single level. + * @param {Function|Object|string} [callback=identity] The function called + * per iteration. If a property name or object is provided it will be used + * to create a "_.pluck" or "_.where" style callback, respectively. + * @param {*} [thisArg] The `this` binding of `callback`. + * @returns {Array} Returns a new flattened array. + * @example + * + * _.flatten([1, [2], [3, [[4]]]]); + * // => [1, 2, 3, 4]; + * + * _.flatten([1, [2], [3, [[4]]]], true); + * // => [1, 2, 3, [[4]]]; + * + * var characters = [ + * { 'name': 'barney', 'age': 30, 'pets': ['hoppy'] }, + * { 'name': 'fred', 'age': 40, 'pets': ['baby puss', 'dino'] } + * ]; + * + * // using "_.pluck" callback shorthand + * _.flatten(characters, 'pets'); + * // => ['hoppy', 'baby puss', 'dino'] + */ + function flatten(array, isShallow, callback, thisArg) { + // juggle arguments + if (typeof isShallow != 'boolean' && isShallow != null) { + thisArg = callback; + callback = (typeof isShallow != 'function' && thisArg && thisArg[isShallow] === array) ? null : isShallow; + isShallow = false; + } + if (callback != null) { + array = map(array, callback, thisArg); + } + return baseFlatten(array, isShallow); + } + + /** + * Gets the index at which the first occurrence of `value` is found using + * strict equality for comparisons, i.e. `===`. If the array is already sorted + * providing `true` for `fromIndex` will run a faster binary search. + * + * @static + * @memberOf _ + * @category Arrays + * @param {Array} array The array to search. + * @param {*} value The value to search for. + * @param {boolean|number} [fromIndex=0] The index to search from or `true` + * to perform a binary search on a sorted array. + * @returns {number} Returns the index of the matched value or `-1`. + * @example + * + * _.indexOf([1, 2, 3, 1, 2, 3], 2); + * // => 1 + * + * _.indexOf([1, 2, 3, 1, 2, 3], 2, 3); + * // => 4 + * + * _.indexOf([1, 1, 2, 2, 3, 3], 2, true); + * // => 2 + */ + function indexOf(array, value, fromIndex) { + if (typeof fromIndex == 'number') { + var length = array ? array.length : 0; + fromIndex = (fromIndex < 0 ? nativeMax(0, length + fromIndex) : fromIndex || 0); + } else if (fromIndex) { + var index = sortedIndex(array, value); + return array[index] === value ? index : -1; + } + return baseIndexOf(array, value, fromIndex); + } + + /** + * Gets all but the last element or last `n` elements of an array. If a + * callback is provided elements at the end of the array are excluded from + * the result as long as the callback returns truey. The callback is bound + * to `thisArg` and invoked with three arguments; (value, index, array). + * + * If a property name is provided for `callback` the created "_.pluck" style + * callback will return the property value of the given element. + * + * If an object is provided for `callback` the created "_.where" style callback + * will return `true` for elements that have the properties of the given object, + * else `false`. + * + * @static + * @memberOf _ + * @category Arrays + * @param {Array} array The array to query. + * @param {Function|Object|number|string} [callback=1] The function called + * per element or the number of elements to exclude. If a property name or + * object is provided it will be used to create a "_.pluck" or "_.where" + * style callback, respectively. + * @param {*} [thisArg] The `this` binding of `callback`. + * @returns {Array} Returns a slice of `array`. + * @example + * + * _.initial([1, 2, 3]); + * // => [1, 2] + * + * _.initial([1, 2, 3], 2); + * // => [1] + * + * _.initial([1, 2, 3], function(num) { + * return num > 1; + * }); + * // => [1] + * + * var characters = [ + * { 'name': 'barney', 'blocked': false, 'employer': 'slate' }, + * { 'name': 'fred', 'blocked': true, 'employer': 'slate' }, + * { 'name': 'pebbles', 'blocked': true, 'employer': 'na' } + * ]; + * + * // using "_.pluck" callback shorthand + * _.initial(characters, 'blocked'); + * // => [{ 'name': 'barney', 'blocked': false, 'employer': 'slate' }] + * + * // using "_.where" callback shorthand + * _.pluck(_.initial(characters, { 'employer': 'na' }), 'name'); + * // => ['barney', 'fred'] + */ + function initial(array, callback, thisArg) { + var n = 0, + length = array ? array.length : 0; + + if (typeof callback != 'number' && callback != null) { + var index = length; + callback = lodash.createCallback(callback, thisArg, 3); + while (index-- && callback(array[index], index, array)) { + n++; + } + } else { + n = (callback == null || thisArg) ? 1 : callback || n; + } + return slice(array, 0, nativeMin(nativeMax(0, length - n), length)); + } + + /** + * Creates an array of unique values present in all provided arrays using + * strict equality for comparisons, i.e. `===`. + * + * @static + * @memberOf _ + * @category Arrays + * @param {...Array} [array] The arrays to inspect. + * @returns {Array} Returns an array of shared values. + * @example + * + * _.intersection([1, 2, 3], [5, 2, 1, 4], [2, 1]); + * // => [1, 2] + */ + function intersection() { + var args = [], + argsIndex = -1, + argsLength = arguments.length, + caches = getArray(), + indexOf = getIndexOf(), + trustIndexOf = indexOf === baseIndexOf, + seen = getArray(); + + while (++argsIndex < argsLength) { + var value = arguments[argsIndex]; + if (isArray(value) || isArguments(value)) { + args.push(value); + caches.push(trustIndexOf && value.length >= largeArraySize && + createCache(argsIndex ? args[argsIndex] : seen)); + } + } + var array = args[0], + index = -1, + length = array ? array.length : 0, + result = []; + + outer: + while (++index < length) { + var cache = caches[0]; + value = array[index]; + + if ((cache ? cacheIndexOf(cache, value) : indexOf(seen, value)) < 0) { + argsIndex = argsLength; + (cache || seen).push(value); + while (--argsIndex) { + cache = caches[argsIndex]; + if ((cache ? cacheIndexOf(cache, value) : indexOf(args[argsIndex], value)) < 0) { + continue outer; + } + } + result.push(value); + } + } + while (argsLength--) { + cache = caches[argsLength]; + if (cache) { + releaseObject(cache); + } + } + releaseArray(caches); + releaseArray(seen); + return result; + } + + /** + * Gets the last element or last `n` elements of an array. If a callback is + * provided elements at the end of the array are returned as long as the + * callback returns truey. The callback is bound to `thisArg` and invoked + * with three arguments; (value, index, array). + * + * If a property name is provided for `callback` the created "_.pluck" style + * callback will return the property value of the given element. + * + * If an object is provided for `callback` the created "_.where" style callback + * will return `true` for elements that have the properties of the given object, + * else `false`. + * + * @static + * @memberOf _ + * @category Arrays + * @param {Array} array The array to query. + * @param {Function|Object|number|string} [callback] The function called + * per element or the number of elements to return. If a property name or + * object is provided it will be used to create a "_.pluck" or "_.where" + * style callback, respectively. + * @param {*} [thisArg] The `this` binding of `callback`. + * @returns {*} Returns the last element(s) of `array`. + * @example + * + * _.last([1, 2, 3]); + * // => 3 + * + * _.last([1, 2, 3], 2); + * // => [2, 3] + * + * _.last([1, 2, 3], function(num) { + * return num > 1; + * }); + * // => [2, 3] + * + * var characters = [ + * { 'name': 'barney', 'blocked': false, 'employer': 'slate' }, + * { 'name': 'fred', 'blocked': true, 'employer': 'slate' }, + * { 'name': 'pebbles', 'blocked': true, 'employer': 'na' } + * ]; + * + * // using "_.pluck" callback shorthand + * _.pluck(_.last(characters, 'blocked'), 'name'); + * // => ['fred', 'pebbles'] + * + * // using "_.where" callback shorthand + * _.last(characters, { 'employer': 'na' }); + * // => [{ 'name': 'pebbles', 'blocked': true, 'employer': 'na' }] + */ + function last(array, callback, thisArg) { + var n = 0, + length = array ? array.length : 0; + + if (typeof callback != 'number' && callback != null) { + var index = length; + callback = lodash.createCallback(callback, thisArg, 3); + while (index-- && callback(array[index], index, array)) { + n++; + } + } else { + n = callback; + if (n == null || thisArg) { + return array ? array[length - 1] : undefined; + } + } + return slice(array, nativeMax(0, length - n)); + } + + /** + * Gets the index at which the last occurrence of `value` is found using strict + * equality for comparisons, i.e. `===`. If `fromIndex` is negative, it is used + * as the offset from the end of the collection. + * + * If a property name is provided for `callback` the created "_.pluck" style + * callback will return the property value of the given element. + * + * If an object is provided for `callback` the created "_.where" style callback + * will return `true` for elements that have the properties of the given object, + * else `false`. + * + * @static + * @memberOf _ + * @category Arrays + * @param {Array} array The array to search. + * @param {*} value The value to search for. + * @param {number} [fromIndex=array.length-1] The index to search from. + * @returns {number} Returns the index of the matched value or `-1`. + * @example + * + * _.lastIndexOf([1, 2, 3, 1, 2, 3], 2); + * // => 4 + * + * _.lastIndexOf([1, 2, 3, 1, 2, 3], 2, 3); + * // => 1 + */ + function lastIndexOf(array, value, fromIndex) { + var index = array ? array.length : 0; + if (typeof fromIndex == 'number') { + index = (fromIndex < 0 ? nativeMax(0, index + fromIndex) : nativeMin(fromIndex, index - 1)) + 1; + } + while (index--) { + if (array[index] === value) { + return index; + } + } + return -1; + } + + /** + * Removes all provided values from the given array using strict equality for + * comparisons, i.e. `===`. + * + * @static + * @memberOf _ + * @category Arrays + * @param {Array} array The array to modify. + * @param {...*} [value] The values to remove. + * @returns {Array} Returns `array`. + * @example + * + * var array = [1, 2, 3, 1, 2, 3]; + * _.pull(array, 2, 3); + * console.log(array); + * // => [1, 1] + */ + function pull(array) { + var args = arguments, + argsIndex = 0, + argsLength = args.length, + length = array ? array.length : 0; + + while (++argsIndex < argsLength) { + var index = -1, + value = args[argsIndex]; + while (++index < length) { + if (array[index] === value) { + splice.call(array, index--, 1); + length--; + } + } + } + return array; + } + + /** + * Creates an array of numbers (positive and/or negative) progressing from + * `start` up to but not including `end`. If `start` is less than `stop` a + * zero-length range is created unless a negative `step` is specified. + * + * @static + * @memberOf _ + * @category Arrays + * @param {number} [start=0] The start of the range. + * @param {number} end The end of the range. + * @param {number} [step=1] The value to increment or decrement by. + * @returns {Array} Returns a new range array. + * @example + * + * _.range(4); + * // => [0, 1, 2, 3] + * + * _.range(1, 5); + * // => [1, 2, 3, 4] + * + * _.range(0, 20, 5); + * // => [0, 5, 10, 15] + * + * _.range(0, -4, -1); + * // => [0, -1, -2, -3] + * + * _.range(1, 4, 0); + * // => [1, 1, 1] + * + * _.range(0); + * // => [] + */ + function range(start, end, step) { + start = +start || 0; + step = typeof step == 'number' ? step : (+step || 1); + + if (end == null) { + end = start; + start = 0; + } + // use `Array(length)` so engines like Chakra and V8 avoid slower modes + // http://youtu.be/XAqIpGU8ZZk#t=17m25s + var index = -1, + length = nativeMax(0, ceil((end - start) / (step || 1))), + result = Array(length); + + while (++index < length) { + result[index] = start; + start += step; + } + return result; + } + + /** + * Removes all elements from an array that the callback returns truey for + * and returns an array of removed elements. The callback is bound to `thisArg` + * and invoked with three arguments; (value, index, array). + * + * If a property name is provided for `callback` the created "_.pluck" style + * callback will return the property value of the given element. + * + * If an object is provided for `callback` the created "_.where" style callback + * will return `true` for elements that have the properties of the given object, + * else `false`. + * + * @static + * @memberOf _ + * @category Arrays + * @param {Array} array The array to modify. + * @param {Function|Object|string} [callback=identity] The function called + * per iteration. If a property name or object is provided it will be used + * to create a "_.pluck" or "_.where" style callback, respectively. + * @param {*} [thisArg] The `this` binding of `callback`. + * @returns {Array} Returns a new array of removed elements. + * @example + * + * var array = [1, 2, 3, 4, 5, 6]; + * var evens = _.remove(array, function(num) { return num % 2 == 0; }); + * + * console.log(array); + * // => [1, 3, 5] + * + * console.log(evens); + * // => [2, 4, 6] + */ + function remove(array, callback, thisArg) { + var index = -1, + length = array ? array.length : 0, + result = []; + + callback = lodash.createCallback(callback, thisArg, 3); + while (++index < length) { + var value = array[index]; + if (callback(value, index, array)) { + result.push(value); + splice.call(array, index--, 1); + length--; + } + } + return result; + } + + /** + * The opposite of `_.initial` this method gets all but the first element or + * first `n` elements of an array. If a callback function is provided elements + * at the beginning of the array are excluded from the result as long as the + * callback returns truey. The callback is bound to `thisArg` and invoked + * with three arguments; (value, index, array). + * + * If a property name is provided for `callback` the created "_.pluck" style + * callback will return the property value of the given element. + * + * If an object is provided for `callback` the created "_.where" style callback + * will return `true` for elements that have the properties of the given object, + * else `false`. + * + * @static + * @memberOf _ + * @alias drop, tail + * @category Arrays + * @param {Array} array The array to query. + * @param {Function|Object|number|string} [callback=1] The function called + * per element or the number of elements to exclude. If a property name or + * object is provided it will be used to create a "_.pluck" or "_.where" + * style callback, respectively. + * @param {*} [thisArg] The `this` binding of `callback`. + * @returns {Array} Returns a slice of `array`. + * @example + * + * _.rest([1, 2, 3]); + * // => [2, 3] + * + * _.rest([1, 2, 3], 2); + * // => [3] + * + * _.rest([1, 2, 3], function(num) { + * return num < 3; + * }); + * // => [3] + * + * var characters = [ + * { 'name': 'barney', 'blocked': true, 'employer': 'slate' }, + * { 'name': 'fred', 'blocked': false, 'employer': 'slate' }, + * { 'name': 'pebbles', 'blocked': true, 'employer': 'na' } + * ]; + * + * // using "_.pluck" callback shorthand + * _.pluck(_.rest(characters, 'blocked'), 'name'); + * // => ['fred', 'pebbles'] + * + * // using "_.where" callback shorthand + * _.rest(characters, { 'employer': 'slate' }); + * // => [{ 'name': 'pebbles', 'blocked': true, 'employer': 'na' }] + */ + function rest(array, callback, thisArg) { + if (typeof callback != 'number' && callback != null) { + var n = 0, + index = -1, + length = array ? array.length : 0; + + callback = lodash.createCallback(callback, thisArg, 3); + while (++index < length && callback(array[index], index, array)) { + n++; + } + } else { + n = (callback == null || thisArg) ? 1 : nativeMax(0, callback); + } + return slice(array, n); + } + + /** + * Uses a binary search to determine the smallest index at which a value + * should be inserted into a given sorted array in order to maintain the sort + * order of the array. If a callback is provided it will be executed for + * `value` and each element of `array` to compute their sort ranking. The + * callback is bound to `thisArg` and invoked with one argument; (value). + * + * If a property name is provided for `callback` the created "_.pluck" style + * callback will return the property value of the given element. + * + * If an object is provided for `callback` the created "_.where" style callback + * will return `true` for elements that have the properties of the given object, + * else `false`. + * + * @static + * @memberOf _ + * @category Arrays + * @param {Array} array The array to inspect. + * @param {*} value The value to evaluate. + * @param {Function|Object|string} [callback=identity] The function called + * per iteration. If a property name or object is provided it will be used + * to create a "_.pluck" or "_.where" style callback, respectively. + * @param {*} [thisArg] The `this` binding of `callback`. + * @returns {number} Returns the index at which `value` should be inserted + * into `array`. + * @example + * + * _.sortedIndex([20, 30, 50], 40); + * // => 2 + * + * // using "_.pluck" callback shorthand + * _.sortedIndex([{ 'x': 20 }, { 'x': 30 }, { 'x': 50 }], { 'x': 40 }, 'x'); + * // => 2 + * + * var dict = { + * 'wordToNumber': { 'twenty': 20, 'thirty': 30, 'fourty': 40, 'fifty': 50 } + * }; + * + * _.sortedIndex(['twenty', 'thirty', 'fifty'], 'fourty', function(word) { + * return dict.wordToNumber[word]; + * }); + * // => 2 + * + * _.sortedIndex(['twenty', 'thirty', 'fifty'], 'fourty', function(word) { + * return this.wordToNumber[word]; + * }, dict); + * // => 2 + */ + function sortedIndex(array, value, callback, thisArg) { + var low = 0, + high = array ? array.length : low; + + // explicitly reference `identity` for better inlining in Firefox + callback = callback ? lodash.createCallback(callback, thisArg, 1) : identity; + value = callback(value); + + while (low < high) { + var mid = (low + high) >>> 1; + (callback(array[mid]) < value) + ? low = mid + 1 + : high = mid; + } + return low; + } + + /** + * Creates an array of unique values, in order, of the provided arrays using + * strict equality for comparisons, i.e. `===`. + * + * @static + * @memberOf _ + * @category Arrays + * @param {...Array} [array] The arrays to inspect. + * @returns {Array} Returns an array of combined values. + * @example + * + * _.union([1, 2, 3], [5, 2, 1, 4], [2, 1]); + * // => [1, 2, 3, 5, 4] + */ + function union() { + return baseUniq(baseFlatten(arguments, true, true)); + } + + /** + * Creates a duplicate-value-free version of an array using strict equality + * for comparisons, i.e. `===`. If the array is sorted, providing + * `true` for `isSorted` will use a faster algorithm. If a callback is provided + * each element of `array` is passed through the callback before uniqueness + * is computed. The callback is bound to `thisArg` and invoked with three + * arguments; (value, index, array). + * + * If a property name is provided for `callback` the created "_.pluck" style + * callback will return the property value of the given element. + * + * If an object is provided for `callback` the created "_.where" style callback + * will return `true` for elements that have the properties of the given object, + * else `false`. + * + * @static + * @memberOf _ + * @alias unique + * @category Arrays + * @param {Array} array The array to process. + * @param {boolean} [isSorted=false] A flag to indicate that `array` is sorted. + * @param {Function|Object|string} [callback=identity] The function called + * per iteration. If a property name or object is provided it will be used + * to create a "_.pluck" or "_.where" style callback, respectively. + * @param {*} [thisArg] The `this` binding of `callback`. + * @returns {Array} Returns a duplicate-value-free array. + * @example + * + * _.uniq([1, 2, 1, 3, 1]); + * // => [1, 2, 3] + * + * _.uniq([1, 1, 2, 2, 3], true); + * // => [1, 2, 3] + * + * _.uniq(['A', 'b', 'C', 'a', 'B', 'c'], function(letter) { return letter.toLowerCase(); }); + * // => ['A', 'b', 'C'] + * + * _.uniq([1, 2.5, 3, 1.5, 2, 3.5], function(num) { return this.floor(num); }, Math); + * // => [1, 2.5, 3] + * + * // using "_.pluck" callback shorthand + * _.uniq([{ 'x': 1 }, { 'x': 2 }, { 'x': 1 }], 'x'); + * // => [{ 'x': 1 }, { 'x': 2 }] + */ + function uniq(array, isSorted, callback, thisArg) { + // juggle arguments + if (typeof isSorted != 'boolean' && isSorted != null) { + thisArg = callback; + callback = (typeof isSorted != 'function' && thisArg && thisArg[isSorted] === array) ? null : isSorted; + isSorted = false; + } + if (callback != null) { + callback = lodash.createCallback(callback, thisArg, 3); + } + return baseUniq(array, isSorted, callback); + } + + /** + * Creates an array excluding all provided values using strict equality for + * comparisons, i.e. `===`. + * + * @static + * @memberOf _ + * @category Arrays + * @param {Array} array The array to filter. + * @param {...*} [value] The values to exclude. + * @returns {Array} Returns a new array of filtered values. + * @example + * + * _.without([1, 2, 1, 0, 3, 1, 4], 0, 1); + * // => [2, 3, 4] + */ + function without(array) { + return baseDifference(array, slice(arguments, 1)); + } + + /** + * Creates an array that is the symmetric difference of the provided arrays. + * See http://en.wikipedia.org/wiki/Symmetric_difference. + * + * @static + * @memberOf _ + * @category Arrays + * @param {...Array} [array] The arrays to inspect. + * @returns {Array} Returns an array of values. + * @example + * + * _.xor([1, 2, 3], [5, 2, 1, 4]); + * // => [3, 5, 4] + * + * _.xor([1, 2, 5], [2, 3, 5], [3, 4, 5]); + * // => [1, 4, 5] + */ + function xor() { + var index = -1, + length = arguments.length; + + while (++index < length) { + var array = arguments[index]; + if (isArray(array) || isArguments(array)) { + var result = result + ? baseUniq(baseDifference(result, array).concat(baseDifference(array, result))) + : array; + } + } + return result || []; + } + + /** + * Creates an array of grouped elements, the first of which contains the first + * elements of the given arrays, the second of which contains the second + * elements of the given arrays, and so on. + * + * @static + * @memberOf _ + * @alias unzip + * @category Arrays + * @param {...Array} [array] Arrays to process. + * @returns {Array} Returns a new array of grouped elements. + * @example + * + * _.zip(['fred', 'barney'], [30, 40], [true, false]); + * // => [['fred', 30, true], ['barney', 40, false]] + */ + function zip() { + var array = arguments.length > 1 ? arguments : arguments[0], + index = -1, + length = array ? max(pluck(array, 'length')) : 0, + result = Array(length < 0 ? 0 : length); + + while (++index < length) { + result[index] = pluck(array, index); + } + return result; + } + + /** + * Creates an object composed from arrays of `keys` and `values`. Provide + * either a single two dimensional array, i.e. `[[key1, value1], [key2, value2]]` + * or two arrays, one of `keys` and one of corresponding `values`. + * + * @static + * @memberOf _ + * @alias object + * @category Arrays + * @param {Array} keys The array of keys. + * @param {Array} [values=[]] The array of values. + * @returns {Object} Returns an object composed of the given keys and + * corresponding values. + * @example + * + * _.zipObject(['fred', 'barney'], [30, 40]); + * // => { 'fred': 30, 'barney': 40 } + */ + function zipObject(keys, values) { + var index = -1, + length = keys ? keys.length : 0, + result = {}; + + if (!values && length && !isArray(keys[0])) { + values = []; + } + while (++index < length) { + var key = keys[index]; + if (values) { + result[key] = values[index]; + } else if (key) { + result[key[0]] = key[1]; + } + } + return result; + } + + /*--------------------------------------------------------------------------*/ + + /** + * Creates a function that executes `func`, with the `this` binding and + * arguments of the created function, only after being called `n` times. + * + * @static + * @memberOf _ + * @category Functions + * @param {number} n The number of times the function must be called before + * `func` is executed. + * @param {Function} func The function to restrict. + * @returns {Function} Returns the new restricted function. + * @example + * + * var saves = ['profile', 'settings']; + * + * var done = _.after(saves.length, function() { + * console.log('Done saving!'); + * }); + * + * _.forEach(saves, function(type) { + * asyncSave({ 'type': type, 'complete': done }); + * }); + * // => logs 'Done saving!', after all saves have completed + */ + function after(n, func) { + if (!isFunction(func)) { + throw new TypeError; + } + return function() { + if (--n < 1) { + return func.apply(this, arguments); + } + }; + } + + /** + * Creates a function that, when called, invokes `func` with the `this` + * binding of `thisArg` and prepends any additional `bind` arguments to those + * provided to the bound function. + * + * @static + * @memberOf _ + * @category Functions + * @param {Function} func The function to bind. + * @param {*} [thisArg] The `this` binding of `func`. + * @param {...*} [arg] Arguments to be partially applied. + * @returns {Function} Returns the new bound function. + * @example + * + * var func = function(greeting) { + * return greeting + ' ' + this.name; + * }; + * + * func = _.bind(func, { 'name': 'fred' }, 'hi'); + * func(); + * // => 'hi fred' + */ + function bind(func, thisArg) { + return arguments.length > 2 + ? createWrapper(func, 17, slice(arguments, 2), null, thisArg) + : createWrapper(func, 1, null, null, thisArg); + } + + /** + * Binds methods of an object to the object itself, overwriting the existing + * method. Method names may be specified as individual arguments or as arrays + * of method names. If no method names are provided all the function properties + * of `object` will be bound. + * + * @static + * @memberOf _ + * @category Functions + * @param {Object} object The object to bind and assign the bound methods to. + * @param {...string} [methodName] The object method names to + * bind, specified as individual method names or arrays of method names. + * @returns {Object} Returns `object`. + * @example + * + * var view = { + * 'label': 'docs', + * 'onClick': function() { console.log('clicked ' + this.label); } + * }; + * + * _.bindAll(view); + * jQuery('#docs').on('click', view.onClick); + * // => logs 'clicked docs', when the button is clicked + */ + function bindAll(object) { + var funcs = arguments.length > 1 ? baseFlatten(arguments, true, false, 1) : functions(object), + index = -1, + length = funcs.length; + + while (++index < length) { + var key = funcs[index]; + object[key] = createWrapper(object[key], 1, null, null, object); + } + return object; + } + + /** + * Creates a function that, when called, invokes the method at `object[key]` + * and prepends any additional `bindKey` arguments to those provided to the bound + * function. This method differs from `_.bind` by allowing bound functions to + * reference methods that will be redefined or don't yet exist. + * See http://michaux.ca/articles/lazy-function-definition-pattern. + * + * @static + * @memberOf _ + * @category Functions + * @param {Object} object The object the method belongs to. + * @param {string} key The key of the method. + * @param {...*} [arg] Arguments to be partially applied. + * @returns {Function} Returns the new bound function. + * @example + * + * var object = { + * 'name': 'fred', + * 'greet': function(greeting) { + * return greeting + ' ' + this.name; + * } + * }; + * + * var func = _.bindKey(object, 'greet', 'hi'); + * func(); + * // => 'hi fred' + * + * object.greet = function(greeting) { + * return greeting + 'ya ' + this.name + '!'; + * }; + * + * func(); + * // => 'hiya fred!' + */ + function bindKey(object, key) { + return arguments.length > 2 + ? createWrapper(key, 19, slice(arguments, 2), null, object) + : createWrapper(key, 3, null, null, object); + } + + /** + * Creates a function that is the composition of the provided functions, + * where each function consumes the return value of the function that follows. + * For example, composing the functions `f()`, `g()`, and `h()` produces `f(g(h()))`. + * Each function is executed with the `this` binding of the composed function. + * + * @static + * @memberOf _ + * @category Functions + * @param {...Function} [func] Functions to compose. + * @returns {Function} Returns the new composed function. + * @example + * + * var realNameMap = { + * 'pebbles': 'penelope' + * }; + * + * var format = function(name) { + * name = realNameMap[name.toLowerCase()] || name; + * return name.charAt(0).toUpperCase() + name.slice(1).toLowerCase(); + * }; + * + * var greet = function(formatted) { + * return 'Hiya ' + formatted + '!'; + * }; + * + * var welcome = _.compose(greet, format); + * welcome('pebbles'); + * // => 'Hiya Penelope!' + */ + function compose() { + var funcs = arguments, + length = funcs.length; + + while (length--) { + if (!isFunction(funcs[length])) { + throw new TypeError; + } + } + return function() { + var args = arguments, + length = funcs.length; + + while (length--) { + args = [funcs[length].apply(this, args)]; + } + return args[0]; + }; + } + + /** + * Creates a function which accepts one or more arguments of `func` that when + * invoked either executes `func` returning its result, if all `func` arguments + * have been provided, or returns a function that accepts one or more of the + * remaining `func` arguments, and so on. The arity of `func` can be specified + * if `func.length` is not sufficient. + * + * @static + * @memberOf _ + * @category Functions + * @param {Function} func The function to curry. + * @param {number} [arity=func.length] The arity of `func`. + * @returns {Function} Returns the new curried function. + * @example + * + * var curried = _.curry(function(a, b, c) { + * console.log(a + b + c); + * }); + * + * curried(1)(2)(3); + * // => 6 + * + * curried(1, 2)(3); + * // => 6 + * + * curried(1, 2, 3); + * // => 6 + */ + function curry(func, arity) { + arity = typeof arity == 'number' ? arity : (+arity || func.length); + return createWrapper(func, 4, null, null, null, arity); + } + + /** + * Creates a function that will delay the execution of `func` until after + * `wait` milliseconds have elapsed since the last time it was invoked. + * Provide an options object to indicate that `func` should be invoked on + * the leading and/or trailing edge of the `wait` timeout. Subsequent calls + * to the debounced function will return the result of the last `func` call. + * + * Note: If `leading` and `trailing` options are `true` `func` will be called + * on the trailing edge of the timeout only if the the debounced function is + * invoked more than once during the `wait` timeout. + * + * @static + * @memberOf _ + * @category Functions + * @param {Function} func The function to debounce. + * @param {number} wait The number of milliseconds to delay. + * @param {Object} [options] The options object. + * @param {boolean} [options.leading=false] Specify execution on the leading edge of the timeout. + * @param {number} [options.maxWait] The maximum time `func` is allowed to be delayed before it's called. + * @param {boolean} [options.trailing=true] Specify execution on the trailing edge of the timeout. + * @returns {Function} Returns the new debounced function. + * @example + * + * // avoid costly calculations while the window size is in flux + * var lazyLayout = _.debounce(calculateLayout, 150); + * jQuery(window).on('resize', lazyLayout); + * + * // execute `sendMail` when the click event is fired, debouncing subsequent calls + * jQuery('#postbox').on('click', _.debounce(sendMail, 300, { + * 'leading': true, + * 'trailing': false + * }); + * + * // ensure `batchLog` is executed once after 1 second of debounced calls + * var source = new EventSource('/stream'); + * source.addEventListener('message', _.debounce(batchLog, 250, { + * 'maxWait': 1000 + * }, false); + */ + function debounce(func, wait, options) { + var args, + maxTimeoutId, + result, + stamp, + thisArg, + timeoutId, + trailingCall, + lastCalled = 0, + maxWait = false, + trailing = true; + + if (!isFunction(func)) { + throw new TypeError; + } + wait = nativeMax(0, wait) || 0; + if (options === true) { + var leading = true; + trailing = false; + } else if (isObject(options)) { + leading = options.leading; + maxWait = 'maxWait' in options && (nativeMax(wait, options.maxWait) || 0); + trailing = 'trailing' in options ? options.trailing : trailing; + } + var delayed = function() { + var remaining = wait - (now() - stamp); + if (remaining <= 0) { + if (maxTimeoutId) { + clearTimeout(maxTimeoutId); + } + var isCalled = trailingCall; + maxTimeoutId = timeoutId = trailingCall = undefined; + if (isCalled) { + lastCalled = now(); + result = func.apply(thisArg, args); + if (!timeoutId && !maxTimeoutId) { + args = thisArg = null; + } + } + } else { + timeoutId = setTimeout(delayed, remaining); + } + }; + + var maxDelayed = function() { + if (timeoutId) { + clearTimeout(timeoutId); + } + maxTimeoutId = timeoutId = trailingCall = undefined; + if (trailing || (maxWait !== wait)) { + lastCalled = now(); + result = func.apply(thisArg, args); + if (!timeoutId && !maxTimeoutId) { + args = thisArg = null; + } + } + }; + + return function() { + args = arguments; + stamp = now(); + thisArg = this; + trailingCall = trailing && (timeoutId || !leading); + + if (maxWait === false) { + var leadingCall = leading && !timeoutId; + } else { + if (!maxTimeoutId && !leading) { + lastCalled = stamp; + } + var remaining = maxWait - (stamp - lastCalled), + isCalled = remaining <= 0; + + if (isCalled) { + if (maxTimeoutId) { + maxTimeoutId = clearTimeout(maxTimeoutId); + } + lastCalled = stamp; + result = func.apply(thisArg, args); + } + else if (!maxTimeoutId) { + maxTimeoutId = setTimeout(maxDelayed, remaining); + } + } + if (isCalled && timeoutId) { + timeoutId = clearTimeout(timeoutId); + } + else if (!timeoutId && wait !== maxWait) { + timeoutId = setTimeout(delayed, wait); + } + if (leadingCall) { + isCalled = true; + result = func.apply(thisArg, args); + } + if (isCalled && !timeoutId && !maxTimeoutId) { + args = thisArg = null; + } + return result; + }; + } + + /** + * Defers executing the `func` function until the current call stack has cleared. + * Additional arguments will be provided to `func` when it is invoked. + * + * @static + * @memberOf _ + * @category Functions + * @param {Function} func The function to defer. + * @param {...*} [arg] Arguments to invoke the function with. + * @returns {number} Returns the timer id. + * @example + * + * _.defer(function(text) { console.log(text); }, 'deferred'); + * // logs 'deferred' after one or more milliseconds + */ + function defer(func) { + if (!isFunction(func)) { + throw new TypeError; + } + var args = slice(arguments, 1); + return setTimeout(function() { func.apply(undefined, args); }, 1); + } + + /** + * Executes the `func` function after `wait` milliseconds. Additional arguments + * will be provided to `func` when it is invoked. + * + * @static + * @memberOf _ + * @category Functions + * @param {Function} func The function to delay. + * @param {number} wait The number of milliseconds to delay execution. + * @param {...*} [arg] Arguments to invoke the function with. + * @returns {number} Returns the timer id. + * @example + * + * _.delay(function(text) { console.log(text); }, 1000, 'later'); + * // => logs 'later' after one second + */ + function delay(func, wait) { + if (!isFunction(func)) { + throw new TypeError; + } + var args = slice(arguments, 2); + return setTimeout(function() { func.apply(undefined, args); }, wait); + } + + /** + * Creates a function that memoizes the result of `func`. If `resolver` is + * provided it will be used to determine the cache key for storing the result + * based on the arguments provided to the memoized function. By default, the + * first argument provided to the memoized function is used as the cache key. + * The `func` is executed with the `this` binding of the memoized function. + * The result cache is exposed as the `cache` property on the memoized function. + * + * @static + * @memberOf _ + * @category Functions + * @param {Function} func The function to have its output memoized. + * @param {Function} [resolver] A function used to resolve the cache key. + * @returns {Function} Returns the new memoizing function. + * @example + * + * var fibonacci = _.memoize(function(n) { + * return n < 2 ? n : fibonacci(n - 1) + fibonacci(n - 2); + * }); + * + * fibonacci(9) + * // => 34 + * + * var data = { + * 'fred': { 'name': 'fred', 'age': 40 }, + * 'pebbles': { 'name': 'pebbles', 'age': 1 } + * }; + * + * // modifying the result cache + * var get = _.memoize(function(name) { return data[name]; }, _.identity); + * get('pebbles'); + * // => { 'name': 'pebbles', 'age': 1 } + * + * get.cache.pebbles.name = 'penelope'; + * get('pebbles'); + * // => { 'name': 'penelope', 'age': 1 } + */ + function memoize(func, resolver) { + if (!isFunction(func)) { + throw new TypeError; + } + var memoized = function() { + var cache = memoized.cache, + key = resolver ? resolver.apply(this, arguments) : keyPrefix + arguments[0]; + + return hasOwnProperty.call(cache, key) + ? cache[key] + : (cache[key] = func.apply(this, arguments)); + } + memoized.cache = {}; + return memoized; + } + + /** + * Creates a function that is restricted to execute `func` once. Repeat calls to + * the function will return the value of the first call. The `func` is executed + * with the `this` binding of the created function. + * + * @static + * @memberOf _ + * @category Functions + * @param {Function} func The function to restrict. + * @returns {Function} Returns the new restricted function. + * @example + * + * var initialize = _.once(createApplication); + * initialize(); + * initialize(); + * // `initialize` executes `createApplication` once + */ + function once(func) { + var ran, + result; + + if (!isFunction(func)) { + throw new TypeError; + } + return function() { + if (ran) { + return result; + } + ran = true; + result = func.apply(this, arguments); + + // clear the `func` variable so the function may be garbage collected + func = null; + return result; + }; + } + + /** + * Creates a function that, when called, invokes `func` with any additional + * `partial` arguments prepended to those provided to the new function. This + * method is similar to `_.bind` except it does **not** alter the `this` binding. + * + * @static + * @memberOf _ + * @category Functions + * @param {Function} func The function to partially apply arguments to. + * @param {...*} [arg] Arguments to be partially applied. + * @returns {Function} Returns the new partially applied function. + * @example + * + * var greet = function(greeting, name) { return greeting + ' ' + name; }; + * var hi = _.partial(greet, 'hi'); + * hi('fred'); + * // => 'hi fred' + */ + function partial(func) { + return createWrapper(func, 16, slice(arguments, 1)); + } + + /** + * This method is like `_.partial` except that `partial` arguments are + * appended to those provided to the new function. + * + * @static + * @memberOf _ + * @category Functions + * @param {Function} func The function to partially apply arguments to. + * @param {...*} [arg] Arguments to be partially applied. + * @returns {Function} Returns the new partially applied function. + * @example + * + * var defaultsDeep = _.partialRight(_.merge, _.defaults); + * + * var options = { + * 'variable': 'data', + * 'imports': { 'jq': $ } + * }; + * + * defaultsDeep(options, _.templateSettings); + * + * options.variable + * // => 'data' + * + * options.imports + * // => { '_': _, 'jq': $ } + */ + function partialRight(func) { + return createWrapper(func, 32, null, slice(arguments, 1)); + } + + /** + * Creates a function that, when executed, will only call the `func` function + * at most once per every `wait` milliseconds. Provide an options object to + * indicate that `func` should be invoked on the leading and/or trailing edge + * of the `wait` timeout. Subsequent calls to the throttled function will + * return the result of the last `func` call. + * + * Note: If `leading` and `trailing` options are `true` `func` will be called + * on the trailing edge of the timeout only if the the throttled function is + * invoked more than once during the `wait` timeout. + * + * @static + * @memberOf _ + * @category Functions + * @param {Function} func The function to throttle. + * @param {number} wait The number of milliseconds to throttle executions to. + * @param {Object} [options] The options object. + * @param {boolean} [options.leading=true] Specify execution on the leading edge of the timeout. + * @param {boolean} [options.trailing=true] Specify execution on the trailing edge of the timeout. + * @returns {Function} Returns the new throttled function. + * @example + * + * // avoid excessively updating the position while scrolling + * var throttled = _.throttle(updatePosition, 100); + * jQuery(window).on('scroll', throttled); + * + * // execute `renewToken` when the click event is fired, but not more than once every 5 minutes + * jQuery('.interactive').on('click', _.throttle(renewToken, 300000, { + * 'trailing': false + * })); + */ + function throttle(func, wait, options) { + var leading = true, + trailing = true; + + if (!isFunction(func)) { + throw new TypeError; + } + if (options === false) { + leading = false; + } else if (isObject(options)) { + leading = 'leading' in options ? options.leading : leading; + trailing = 'trailing' in options ? options.trailing : trailing; + } + debounceOptions.leading = leading; + debounceOptions.maxWait = wait; + debounceOptions.trailing = trailing; + + return debounce(func, wait, debounceOptions); + } + + /** + * Creates a function that provides `value` to the wrapper function as its + * first argument. Additional arguments provided to the function are appended + * to those provided to the wrapper function. The wrapper is executed with + * the `this` binding of the created function. + * + * @static + * @memberOf _ + * @category Functions + * @param {*} value The value to wrap. + * @param {Function} wrapper The wrapper function. + * @returns {Function} Returns the new function. + * @example + * + * var p = _.wrap(_.escape, function(func, text) { + * return '

' + func(text) + '

'; + * }); + * + * p('Fred, Wilma, & Pebbles'); + * // => '

Fred, Wilma, & Pebbles

' + */ + function wrap(value, wrapper) { + return createWrapper(wrapper, 16, [value]); + } + + /*--------------------------------------------------------------------------*/ + + /** + * Creates a function that returns `value`. + * + * @static + * @memberOf _ + * @category Utilities + * @param {*} value The value to return from the new function. + * @returns {Function} Returns the new function. + * @example + * + * var object = { 'name': 'fred' }; + * var getter = _.constant(object); + * getter() === object; + * // => true + */ + function constant(value) { + return function() { + return value; + }; + } + + /** + * Produces a callback bound to an optional `thisArg`. If `func` is a property + * name the created callback will return the property value for a given element. + * If `func` is an object the created callback will return `true` for elements + * that contain the equivalent object properties, otherwise it will return `false`. + * + * @static + * @memberOf _ + * @category Utilities + * @param {*} [func=identity] The value to convert to a callback. + * @param {*} [thisArg] The `this` binding of the created callback. + * @param {number} [argCount] The number of arguments the callback accepts. + * @returns {Function} Returns a callback function. + * @example + * + * var characters = [ + * { 'name': 'barney', 'age': 36 }, + * { 'name': 'fred', 'age': 40 } + * ]; + * + * // wrap to create custom callback shorthands + * _.createCallback = _.wrap(_.createCallback, function(func, callback, thisArg) { + * var match = /^(.+?)__([gl]t)(.+)$/.exec(callback); + * return !match ? func(callback, thisArg) : function(object) { + * return match[2] == 'gt' ? object[match[1]] > match[3] : object[match[1]] < match[3]; + * }; + * }); + * + * _.filter(characters, 'age__gt38'); + * // => [{ 'name': 'fred', 'age': 40 }] + */ + function createCallback(func, thisArg, argCount) { + var type = typeof func; + if (func == null || type == 'function') { + return baseCreateCallback(func, thisArg, argCount); + } + // handle "_.pluck" style callback shorthands + if (type != 'object') { + return property(func); + } + var props = keys(func), + key = props[0], + a = func[key]; + + // handle "_.where" style callback shorthands + if (props.length == 1 && a === a && !isObject(a)) { + // fast path the common case of providing an object with a single + // property containing a primitive value + return function(object) { + var b = object[key]; + return a === b && (a !== 0 || (1 / a == 1 / b)); + }; + } + return function(object) { + var length = props.length, + result = false; + + while (length--) { + if (!(result = baseIsEqual(object[props[length]], func[props[length]], null, true))) { + break; + } + } + return result; + }; + } + + /** + * Converts the characters `&`, `<`, `>`, `"`, and `'` in `string` to their + * corresponding HTML entities. + * + * @static + * @memberOf _ + * @category Utilities + * @param {string} string The string to escape. + * @returns {string} Returns the escaped string. + * @example + * + * _.escape('Fred, Wilma, & Pebbles'); + * // => 'Fred, Wilma, & Pebbles' + */ + function escape(string) { + return string == null ? '' : String(string).replace(reUnescapedHtml, escapeHtmlChar); + } + + /** + * This method returns the first argument provided to it. + * + * @static + * @memberOf _ + * @category Utilities + * @param {*} value Any value. + * @returns {*} Returns `value`. + * @example + * + * var object = { 'name': 'fred' }; + * _.identity(object) === object; + * // => true + */ + function identity(value) { + return value; + } + + /** + * Adds function properties of a source object to the destination object. + * If `object` is a function methods will be added to its prototype as well. + * + * @static + * @memberOf _ + * @category Utilities + * @param {Function|Object} [object=lodash] object The destination object. + * @param {Object} source The object of functions to add. + * @param {Object} [options] The options object. + * @param {boolean} [options.chain=true] Specify whether the functions added are chainable. + * @example + * + * function capitalize(string) { + * return string.charAt(0).toUpperCase() + string.slice(1).toLowerCase(); + * } + * + * _.mixin({ 'capitalize': capitalize }); + * _.capitalize('fred'); + * // => 'Fred' + * + * _('fred').capitalize().value(); + * // => 'Fred' + * + * _.mixin({ 'capitalize': capitalize }, { 'chain': false }); + * _('fred').capitalize(); + * // => 'Fred' + */ + function mixin(object, source, options) { + var chain = true, + methodNames = source && functions(source); + + if (!source || (!options && !methodNames.length)) { + if (options == null) { + options = source; + } + ctor = lodashWrapper; + source = object; + object = lodash; + methodNames = functions(source); + } + if (options === false) { + chain = false; + } else if (isObject(options) && 'chain' in options) { + chain = options.chain; + } + var ctor = object, + isFunc = isFunction(ctor); + + forEach(methodNames, function(methodName) { + var func = object[methodName] = source[methodName]; + if (isFunc) { + ctor.prototype[methodName] = function() { + var chainAll = this.__chain__, + value = this.__wrapped__, + args = [value]; + + push.apply(args, arguments); + var result = func.apply(object, args); + if (chain || chainAll) { + if (value === result && isObject(result)) { + return this; + } + result = new ctor(result); + result.__chain__ = chainAll; + } + return result; + }; + } + }); + } + + /** + * Reverts the '_' variable to its previous value and returns a reference to + * the `lodash` function. + * + * @static + * @memberOf _ + * @category Utilities + * @returns {Function} Returns the `lodash` function. + * @example + * + * var lodash = _.noConflict(); + */ + function noConflict() { + context._ = oldDash; + return this; + } + + /** + * A no-operation function. + * + * @static + * @memberOf _ + * @category Utilities + * @example + * + * var object = { 'name': 'fred' }; + * _.noop(object) === undefined; + * // => true + */ + function noop() { + // no operation performed + } + + /** + * Gets the number of milliseconds that have elapsed since the Unix epoch + * (1 January 1970 00:00:00 UTC). + * + * @static + * @memberOf _ + * @category Utilities + * @example + * + * var stamp = _.now(); + * _.defer(function() { console.log(_.now() - stamp); }); + * // => logs the number of milliseconds it took for the deferred function to be called + */ + var now = isNative(now = Date.now) && now || function() { + return new Date().getTime(); + }; + + /** + * Converts the given value into an integer of the specified radix. + * If `radix` is `undefined` or `0` a `radix` of `10` is used unless the + * `value` is a hexadecimal, in which case a `radix` of `16` is used. + * + * Note: This method avoids differences in native ES3 and ES5 `parseInt` + * implementations. See http://es5.github.io/#E. + * + * @static + * @memberOf _ + * @category Utilities + * @param {string} value The value to parse. + * @param {number} [radix] The radix used to interpret the value to parse. + * @returns {number} Returns the new integer value. + * @example + * + * _.parseInt('08'); + * // => 8 + */ + var parseInt = nativeParseInt(whitespace + '08') == 8 ? nativeParseInt : function(value, radix) { + // Firefox < 21 and Opera < 15 follow the ES3 specified implementation of `parseInt` + return nativeParseInt(isString(value) ? value.replace(reLeadingSpacesAndZeros, '') : value, radix || 0); + }; + + /** + * Creates a "_.pluck" style function, which returns the `key` value of a + * given object. + * + * @static + * @memberOf _ + * @category Utilities + * @param {string} key The name of the property to retrieve. + * @returns {Function} Returns the new function. + * @example + * + * var characters = [ + * { 'name': 'fred', 'age': 40 }, + * { 'name': 'barney', 'age': 36 } + * ]; + * + * var getName = _.property('name'); + * + * _.map(characters, getName); + * // => ['barney', 'fred'] + * + * _.sortBy(characters, getName); + * // => [{ 'name': 'barney', 'age': 36 }, { 'name': 'fred', 'age': 40 }] + */ + function property(key) { + return function(object) { + return object[key]; + }; + } + + /** + * Produces a random number between `min` and `max` (inclusive). If only one + * argument is provided a number between `0` and the given number will be + * returned. If `floating` is truey or either `min` or `max` are floats a + * floating-point number will be returned instead of an integer. + * + * @static + * @memberOf _ + * @category Utilities + * @param {number} [min=0] The minimum possible value. + * @param {number} [max=1] The maximum possible value. + * @param {boolean} [floating=false] Specify returning a floating-point number. + * @returns {number} Returns a random number. + * @example + * + * _.random(0, 5); + * // => an integer between 0 and 5 + * + * _.random(5); + * // => also an integer between 0 and 5 + * + * _.random(5, true); + * // => a floating-point number between 0 and 5 + * + * _.random(1.2, 5.2); + * // => a floating-point number between 1.2 and 5.2 + */ + function random(min, max, floating) { + var noMin = min == null, + noMax = max == null; + + if (floating == null) { + if (typeof min == 'boolean' && noMax) { + floating = min; + min = 1; + } + else if (!noMax && typeof max == 'boolean') { + floating = max; + noMax = true; + } + } + if (noMin && noMax) { + max = 1; + } + min = +min || 0; + if (noMax) { + max = min; + min = 0; + } else { + max = +max || 0; + } + if (floating || min % 1 || max % 1) { + var rand = nativeRandom(); + return nativeMin(min + (rand * (max - min + parseFloat('1e-' + ((rand +'').length - 1)))), max); + } + return baseRandom(min, max); + } + + /** + * Resolves the value of property `key` on `object`. If `key` is a function + * it will be invoked with the `this` binding of `object` and its result returned, + * else the property value is returned. If `object` is falsey then `undefined` + * is returned. + * + * @static + * @memberOf _ + * @category Utilities + * @param {Object} object The object to inspect. + * @param {string} key The name of the property to resolve. + * @returns {*} Returns the resolved value. + * @example + * + * var object = { + * 'cheese': 'crumpets', + * 'stuff': function() { + * return 'nonsense'; + * } + * }; + * + * _.result(object, 'cheese'); + * // => 'crumpets' + * + * _.result(object, 'stuff'); + * // => 'nonsense' + */ + function result(object, key) { + if (object) { + var value = object[key]; + return isFunction(value) ? object[key]() : value; + } + } + + /** + * A micro-templating method that handles arbitrary delimiters, preserves + * whitespace, and correctly escapes quotes within interpolated code. + * + * Note: In the development build, `_.template` utilizes sourceURLs for easier + * debugging. See http://www.html5rocks.com/en/tutorials/developertools/sourcemaps/#toc-sourceurl + * + * For more information on precompiling templates see: + * http://lodash.com/custom-builds + * + * For more information on Chrome extension sandboxes see: + * http://developer.chrome.com/stable/extensions/sandboxingEval.html + * + * @static + * @memberOf _ + * @category Utilities + * @param {string} text The template text. + * @param {Object} data The data object used to populate the text. + * @param {Object} [options] The options object. + * @param {RegExp} [options.escape] The "escape" delimiter. + * @param {RegExp} [options.evaluate] The "evaluate" delimiter. + * @param {Object} [options.imports] An object to import into the template as local variables. + * @param {RegExp} [options.interpolate] The "interpolate" delimiter. + * @param {string} [sourceURL] The sourceURL of the template's compiled source. + * @param {string} [variable] The data object variable name. + * @returns {Function|string} Returns a compiled function when no `data` object + * is given, else it returns the interpolated text. + * @example + * + * // using the "interpolate" delimiter to create a compiled template + * var compiled = _.template('hello <%= name %>'); + * compiled({ 'name': 'fred' }); + * // => 'hello fred' + * + * // using the "escape" delimiter to escape HTML in data property values + * _.template('<%- value %>', { 'value': ' + + + + diff -r a7b72620d227 -r 3e075a48e19e annot-server/webapp/templates/pianoroll.html --- a/annot-server/webapp/templates/pianoroll.html Thu Jan 22 09:26:43 2015 +0100 +++ b/annot-server/webapp/templates/pianoroll.html Thu Jan 22 09:29:49 2015 +0100 @@ -2,37 +2,51 @@ Piano Roll {{event.label}} - +

Piano Roll {{event.label}}

- stop intervals - - temps écoulé : -

+ stop intervals - + start intervals - + temps écoulé : +


-    
+    
+    
     
-    
-    
-
 
 
diff -r a7b72620d227 -r 3e075a48e19e annot-server/webapp/templates/pianoroll_index.html
--- a/annot-server/webapp/templates/pianoroll_index.html	Thu Jan 22 09:26:43 2015 +0100
+++ b/annot-server/webapp/templates/pianoroll_index.html	Thu Jan 22 09:29:49 2015 +0100
@@ -12,7 +12,7 @@
       

Évènements actifs :

{% for event in events %}
- +
{{ event.label }} : pianoroll, annotviz
Code : {{ event.code }}
Description : {{ event.description }}
diff -r a7b72620d227 -r 3e075a48e19e annot-server/webapp/views.py --- a/annot-server/webapp/views.py Thu Jan 22 09:26:43 2015 +0100 +++ b/annot-server/webapp/views.py Thu Jan 22 09:29:49 2015 +0100 @@ -60,7 +60,7 @@ events = db_session.query(models.Event).filter(models.Event.active == True).order_by(models.Event.start_date, models.Event.code).all() return render_template('pianoroll_index.html', events=events) -@app.route('/pianoroll/') +@app.route('/pianoroll/pr/') def page_pianoroll_event_code(event_code): event = db_session.query(models.Event).filter(models.Event.code == event_code, models.Event.active == True).first() if not event: @@ -68,6 +68,14 @@ return render_template('pianoroll.html', logging=True, event=event) +@app.route('/pianoroll/av/') +def page_annotviz_event_code(event_code): + event = db_session.query(models.Event).filter(models.Event.code == event_code, models.Event.active == True).first() + if not event: + abort(404) + + return render_template('annotviz.html', logging=True, event=event) + @app.route('/api/test', methods=['PUT', 'POST']) diff -r a7b72620d227 -r 3e075a48e19e client/annot-client/app/annotationclient.html --- a/client/annot-client/app/annotationclient.html Thu Jan 22 09:26:43 2015 +0100 +++ b/client/annot-client/app/annotationclient.html Thu Jan 22 09:29:49 2015 +0100 @@ -16,13 +16,18 @@
+
- - - +
+
+
+ +
+
+
@@ -33,7 +38,7 @@
-
+
@@ -47,7 +52,7 @@
+ ng-click="selectLevel(c.label, c.code, catLabel, catText, c.color, c)" ng-class="{'success-border':c.sendSuccess, 'error-border':c.sendError}">
@@ -60,7 +65,7 @@
+ ng-click="sendAnnotation({% raw %} (c.prelabel !== '') ? (c.prelabel + ': ' + c.label) : c.label {% endraw %}, c.code, catLabel, catText, c.color, c)" ng-class="{'success-border':c.sendSuccess, 'error-border':c.sendError}">
diff -r a7b72620d227 -r 3e075a48e19e client/annot-client/app/app.css --- a/client/annot-client/app/app.css Thu Jan 22 09:26:43 2015 +0100 +++ b/client/annot-client/app/app.css Thu Jan 22 09:29:49 2015 +0100 @@ -63,7 +63,7 @@ font-size: 2vw; } .send{ - background-color: #4cae4c; + background-color: #536991; } .return{ background-color: #e6e6e6; diff -r a7b72620d227 -r 3e075a48e19e client/annot-client/app/app.js --- a/client/annot-client/app/app.js Thu Jan 22 09:26:43 2015 +0100 +++ b/client/annot-client/app/app.js Thu Jan 22 09:29:49 2015 +0100 @@ -35,6 +35,11 @@ ); } + 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; + } + $scope.data = dataModel.data; var process_categories = function(data) { @@ -50,7 +55,12 @@ } } } - $scope.allCatLabels = cats; + if(typeof data.autocomplete === 'undefined' || data.autocomplete.length === 0) { + $scope.allCatLabels = cats; + } + else { + $scope.allCatLabels = data.autocomplete; + } } }; @@ -101,7 +111,7 @@ } else { wsuri = 'ws://' + window.location.hostname + ':8090/annot'; } - + var eventCode = context.event_code; if(typeof eventCode==='undefined' || eventCode===''){ eventCode = $location.search().event; @@ -173,7 +183,12 @@ }; } - $scope.sendAnnotation = function(label, code, c){ + $scope.sendFreeAnnotation = function(label, text) { + $scope.sendAnnotation(label, window.S(label).slugify().s, label, text, $scope.data.defaultColor || "#536991"); + }; + + $scope.sendAnnotation = function(label, code, freeLabel, freetext, color, c){ + if($scope.username==='' || typeof $scope.username==='undefined'){ showAlert('Vous devez indiquer un nom d\'utilisateur.', false); return; @@ -187,12 +202,24 @@ if(typeof code==='undefined' || code===''){ code = window.S(label).slugify().s; } + + var hexc; + if(color.substring(0, 4) === 'rgb(') { + hexc = colorToHex(color); + } + else { + hexc = color; + } + + var new_annot = { category: {code: code, label: label}, + text: freetext, + color: hexc, user : $scope.username }; sock.send(JSON.stringify(new_annot)); - if(context.logging===true){ + if(context.logging===true){ log('Sent: ' + JSON.stringify(new_annot)); } if(typeof c==='undefined'){ @@ -201,6 +228,8 @@ else{ $scope.annotPile.push(c); } + $scope.catText = ""; + $scope.catLabel = ""; } else { showAlert('La socket ne fonctionne pas.', false); if(context.logging===true){ @@ -210,7 +239,7 @@ }; // Interface management - $scope.selectLevel = function(label, code, c){ + $scope.selectLevel = function(label, code, freelabel, freetext, color, c){ if(typeof c==='undefined'){ $scope.returnVisStyle = {visibility:'hidden'}; $scope.selectedlevel = false; @@ -223,7 +252,7 @@ else{ // Send query //console.log('send ntm', c); - $scope.sendAnnotation(label, code, c); + $scope.sendAnnotation(label, code, freelabel, freetext, color, c); } }; diff -r a7b72620d227 -r 3e075a48e19e client/annotviz/app/annotsvizview.html --- a/client/annotviz/app/annotsvizview.html Thu Jan 22 09:26:43 2015 +0100 +++ b/client/annotviz/app/annotsvizview.html Thu Jan 22 09:29:49 2015 +0100 @@ -21,9 +21,58 @@