mirror of https://github.com/OCA/web.git
1539 lines
56 KiB
JavaScript
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); |