UNPKG

chart

Version:

event based time series charting API

1,559 lines (1,386 loc) 82.5 kB
(function(){var require = function (file, cwd) { var resolved = require.resolve(file, cwd || '/'); var mod = require.modules[resolved]; if (!mod) throw new Error( 'Failed to resolve module ' + file + ', tried ' + resolved ); var cached = require.cache[resolved]; var res = cached? cached.exports : mod(); return res; }; require.paths = []; require.modules = {}; require.cache = {}; require.extensions = [".js",".coffee"]; require._core = { 'assert': true, 'events': true, 'fs': true, 'path': true, 'vm': true }; require.resolve = (function () { return function (x, cwd) { if (!cwd) cwd = '/'; if (require._core[x]) return x; var path = require.modules.path(); cwd = path.resolve('/', cwd); var y = cwd || '/'; if (x.match(/^(?:\.\.?\/|\/)/)) { var m = loadAsFileSync(path.resolve(y, x)) || loadAsDirectorySync(path.resolve(y, x)); if (m) return m; } var n = loadNodeModulesSync(x, y); if (n) return n; throw new Error("Cannot find module '" + x + "'"); function loadAsFileSync (x) { x = path.normalize(x); if (require.modules[x]) { return x; } for (var i = 0; i < require.extensions.length; i++) { var ext = require.extensions[i]; if (require.modules[x + ext]) return x + ext; } } function loadAsDirectorySync (x) { x = x.replace(/\/+$/, ''); var pkgfile = path.normalize(x + '/package.json'); if (require.modules[pkgfile]) { var pkg = require.modules[pkgfile](); var b = pkg.browserify; if (typeof b === 'object' && b.main) { var m = loadAsFileSync(path.resolve(x, b.main)); if (m) return m; } else if (typeof b === 'string') { var m = loadAsFileSync(path.resolve(x, b)); if (m) return m; } else if (pkg.main) { var m = loadAsFileSync(path.resolve(x, pkg.main)); if (m) return m; } } return loadAsFileSync(x + '/index'); } function loadNodeModulesSync (x, start) { var dirs = nodeModulesPathsSync(start); for (var i = 0; i < dirs.length; i++) { var dir = dirs[i]; var m = loadAsFileSync(dir + '/' + x); if (m) return m; var n = loadAsDirectorySync(dir + '/' + x); if (n) return n; } var m = loadAsFileSync(x); if (m) return m; } function nodeModulesPathsSync (start) { var parts; if (start === '/') parts = [ '' ]; else parts = path.normalize(start).split('/'); var dirs = []; for (var i = parts.length - 1; i >= 0; i--) { if (parts[i] === 'node_modules') continue; var dir = parts.slice(0, i + 1).join('/') + '/node_modules'; dirs.push(dir); } return dirs; } }; })(); require.alias = function (from, to) { var path = require.modules.path(); var res = null; try { res = require.resolve(from + '/package.json', '/'); } catch (err) { res = require.resolve(from, '/'); } var basedir = path.dirname(res); var keys = (Object.keys || function (obj) { var res = []; for (var key in obj) res.push(key); return res; })(require.modules); for (var i = 0; i < keys.length; i++) { var key = keys[i]; if (key.slice(0, basedir.length + 1) === basedir + '/') { var f = key.slice(basedir.length); require.modules[to + f] = require.modules[basedir + f]; } else if (key === basedir) { require.modules[to] = require.modules[basedir]; } } }; (function () { var process = {}; require.define = function (filename, fn) { if (require.modules.__browserify_process) { process = require.modules.__browserify_process(); } var dirname = require._core[filename] ? '' : require.modules.path().dirname(filename) ; var require_ = function (file) { var requiredModule = require(file, dirname); var cached = require.cache[require.resolve(file, dirname)]; if (cached && cached.parent === null) { cached.parent = module_; } return requiredModule; }; require_.resolve = function (name) { return require.resolve(name, dirname); }; require_.modules = require.modules; require_.define = require.define; require_.cache = require.cache; var module_ = { id : filename, filename: filename, exports : {}, loaded : false, parent: null }; require.modules[filename] = function () { require.cache[filename] = module_; fn.call( module_.exports, require_, module_, module_.exports, dirname, filename, process ); module_.loaded = true; return module_.exports; }; }; })(); require.define("path",function(require,module,exports,__dirname,__filename,process){function filter (xs, fn) { var res = []; for (var i = 0; i < xs.length; i++) { if (fn(xs[i], i, xs)) res.push(xs[i]); } return res; } // resolves . and .. elements in a path array with directory names there // must be no slashes, empty elements, or device names (c:\) in the array // (so also no leading and trailing slashes - it does not distinguish // relative and absolute paths) function normalizeArray(parts, allowAboveRoot) { // if the path tries to go above the root, `up` ends up > 0 var up = 0; for (var i = parts.length; i >= 0; i--) { var last = parts[i]; if (last == '.') { parts.splice(i, 1); } else if (last === '..') { parts.splice(i, 1); up++; } else if (up) { parts.splice(i, 1); up--; } } // if the path is allowed to go above the root, restore leading ..s if (allowAboveRoot) { for (; up--; up) { parts.unshift('..'); } } return parts; } // Regex to split a filename into [*, dir, basename, ext] // posix version var splitPathRe = /^(.+\/(?!$)|\/)?((?:.+?)?(\.[^.]*)?)$/; // path.resolve([from ...], to) // posix version exports.resolve = function() { var resolvedPath = '', resolvedAbsolute = false; for (var i = arguments.length; i >= -1 && !resolvedAbsolute; i--) { var path = (i >= 0) ? arguments[i] : process.cwd(); // Skip empty and invalid entries if (typeof path !== 'string' || !path) { continue; } resolvedPath = path + '/' + resolvedPath; resolvedAbsolute = path.charAt(0) === '/'; } // At this point the path should be resolved to a full absolute path, but // handle relative paths to be safe (might happen when process.cwd() fails) // Normalize the path resolvedPath = normalizeArray(filter(resolvedPath.split('/'), function(p) { return !!p; }), !resolvedAbsolute).join('/'); return ((resolvedAbsolute ? '/' : '') + resolvedPath) || '.'; }; // path.normalize(path) // posix version exports.normalize = function(path) { var isAbsolute = path.charAt(0) === '/', trailingSlash = path.slice(-1) === '/'; // Normalize the path path = normalizeArray(filter(path.split('/'), function(p) { return !!p; }), !isAbsolute).join('/'); if (!path && !isAbsolute) { path = '.'; } if (path && trailingSlash) { path += '/'; } return (isAbsolute ? '/' : '') + path; }; // posix version exports.join = function() { var paths = Array.prototype.slice.call(arguments, 0); return exports.normalize(filter(paths, function(p, index) { return p && typeof p === 'string'; }).join('/')); }; exports.dirname = function(path) { var dir = splitPathRe.exec(path)[1] || ''; var isWindows = false; if (!dir) { // No dirname return '.'; } else if (dir.length === 1 || (isWindows && dir.length <= 3 && dir.charAt(1) === ':')) { // It is just a slash or a drive letter with a slash return dir; } else { // It is a full dirname, strip trailing slash return dir.substring(0, dir.length - 1); } }; exports.basename = function(path, ext) { var f = splitPathRe.exec(path)[2] || ''; // TODO: make this comparison case-insensitive on windows? if (ext && f.substr(-1 * ext.length) === ext) { f = f.substr(0, f.length - ext.length); } return f; }; exports.extname = function(path) { return splitPathRe.exec(path)[3] || ''; }; }); require.define("__browserify_process",function(require,module,exports,__dirname,__filename,process){var process = module.exports = {}; process.nextTick = (function () { var queue = []; var canPost = typeof window !== 'undefined' && window.postMessage && window.addEventListener ; if (canPost) { window.addEventListener('message', function (ev) { if (ev.source === window && ev.data === 'browserify-tick') { ev.stopPropagation(); if (queue.length > 0) { var fn = queue.shift(); fn(); } } }, true); } return function (fn) { if (canPost) { queue.push(fn); window.postMessage('browserify-tick', '*'); } else setTimeout(fn, 0); }; })(); process.title = 'browser'; process.browser = true; process.env = {}; process.argv = []; process.binding = function (name) { if (name === 'evals') return (require)('vm') else throw new Error('No such module. (Possibly not yet loaded)') }; (function () { var cwd = '/'; var path; process.cwd = function () { return cwd }; process.chdir = function (dir) { if (!path) path = require('path'); cwd = path.resolve(dir, cwd); }; })(); }); require.define("vm",function(require,module,exports,__dirname,__filename,process){module.exports = require("vm-browserify")}); require.define("/node_modules/vm-browserify/package.json",function(require,module,exports,__dirname,__filename,process){module.exports = {"main":"index.js"}}); require.define("/node_modules/vm-browserify/index.js",function(require,module,exports,__dirname,__filename,process){var Object_keys = function (obj) { if (Object.keys) return Object.keys(obj) else { var res = []; for (var key in obj) res.push(key) return res; } }; var forEach = function (xs, fn) { if (xs.forEach) return xs.forEach(fn) else for (var i = 0; i < xs.length; i++) { fn(xs[i], i, xs); } }; var Script = exports.Script = function NodeScript (code) { if (!(this instanceof Script)) return new Script(code); this.code = code; }; Script.prototype.runInNewContext = function (context) { if (!context) context = {}; var iframe = document.createElement('iframe'); if (!iframe.style) iframe.style = {}; iframe.style.display = 'none'; document.body.appendChild(iframe); var win = iframe.contentWindow; forEach(Object_keys(context), function (key) { win[key] = context[key]; }); if (!win.eval && win.execScript) { // win.eval() magically appears when this is called in IE: win.execScript('null'); } var res = win.eval(this.code); forEach(Object_keys(win), function (key) { context[key] = win[key]; }); document.body.removeChild(iframe); return res; }; Script.prototype.runInThisContext = function () { return eval(this.code); // maybe... }; Script.prototype.runInContext = function (context) { // seems to be just runInNewContext on magical context objects which are // otherwise indistinguishable from objects except plain old objects // for the parameter segfaults node return this.runInNewContext(context); }; forEach(Object_keys(Script.prototype), function (name) { exports[name] = Script[name] = function (code) { var s = Script(code); return s[name].apply(s, [].slice.call(arguments, 1)); }; }); exports.createScript = function (code) { return exports.Script(code); }; exports.createContext = Script.createContext = function (context) { // not really sure what this one does // seems to just make a shallow copy var copy = {}; if(typeof context === 'object') { forEach(Object_keys(context), function (key) { copy[key] = context[key]; }); } return copy; }; }); require.define("events",function(require,module,exports,__dirname,__filename,process){if (!process.EventEmitter) process.EventEmitter = function () {}; var EventEmitter = exports.EventEmitter = process.EventEmitter; var isArray = typeof Array.isArray === 'function' ? Array.isArray : function (xs) { return Object.prototype.toString.call(xs) === '[object Array]' } ; // By default EventEmitters will print a warning if more than // 10 listeners are added to it. This is a useful default which // helps finding memory leaks. // // Obviously not all Emitters should be limited to 10. This function allows // that to be increased. Set to zero for unlimited. var defaultMaxListeners = 10; EventEmitter.prototype.setMaxListeners = function(n) { if (!this._events) this._events = {}; this._events.maxListeners = n; }; EventEmitter.prototype.emit = function(type) { // If there is no 'error' event listener then throw. if (type === 'error') { if (!this._events || !this._events.error || (isArray(this._events.error) && !this._events.error.length)) { if (arguments[1] instanceof Error) { throw arguments[1]; // Unhandled 'error' event } else { throw new Error("Uncaught, unspecified 'error' event."); } return false; } } if (!this._events) return false; var handler = this._events[type]; if (!handler) return false; if (typeof handler == 'function') { switch (arguments.length) { // fast cases case 1: handler.call(this); break; case 2: handler.call(this, arguments[1]); break; case 3: handler.call(this, arguments[1], arguments[2]); break; // slower default: var args = Array.prototype.slice.call(arguments, 1); handler.apply(this, args); } return true; } else if (isArray(handler)) { var args = Array.prototype.slice.call(arguments, 1); var listeners = handler.slice(); for (var i = 0, l = listeners.length; i < l; i++) { listeners[i].apply(this, args); } return true; } else { return false; } }; // EventEmitter is defined in src/node_events.cc // EventEmitter.prototype.emit() is also defined there. EventEmitter.prototype.addListener = function(type, listener) { if ('function' !== typeof listener) { throw new Error('addListener only takes instances of Function'); } if (!this._events) this._events = {}; // To avoid recursion in the case that type == "newListeners"! Before // adding it to the listeners, first emit "newListeners". this.emit('newListener', type, listener); if (!this._events[type]) { // Optimize the case of one listener. Don't need the extra array object. this._events[type] = listener; } else if (isArray(this._events[type])) { // Check for listener leak if (!this._events[type].warned) { var m; if (this._events.maxListeners !== undefined) { m = this._events.maxListeners; } else { m = defaultMaxListeners; } if (m && m > 0 && this._events[type].length > m) { this._events[type].warned = true; console.error('(node) warning: possible EventEmitter memory ' + 'leak detected. %d listeners added. ' + 'Use emitter.setMaxListeners() to increase limit.', this._events[type].length); console.trace(); } } // If we've already got an array, just append. this._events[type].push(listener); } else { // Adding the second element, need to change to array. this._events[type] = [this._events[type], listener]; } return this; }; EventEmitter.prototype.on = EventEmitter.prototype.addListener; EventEmitter.prototype.once = function(type, listener) { var self = this; self.on(type, function g() { self.removeListener(type, g); listener.apply(this, arguments); }); return this; }; EventEmitter.prototype.removeListener = function(type, listener) { if ('function' !== typeof listener) { throw new Error('removeListener only takes instances of Function'); } // does not use listeners(), so no side effect of creating _events[type] if (!this._events || !this._events[type]) return this; var list = this._events[type]; if (isArray(list)) { var i = list.indexOf(listener); if (i < 0) return this; list.splice(i, 1); if (list.length == 0) delete this._events[type]; } else if (this._events[type] === listener) { delete this._events[type]; } return this; }; EventEmitter.prototype.removeAllListeners = function(type) { // does not use listeners(), so no side effect of creating _events[type] if (type && this._events && this._events[type]) this._events[type] = null; return this; }; EventEmitter.prototype.listeners = function(type) { if (!this._events) this._events = {}; if (!this._events[type]) this._events[type] = []; if (!isArray(this._events[type])) { this._events[type] = [this._events[type]]; } return this._events[type]; }; }); require.define("/package.json",function(require,module,exports,__dirname,__filename,process){module.exports = {"main":"index.js"}}); require.define("/index.js",function(require,module,exports,__dirname,__filename,process){var lib = require('./lib'); var legend = require('./lib/legend'); var Interaction = require('./lib/interaction'); var hat = require('hat'); var rack = hat.rack(); var series = function() { var args = [].slice.call(arguments,0); for (var i = 0;i < args.length; i++) { var source = args[i]; var id = rack(); source.id = id; this.buffer[id] = document.createElement('canvas'); this.bufferctx[id] = this.buffer[id].getContext('2d'); this.sources.push(source); } }; var to = function(el) { // wrap canvas in a div, set this.canvas and this.ctx this.wrappingDivId = "_".concat(rack()).slice(0,10); lib.setCanvas(el,this) this.sources.forEach(lib.setSource.bind(this)); this.sources.forEach(function(source) { var that = this; source.on('data',function(data) { that.currentdata = data; }); },this); // this.interaction refers to the element created during new Chart $(this.interaction).css('position','absolute'); this.interaction.width = el.width; this.interaction.height = el.height; $(el).before(this.interaction); // wrappingDivId happens during setcanvas (TODO : correct for ref transparency) var interaction = new Interaction({ctx:this.interactionctx,canvas:this.interaction,sources:this.sources,color:this.color,wrappingDivId:this.wrappingDivId}); lib.setInteraction(interaction); $('#'.concat(this.wrappingDivId)).mousemove(interaction.mousemove); $('#'.concat(this.wrappingDivId)).mouseout(interaction.stop); }; var legendfn = function(el) { this.legend_el = el; legendfn.clear = lib.legendClear.bind({legend_el:this.legend_el}) }; var inspect = function() { return this.currentdata; }; var chart = function() { this.buffer = {}; this.bufferctx = {}; this.currentdata = undefined; this.sources = []; this.to = to; this.series = series; this.legend = legendfn; this.inspect = inspect; this.legendobj = new legend; this.interaction = document.createElement('canvas'); this.interactionctx = this.interaction.getContext('2d'); this.bgcolor = undefined; this.color = {grid:'#c9d6de',bg:'#FFF',xlabel:'#000',xline:'#000',ylabel:'#000',yline:'#000',interactionline:'#000',line:undefined}; this.rendermode = "line"; // linefill, line, bar this.custom = {boundaries : {left:undefined,right:undefined}, cropFn : undefined}; this.pause = false; }; exports = module.exports = chart; }); require.define("/lib/index.js",function(require,module,exports,__dirname,__filename,process){var util = require('./util'); var Hash = require('hashish'); var interaction = undefined; var config = { padding : { left : 10, top : 20, bottom : 30 }, axispadding : { left : 50, // yaxis bottom : 20 // xaxis } }; exports.displayConfig = function(params) { if (params !== undefined) { Hash(config).update(params); } }; exports.setInteraction = function(obj) { interaction = obj; interaction.config = config; } exports.setCanvas = function(el,that) { that.canvas = el; // transfer inline style to wrapping div var style = $(el).attr('style'); var wrappingDiv = document.createElement('div'); $(wrappingDiv).attr('style',style); $(el).removeAttr('style'); wrappingDiv.id = that.wrappingDivId; wrappingDiv.height = that.canvas.height; $(that.canvas).wrap(wrappingDiv); that.ctx = el.getContext('2d'); that.ctx.fillStyle = that.color.bg; that.ctx.fillRect(0,0,that.canvas.width,that.canvas.height); }; exports.legendClear = function() { legend.clear(this.legend_el); }; exports.setSource = function(source) { var id = source.id; this.buffer[id].width = this.canvas.width; this.buffer[id].height = this.canvas.height; $(this.buffer[id]).css('position','absolute'); $(this.canvas).before(this.buffer[id]); var onDataGraph = function(data,flags) { // timestamp data.date = new Date().getTime(); // actual timestamp if ((source.dataset === undefined) || (flags && (flags.multiple == true) && (flags.clear && flags.clear == true))) { source.dataset = []; } source.dataset.push(data); if (this.pause === true) return var windowsize = source.windowsize || data.windowsize || 10; var datatodisplay = (this.custom.cropFn) ? this.custom.cropFn(source.dataset,windowsize,this.custom.boundaries) : util.cropData(source.dataset,windowsize); var startx = util.getStartX(datatodisplay.length,windowsize,this.canvas.width); var spacing = util.getSpacing(windowsize,this.canvas.width); var yaxises = this.legendobj.update(datatodisplay,this.color.line); if (this.legend_el !== undefined) this.legendobj.updateHTML({el:this.legend_el}); this.ctx.fillStyle = this.color.bg; this.ctx.fillRect(0,0,this.canvas.width,this.canvas.height); if (flags && flags.multiple && (flags.multiple === true)) { Hash(yaxises).forEach(function(axis,key) { axis.range = util.rangeY(datatodisplay,key); }); util.drawYaxisMultiple(this.canvas,this.ctx,yaxises,config); // util.drawHorizontalGrid(this.canvas.width,this.canvas.height,this.ctx); // util.drawVerticalGrid(datatodisplay,this.ctx,spacing,startx,this.canvas.height); util.draw_multiple({startx:startx,datatodisplay:datatodisplay,spacing:spacing,buffer:this.buffer[id],bufferctx:this.bufferctx[id],yaxises:yaxises}); } else { var range = util.filterDynamicRangeY(datatodisplay,yaxises); // util.drawHorizontalGrid(this.canvas.width,this.canvas.height,this.ctx); util.drawXaxis({datatodisplay:datatodisplay,ctx:this.ctx,spacing:spacing,startx:startx,height:this.canvas.height,width:this.canvas.width,config:config,gridcolor:this.color.grid,xlabel:this.color.xlabel,xline:this.color.xline,doVertGrid:true}); util.draw({startx:startx,datatodisplay:datatodisplay,spacing:spacing,buffer:this.buffer[id],bufferctx:this.bufferctx[id],yaxises:yaxises,config:config,rendermode:source.rendermode || this.rendermode || "line", range:range}); util.clip({ctx:this.bufferctx[id],config:config,height:this.buffer[id].height,type:'clear',clipcolor:this.color.bg}); util.clip({ctx:this.ctx,config:config,height:this.canvas.height,type:'fill',clipcolor:this.color.bg}); util.drawYaxis({canvas:this.canvas,ctx:this.ctx,range:range,config:config,yline:this.color.yline,ylabel:this.color.ylabel}); source.displayData = util.getDisplayPoints({startx:startx,datatodisplay:datatodisplay,spacing:spacing,height:this.buffer[id].height,yaxises:yaxises,config:config,range:range}); } if (interaction !== undefined) { interaction.redraw(); } }; source.on('data',onDataGraph.bind(this)); }; }); require.define("/lib/util.js",function(require,module,exports,__dirname,__filename,process){var Hash = require('hashish'); var mr = require('mrcolor'); var getSpacing = function(windowsize,canvaswidth) { return Math.floor(canvaswidth / (windowsize-1)); } exports.getSpacing = getSpacing; exports.getStartX = function(length,windowsize,canvaswidth) { var x = undefined; var spacing = getSpacing(windowsize,canvaswidth); if (length <= windowsize) { x = canvaswidth - (spacing * (length-1)); } else x = 0; return x; }; exports.cropData = function(list,windowsize) { if (list.length < windowsize) return list else return list.slice(list.length - windowsize) }; var colorToString = function(colorobj,alpha) { var color = colorobj.rgb(); if (alpha !== undefined) return 'rgba('+color[0]+','+color[1]+','+color[2]+','+alpha+')'; else return 'rgb('+color[0]+','+color[1]+','+color[2]+')'; }; exports.colorToString = colorToString; var drawDot = function(params) { params.ctx.beginPath(); params.ctx.strokeStyle = colorToString(params.color); params.ctx.arc(params.x, params.y, params.radius, 0, Math.PI*2, false); params.ctx.stroke(); }; exports.drawDot = drawDot; exports.drawLine = function(params) { params.ctx.beginPath(); params.ctx.arc(params.x, params.y, params.radius, 0, Math.PI*2, false); params.ctx.strokeStyle = params.color; params.ctx.stroke(); }; exports.drawHorizontalGrid = function(width,height,ctx,color){ var heightchunks = Math.floor(height / 10); for (var i = 0; i < heightchunks; i++) { ctx.strokeStyle = color; ctx.beginPath(); ctx.moveTo(0,i*heightchunks); ctx.lineTo(width,i*heightchunks); ctx.stroke(); } } var getDateString = function(ms) { var date = new Date(ms); var pad = function(str) { if (str.length == 1) return '0'.concat(str) if (str.length === 0) return '00' else return str }; var hours = date.getHours() % 12; if (hours === 0) hours = '12'; var seconds = pad(date.getSeconds()); var minutes = pad(date.getMinutes()); var meridian = date.getHours() >= 12 ? 'pm' : 'am'; return hours +':'.concat(minutes) + ':'.concat(seconds) + meridian; }; // if specialkey is defined, then we only look at members of list are specialkey // i.e. list = [{foo:3,bar:9},{foo:4,bar:19}] rangeY(list,'foo'), gets range for just foo. exports.rangeY = function(list,specialkey) { // % to top pad so the "peak" isn't at the top of the viewport, but we allow some extra space for better visualization // var padding = 0.10; // 0.10 = 10%; var padding = 0; var minY = undefined; var maxY = undefined; for (var i = 0; i < list.length; i++) { Hash(list[i]) .filter(function(val,key) { if (specialkey !== undefined) return (key == specialkey) return (key !== 'date') }) .forEach(function(val,key) { if (minY == undefined) minY = val; if (maxY == undefined) maxY = val; if (val < minY) minY = val; if (val > maxY) maxY = val; }); } maxY = (1 + padding)*maxY; var spread = undefined; if ((minY!== undefined) && (maxY !== undefined)) { spread = maxY - minY; } // shift is the amount any value in the interval needs to be shifted by to fall with the interval [0,spread] var shift = undefined; if ((minY < 0) && (maxY >= 0)) { shift = Math.abs(minY); } if ((minY < 0) && (maxY < 0)) { shift = Math.abs(maxY) + Math.abs(minY); } if (minY > 0) { shift = -minY; } if (minY == 0) shift = 0; return {min:minY,max:maxY,spread:spread,shift:shift} }; var tick = function() { var dash = function(ctx,x,y,offset,value,linecolor,labelcolor) { ctx.fillStyle = labelcolor; ctx.strokeStyle = linecolor; ctx.beginPath() ctx.moveTo(x-offset,y) ctx.lineTo(x+offset,y); ctx.stroke(); ctx.fillText(value.toFixed(2),x-40,y+3); } var large = function(ctx,x,y,value,linecolor,labelcolor) { dash(ctx,x,y,6,value,linecolor,labelcolor); } var small = function(ctx,x,y,value,linecolor,labelcolor) { dash(ctx,x,y,2,value,linecolor,labelcolor); } return { large: large, small: small } }; exports.drawYaxis = function(params) { var canvas = params.canvas; var ctx = params.ctx; var range = params.range; var config = params.config; var yline = params.yline; var ylabel = params.ylabel; var availableHeight = canvas.height - config.padding.top - config.padding.bottom; ctx.strokeStyle = yline; ctx.beginPath(); ctx.moveTo(config.axispadding.left,canvas.height-config.padding.bottom); ctx.lineTo(config.axispadding.left,config.padding.top); ctx.stroke(); var majordivisions = 4; var step = range.spread / majordivisions; for (var i = 0; i <= majordivisions; i++) { var ticky = (availableHeight) - ((i / majordivisions) * availableHeight); ticky += config.padding.top; var value = range.min + (i*step); tick().large(ctx,config.axispadding.left,ticky,value,yline,ylabel); } }; exports.drawYaxisMultiple = function(canvas,ctx,yaxises) { var idx = 0; Hash(yaxises).forEach(function(axis,key) { var x = 5 + (35*idx); ctx.fillStyle = '#FFF'; ctx.font = '10px sans-serif'; ctx.fillText(axis.range.min.toFixed(2),x,canvas.height); ctx.fillText(axis.range.max.toFixed(2),x,10); ctx.strokeStyle = colorToString(axis.color); ctx.beginPath(); ctx.moveTo(x,canvas.height); ctx.lineTo(x,0); ctx.stroke(); var majordivisions = 4; var step = axis.range.spread / majordivisions; for (var i = 0; i < majordivisions; i++) { var ticky = (canvas.height) - ((i / majordivisions) * canvas.height); var value = axis.range.min + (i*step); tick().large(ctx,x,ticky,value); } idx++; }); }; exports.clip = function(params) { var ctx = params.ctx; var height = params.height; var config = params.config; var clipcolor = params.clipcolor; if (params.type == 'clear') ctx.clearRect(0,0,config.axispadding.left,height); if (params.type == 'fill') { ctx.fillStyle = clipcolor; ctx.fillRect(0,0,config.axispadding.left,height); } }; exports.drawXaxis = function(params) { var datatodisplay = params.datatodisplay; var ctx = params.ctx; var spacing = params.spacing; var startx = params.startx; var height = params.height; var width = params.width; var config = params.config; var gridcolor = params.gridcolor; var xlabel = params.xlabel; var xline = params.xline; var doVertGrid = params.doVertGrid || false; // draw x-axis ctx.strokeStyle = params.xline; ctx.beginPath(); ctx.moveTo(0,height - config.padding.bottom); ctx.lineTo(width,height - config.padding.bottom); ctx.stroke(); // draw vertical grid if (doVertGrid === true) { ctx.fillStyle = xlabel; ctx.lineWidth = 1; for (var i = 0; i < datatodisplay.length;i++) { ctx.strokeStyle = gridcolor; ctx.beginPath(); var x = startx+i*spacing; x += 0.5; ctx.moveTo(x,0); ctx.lineTo(x,height); ctx.stroke(); var datestring = getDateString(datatodisplay[i].date); ctx.fillText(datestring,startx+i*spacing,height-5); } } }; var lastsavedparams = {}; exports.getDisplayPoints = function(params) { var datatodisplay = params.datatodisplay; var startx = params.startx; var spacing = params.spacing; var height = params.height; var yaxises = params.yaxises; var range = params.range; var config = params.config; var displayPoints = {}; Hash(yaxises) .filter(function(obj) { return (obj.display && obj.display === true) }) .forEach(function(yaxis,key) { displayPoints[key] = {}; displayPoints[key].yaxis = yaxis; displayPoints[key].list = []; datatodisplay.forEach(function(data,idx) { var yval = 0; var ratio = (data[key] + range.shift) / range.spread; var availableHeight = height - config.padding.top - config.padding.bottom; if (range.spread !== 0) { yval = ratio * availableHeight; } var displayY = height - yval - config.padding.bottom; displayPoints[key].list.push({x:startx+(idx*spacing),y:displayY}); },this); }) ; return displayPoints; }; // filters datatodisplay for dyanmic ranges based on legend select/deselect exports.filterDynamicRangeY = function(datatodisplay,yaxises) { var filtered_list = []; // specifically for dynamic ranges for (var i = 0; i < datatodisplay.length; i++) { var item = Hash(datatodisplay[i]) .filter(function(val,key) { return (key == 'date') || (yaxises[key].display == true) }) .end; filtered_list.push(item); } var range = exports.rangeY(filtered_list); return range; } exports.draw = function (params) { lastsavedparams = params; var datatodisplay = params.datatodisplay; var startx = params.startx; var spacing = params.spacing; var buffer = params.buffer; var bufferctx = params.bufferctx; var yaxises = params.yaxises; var config = params.config; var rendermode = params.rendermode; bufferctx.clearRect(0,0,buffer.width,buffer.height); var range = params.range; Hash(yaxises) .filter(function(obj) { return (obj.display && obj.display === true) }) .forEach(function(yaxis,key) { // draw lines bufferctx.strokeStyle = colorToString(yaxis.color); bufferctx.fillStyle = colorToString(mr.lighten(yaxis.color),0.5); datatodisplay.forEach(function(data,idx) { var yval = 0; var ratio = (data[key] + range.shift) / range.spread; var availableHeight = buffer.height - config.padding.top - config.padding.bottom; if (range.spread !== 0) { yval = ratio * availableHeight; } var displayY = buffer.height - yval - config.padding.bottom; if (rendermode == 'line' || rendermode == 'linefill') { if (idx === 0) { bufferctx.beginPath(); bufferctx.moveTo(startx+idx*spacing,displayY); } else { bufferctx.lineTo(startx+(idx*spacing),displayY); } if (idx == (datatodisplay.length -1)) { if (rendermode == 'linefill') { bufferctx.lineTo(startx+(idx*spacing),buffer.height-config.padding.bottom); bufferctx.lineTo(startx,buffer.height-config.padding.bottom); bufferctx.fill(); } bufferctx.stroke(); } } if (rendermode == 'bar') { bufferctx.beginPath(); var centerx = startx + idx*spacing; bufferctx.moveTo(centerx-10,displayY); bufferctx.lineTo(centerx+10,displayY); bufferctx.lineTo(centerx+10,buffer.height-config.padding.bottom); bufferctx.lineTo(centerx-10,buffer.height-config.padding.bottom); bufferctx.lineTo(centerx-10,displayY); bufferctx.stroke(); bufferctx.fill(); } },this); // draw dots datatodisplay.forEach(function(data,idx) { var yval = 0; var ratio = (data[key] + range.shift) / range.spread; var availableHeight = buffer.height - config.padding.top - config.padding.bottom; if (range.spread !== 0) { yval = ratio * availableHeight; } var displayY = buffer.height - yval - config.padding.bottom; drawDot({ x:startx+(idx*spacing), y:displayY, radius:3, ctx:bufferctx, color:yaxis.color }); },this); }) ; }; exports.redraw = function(params) { lastsavedparams.yaxises = params.yaxises; exports.draw(lastsavedparams); }; // completely parallel implementation for multiple y-axises. // diff log // changed functions/variables to _multiple // commented out portions of code are there to indicate the strikethrus from the single axis var lastsavedparams_multiple = {}; exports.draw_multiple = function (params) { lastsavedparams_multiple = params; var datatodisplay = params.datatodisplay; var startx = params.startx; var spacing = params.spacing; var buffer = params.buffer; var bufferctx = params.bufferctx; var yaxises = params.yaxises; bufferctx.clearRect(0,0,buffer.width,buffer.height); // commmented out because range now comes on the axis // var range = exports.rangeY(datatodisplay); Hash(yaxises) .filter(function(obj) { return (obj.display && obj.display === true) }) .forEach(function(yaxis,key) { // draw lines bufferctx.strokeStyle = colorToString(yaxis.color); datatodisplay.forEach(function(data,idx) { var yval = 0; // var ratio = (data[key] + range.shift) / range.spread; var ratio = (data[key] + yaxis.range.shift) / yaxis.range.spread; if (yaxis.range.spread !== 0) { yval = ratio * buffer.height; } if (idx === 0) { bufferctx.beginPath(); bufferctx.moveTo(startx+idx*spacing,buffer.height - yval); } else { bufferctx.lineTo(startx+(idx*spacing),buffer.height - yval); } if (idx == (datatodisplay.length -1)) { bufferctx.stroke(); } },this); // draw dots datatodisplay.forEach(function(data,idx) { var yval = 0; if (yaxis.range.spread !== 0) { yval = ((data[key] + yaxis.range.shift) / yaxis.range.spread) * buffer.height; } drawDot({ x:startx+(idx*spacing), y:buffer.height - yval, radius:3, ctx:bufferctx, color:yaxis.color }); },this); }) ; }; exports.redraw_multiple = function(params) { lastsavedparams_multiple.yaxises = params.yaxises; exports.draw_multiple(lastsavedparams_multiple); }; }); require.define("/node_modules/hashish/package.json",function(require,module,exports,__dirname,__filename,process){module.exports = {"main":"./index.js"}}); require.define("/node_modules/hashish/index.js",function(require,module,exports,__dirname,__filename,process){module.exports = Hash; var Traverse = require('traverse'); function Hash (hash, xs) { if (Array.isArray(hash) && Array.isArray(xs)) { var to = Math.min(hash.length, xs.length); var acc = {}; for (var i = 0; i < to; i++) { acc[hash[i]] = xs[i]; } return Hash(acc); } if (hash === undefined) return Hash({}); var self = { map : function (f) { var acc = { __proto__ : hash.__proto__ }; Object.keys(hash).forEach(function (key) { acc[key] = f.call(self, hash[key], key); }); return Hash(acc); }, forEach : function (f) { Object.keys(hash).forEach(function (key) { f.call(self, hash[key], key); }); return self; }, filter : function (f) { var acc = { __proto__ : hash.__proto__ }; Object.keys(hash).forEach(function (key) { if (f.call(self, hash[key], key)) { acc[key] = hash[key]; } }); return Hash(acc); }, detect : function (f) { for (var key in hash) { if (f.call(self, hash[key], key)) { return hash[key]; } } return undefined; }, reduce : function (f, acc) { var keys = Object.keys(hash); if (acc === undefined) acc = keys.shift(); keys.forEach(function (key) { acc = f.call(self, acc, hash[key], key); }); return acc; }, some : function (f) { for (var key in hash) { if (f.call(self, hash[key], key)) return true; } return false; }, update : function (obj) { if (arguments.length > 1) { self.updateAll([].slice.call(arguments)); } else { Object.keys(obj).forEach(function (key) { hash[key] = obj[key]; }); } return self; }, updateAll : function (xs) { xs.filter(Boolean).forEach(function (x) { self.update(x); }); return self; }, merge : function (obj) { if (arguments.length > 1) { return self.copy.updateAll([].slice.call(arguments)); } else { return self.copy.update(obj); } }, mergeAll : function (xs) { return self.copy.updateAll(xs); }, has : function (key) { // only operates on enumerables return Array.isArray(key) ? key.every(function (k) { return self.has(k) }) : self.keys.indexOf(key.toString()) >= 0; }, valuesAt : function (keys) { return Array.isArray(keys) ? keys.map(function (key) { return hash[key] }) : hash[keys] ; }, tap : function (f) { f.call(self, hash); return self; }, extract : function (keys) { var acc = {}; keys.forEach(function (key) { acc[key] = hash[key]; }); return Hash(acc); }, exclude : function (keys) { return self.filter(function (_, key) { return keys.indexOf(key) < 0 }); }, end : hash, items : hash }; var props = { keys : function () { return Object.keys(hash) }, values : function () { return Object.keys(hash).map(function (key) { return hash[key] }); }, compact : function () { return self.filter(function (x) { return x !== undefined }); }, clone : function () { return Hash(Hash.clone(hash)) }, copy : function () { return Hash(Hash.copy(hash)) }, length : function () { return Object.keys(hash).length }, size : function () { return self.length } }; if (Object.defineProperty) { // es5-shim has an Object.defineProperty but it throws for getters try { for (var key in props) { Object.defineProperty(self, key, { get : props[key] }); } } catch (err) { for (var key in props) { if (key !== 'clone' && key !== 'copy' && key !== 'compact') { // ^ those keys use Hash() so can't call them without // a stack overflow self[key] = props[key](); } } } } else if (self.__defineGetter__) { for (var key in props) { self.__defineGetter__(key, props[key]); } } else { // non-lazy version for browsers that suck >_< for (var key in props) { self[key] = props[key](); } } return self; }; // deep copy Hash.clone = function (ref) { return Traverse.clone(ref); }; // shallow copy Hash.copy = function (ref) { var hash = { __proto__ : ref.__proto__ }; Object.keys(ref).forEach(function (key) { hash[key] = ref[key]; }); return hash; }; Hash.map = function (ref, f) { return Hash(ref).map(f).items; }; Hash.forEach = function (ref, f) { Hash(ref).forEach(f); }; Hash.filter = function (ref, f) { return Hash(ref).filter(f).items; }; Hash.detect = function (ref, f) { return Hash(ref).detect(f); }; Hash.reduce = function (ref, f, acc) { return Hash(ref).reduce(f, acc); }; Hash.some = function (ref, f) { return Hash(ref).some(f); }; Hash.update = function (a /*, b, c, ... */) { var args = Array.prototype.slice.call(arguments, 1); var hash = Hash(a); return hash.update.apply(hash, args).items; }; Hash.merge = function (a /*, b, c, ... */) { var args = Array.prototype.slice.call(arguments, 1); var hash = Hash(a); return hash.merge.apply(hash, args).items; }; Hash.has = function (ref, key) { return Hash(ref).has(key); }; Hash.valuesAt = function (ref, keys) { return Hash(ref).valuesAt(keys); }; Hash.tap = function (ref, f) { return Hash(ref).tap(f).items; }; Hash.extract = function (ref, keys) { return Hash(ref).extract(keys).items; }; Hash.exclude = function (ref, keys) { return Hash(ref).exclude(keys).items; }; Hash.concat = function (xs) { var hash = Hash({}); xs.forEach(function (x) { hash.update(x) }); return hash.items; }; Hash.zip = function (xs, ys) { return Hash(xs, ys).items; }; // .length is already defined for function prototypes Hash.size = function (ref) { return Hash(ref).size; }; Hash.compact = function (ref) { return Hash(ref).compact.items; }; }); require.define("/node_modules/hashish/node_modules/traverse/package.json",function(require,module,exports,__dirname,__filename,process){module.exports = {"main":"index.js"}}); require.define("/node_modules/hashish/node_modules/traverse/index.js",function(require,module,exports,__dirname,__filename,process){var traverse = module.exports = function (obj) { return new Traverse(obj); }; function Traverse (obj) { this.value = obj; } Traverse.prototype.get = function (ps) { var node = this.value; for (var i = 0; i < ps.length; i ++) { var key = ps[i]; if (!Object.hasOwnProperty.call(node, key)) { node = undefined; break; } node = node[key]; } return node; }; Traverse.prototype.has = function (ps) { var node = this.value; for (var i = 0; i < ps.length; i ++) { var key = ps[i]; if (!Object.hasOwnProperty.call(node, key)) { return false; } node = node[key]; } return true; }; Traverse.prototype.set = function (ps, value) { var node = this.value; for (var i = 0; i < ps.length - 1; i ++) { var key = ps[i]; if (!Object.hasOwnProperty.call(node, key)) node[key] = {}; node = node[key]; } node[ps[i]] = value; return value; }; Traverse.prototype.map = function (cb) { return walk(this.value, cb, true); }; Traverse.prototype.forEach = function (cb) { this.value = walk(this.value, cb, false); return this.value; }; Trave