sjy-ecos/public/lib/bokeh/js/tree/models/axes/axis.js

468 lines
17 KiB
JavaScript

var Axis, AxisView, GE, GuideRenderer, Renderer, SidePanel, _, logger, p,
extend = function(child, parent) { for (var key in parent) { if (hasProp.call(parent, key)) child[key] = parent[key]; } function ctor() { this.constructor = child; } ctor.prototype = parent.prototype; child.prototype = new ctor(); child.__super__ = parent.prototype; return child; },
hasProp = {}.hasOwnProperty;
_ = require("underscore");
SidePanel = require("../../core/layout/side_panel");
GuideRenderer = require("../renderers/guide_renderer");
Renderer = require("../renderers/renderer");
GE = require("../../core/layout/solver").GE;
logger = require("../../core/logging").logger;
p = require("../../core/properties");
AxisView = (function(superClass) {
extend(AxisView, superClass);
function AxisView() {
return AxisView.__super__.constructor.apply(this, arguments);
}
AxisView.prototype.initialize = function(options) {
AxisView.__super__.initialize.call(this, options);
this._x_range_name = this.mget('x_range_name');
return this._y_range_name = this.mget('y_range_name');
};
AxisView.prototype.render = function() {
var ctx;
if (this.model.visible === false) {
return;
}
ctx = this.plot_view.canvas_view.ctx;
ctx.save();
this._draw_rule(ctx);
this._draw_major_ticks(ctx);
this._draw_minor_ticks(ctx);
this._draw_major_labels(ctx);
this._draw_axis_label(ctx);
return ctx.restore();
};
AxisView.prototype.bind_bokeh_events = function() {
return this.listenTo(this.model, 'change', this.plot_view.request_render);
};
AxisView.prototype._get_size = function() {
return this._tick_extent() + this._tick_label_extent() + this._axis_label_extent();
};
AxisView.prototype._draw_rule = function(ctx) {
var coords, i, k, nx, ny, ref, ref1, ref2, ref3, ref4, sx, sy, x, xoff, y, yoff;
if (!this.visuals.axis_line.doit) {
return;
}
ref = coords = this.mget('rule_coords'), x = ref[0], y = ref[1];
ref1 = this.plot_view.map_to_screen(x, y, this._x_range_name, this._y_range_name), sx = ref1[0], sy = ref1[1];
ref2 = this.mget('normals'), nx = ref2[0], ny = ref2[1];
ref3 = this.mget('offsets'), xoff = ref3[0], yoff = ref3[1];
this.visuals.axis_line.set_value(ctx);
ctx.beginPath();
ctx.moveTo(Math.round(sx[0] + nx * xoff), Math.round(sy[0] + ny * yoff));
for (i = k = 1, ref4 = sx.length; 1 <= ref4 ? k < ref4 : k > ref4; i = 1 <= ref4 ? ++k : --k) {
ctx.lineTo(Math.round(sx[i] + nx * xoff), Math.round(sy[i] + ny * yoff));
}
return ctx.stroke();
};
AxisView.prototype._draw_major_ticks = function(ctx) {
var coords, i, k, nx, ny, ref, ref1, ref2, ref3, ref4, results, sx, sy, tin, tout, x, xoff, y, yoff;
if (!this.visuals.major_tick_line.doit) {
return;
}
coords = this.mget('tick_coords');
ref = coords.major, x = ref[0], y = ref[1];
ref1 = this.plot_view.map_to_screen(x, y, this._x_range_name, this._y_range_name), sx = ref1[0], sy = ref1[1];
ref2 = this.mget('normals'), nx = ref2[0], ny = ref2[1];
ref3 = this.mget('offsets'), xoff = ref3[0], yoff = ref3[1];
tin = this.mget('major_tick_in');
tout = this.mget('major_tick_out');
this.visuals.major_tick_line.set_value(ctx);
results = [];
for (i = k = 0, ref4 = sx.length; 0 <= ref4 ? k < ref4 : k > ref4; i = 0 <= ref4 ? ++k : --k) {
ctx.beginPath();
ctx.moveTo(Math.round(sx[i] + nx * tout + nx * xoff), Math.round(sy[i] + ny * tout + ny * yoff));
ctx.lineTo(Math.round(sx[i] - nx * tin + nx * xoff), Math.round(sy[i] - ny * tin + ny * yoff));
results.push(ctx.stroke());
}
return results;
};
AxisView.prototype._draw_minor_ticks = function(ctx) {
var coords, i, k, nx, ny, ref, ref1, ref2, ref3, ref4, results, sx, sy, tin, tout, x, xoff, y, yoff;
if (!this.visuals.minor_tick_line.doit) {
return;
}
coords = this.mget('tick_coords');
ref = coords.minor, x = ref[0], y = ref[1];
ref1 = this.plot_view.map_to_screen(x, y, this._x_range_name, this._y_range_name), sx = ref1[0], sy = ref1[1];
ref2 = this.mget('normals'), nx = ref2[0], ny = ref2[1];
ref3 = this.mget('offsets'), xoff = ref3[0], yoff = ref3[1];
tin = this.mget('minor_tick_in');
tout = this.mget('minor_tick_out');
this.visuals.minor_tick_line.set_value(ctx);
results = [];
for (i = k = 0, ref4 = sx.length; 0 <= ref4 ? k < ref4 : k > ref4; i = 0 <= ref4 ? ++k : --k) {
ctx.beginPath();
ctx.moveTo(Math.round(sx[i] + nx * tout + nx * xoff), Math.round(sy[i] + ny * tout + ny * yoff));
ctx.lineTo(Math.round(sx[i] - nx * tin + nx * xoff), Math.round(sy[i] - ny * tin + ny * yoff));
results.push(ctx.stroke());
}
return results;
};
AxisView.prototype._draw_major_labels = function(ctx) {
var angle, coords, dim, i, k, labels, nx, ny, orient, ref, ref1, ref2, ref3, ref4, results, side, standoff, sx, sy, x, xoff, y, yoff;
coords = this.mget('tick_coords');
ref = coords.major, x = ref[0], y = ref[1];
ref1 = this.plot_view.map_to_screen(x, y, this._x_range_name, this._y_range_name), sx = ref1[0], sy = ref1[1];
ref2 = this.mget('normals'), nx = ref2[0], ny = ref2[1];
ref3 = this.mget('offsets'), xoff = ref3[0], yoff = ref3[1];
dim = this.mget('dimension');
side = this.mget('panel_side');
orient = this.mget('major_label_orientation');
if (_.isString(orient)) {
angle = this.model.panel.get_label_angle_heuristic(orient);
} else {
angle = -orient;
}
standoff = this._tick_extent() + this.mget('major_label_standoff');
labels = this.mget('formatter').doFormat(coords.major[dim]);
this.visuals.major_label_text.set_value(ctx);
this.model.panel.apply_label_text_heuristics(ctx, orient);
results = [];
for (i = k = 0, ref4 = sx.length; 0 <= ref4 ? k < ref4 : k > ref4; i = 0 <= ref4 ? ++k : --k) {
if (angle) {
ctx.translate(sx[i] + nx * standoff + nx * xoff, sy[i] + ny * standoff + ny * yoff);
ctx.rotate(angle);
ctx.fillText(labels[i], 0, 0);
ctx.rotate(-angle);
results.push(ctx.translate(-sx[i] - nx * standoff + nx * xoff, -sy[i] - ny * standoff + ny * yoff));
} else {
results.push(ctx.fillText(labels[i], Math.round(sx[i] + nx * standoff + nx * xoff), Math.round(sy[i] + ny * standoff + ny * yoff)));
}
}
return results;
};
AxisView.prototype._draw_axis_label = function(ctx) {
var angle, label, nx, ny, orient, ref, ref1, ref2, ref3, side, standoff, sx, sy, x, xoff, y, yoff;
label = this.mget('axis_label');
if (label == null) {
return;
}
ref = this.mget('rule_coords'), x = ref[0], y = ref[1];
ref1 = this.plot_view.map_to_screen(x, y, this._x_range_name, this._y_range_name), sx = ref1[0], sy = ref1[1];
ref2 = this.mget('normals'), nx = ref2[0], ny = ref2[1];
ref3 = this.mget('offsets'), xoff = ref3[0], yoff = ref3[1];
side = this.mget('panel_side');
orient = 'parallel';
angle = this.model.panel.get_label_angle_heuristic(orient);
standoff = this._tick_extent() + this._tick_label_extent() + this.mget('axis_label_standoff');
sx = (sx[0] + sx[sx.length - 1]) / 2;
sy = (sy[0] + sy[sy.length - 1]) / 2;
this.visuals.axis_label_text.set_value(ctx);
this.model.panel.apply_label_text_heuristics(ctx, orient);
x = sx + nx * standoff + nx * xoff;
y = sy + ny * standoff + ny * yoff;
if (isNaN(x) || isNaN(y)) {
return;
}
if (angle) {
ctx.translate(x, y);
ctx.rotate(angle);
ctx.fillText(label, 0, 0);
ctx.rotate(-angle);
return ctx.translate(-x, -y);
} else {
return ctx.fillText(label, x, y);
}
};
AxisView.prototype._tick_extent = function() {
return this.mget('major_tick_out');
};
AxisView.prototype._tick_label_extent = function() {
var angle, c, coords, ctx, dim, extent, h, hfactor, hscale, i, k, labels, orient, ref, s, side, val, w, wfactor;
extent = 0;
ctx = this.plot_view.canvas_view.ctx;
dim = this.mget('dimension');
coords = this.mget('tick_coords').major;
side = this.mget('panel_side');
orient = this.mget('major_label_orientation');
labels = this.mget('formatter').doFormat(coords[dim]);
this.visuals.major_label_text.set_value(ctx);
if (_.isString(orient)) {
hscale = 1;
angle = this.model.panel.get_label_angle_heuristic(orient);
} else {
hscale = 2;
angle = -orient;
}
angle = Math.abs(angle);
c = Math.cos(angle);
s = Math.sin(angle);
if (side === "above" || side === "below") {
wfactor = s;
hfactor = c;
} else {
wfactor = c;
hfactor = s;
}
for (i = k = 0, ref = labels.length; 0 <= ref ? k < ref : k > ref; i = 0 <= ref ? ++k : --k) {
if (labels[i] == null) {
continue;
}
w = ctx.measureText(labels[i]).width * 1.1;
h = ctx.measureText(labels[i]).ascent * 0.9;
val = w * wfactor + (h / hscale) * hfactor;
if (val > extent) {
extent = val;
}
}
if (extent > 0) {
extent += this.mget('major_label_standoff');
}
return extent;
};
AxisView.prototype._axis_label_extent = function() {
var angle, axis_label, c, ctx, extent, h, orient, s, side, w;
extent = 0;
side = this.mget('panel_side');
axis_label = this.mget('axis_label');
orient = 'parallel';
ctx = this.plot_view.canvas_view.ctx;
this.visuals.axis_label_text.set_value(ctx);
angle = Math.abs(this.model.panel.get_label_angle_heuristic(orient));
c = Math.cos(angle);
s = Math.sin(angle);
if (axis_label) {
extent += this.mget('axis_label_standoff');
this.visuals.axis_label_text.set_value(ctx);
w = ctx.measureText(axis_label).width * 1.1;
h = ctx.measureText(axis_label).ascent * 0.9;
if (side === "above" || side === "below") {
extent += w * s + h * c;
} else {
extent += w * c + h * s;
}
}
return extent;
};
return AxisView;
})(Renderer.View);
Axis = (function(superClass) {
extend(Axis, superClass);
function Axis() {
return Axis.__super__.constructor.apply(this, arguments);
}
Axis.prototype.default_view = AxisView;
Axis.prototype.type = 'Axis';
Axis.mixins(['line:axis_', 'line:major_tick_', 'line:minor_tick_', 'text:major_label_', 'text:axis_label_']);
Axis.define({
bounds: [p.Any, 'auto'],
ticker: [p.Instance, null],
formatter: [p.Instance, null],
x_range_name: [p.String, 'default'],
y_range_name: [p.String, 'default'],
axis_label: [p.String, ''],
axis_label_standoff: [p.Int, 5],
major_label_standoff: [p.Int, 5],
major_label_orientation: [p.Any, "horizontal"],
major_tick_in: [p.Number, 2],
major_tick_out: [p.Number, 6],
minor_tick_in: [p.Number, 0],
minor_tick_out: [p.Number, 4]
});
Axis.override({
axis_line_color: 'black',
major_tick_line_color: 'black',
minor_tick_line_color: 'black',
major_label_text_font_size: "8pt",
major_label_text_align: "center",
major_label_text_baseline: "alphabetic",
axis_label_text_font_size: "10pt",
axis_label_text_font_style: "italic"
});
Axis.internal({
panel_side: [p.Any]
});
Axis.prototype.initialize = function(attrs, options) {
Axis.__super__.initialize.call(this, attrs, options);
this.define_computed_property('computed_bounds', this._computed_bounds, false);
this.add_dependencies('computed_bounds', this, ['bounds']);
this.add_dependencies('computed_bounds', this.get('plot'), ['x_range', 'y_range']);
this.define_computed_property('rule_coords', this._rule_coords, false);
this.add_dependencies('rule_coords', this, ['computed_bounds', 'side']);
this.define_computed_property('tick_coords', this._tick_coords, false);
this.add_dependencies('tick_coords', this, ['computed_bounds', 'panel_side']);
this.define_computed_property('ranges', this._ranges, true);
this.define_computed_property('normals', (function() {
return this.panel._normals;
}), true);
this.define_computed_property('dimension', (function() {
return this.panel._dim;
}), true);
return this.define_computed_property('offsets', this._offsets, true);
};
Axis.prototype.add_panel = function(side) {
this.panel = new SidePanel.Model({
side: side
});
this.panel.attach_document(this.document);
return this.set('panel_side', side);
};
Axis.prototype._offsets = function() {
var frame, ref, side, xoff, yoff;
side = this.get('panel_side');
ref = [0, 0], xoff = ref[0], yoff = ref[1];
frame = this.plot.plot_canvas.get('frame');
if (side === "below") {
yoff = Math.abs(this.panel.get("top") - frame.get("bottom"));
} else if (side === "above") {
yoff = Math.abs(this.panel.get("bottom") - frame.get("top"));
} else if (side === "right") {
xoff = Math.abs(this.panel.get("left") - frame.get("right"));
} else if (side === "left") {
xoff = Math.abs(this.panel.get("right") - frame.get("left"));
}
return [xoff, yoff];
};
Axis.prototype._ranges = function() {
var frame, i, j, ranges;
i = this.get('dimension');
j = (i + 1) % 2;
frame = this.plot.plot_canvas.get('frame');
ranges = [frame.get('x_ranges')[this.get('x_range_name')], frame.get('y_ranges')[this.get('y_range_name')]];
return [ranges[i], ranges[j]];
};
Axis.prototype._computed_bounds = function() {
var cross_range, end, range, range_bounds, ref, ref1, start, user_bounds;
ref = this.get('ranges'), range = ref[0], cross_range = ref[1];
user_bounds = (ref1 = this.get('bounds')) != null ? ref1 : 'auto';
range_bounds = [range.get('min'), range.get('max')];
if (user_bounds === 'auto') {
return range_bounds;
}
if (_.isArray(user_bounds)) {
if (Math.abs(user_bounds[0] - user_bounds[1]) > Math.abs(range_bounds[0] - range_bounds[1])) {
start = Math.max(Math.min(user_bounds[0], user_bounds[1]), range_bounds[0]);
end = Math.min(Math.max(user_bounds[0], user_bounds[1]), range_bounds[1]);
} else {
start = Math.min(user_bounds[0], user_bounds[1]);
end = Math.max(user_bounds[0], user_bounds[1]);
}
return [start, end];
}
logger.error("user bounds '" + user_bounds + "' not understood");
return null;
};
Axis.prototype._rule_coords = function() {
var coords, cross_range, end, i, j, loc, range, ref, ref1, start, xs, ys;
i = this.get('dimension');
j = (i + 1) % 2;
ref = this.get('ranges'), range = ref[0], cross_range = ref[1];
ref1 = this.get('computed_bounds'), start = ref1[0], end = ref1[1];
xs = new Array(2);
ys = new Array(2);
coords = [xs, ys];
loc = this._get_loc(cross_range);
coords[i][0] = Math.max(start, range.get('min'));
coords[i][1] = Math.min(end, range.get('max'));
if (coords[i][0] > coords[i][1]) {
coords[i][0] = coords[i][1] = NaN;
}
coords[j][0] = loc;
coords[j][1] = loc;
return coords;
};
Axis.prototype._tick_coords = function() {
var coords, cross_range, end, i, ii, j, k, l, loc, m, majors, minor_coords, minor_xs, minor_ys, minors, range, range_max, range_min, ref, ref1, ref2, ref3, ref4, ref5, start, ticks, xs, ys;
i = this.get('dimension');
j = (i + 1) % 2;
ref = this.get('ranges'), range = ref[0], cross_range = ref[1];
ref1 = this.get('computed_bounds'), start = ref1[0], end = ref1[1];
ticks = this.get('ticker').get_ticks(start, end, range, {});
majors = ticks.major;
minors = ticks.minor;
loc = this._get_loc(cross_range);
xs = [];
ys = [];
coords = [xs, ys];
minor_xs = [];
minor_ys = [];
minor_coords = [minor_xs, minor_ys];
if (range.type === "FactorRange") {
for (ii = k = 0, ref2 = majors.length; 0 <= ref2 ? k < ref2 : k > ref2; ii = 0 <= ref2 ? ++k : --k) {
coords[i].push(majors[ii]);
coords[j].push(loc);
}
} else {
ref3 = [range.get('min'), range.get('max')], range_min = ref3[0], range_max = ref3[1];
for (ii = l = 0, ref4 = majors.length; 0 <= ref4 ? l < ref4 : l > ref4; ii = 0 <= ref4 ? ++l : --l) {
if (majors[ii] < range_min || majors[ii] > range_max) {
continue;
}
coords[i].push(majors[ii]);
coords[j].push(loc);
}
for (ii = m = 0, ref5 = minors.length; 0 <= ref5 ? m < ref5 : m > ref5; ii = 0 <= ref5 ? ++m : --m) {
if (minors[ii] < range_min || minors[ii] > range_max) {
continue;
}
minor_coords[i].push(minors[ii]);
minor_coords[j].push(loc);
}
}
return {
"major": coords,
"minor": minor_coords
};
};
Axis.prototype._get_loc = function(cross_range) {
var cend, cstart, loc, side;
cstart = cross_range.get('start');
cend = cross_range.get('end');
side = this.get('panel_side');
if (side === 'left' || side === 'below') {
loc = 'start';
} else if (side === 'right' || side === 'above') {
loc = 'end';
}
return cross_range.get(loc);
};
return Axis;
})(GuideRenderer.Model);
module.exports = {
Model: Axis,
View: AxisView
};