chart
Version:
event based time series charting API
1,559 lines (1,386 loc) • 82.3 kB
JavaScript
(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