web/web_widget_mpld3_chart/static/src/lib/mpld3/mpld3.v0.3.1.dev1.js

1539 lines
56 KiB
JavaScript

!function(d3) {
var mpld3 = {
_mpld3IsLoaded: true,
figures: [],
plugin_map: {}
};
mpld3.version = "0.3.1.dev1";
mpld3.register_plugin = function(name, obj) {
mpld3.plugin_map[name] = obj;
};
mpld3.draw_figure = function(figid, spec) {
var element = document.getElementById(figid);
if (element === null) {
throw figid + " is not a valid id";
}
var fig = new mpld3.Figure(figid, spec);
mpld3.figures.push(fig);
fig.draw();
return fig;
};
mpld3.cloneObj = mpld3_cloneObj;
function mpld3_cloneObj(oldObj) {
var newObj = {};
for (var key in oldObj) {
newObj[key] = oldObj[key];
}
return newObj;
}
mpld3.merge_objects = function(_) {
var output = {};
var obj;
for (var i = 0; i < arguments.length; i++) {
obj = arguments[i];
for (var attr in obj) {
output[attr] = obj[attr];
}
}
return output;
};
mpld3.generate_id = function(N, chars) {
console.warn("mpld3.generate_id is deprecated. " + "Use mpld3.generateId instead.");
return mpld3_generateId(N, chars);
};
mpld3.generateId = mpld3_generateId;
function mpld3_generateId(N, chars) {
N = typeof N !== "undefined" ? N : 10;
chars = typeof chars !== "undefined" ? chars : "abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ0123456789";
var id = chars.charAt(Math.round(Math.random() * (chars.length - 11)));
for (var i = 1; i < N; i++) id += chars.charAt(Math.round(Math.random() * (chars.length - 1)));
return id;
}
mpld3.get_element = function(id, fig) {
var figs_to_search, ax, el;
if (typeof fig === "undefined") {
figs_to_search = mpld3.figures;
} else if (typeof fig.length === "undefined") {
figs_to_search = [ fig ];
} else {
figs_to_search = fig;
}
for (var i = 0; i < figs_to_search.length; i++) {
fig = figs_to_search[i];
if (fig.props.id === id) {
return fig;
}
for (var j = 0; j < fig.axes.length; j++) {
ax = fig.axes[j];
if (ax.props.id === id) {
return ax;
}
for (var k = 0; k < ax.elements.length; k++) {
el = ax.elements[k];
if (el.props.id === id) {
return el;
}
}
}
}
return null;
};
mpld3.insert_css = function(selector, attributes) {
var head = document.head || document.getElementsByTagName("head")[0];
var style = document.createElement("style");
var css = selector + " {";
for (var prop in attributes) {
css += prop + ":" + attributes[prop] + "; ";
}
css += "}";
style.type = "text/css";
if (style.styleSheet) {
style.styleSheet.cssText = css;
} else {
style.appendChild(document.createTextNode(css));
}
head.appendChild(style);
};
mpld3.process_props = function(obj, properties, defaults, required) {
console.warn("mpld3.process_props is deprecated. " + "Plot elements should derive from mpld3.PlotElement");
Element.prototype = Object.create(mpld3_PlotElement.prototype);
Element.prototype.constructor = Element;
Element.prototype.requiredProps = required;
Element.prototype.defaultProps = defaults;
function Element(props) {
mpld3_PlotElement.call(this, null, props);
}
var el = new Element(properties);
return el.props;
};
mpld3.interpolateDates = mpld3_interpolateDates;
function mpld3_interpolateDates(a, b) {
var interp = d3.interpolate([ a[0].valueOf(), a[1].valueOf() ], [ b[0].valueOf(), b[1].valueOf() ]);
return function(t) {
var i = interp(t);
return [ new Date(i[0]), new Date(i[1]) ];
};
}
function isUndefined(x) {
return typeof x === "undefined";
}
function isUndefinedOrNull(x) {
return x == null || isUndefined(x);
}
function getMod(L, i) {
return L.length > 0 ? L[i % L.length] : null;
}
mpld3.path = function() {
return mpld3_path();
};
function mpld3_path(_) {
var x = function(d, i) {
return d[0];
};
var y = function(d, i) {
return d[1];
};
var defined = function(d, i) {
return true;
};
var n_vertices = {
M: 1,
m: 1,
L: 1,
l: 1,
Q: 2,
q: 2,
T: 1,
t: 1,
S: 2,
s: 2,
C: 3,
c: 3,
Z: 0,
z: 0
};
function path(vertices, pathcodes) {
var fx = d3.functor(x), fy = d3.functor(y);
var points = [], segments = [], i_v = 0, i_c = -1, halt = 0, nullpath = false;
if (!pathcodes) {
pathcodes = [ "M" ];
for (var i = 1; i < vertices.length; i++) pathcodes.push("L");
}
while (++i_c < pathcodes.length) {
halt = i_v + n_vertices[pathcodes[i_c]];
points = [];
while (i_v < halt) {
if (defined.call(this, vertices[i_v], i_v)) {
points.push(fx.call(this, vertices[i_v], i_v), fy.call(this, vertices[i_v], i_v));
i_v++;
} else {
points = null;
i_v = halt;
}
}
if (!points) {
nullpath = true;
} else if (nullpath && points.length > 0) {
segments.push("M", points[0], points[1]);
nullpath = false;
} else {
segments.push(pathcodes[i_c]);
segments = segments.concat(points);
}
}
if (i_v != vertices.length) console.warn("Warning: not all vertices used in Path");
return segments.join(" ");
}
path.x = function(_) {
if (!arguments.length) return x;
x = _;
return path;
};
path.y = function(_) {
if (!arguments.length) return y;
y = _;
return path;
};
path.defined = function(_) {
if (!arguments.length) return defined;
defined = _;
return path;
};
path.call = path;
return path;
}
mpld3.multiscale = mpld3_multiscale;
function mpld3_multiscale(_) {
var args = Array.prototype.slice.call(arguments, 0);
var N = args.length;
function scale(x) {
args.forEach(function(mapping) {
x = mapping(x);
});
return x;
}
scale.domain = function(x) {
if (!arguments.length) return args[0].domain();
args[0].domain(x);
return scale;
};
scale.range = function(x) {
if (!arguments.length) return args[N - 1].range();
args[N - 1].range(x);
return scale;
};
scale.step = function(i) {
return args[i];
};
return scale;
}
mpld3.icons = {
reset: "\nWXMAAAsTAAALEwEAmpwYAAAAB3RJTUUH3gIcACMoD/OzIwAAAJhJREFUOMtjYKAx4KDUgNsMDAx7\nyNV8i4GB4T8U76VEM8mGYNNMtCH4NBM0hBjNMIwSsMzQ0MamcDkDA8NmQi6xggpUoikwQbIkHk2u\nE0rLI7vCBknBSyxeRDZAE6qHgQkq+ZeBgYERSfFPAoHNDNUDN4BswIRmKgxwEasP2dlsDAwMYlA/\n/mVgYHiBpkkGKscIDaPfVMmuAGnOTaGsXF0MAAAAAElFTkSuQmCC\n",
move: "\nWXMAAAsTAAALEwEAmpwYAAAAB3RJTUUH3gIcACQMfLHBNQAAANZJREFUOMud07FKA0EQBuAviaKB\nlFr7COJrpAyYRlKn8hECEkFEn8ROCCm0sBMRYgh5EgVFtEhsRjiO27vkBoZd/vn5d3b+XcrjFI9q\nxgXWkc8pUjOB93GMd3zgB9d1unjDSxmhWSHQqOJki+MtOuv/b3ZifUqctIrMxwhHuG1gim4Ma5kR\nWuEkXFgU4B0MW1Ho4TeyjX3s4TDq3zn8ALvZ7q5wX9DqLOHCDA95cFBAnOO1AL/ZdNopgY3fQcqF\nyriMe37hM9w521ZkkvlMo7o/8g7nZYQ/QDctp1nTCf0AAAAASUVORK5CYII=\n",
zoom: "\nWXMAAAsTAAALEwEAmpwYAAAAB3RJTUUH3gMPDiIRPL/2oQAAANBJREFUOMvF0b9KgzEcheHHVnCT\nKoI4uXbtLXgB3oJDJxevw1VwkoJ/NjepQ2/BrZRCx0ILFURQKV2kyOeSQpAmn7WDB0Lg955zEhLy\n2scdXlBggits+4WOQqjAJ3qYR7NGLrwXGU9+sGbEtlIF18FwmuBngZ+nCt6CIacC3Rx8LSl4xzgF\nn0tusBn4UyVhuA/7ZYIv5g+pE3ail25hN/qdmzCfpsJVjKKCZesDBwtzrAqGOMQj6vhCDRsY4ALH\nmOVObltR/xeG/jph6OD2r+Fv5lZBWEhMx58AAAAASUVORK5CYII=\n",
brush: "\nWXMAAEQkAABEJAFAZ8RUAAAAB3RJTUUH3gMCEiQKB9YaAgAAAWtJREFUOMuN0r1qVVEQhuFn700k\nnfEvBq0iNiIiOKXgH4KCaBeIhWARK/EibLwFCwVLjyAWaQzRGG9grC3URkHUBKKgRuWohWvL5pjj\nyTSLxcz7rZlZHyMiItqzFxGTEVF18/UoODNFxDIO4x12dkXqTcBPsCUzD+AK3ndFqhHwEsYz82gn\nN4dbmMRK9R/4KY7jAvbiWmYeHBT5Z4QCP8J1rGAeN3GvU3Mbl/Gq3qCDcxjLzOV+v78fq/iFIxFx\nPyJ2lNJpfBy2g59YzMyzEbEVLzGBJjOriLiBq5gaJrCIU3hcRCbwAtuwjm/Yg/V6I9NgDA1OR8RC\nZq6Vcd7iUwtn5h8fdMBdETGPE+Xe4ExELDRNs4bX2NfCUHe+7UExyfkCP8MhzOA7PuAkvrbwXyNF\nxF3MDqxiqlhXC7SPdaOKiN14g0u4g3H0MvOiTUSNY3iemb0ywmfMdfYyUmAJ2yPiBx6Wr/oy2Oqw\n+A1SupBzAOuE/AAAAABJRU5ErkJggg==\n"
};
mpld3.Grid = mpld3_Grid;
mpld3_Grid.prototype = Object.create(mpld3_PlotElement.prototype);
mpld3_Grid.prototype.constructor = mpld3_Grid;
mpld3_Grid.prototype.requiredProps = [ "xy" ];
mpld3_Grid.prototype.defaultProps = {
color: "gray",
dasharray: "2,2",
alpha: "0.5",
nticks: 10,
gridOn: true,
tickvalues: null,
zorder: 0
};
function mpld3_Grid(ax, prop) {
mpld3_PlotElement.call(this, ax, prop);
this.cssclass = "mpld3-" + this.props.xy + "grid";
if (this.props.xy == "x") {
this.transform = "translate(0," + this.ax.height + ")";
this.position = "bottom";
this.scale = this.ax.xdom;
this.tickSize = -this.ax.height;
} else if (this.props.xy == "y") {
this.transform = "translate(0,0)";
this.position = "left";
this.scale = this.ax.ydom;
this.tickSize = -this.ax.width;
} else {
throw "unrecognized grid xy specifier: should be 'x' or 'y'";
}
}
mpld3_Grid.prototype.draw = function() {
this.grid = d3.svg.axis().scale(this.scale).orient(this.position).ticks(this.props.nticks).tickValues(this.props.tickvalues).tickSize(this.tickSize, 0, 0).tickFormat("");
this.elem = this.ax.axes.append("g").attr("class", this.cssclass).attr("transform", this.transform).call(this.grid);
mpld3.insert_css("div#" + this.ax.fig.figid + " ." + this.cssclass + " .tick", {
stroke: this.props.color,
"stroke-dasharray": this.props.dasharray,
"stroke-opacity": this.props.alpha
});
mpld3.insert_css("div#" + this.ax.fig.figid + " ." + this.cssclass + " path", {
"stroke-width": 0
});
};
mpld3_Grid.prototype.zoomed = function() {
this.elem.call(this.grid);
};
mpld3.Axis = mpld3_Axis;
mpld3_Axis.prototype = Object.create(mpld3_PlotElement.prototype);
mpld3_Axis.prototype.constructor = mpld3_Axis;
mpld3_Axis.prototype.requiredProps = [ "position" ];
mpld3_Axis.prototype.defaultProps = {
nticks: 10,
tickvalues: null,
tickformat: null,
fontsize: "11px",
fontcolor: "black",
axiscolor: "black",
scale: "linear",
grid: {},
zorder: 0
};
function mpld3_Axis(ax, props) {
mpld3_PlotElement.call(this, ax, props);
var trans = {
bottom: [ 0, this.ax.height ],
top: [ 0, 0 ],
left: [ 0, 0 ],
right: [ this.ax.width, 0 ]
};
var xy = {
bottom: "x",
top: "x",
left: "y",
right: "y"
};
this.transform = "translate(" + trans[this.props.position] + ")";
this.props.xy = xy[this.props.position];
this.cssclass = "mpld3-" + this.props.xy + "axis";
this.scale = this.ax[this.props.xy + "dom"];
}
mpld3_Axis.prototype.getGrid = function() {
var gridprop = {
nticks: this.props.nticks,
zorder: this.props.zorder,
tickvalues: this.props.tickvalues,
xy: this.props.xy
};
if (this.props.grid) {
for (var key in this.props.grid) {
gridprop[key] = this.props.grid[key];
}
}
return new mpld3_Grid(this.ax, gridprop);
};
mpld3_Axis.prototype.draw = function() {
var scale = this.props.xy === "x" ? this.parent.props.xscale : this.parent.props.yscale;
if (scale === "date" && this.props.tickvalues) {
var domain = this.props.xy === "x" ? this.parent.x.domain() : this.parent.y.domain();
var range = this.props.xy === "x" ? this.parent.xdom.domain() : this.parent.ydom.domain();
var ordinal_to_js_date = d3.scale.linear().domain(domain).range(range);
this.props.tickvalues = this.props.tickvalues.map(function(value) {
return new Date(ordinal_to_js_date(value));
});
}
var tickformat = mpld3_tickFormat(this.props.tickformat, this.props.tickvalues);
this.axis = d3.svg.axis().scale(this.scale).orient(this.props.position).ticks(this.props.nticks).tickValues(this.props.tickvalues).tickFormat(tickformat);
this.filter_ticks(this.axis.tickValues, this.axis.scale().domain());
this.elem = this.ax.baseaxes.append("g").attr("transform", this.transform).attr("class", this.cssclass).call(this.axis);
mpld3.insert_css("div#" + this.ax.fig.figid + " ." + this.cssclass + " line, " + " ." + this.cssclass + " path", {
"shape-rendering": "crispEdges",
stroke: this.props.axiscolor,
fill: "none"
});
mpld3.insert_css("div#" + this.ax.fig.figid + " ." + this.cssclass + " text", {
"font-family": "sans-serif",
"font-size": this.props.fontsize + "px",
fill: this.props.fontcolor,
stroke: "none"
});
};
function mpld3_tickFormat(tickformat, tickvalues) {
if (tickformat === "" || tickformat === null) {
return tickformat;
} else {
return d3.scale.threshold().domain(tickvalues.slice(1)).range(tickformat);
}
}
mpld3_Axis.prototype.zoomed = function() {
this.filter_ticks(this.axis.tickValues, this.axis.scale().domain());
this.elem.call(this.axis);
};
mpld3_Axis.prototype.filter_ticks = function(tickValues, domain) {
if (this.props.tickvalues != null) {
tickValues(this.props.tickvalues.filter(function(v) {
return v >= domain[0] && v <= domain[1];
}));
}
};
mpld3.Coordinates = mpld3_Coordinates;
function mpld3_Coordinates(trans, ax) {
if (typeof ax === "undefined") {
this.ax = null;
this.fig = null;
if (this.trans !== "display") throw "ax must be defined if transform != 'display'";
} else {
this.ax = ax;
this.fig = ax.fig;
}
this.zoomable = trans === "data";
this.x = this["x_" + trans];
this.y = this["y_" + trans];
if (typeof this.x === "undefined" || typeof this.y === "undefined") throw "unrecognized coordinate code: " + trans;
}
mpld3_Coordinates.prototype.xy = function(d, ix, iy) {
ix = typeof ix === "undefined" ? 0 : ix;
iy = typeof iy === "undefined" ? 1 : iy;
return [ this.x(d[ix]), this.y(d[iy]) ];
};
mpld3_Coordinates.prototype.x_data = function(x) {
return this.ax.x(x);
};
mpld3_Coordinates.prototype.y_data = function(y) {
return this.ax.y(y);
};
mpld3_Coordinates.prototype.x_display = function(x) {
return x;
};
mpld3_Coordinates.prototype.y_display = function(y) {
return y;
};
mpld3_Coordinates.prototype.x_axes = function(x) {
return x * this.ax.width;
};
mpld3_Coordinates.prototype.y_axes = function(y) {
return this.ax.height * (1 - y);
};
mpld3_Coordinates.prototype.x_figure = function(x) {
return x * this.fig.width - this.ax.position[0];
};
mpld3_Coordinates.prototype.y_figure = function(y) {
return (1 - y) * this.fig.height - this.ax.position[1];
};
mpld3.Path = mpld3_Path;
mpld3_Path.prototype = Object.create(mpld3_PlotElement.prototype);
mpld3_Path.prototype.constructor = mpld3_Path;
mpld3_Path.prototype.requiredProps = [ "data" ];
mpld3_Path.prototype.defaultProps = {
xindex: 0,
yindex: 1,
coordinates: "data",
facecolor: "green",
edgecolor: "black",
edgewidth: 1,
dasharray: "none",
pathcodes: null,
offset: null,
offsetcoordinates: "data",
alpha: 1,
zorder: 1
};
function mpld3_Path(ax, props) {
mpld3_PlotElement.call(this, ax, props);
this.data = ax.fig.get_data(this.props.data);
this.pathcodes = this.props.pathcodes;
this.pathcoords = new mpld3_Coordinates(this.props.coordinates, this.ax);
this.offsetcoords = new mpld3_Coordinates(this.props.offsetcoordinates, this.ax);
this.datafunc = mpld3_path();
}
mpld3_Path.prototype.finiteFilter = function(d, i) {
return isFinite(this.pathcoords.x(d[this.props.xindex])) && isFinite(this.pathcoords.y(d[this.props.yindex]));
};
mpld3_Path.prototype.draw = function() {
this.datafunc.defined(this.finiteFilter.bind(this)).x(function(d) {
return this.pathcoords.x(d[this.props.xindex]);
}).y(function(d) {
return this.pathcoords.y(d[this.props.yindex]);
});
this.path = this.ax.axes.append("svg:path").attr("d", this.datafunc(this.data, this.pathcodes)).attr("class", "mpld3-path").style("stroke", this.props.edgecolor).style("stroke-width", this.props.edgewidth).style("stroke-dasharray", this.props.dasharray).style("stroke-opacity", this.props.alpha).style("fill", this.props.facecolor).style("fill-opacity", this.props.alpha).attr("vector-effect", "non-scaling-stroke");
if (this.props.offset !== null) {
var offset = this.offsetcoords.xy(this.props.offset);
this.path.attr("transform", "translate(" + offset + ")");
}
};
mpld3_Path.prototype.elements = function(d) {
return this.path;
};
mpld3_Path.prototype.zoomed = function() {
if (this.pathcoords.zoomable) {
this.path.attr("d", this.datafunc(this.data, this.pathcodes));
}
if (this.props.offset !== null && this.offsetcoords.zoomable) {
var offset = this.offsetcoords.xy(this.props.offset);
this.path.attr("transform", "translate(" + offset + ")");
}
};
mpld3.PathCollection = mpld3_PathCollection;
mpld3_PathCollection.prototype = Object.create(mpld3_PlotElement.prototype);
mpld3_PathCollection.prototype.constructor = mpld3_PathCollection;
mpld3_PathCollection.prototype.requiredProps = [ "paths", "offsets" ];
mpld3_PathCollection.prototype.defaultProps = {
xindex: 0,
yindex: 1,
pathtransforms: [],
pathcoordinates: "display",
offsetcoordinates: "data",
offsetorder: "before",
edgecolors: [ "#000000" ],
edgewidths: [ 1 ],
facecolors: [ "#0000FF" ],
alphas: [ 1 ],
zorder: 2
};
function mpld3_PathCollection(ax, props) {
mpld3_PlotElement.call(this, ax, props);
if (this.props.facecolors == null || this.props.facecolors.length == 0) {
this.props.facecolors = [ "none" ];
}
if (this.props.edgecolors == null || this.props.edgecolors.length == 0) {
this.props.edgecolors = [ "none" ];
}
var offsets = this.ax.fig.get_data(this.props.offsets);
if (offsets === null || offsets.length === 0) offsets = [ null ];
var N = Math.max(this.props.paths.length, offsets.length);
if (offsets.length === N) {
this.offsets = offsets;
} else {
this.offsets = [];
for (var i = 0; i < N; i++) this.offsets.push(getMod(offsets, i));
}
this.pathcoords = new mpld3_Coordinates(this.props.pathcoordinates, this.ax);
this.offsetcoords = new mpld3_Coordinates(this.props.offsetcoordinates, this.ax);
}
mpld3_PathCollection.prototype.transformFunc = function(d, i) {
var t = this.props.pathtransforms;
var transform = t.length == 0 ? "" : d3.transform("matrix(" + getMod(t, i) + ")").toString();
var offset = d === null || typeof d === "undefined" ? "translate(0, 0)" : "translate(" + this.offsetcoords.xy(d, this.props.xindex, this.props.yindex) + ")";
return this.props.offsetorder === "after" ? transform + offset : offset + transform;
};
mpld3_PathCollection.prototype.pathFunc = function(d, i) {
return mpld3_path().x(function(d) {
return this.pathcoords.x(d[0]);
}.bind(this)).y(function(d) {
return this.pathcoords.y(d[1]);
}.bind(this)).apply(this, getMod(this.props.paths, i));
};
mpld3_PathCollection.prototype.styleFunc = function(d, i) {
var styles = {
stroke: getMod(this.props.edgecolors, i),
"stroke-width": getMod(this.props.edgewidths, i),
"stroke-opacity": getMod(this.props.alphas, i),
fill: getMod(this.props.facecolors, i),
"fill-opacity": getMod(this.props.alphas, i)
};
var ret = "";
for (var key in styles) {
ret += key + ":" + styles[key] + ";";
}
return ret;
};
mpld3_PathCollection.prototype.allFinite = function(d) {
if (d instanceof Array) {
return d.length == d.filter(isFinite).length;
} else {
return true;
}
};
mpld3_PathCollection.prototype.draw = function() {
this.group = this.ax.axes.append("svg:g");
this.pathsobj = this.group.selectAll("paths").data(this.offsets.filter(this.allFinite)).enter().append("svg:path").attr("d", this.pathFunc.bind(this)).attr("class", "mpld3-path").attr("transform", this.transformFunc.bind(this)).attr("style", this.styleFunc.bind(this)).attr("vector-effect", "non-scaling-stroke");
};
mpld3_PathCollection.prototype.elements = function(d) {
return this.group.selectAll("path");
};
mpld3_PathCollection.prototype.zoomed = function() {
if (this.props.pathcoordinates === "data") {
this.pathsobj.attr("d", this.pathFunc.bind(this));
}
if (this.props.offsetcoordinates === "data") {
this.pathsobj.attr("transform", this.transformFunc.bind(this));
}
};
mpld3.Line = mpld3_Line;
mpld3_Line.prototype = Object.create(mpld3_Path.prototype);
mpld3_Line.prototype.constructor = mpld3_Line;
mpld3_Line.prototype.requiredProps = [ "data" ];
mpld3_Line.prototype.defaultProps = {
xindex: 0,
yindex: 1,
coordinates: "data",
color: "salmon",
linewidth: 2,
dasharray: "none",
alpha: 1,
zorder: 2,
drawstyle: "none"
};
function mpld3_Line(ax, props) {
mpld3_PlotElement.call(this, ax, props);
var pathProps = this.props;
pathProps.facecolor = "none";
pathProps.edgecolor = pathProps.color;
delete pathProps.color;
pathProps.edgewidth = pathProps.linewidth;
delete pathProps.linewidth;
drawstyle = pathProps.drawstyle;
delete pathProps.drawstyle;
this.defaultProps = mpld3_Path.prototype.defaultProps;
mpld3_Path.call(this, ax, pathProps);
switch (drawstyle) {
case "steps":
case "steps-pre":
this.datafunc = d3.svg.line().interpolate("step-before");
break;
case "steps-post":
this.datafunc = d3.svg.line().interpolate("step-after");
break;
case "steps-mid":
this.datafunc = d3.svg.line().interpolate("step");
break;
default:
this.datafunc = d3.svg.line().interpolate("linear");
}
}
mpld3.Markers = mpld3_Markers;
mpld3_Markers.prototype = Object.create(mpld3_PathCollection.prototype);
mpld3_Markers.prototype.constructor = mpld3_Markers;
mpld3_Markers.prototype.requiredProps = [ "data" ];
mpld3_Markers.prototype.defaultProps = {
xindex: 0,
yindex: 1,
coordinates: "data",
facecolor: "salmon",
edgecolor: "black",
edgewidth: 1,
alpha: 1,
markersize: 6,
markername: "circle",
markerpath: null,
zorder: 3
};
function mpld3_Markers(ax, props) {
mpld3_PlotElement.call(this, ax, props);
if (this.props.markerpath !== null) {
this.marker = this.props.markerpath[0].length == 0 ? null : mpld3.path().call(this.props.markerpath[0], this.props.markerpath[1]);
} else {
this.marker = this.props.markername === null ? null : d3.svg.symbol(this.props.markername).size(Math.pow(this.props.markersize, 2))();
}
var PCprops = {
paths: [ this.props.markerpath ],
offsets: ax.fig.get_data(this.props.data),
xindex: this.props.xindex,
yindex: this.props.yindex,
offsetcoordinates: this.props.coordinates,
edgecolors: [ this.props.edgecolor ],
edgewidths: [ this.props.edgewidth ],
facecolors: [ this.props.facecolor ],
alphas: [ this.props.alpha ],
zorder: this.props.zorder,
id: this.props.id
};
this.requiredProps = mpld3_PathCollection.prototype.requiredProps;
this.defaultProps = mpld3_PathCollection.prototype.defaultProps;
mpld3_PathCollection.call(this, ax, PCprops);
}
mpld3_Markers.prototype.pathFunc = function(d, i) {
return this.marker;
};
mpld3.Image = mpld3_Image;
mpld3_Image.prototype = Object.create(mpld3_PlotElement.prototype);
mpld3_Image.prototype.constructor = mpld3_Image;
mpld3_Image.prototype.requiredProps = [ "data", "extent" ];
mpld3_Image.prototype.defaultProps = {
alpha: 1,
coordinates: "data",
zorder: 1
};
function mpld3_Image(ax, props) {
mpld3_PlotElement.call(this, ax, props);
this.coords = new mpld3_Coordinates(this.props.coordinates, this.ax);
}
mpld3_Image.prototype.draw = function() {
this.image = this.ax.axes.append("svg:image").attr("class", "mpld3-image").attr("xlink:href", "data:image/png;base64," + this.props.data).style({
opacity: this.props.alpha
}).attr("preserveAspectRatio", "none");
this.zoomed();
};
mpld3_Image.prototype.elements = function(d) {
return d3.select(this.image);
};
mpld3_Image.prototype.zoomed = function() {
var extent = this.props.extent;
this.image.attr("x", this.coords.x(extent[0])).attr("y", this.coords.y(extent[3])).attr("width", this.coords.x(extent[1]) - this.coords.x(extent[0])).attr("height", this.coords.y(extent[2]) - this.coords.y(extent[3]));
};
mpld3.Text = mpld3_Text;
mpld3_Text.prototype = Object.create(mpld3_PlotElement.prototype);
mpld3_Text.prototype.constructor = mpld3_Text;
mpld3_Text.prototype.requiredProps = [ "text", "position" ];
mpld3_Text.prototype.defaultProps = {
coordinates: "data",
h_anchor: "start",
v_baseline: "auto",
rotation: 0,
fontsize: 11,
color: "black",
alpha: 1,
zorder: 3
};
function mpld3_Text(ax, props) {
mpld3_PlotElement.call(this, ax, props);
this.text = this.props.text;
this.position = this.props.position;
this.coords = new mpld3_Coordinates(this.props.coordinates, this.ax);
}
mpld3_Text.prototype.draw = function() {
if (this.props.coordinates == "data") {
this.obj = this.ax.axes.append("text");
} else {
this.obj = this.ax.baseaxes.append("text");
}
this.obj.attr("class", "mpld3-text").text(this.text).style("text-anchor", this.props.h_anchor).style("dominant-baseline", this.props.v_baseline).style("font-size", this.props.fontsize).style("fill", this.props.color).style("opacity", this.props.alpha);
this.applyTransform();
};
mpld3_Text.prototype.elements = function(d) {
return d3.select(this.obj);
};
mpld3_Text.prototype.applyTransform = function() {
var pos = this.coords.xy(this.position);
this.obj.attr("x", pos[0]).attr("y", pos[1]);
if (this.props.rotation) this.obj.attr("transform", "rotate(" + this.props.rotation + "," + pos + ")");
};
mpld3_Text.prototype.zoomed = function() {
if (this.coords.zoomable) this.applyTransform();
};
mpld3.Axes = mpld3_Axes;
mpld3_Axes.prototype = Object.create(mpld3_PlotElement.prototype);
mpld3_Axes.prototype.constructor = mpld3_Axes;
mpld3_Axes.prototype.requiredProps = [ "xlim", "ylim" ];
mpld3_Axes.prototype.defaultProps = {
bbox: [ .1, .1, .8, .8 ],
axesbg: "#FFFFFF",
axesbgalpha: 1,
gridOn: false,
xdomain: null,
ydomain: null,
xscale: "linear",
yscale: "linear",
zoomable: true,
axes: [ {
position: "left"
}, {
position: "bottom"
} ],
lines: [],
paths: [],
markers: [],
texts: [],
collections: [],
sharex: [],
sharey: [],
images: []
};
function mpld3_Axes(fig, props) {
mpld3_PlotElement.call(this, fig, props);
this.axnum = this.fig.axes.length;
this.axid = this.fig.figid + "_ax" + (this.axnum + 1);
this.clipid = this.axid + "_clip";
this.props.xdomain = this.props.xdomain || this.props.xlim;
this.props.ydomain = this.props.ydomain || this.props.ylim;
this.sharex = [];
this.sharey = [];
this.elements = [];
var bbox = this.props.bbox;
this.position = [ bbox[0] * this.fig.width, (1 - bbox[1] - bbox[3]) * this.fig.height ];
this.width = bbox[2] * this.fig.width;
this.height = bbox[3] * this.fig.height;
function buildDate(d) {
return new Date(d[0], d[1], d[2], d[3], d[4], d[5]);
}
function setDomain(scale, domain) {
return scale !== "date" ? domain : [ buildDate(domain[0]), buildDate(domain[1]) ];
}
this.props.xdomain = setDomain(this.props.xscale, this.props.xdomain);
this.props.ydomain = setDomain(this.props.yscale, this.props.ydomain);
function build_scale(scale, domain, range) {
var dom = scale === "date" ? d3.time.scale() : scale === "log" ? d3.scale.log() : d3.scale.linear();
return dom.domain(domain).range(range);
}
this.x = this.xdom = build_scale(this.props.xscale, this.props.xdomain, [ 0, this.width ]);
this.y = this.ydom = build_scale(this.props.yscale, this.props.ydomain, [ this.height, 0 ]);
if (this.props.xscale === "date") {
this.x = mpld3.multiscale(d3.scale.linear().domain(this.props.xlim).range(this.props.xdomain.map(Number)), this.xdom);
}
if (this.props.yscale === "date") {
this.y = mpld3.multiscale(d3.scale.linear().domain(this.props.ylim).range(this.props.ydomain.map(Number)), this.ydom);
}
var axes = this.props.axes;
for (var i = 0; i < axes.length; i++) {
var axis = new mpld3.Axis(this, axes[i]);
this.elements.push(axis);
if (this.props.gridOn || axis.props.grid.gridOn) {
this.elements.push(axis.getGrid());
}
}
var paths = this.props.paths;
for (var i = 0; i < paths.length; i++) {
this.elements.push(new mpld3.Path(this, paths[i]));
}
var lines = this.props.lines;
for (var i = 0; i < lines.length; i++) {
this.elements.push(new mpld3.Line(this, lines[i]));
}
var markers = this.props.markers;
for (var i = 0; i < markers.length; i++) {
this.elements.push(new mpld3.Markers(this, markers[i]));
}
var texts = this.props.texts;
for (var i = 0; i < texts.length; i++) {
this.elements.push(new mpld3.Text(this, texts[i]));
}
var collections = this.props.collections;
for (var i = 0; i < collections.length; i++) {
this.elements.push(new mpld3.PathCollection(this, collections[i]));
}
var images = this.props.images;
for (var i = 0; i < images.length; i++) {
this.elements.push(new mpld3.Image(this, images[i]));
}
this.elements.sort(function(a, b) {
return a.props.zorder - b.props.zorder;
});
}
mpld3_Axes.prototype.draw = function() {
for (var i = 0; i < this.props.sharex.length; i++) {
this.sharex.push(mpld3.get_element(this.props.sharex[i]));
}
for (var i = 0; i < this.props.sharey.length; i++) {
this.sharey.push(mpld3.get_element(this.props.sharey[i]));
}
this.zoom = d3.behavior.zoom();
this.zoom.last_t = this.zoom.translate();
this.zoom.last_s = this.zoom.scale();
this.zoom_x = d3.behavior.zoom().x(this.xdom);
this.zoom_y = d3.behavior.zoom().y(this.ydom);
this.baseaxes = this.fig.canvas.append("g").attr("transform", "translate(" + this.position[0] + "," + this.position[1] + ")").attr("width", this.width).attr("height", this.height).attr("class", "mpld3-baseaxes");
this.clip = this.baseaxes.append("svg:clipPath").attr("id", this.clipid).append("svg:rect").attr("x", 0).attr("y", 0).attr("width", this.width).attr("height", this.height);
this.axes = this.baseaxes.append("g").attr("class", "mpld3-axes").attr("clip-path", "url(#" + this.clipid + ")");
this.axesbg = this.axes.append("svg:rect").attr("width", this.width).attr("height", this.height).attr("class", "mpld3-axesbg").style("fill", this.props.axesbg).style("fill-opacity", this.props.axesbgalpha);
for (var i = 0; i < this.elements.length; i++) {
this.elements[i].draw();
}
};
mpld3_Axes.prototype.enable_zoom = function() {
if (this.props.zoomable) {
this.zoom.on("zoom", this.zoomed.bind(this, true));
this.axes.call(this.zoom);
this.axes.style("cursor", "move");
}
};
mpld3_Axes.prototype.disable_zoom = function() {
if (this.props.zoomable) {
this.zoom.on("zoom", null);
this.axes.on(".zoom", null);
this.axes.style("cursor", null);
}
};
mpld3_Axes.prototype.zoomed = function(propagate) {
propagate = typeof propagate == "undefined" ? true : propagate;
if (propagate) {
var dt0 = this.zoom.translate()[0] - this.zoom.last_t[0];
var dt1 = this.zoom.translate()[1] - this.zoom.last_t[1];
var ds = this.zoom.scale() / this.zoom.last_s;
this.zoom_x.translate([ this.zoom_x.translate()[0] + dt0, 0 ]);
this.zoom_x.scale(this.zoom_x.scale() * ds);
this.zoom_y.translate([ 0, this.zoom_y.translate()[1] + dt1 ]);
this.zoom_y.scale(this.zoom_y.scale() * ds);
this.zoom.last_t = this.zoom.translate();
this.zoom.last_s = this.zoom.scale();
this.sharex.forEach(function(ax) {
ax.zoom_x.translate(this.zoom_x.translate()).scale(this.zoom_x.scale());
}.bind(this));
this.sharey.forEach(function(ax) {
ax.zoom_y.translate(this.zoom_y.translate()).scale(this.zoom_y.scale());
}.bind(this));
this.sharex.forEach(function(ax) {
ax.zoomed(false);
});
this.sharey.forEach(function(ax) {
ax.zoomed(false);
});
}
for (var i = 0; i < this.elements.length; i++) {
this.elements[i].zoomed();
}
};
mpld3_Axes.prototype.reset = function(duration, propagate) {
this.set_axlim(this.props.xdomain, this.props.ydomain, duration, propagate);
};
mpld3_Axes.prototype.set_axlim = function(xlim, ylim, duration, propagate) {
xlim = isUndefinedOrNull(xlim) ? this.xdom.domain() : xlim;
ylim = isUndefinedOrNull(ylim) ? this.ydom.domain() : ylim;
duration = isUndefinedOrNull(duration) ? 750 : duration;
propagate = isUndefined(propagate) ? true : propagate;
var interpX = this.props.xscale === "date" ? mpld3.interpolateDates(this.xdom.domain(), xlim) : d3.interpolate(this.xdom.domain(), xlim);
var interpY = this.props.yscale === "date" ? mpld3.interpolateDates(this.ydom.domain(), ylim) : d3.interpolate(this.ydom.domain(), ylim);
var transition = function(t) {
this.zoom_x.x(this.xdom.domain(interpX(t)));
this.zoom_y.y(this.ydom.domain(interpY(t)));
this.zoomed(false);
}.bind(this);
d3.select({}).transition().duration(duration).tween("zoom", function() {
return transition;
});
if (propagate) {
this.sharex.forEach(function(ax) {
ax.set_axlim(xlim, null, duration, false);
});
this.sharey.forEach(function(ax) {
ax.set_axlim(null, ylim, duration, false);
});
}
this.zoom.scale(1).translate([ 0, 0 ]);
this.zoom.last_t = this.zoom.translate();
this.zoom.last_s = this.zoom.scale();
this.zoom_x.scale(1).translate([ 0, 0 ]);
this.zoom_y.scale(1).translate([ 0, 0 ]);
};
mpld3.Toolbar = mpld3_Toolbar;
mpld3_Toolbar.prototype = Object.create(mpld3_PlotElement.prototype);
mpld3_Toolbar.prototype.constructor = mpld3_Toolbar;
mpld3_Toolbar.prototype.defaultProps = {
buttons: [ "reset", "move" ]
};
function mpld3_Toolbar(fig, props) {
mpld3_PlotElement.call(this, fig, props);
this.buttons = [];
this.props.buttons.forEach(this.addButton.bind(this));
}
mpld3_Toolbar.prototype.addButton = function(button) {
this.buttons.push(new button(this));
};
mpld3_Toolbar.prototype.draw = function() {
mpld3.insert_css("div#" + this.fig.figid + " .mpld3-toolbar image", {
cursor: "pointer",
opacity: .2,
display: "inline-block",
margin: "0px"
});
mpld3.insert_css("div#" + this.fig.figid + " .mpld3-toolbar image.active", {
opacity: .4
});
mpld3.insert_css("div#" + this.fig.figid + " .mpld3-toolbar image.pressed", {
opacity: .6
});
function showButtons() {
this.buttonsobj.transition(750).attr("y", 0);
}
function hideButtons() {
this.buttonsobj.transition(750).delay(250).attr("y", 16);
}
this.fig.canvas.on("mouseenter", showButtons.bind(this)).on("mouseleave", hideButtons.bind(this)).on("touchenter", showButtons.bind(this)).on("touchstart", showButtons.bind(this));
this.toolbar = this.fig.canvas.append("svg:svg").attr("width", 16 * this.buttons.length).attr("height", 16).attr("x", 2).attr("y", this.fig.height - 16 - 2).attr("class", "mpld3-toolbar");
this.buttonsobj = this.toolbar.append("svg:g").selectAll("buttons").data(this.buttons).enter().append("svg:image").attr("class", function(d) {
return d.cssclass;
}).attr("xlink:href", function(d) {
return d.icon();
}).attr("width", 16).attr("height", 16).attr("x", function(d, i) {
return i * 16;
}).attr("y", 16).on("click", function(d) {
d.click();
}).on("mouseenter", function() {
d3.select(this).classed({
active: 1
});
}).on("mouseleave", function() {
d3.select(this).classed({
active: 0
});
});
for (var i = 0; i < this.buttons.length; i++) this.buttons[i].onDraw();
};
mpld3_Toolbar.prototype.deactivate_all = function() {
this.buttons.forEach(function(b) {
b.deactivate();
});
};
mpld3_Toolbar.prototype.deactivate_by_action = function(actions) {
function filt(e) {
return actions.indexOf(e) !== -1;
}
if (actions.length > 0) {
this.buttons.forEach(function(button) {
if (button.actions.filter(filt).length > 0) button.deactivate();
});
}
};
mpld3.Button = mpld3_Button;
mpld3_Button.prototype = Object.create(mpld3_PlotElement.prototype);
mpld3_Button.prototype.constructor = mpld3_Button;
function mpld3_Button(toolbar, key) {
mpld3_PlotElement.call(this, toolbar);
this.toolbar = toolbar;
this.fig = this.toolbar.fig;
this.cssclass = "mpld3-" + key + "button";
this.active = false;
}
mpld3_Button.prototype.setState = function(state) {
state ? this.activate() : this.deactivate();
};
mpld3_Button.prototype.click = function() {
this.active ? this.deactivate() : this.activate();
};
mpld3_Button.prototype.activate = function() {
this.toolbar.deactivate_by_action(this.actions);
this.onActivate();
this.active = true;
this.toolbar.toolbar.select("." + this.cssclass).classed({
pressed: true
});
if (!this.sticky) this.deactivate();
};
mpld3_Button.prototype.deactivate = function() {
this.onDeactivate();
this.active = false;
this.toolbar.toolbar.select("." + this.cssclass).classed({
pressed: false
});
};
mpld3_Button.prototype.sticky = false;
mpld3_Button.prototype.actions = [];
mpld3_Button.prototype.icon = function() {
return "";
};
mpld3_Button.prototype.onActivate = function() {};
mpld3_Button.prototype.onDeactivate = function() {};
mpld3_Button.prototype.onDraw = function() {};
mpld3.ButtonFactory = function(members) {
if (typeof members.buttonID !== "string") {
throw "ButtonFactory: buttonID must be present and be a string";
}
function B(toolbar) {
mpld3_Button.call(this, toolbar, this.buttonID);
}
B.prototype = Object.create(mpld3_Button.prototype);
B.prototype.constructor = B;
for (var key in members) {
B.prototype[key] = members[key];
}
return B;
};
mpld3.Plugin = mpld3_Plugin;
mpld3_Plugin.prototype = Object.create(mpld3_PlotElement.prototype);
mpld3_Plugin.prototype.constructor = mpld3_Plugin;
mpld3_Plugin.prototype.requiredProps = [];
mpld3_Plugin.prototype.defaultProps = {};
function mpld3_Plugin(fig, props) {
mpld3_PlotElement.call(this, fig, props);
}
mpld3_Plugin.prototype.draw = function() {};
mpld3.ResetPlugin = mpld3_ResetPlugin;
mpld3.register_plugin("reset", mpld3_ResetPlugin);
mpld3_ResetPlugin.prototype = Object.create(mpld3_Plugin.prototype);
mpld3_ResetPlugin.prototype.constructor = mpld3_ResetPlugin;
mpld3_ResetPlugin.prototype.requiredProps = [];
mpld3_ResetPlugin.prototype.defaultProps = {};
function mpld3_ResetPlugin(fig, props) {
mpld3_Plugin.call(this, fig, props);
var ResetButton = mpld3.ButtonFactory({
buttonID: "reset",
sticky: false,
onActivate: function() {
this.toolbar.fig.reset();
},
icon: function() {
return mpld3.icons["reset"];
}
});
this.fig.buttons.push(ResetButton);
}
mpld3.ZoomPlugin = mpld3_ZoomPlugin;
mpld3.register_plugin("zoom", mpld3_ZoomPlugin);
mpld3_ZoomPlugin.prototype = Object.create(mpld3_Plugin.prototype);
mpld3_ZoomPlugin.prototype.constructor = mpld3_ZoomPlugin;
mpld3_ZoomPlugin.prototype.requiredProps = [];
mpld3_ZoomPlugin.prototype.defaultProps = {
button: true,
enabled: null
};
function mpld3_ZoomPlugin(fig, props) {
mpld3_Plugin.call(this, fig, props);
if (this.props.enabled === null) {
this.props.enabled = !this.props.button;
}
var enabled = this.props.enabled;
if (this.props.button) {
var ZoomButton = mpld3.ButtonFactory({
buttonID: "zoom",
sticky: true,
actions: [ "scroll", "drag" ],
onActivate: this.activate.bind(this),
onDeactivate: this.deactivate.bind(this),
onDraw: function() {
this.setState(enabled);
},
icon: function() {
return mpld3.icons["move"];
}
});
this.fig.buttons.push(ZoomButton);
}
}
mpld3_ZoomPlugin.prototype.activate = function() {
this.fig.enable_zoom();
};
mpld3_ZoomPlugin.prototype.deactivate = function() {
this.fig.disable_zoom();
};
mpld3_ZoomPlugin.prototype.draw = function() {
if (this.props.enabled) this.fig.enable_zoom(); else this.fig.disable_zoom();
};
mpld3.BoxZoomPlugin = mpld3_BoxZoomPlugin;
mpld3.register_plugin("boxzoom", mpld3_BoxZoomPlugin);
mpld3_BoxZoomPlugin.prototype = Object.create(mpld3_Plugin.prototype);
mpld3_BoxZoomPlugin.prototype.constructor = mpld3_BoxZoomPlugin;
mpld3_BoxZoomPlugin.prototype.requiredProps = [];
mpld3_BoxZoomPlugin.prototype.defaultProps = {
button: true,
enabled: null
};
function mpld3_BoxZoomPlugin(fig, props) {
mpld3_Plugin.call(this, fig, props);
if (this.props.enabled === null) {
this.props.enabled = !this.props.button;
}
var enabled = this.props.enabled;
if (this.props.button) {
var BoxZoomButton = mpld3.ButtonFactory({
buttonID: "boxzoom",
sticky: true,
actions: [ "drag" ],
onActivate: this.activate.bind(this),
onDeactivate: this.deactivate.bind(this),
onDraw: function() {
this.setState(enabled);
},
icon: function() {
return mpld3.icons["zoom"];
}
});
this.fig.buttons.push(BoxZoomButton);
}
this.extentClass = "boxzoombrush";
}
mpld3_BoxZoomPlugin.prototype.activate = function() {
if (this.enable) this.enable();
};
mpld3_BoxZoomPlugin.prototype.deactivate = function() {
if (this.disable) this.disable();
};
mpld3_BoxZoomPlugin.prototype.draw = function() {
mpld3.insert_css("#" + this.fig.figid + " rect.extent." + this.extentClass, {
fill: "#fff",
"fill-opacity": 0,
stroke: "#999"
});
var brush = this.fig.getBrush();
this.enable = function() {
this.fig.showBrush(this.extentClass);
brush.on("brushend", brushend.bind(this));
this.enabled = true;
};
this.disable = function() {
this.fig.hideBrush(this.extentClass);
this.enabled = false;
};
this.toggle = function() {
this.enabled ? this.disable() : this.enable();
};
function brushend(d) {
if (this.enabled) {
var extent = brush.extent();
if (!brush.empty()) {
d.set_axlim([ extent[0][0], extent[1][0] ], [ extent[0][1], extent[1][1] ]);
}
}
d.axes.call(brush.clear());
}
this.disable();
};
mpld3.TooltipPlugin = mpld3_TooltipPlugin;
mpld3.register_plugin("tooltip", mpld3_TooltipPlugin);
mpld3_TooltipPlugin.prototype = Object.create(mpld3_Plugin.prototype);
mpld3_TooltipPlugin.prototype.constructor = mpld3_TooltipPlugin;
mpld3_TooltipPlugin.prototype.requiredProps = [ "id" ];
mpld3_TooltipPlugin.prototype.defaultProps = {
labels: null,
hoffset: 0,
voffset: 10,
location: "mouse"
};
function mpld3_TooltipPlugin(fig, props) {
mpld3_Plugin.call(this, fig, props);
}
mpld3_TooltipPlugin.prototype.draw = function() {
var obj = mpld3.get_element(this.props.id, this.fig);
var labels = this.props.labels;
var loc = this.props.location;
this.tooltip = this.fig.canvas.append("text").attr("class", "mpld3-tooltip-text").attr("x", 0).attr("y", 0).text("").style("visibility", "hidden");
if (loc == "bottom left" || loc == "top left") {
this.x = obj.ax.position[0] + 5 + this.props.hoffset;
this.tooltip.style("text-anchor", "beginning");
} else if (loc == "bottom right" || loc == "top right") {
this.x = obj.ax.position[0] + obj.ax.width - 5 + this.props.hoffset;
this.tooltip.style("text-anchor", "end");
} else {
this.tooltip.style("text-anchor", "middle");
}
if (loc == "bottom left" || loc == "bottom right") {
this.y = obj.ax.position[1] + obj.ax.height - 5 + this.props.voffset;
} else if (loc == "top left" || loc == "top right") {
this.y = obj.ax.position[1] + 5 + this.props.voffset;
}
function mouseover(d, i) {
this.tooltip.style("visibility", "visible").text(labels === null ? "(" + d + ")" : getMod(labels, i));
}
function mousemove(d, i) {
if (loc === "mouse") {
var pos = d3.mouse(this.fig.canvas.node());
this.x = pos[0] + this.props.hoffset;
this.y = pos[1] - this.props.voffset;
}
this.tooltip.attr("x", this.x).attr("y", this.y);
}
function mouseout(d, i) {
this.tooltip.style("visibility", "hidden");
}
obj.elements().on("mouseover", mouseover.bind(this)).on("mousemove", mousemove.bind(this)).on("mouseout", mouseout.bind(this));
};
mpld3.LinkedBrushPlugin = mpld3_LinkedBrushPlugin;
mpld3.register_plugin("linkedbrush", mpld3_LinkedBrushPlugin);
mpld3_LinkedBrushPlugin.prototype = Object.create(mpld3.Plugin.prototype);
mpld3_LinkedBrushPlugin.prototype.constructor = mpld3_LinkedBrushPlugin;
mpld3_LinkedBrushPlugin.prototype.requiredProps = [ "id" ];
mpld3_LinkedBrushPlugin.prototype.defaultProps = {
button: true,
enabled: null
};
function mpld3_LinkedBrushPlugin(fig, props) {
mpld3.Plugin.call(this, fig, props);
if (this.props.enabled === null) {
this.props.enabled = !this.props.button;
}
var enabled = this.props.enabled;
if (this.props.button) {
var BrushButton = mpld3.ButtonFactory({
buttonID: "linkedbrush",
sticky: true,
actions: [ "drag" ],
onActivate: this.activate.bind(this),
onDeactivate: this.deactivate.bind(this),
onDraw: function() {
this.setState(enabled);
},
icon: function() {
return mpld3.icons["brush"];
}
});
this.fig.buttons.push(BrushButton);
}
this.extentClass = "linkedbrush";
}
mpld3_LinkedBrushPlugin.prototype.activate = function() {
if (this.enable) this.enable();
};
mpld3_LinkedBrushPlugin.prototype.deactivate = function() {
if (this.disable) this.disable();
};
mpld3_LinkedBrushPlugin.prototype.draw = function() {
var obj = mpld3.get_element(this.props.id);
if (obj === null) {
throw "LinkedBrush: no object with id='" + this.props.id + "' was found";
}
var fig = this.fig;
if (!("offsets" in obj.props)) {
throw "Plot object with id='" + this.props.id + "' is not a scatter plot";
}
var dataKey = "offsets" in obj.props ? "offsets" : "data";
mpld3.insert_css("#" + fig.figid + " rect.extent." + this.extentClass, {
fill: "#000",
"fill-opacity": .125,
stroke: "#fff"
});
mpld3.insert_css("#" + fig.figid + " path.mpld3-hidden", {
stroke: "#ccc !important",
fill: "#ccc !important"
});
var dataClass = "mpld3data-" + obj.props[dataKey];
var brush = fig.getBrush();
var dataByAx = [];
fig.axes.forEach(function(ax) {
var axData = [];
ax.elements.forEach(function(el) {
if (el.props[dataKey] === obj.props[dataKey]) {
el.group.classed(dataClass, true);
axData.push(el);
}
});
dataByAx.push(axData);
});
var allData = [];
var dataToBrush = fig.canvas.selectAll("." + dataClass);
var currentAxes;
function brushstart(d) {
if (currentAxes != this) {
d3.select(currentAxes).call(brush.clear());
currentAxes = this;
brush.x(d.xdom).y(d.ydom);
}
}
function brushmove(d) {
var data = dataByAx[d.axnum];
if (data.length > 0) {
var ix = data[0].props.xindex;
var iy = data[0].props.yindex;
var e = brush.extent();
if (brush.empty()) {
dataToBrush.selectAll("path").classed("mpld3-hidden", false);
} else {
dataToBrush.selectAll("path").classed("mpld3-hidden", function(p) {
return e[0][0] > p[ix] || e[1][0] < p[ix] || e[0][1] > p[iy] || e[1][1] < p[iy];
});
}
}
}
function brushend(d) {
if (brush.empty()) {
dataToBrush.selectAll("path").classed("mpld3-hidden", false);
}
}
this.enable = function() {
this.fig.showBrush(this.extentClass);
brush.on("brushstart", brushstart).on("brush", brushmove).on("brushend", brushend);
this.enabled = true;
};
this.disable = function() {
d3.select(currentAxes).call(brush.clear());
this.fig.hideBrush(this.extentClass);
this.enabled = false;
};
this.disable();
};
mpld3.register_plugin("mouseposition", MousePositionPlugin);
MousePositionPlugin.prototype = Object.create(mpld3.Plugin.prototype);
MousePositionPlugin.prototype.constructor = MousePositionPlugin;
MousePositionPlugin.prototype.requiredProps = [];
MousePositionPlugin.prototype.defaultProps = {
fontsize: 12,
fmt: ".3g"
};
function MousePositionPlugin(fig, props) {
mpld3.Plugin.call(this, fig, props);
}
MousePositionPlugin.prototype.draw = function() {
var fig = this.fig;
var fmt = d3.format(this.props.fmt);
var coords = fig.canvas.append("text").attr("class", "mpld3-coordinates").style("text-anchor", "end").style("font-size", this.props.fontsize).attr("x", this.fig.width - 5).attr("y", this.fig.height - 5);
for (var i = 0; i < this.fig.axes.length; i++) {
var update_coords = function() {
var ax = fig.axes[i];
return function() {
var pos = d3.mouse(this), x = ax.x.invert(pos[0]), y = ax.y.invert(pos[1]);
coords.text("(" + fmt(x) + ", " + fmt(y) + ")");
};
}();
fig.axes[i].baseaxes.on("mousemove", update_coords).on("mouseout", function() {
coords.text("");
});
}
};
mpld3.Figure = mpld3_Figure;
mpld3_Figure.prototype = Object.create(mpld3_PlotElement.prototype);
mpld3_Figure.prototype.constructor = mpld3_Figure;
mpld3_Figure.prototype.requiredProps = [ "width", "height" ];
mpld3_Figure.prototype.defaultProps = {
data: {},
axes: [],
plugins: [ {
type: "reset"
}, {
type: "zoom"
}, {
type: "boxzoom"
} ]
};
function mpld3_Figure(figid, props) {
mpld3_PlotElement.call(this, null, props);
this.figid = figid;
this.width = this.props.width;
this.height = this.props.height;
this.data = this.props.data;
this.buttons = [];
this.root = d3.select("#" + figid).append("div").style("position", "relative");
this.axes = [];
for (var i = 0; i < this.props.axes.length; i++) this.axes.push(new mpld3_Axes(this, this.props.axes[i]));
this.plugins = [];
for (var i = 0; i < this.props.plugins.length; i++) this.add_plugin(this.props.plugins[i]);
this.toolbar = new mpld3.Toolbar(this, {
buttons: this.buttons
});
}
mpld3_Figure.prototype.getBrush = function() {
if (typeof this._brush === "undefined") {
var brush = d3.svg.brush().x(d3.scale.linear()).y(d3.scale.linear());
this.root.selectAll(".mpld3-axes").data(this.axes).call(brush);
this.axes.forEach(function(ax) {
brush.x(ax.xdom).y(ax.ydom);
ax.axes.call(brush);
});
this._brush = brush;
this.hideBrush();
}
return this._brush;
};
mpld3_Figure.prototype.showBrush = function(extentClass) {
extentClass = typeof extentClass === "undefined" ? "" : extentClass;
var brush = this.getBrush();
brush.on("brushstart", function(d) {
brush.x(d.xdom).y(d.ydom);
});
this.canvas.selectAll("rect.background").style("cursor", "crosshair").style("pointer-events", null);
this.canvas.selectAll("rect.extent, rect.resize").style("display", null).classed(extentClass, true);
};
mpld3_Figure.prototype.hideBrush = function(extentClass) {
extentClass = typeof extentClass === "undefined" ? "" : extentClass;
var brush = this.getBrush();
brush.on("brushstart", null).on("brush", null).on("brushend", function(d) {
d.axes.call(brush.clear());
});
this.canvas.selectAll("rect.background").style("cursor", null).style("pointer-events", "visible");
this.canvas.selectAll("rect.extent, rect.resize").style("display", "none").classed(extentClass, false);
};
mpld3_Figure.prototype.add_plugin = function(props) {
var plug = props.type;
if (typeof plug === "undefined") {
console.warn("unspecified plugin type. Skipping this");
return;
}
props = mpld3_cloneObj(props);
delete props.type;
if (plug in mpld3.plugin_map) plug = mpld3.plugin_map[plug];
if (typeof plug !== "function") {
console.warn("Skipping unrecognized plugin: " + plug);
return;
}
if (props.clear_toolbar) {
this.props.toolbar = [];
}
if ("buttons" in props) {
if (typeof props.buttons === "string") {
this.props.toolbar.push(props.buttons);
} else {
for (var i = 0; i < props.buttons.length; i++) {
this.props.toolbar.push(props.buttons[i]);
}
}
}
this.plugins.push(new plug(this, props));
};
mpld3_Figure.prototype.draw = function() {
this.canvas = this.root.append("svg:svg").attr("class", "mpld3-figure").attr("width", this.width).attr("height", this.height);
for (var i = 0; i < this.axes.length; i++) {
this.axes[i].draw();
}
this.disable_zoom();
for (var i = 0; i < this.plugins.length; i++) {
this.plugins[i].draw();
}
this.toolbar.draw();
};
mpld3_Figure.prototype.reset = function(duration) {
this.axes.forEach(function(ax) {
ax.reset(duration, false);
});
};
mpld3_Figure.prototype.enable_zoom = function() {
for (var i = 0; i < this.axes.length; i++) {
this.axes[i].enable_zoom();
}
this.zoom_on = true;
};
mpld3_Figure.prototype.disable_zoom = function() {
for (var i = 0; i < this.axes.length; i++) {
this.axes[i].disable_zoom();
}
this.zoom_on = false;
};
mpld3_Figure.prototype.toggle_zoom = function() {
if (this.zoom_on) {
this.disable_zoom();
} else {
this.enable_zoom();
}
};
mpld3_Figure.prototype.get_data = function(data) {
if (data === null || typeof data === "undefined") {
return null;
} else if (typeof data === "string") {
return this.data[data];
} else {
return data;
}
};
mpld3.PlotElement = mpld3_PlotElement;
function mpld3_PlotElement(parent, props) {
this.parent = isUndefinedOrNull(parent) ? null : parent;
this.props = isUndefinedOrNull(props) ? {} : this.processProps(props);
this.fig = parent instanceof mpld3_Figure ? parent : parent && "fig" in parent ? parent.fig : null;
this.ax = parent instanceof mpld3_Axes ? parent : parent && "ax" in parent ? parent.ax : null;
}
mpld3_PlotElement.prototype.requiredProps = [];
mpld3_PlotElement.prototype.defaultProps = {};
mpld3_PlotElement.prototype.processProps = function(props) {
props = mpld3_cloneObj(props);
var finalProps = {};
var this_name = this.name();
this.requiredProps.forEach(function(p) {
if (!(p in props)) {
throw "property '" + p + "' " + "must be specified for " + this_name;
}
finalProps[p] = props[p];
delete props[p];
});
for (var p in this.defaultProps) {
if (p in props) {
finalProps[p] = props[p];
delete props[p];
} else {
finalProps[p] = this.defaultProps[p];
}
}
if ("id" in props) {
finalProps.id = props.id;
delete props.id;
} else if (!("id" in finalProps)) {
finalProps.id = mpld3.generateId();
}
for (var p in props) {
console.warn("Unrecognized property '" + p + "' " + "for object " + this.name() + " (value = " + props[p] + ").");
}
return finalProps;
};
mpld3_PlotElement.prototype.name = function() {
var funcNameRegex = /function (.{1,})\(/;
var results = funcNameRegex.exec(this.constructor.toString());
return results && results.length > 1 ? results[1] : "";
};
if (typeof module === "object" && module.exports) {
module.exports = mpld3;
} else {
this.mpld3 = mpld3;
}
console.log("Loaded mpld3 version " + mpld3.version);
}(d3);