2015-06-12 22:00:48 +02:00
|
|
|
(function() {
|
|
|
|
|
2015-06-13 16:15:21 +02:00
|
|
|
"use strict";
|
|
|
|
|
|
|
|
var root = this,
|
|
|
|
Chart = root.Chart,
|
|
|
|
helpers = Chart.helpers;
|
|
|
|
|
|
|
|
Chart.defaults.global.tooltips = {
|
|
|
|
enabled: true,
|
|
|
|
custom: null,
|
2015-10-07 04:40:25 +02:00
|
|
|
mode: 'single',
|
2015-06-13 16:15:21 +02:00
|
|
|
backgroundColor: "rgba(0,0,0,0.8)",
|
|
|
|
titleFontFamily: "'Helvetica Neue', 'Helvetica', 'Arial', sans-serif",
|
|
|
|
titleFontSize: 12,
|
|
|
|
titleFontStyle: "bold",
|
2015-10-12 22:51:00 +02:00
|
|
|
titleSpacing: 2,
|
|
|
|
titleMarginBottom: 6,
|
2015-10-07 04:40:25 +02:00
|
|
|
titleColor: "#fff",
|
|
|
|
titleAlign: "left",
|
|
|
|
bodyFontFamily: "'Helvetica Neue', 'Helvetica', 'Arial', sans-serif",
|
|
|
|
bodyFontSize: 12,
|
|
|
|
bodyFontStyle: "normal",
|
2015-10-12 22:51:00 +02:00
|
|
|
bodySpacing: 2,
|
2015-10-07 04:40:25 +02:00
|
|
|
bodyColor: "#fff",
|
|
|
|
bodyAlign: "left",
|
|
|
|
footerFontFamily: "'Helvetica Neue', 'Helvetica', 'Arial', sans-serif",
|
|
|
|
footerFontSize: 12,
|
|
|
|
footerFontStyle: "bold",
|
2015-10-12 22:51:00 +02:00
|
|
|
footerSpacing: 2,
|
|
|
|
footerMarginTop: 6,
|
2015-10-07 04:40:25 +02:00
|
|
|
footerColor: "#fff",
|
|
|
|
footerAlign: "left",
|
2015-06-13 16:15:21 +02:00
|
|
|
yPadding: 6,
|
|
|
|
xPadding: 6,
|
2015-10-07 04:40:25 +02:00
|
|
|
caretSize: 5,
|
2015-06-13 16:15:21 +02:00
|
|
|
cornerRadius: 6,
|
|
|
|
xOffset: 10,
|
|
|
|
multiKeyBackground: '#fff',
|
2015-10-07 04:40:25 +02:00
|
|
|
callbacks: {
|
|
|
|
beforeTitle: helpers.noop,
|
|
|
|
title: function(xLabel, yLabel, index, datasetIndex, data) {
|
2015-10-12 22:51:00 +02:00
|
|
|
return this._options.tooltips.mode == 'single' ? data.datasets[datasetIndex].label : data.labels[index];
|
2015-10-07 04:40:25 +02:00
|
|
|
},
|
|
|
|
afterTitle: helpers.noop,
|
|
|
|
|
|
|
|
beforeBody: helpers.noop,
|
|
|
|
|
|
|
|
beforeLabel: helpers.noop,
|
|
|
|
label: function(xLabel, yLabel, index, datasetIndex, data) {
|
|
|
|
return xLabel + ': ' + yLabel;
|
|
|
|
},
|
|
|
|
afterLabel: helpers.noop,
|
|
|
|
|
|
|
|
afterBody: helpers.noop,
|
|
|
|
|
|
|
|
beforeFooter: helpers.noop,
|
|
|
|
footer: helpers.noop,
|
|
|
|
afterFooter: helpers.noop,
|
|
|
|
},
|
2015-06-13 16:15:21 +02:00
|
|
|
};
|
|
|
|
|
|
|
|
Chart.Tooltip = Chart.Element.extend({
|
|
|
|
initialize: function() {
|
|
|
|
var options = this._options;
|
|
|
|
helpers.extend(this, {
|
|
|
|
_model: {
|
|
|
|
// Positioning
|
|
|
|
xPadding: options.tooltips.xPadding,
|
|
|
|
yPadding: options.tooltips.yPadding,
|
|
|
|
xOffset: options.tooltips.xOffset,
|
|
|
|
|
2015-10-07 04:40:25 +02:00
|
|
|
// Body
|
|
|
|
bodyColor: options.tooltips.bodyColor,
|
|
|
|
_bodyFontFamily: options.tooltips.bodyFontFamily,
|
|
|
|
_bodyFontStyle: options.tooltips.bodyFontStyle,
|
|
|
|
bodyFontSize: options.tooltips.bodyFontSize,
|
2015-10-12 22:51:00 +02:00
|
|
|
bodySpacing: options.tooltips.bodySpacing,
|
2015-10-07 04:40:25 +02:00
|
|
|
_bodposition: options.tooltips.bodposition,
|
2015-06-13 16:15:21 +02:00
|
|
|
|
|
|
|
// Title
|
2015-10-07 04:40:25 +02:00
|
|
|
titleColor: options.tooltips.titleColor,
|
2015-06-13 16:15:21 +02:00
|
|
|
_titleFontFamily: options.tooltips.titleFontFamily,
|
|
|
|
_titleFontStyle: options.tooltips.titleFontStyle,
|
|
|
|
titleFontSize: options.tooltips.titleFontSize,
|
2015-10-07 04:40:25 +02:00
|
|
|
_titleAlign: options.tooltips.titleAlign,
|
2015-10-12 22:51:00 +02:00
|
|
|
titleSpacing: options.tooltips.titleSpacing,
|
|
|
|
titleMarginBottom: options.tooltips.titleMarginBottom,
|
2015-10-07 04:40:25 +02:00
|
|
|
|
|
|
|
// Footer
|
|
|
|
footerColor: options.tooltips.footerColor,
|
|
|
|
_footerFontFamily: options.tooltips.footerFontFamily,
|
|
|
|
_footerFontStyle: options.tooltips.footerFontStyle,
|
|
|
|
footerFontSize: options.tooltips.footerFontSize,
|
|
|
|
_footerAlign: options.tooltips.footerAlign,
|
2015-10-12 22:51:00 +02:00
|
|
|
footerSpacing: options.tooltips.footerSpacing,
|
|
|
|
footerMarginTop: options.tooltips.footerMarginTop,
|
2015-06-13 16:15:21 +02:00
|
|
|
|
|
|
|
// Appearance
|
2015-10-07 04:40:25 +02:00
|
|
|
caretSize: options.tooltips.caretSize,
|
2015-06-13 16:15:21 +02:00
|
|
|
cornerRadius: options.tooltips.cornerRadius,
|
|
|
|
backgroundColor: options.tooltips.backgroundColor,
|
|
|
|
opacity: 0,
|
|
|
|
legendColorBackground: options.tooltips.multiKeyBackground,
|
|
|
|
},
|
|
|
|
});
|
|
|
|
},
|
|
|
|
|
2015-10-07 04:40:25 +02:00
|
|
|
getTitle: function() {
|
|
|
|
var beforeTitle = this._options.tooltips.callbacks.beforeTitle.apply(this, arguments),
|
|
|
|
title = this._options.tooltips.callbacks.title.apply(this, arguments),
|
|
|
|
afterTitle = this._options.tooltips.callbacks.afterTitle.apply(this, arguments);
|
2015-06-13 16:15:21 +02:00
|
|
|
|
2015-10-07 04:40:25 +02:00
|
|
|
var lines = [];
|
2015-06-13 16:15:21 +02:00
|
|
|
|
2015-10-07 04:40:25 +02:00
|
|
|
if (beforeTitle) {
|
|
|
|
lines.push(beforeTitle);
|
|
|
|
}
|
|
|
|
if (title) {
|
|
|
|
lines.push(title);
|
|
|
|
}
|
|
|
|
if (afterTitle) {
|
|
|
|
lines.push(afterTitle);
|
|
|
|
}
|
|
|
|
return lines;
|
|
|
|
},
|
2015-06-13 16:15:21 +02:00
|
|
|
|
2015-10-07 04:40:25 +02:00
|
|
|
getBody: function(xLabel, yLabel, index, datasetIndex) {
|
2015-06-13 16:15:21 +02:00
|
|
|
|
2015-10-07 04:40:25 +02:00
|
|
|
var lines = [];
|
2015-06-13 16:15:21 +02:00
|
|
|
|
2015-10-07 04:40:25 +02:00
|
|
|
var beforeLabel,
|
|
|
|
afterLabel,
|
|
|
|
label;
|
2015-06-13 16:15:21 +02:00
|
|
|
|
2015-10-07 04:40:25 +02:00
|
|
|
if (helpers.isArray(xLabel)) {
|
2015-06-13 16:15:21 +02:00
|
|
|
|
2015-10-07 04:40:25 +02:00
|
|
|
var labels = [];
|
2015-06-13 16:15:21 +02:00
|
|
|
|
2015-10-07 04:40:25 +02:00
|
|
|
// Run EACH label pair through the label callback this time.
|
|
|
|
for (var i = 0; i < xLabel.length; i++) {
|
|
|
|
|
2015-10-12 22:51:00 +02:00
|
|
|
beforeLabel = this._options.tooltips.callbacks.beforeLabel.call(this, xLabel[i], yLabel[i], index, datasetIndex);
|
|
|
|
afterLabel = this._options.tooltips.callbacks.afterLabel.call(this, xLabel[i], yLabel[i], index, datasetIndex);
|
2015-10-07 04:40:25 +02:00
|
|
|
|
2015-10-12 22:51:00 +02:00
|
|
|
labels.push((beforeLabel ? beforeLabel : '') + this._options.tooltips.callbacks.label.call(this, xLabel[i], yLabel[i], index, datasetIndex) + (afterLabel ? afterLabel : ''));
|
2015-10-07 04:40:25 +02:00
|
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
if (labels.length) {
|
|
|
|
lines = lines.concat(labels);
|
|
|
|
}
|
|
|
|
|
|
|
|
} else {
|
|
|
|
|
|
|
|
// Run the single label through the callback
|
|
|
|
|
|
|
|
beforeLabel = this._options.tooltips.callbacks.beforeLabel.apply(this, arguments);
|
|
|
|
label = this._options.tooltips.callbacks.label.apply(this, arguments);
|
|
|
|
afterLabel = this._options.tooltips.callbacks.afterLabel.apply(this, arguments);
|
|
|
|
|
|
|
|
if (beforeLabel || label || afterLabel) {
|
|
|
|
lines.push((beforeLabel ? afterLabel : '') + label + (afterLabel ? afterLabel : ''));
|
|
|
|
}
|
2015-06-13 16:15:21 +02:00
|
|
|
}
|
|
|
|
|
2015-10-07 04:40:25 +02:00
|
|
|
return lines;
|
|
|
|
},
|
|
|
|
|
|
|
|
getFooter: function() {
|
|
|
|
var beforeFooter = this._options.tooltips.callbacks.beforeFooter.apply(this, arguments),
|
|
|
|
footer = this._options.tooltips.callbacks.footer.apply(this, arguments),
|
|
|
|
afterFooter = this._options.tooltips.callbacks.afterFooter.apply(this, arguments);
|
|
|
|
|
|
|
|
var lines = [];
|
|
|
|
|
|
|
|
if (beforeFooter) {
|
|
|
|
lines.push(beforeFooter);
|
|
|
|
}
|
|
|
|
if (footer) {
|
|
|
|
lines.push(footer);
|
|
|
|
}
|
|
|
|
if (afterFooter) {
|
|
|
|
lines.push(afterFooter);
|
|
|
|
}
|
|
|
|
|
|
|
|
return lines;
|
|
|
|
},
|
|
|
|
|
|
|
|
update: function() {
|
|
|
|
|
|
|
|
var ctx = this._chart.ctx;
|
|
|
|
|
|
|
|
var element = this._active[0],
|
|
|
|
xLabel,
|
|
|
|
yLabel,
|
2015-10-12 22:51:00 +02:00
|
|
|
labelColors = [],
|
2015-10-07 04:40:25 +02:00
|
|
|
tooltipPosition;
|
|
|
|
|
|
|
|
if (this._options.tooltips.mode == 'single') {
|
|
|
|
|
|
|
|
xLabel = element._xScale.getLabelForIndex(element._index, element._datasetIndex);
|
|
|
|
yLabel = element._yScale.getLabelForIndex(element._index, element._datasetIndex);
|
|
|
|
tooltipPosition = this._active[0].tooltipPosition();
|
|
|
|
|
|
|
|
} else {
|
|
|
|
|
|
|
|
xLabel = [];
|
|
|
|
yLabel = [];
|
2015-10-12 22:51:00 +02:00
|
|
|
|
|
|
|
console.log(this._active);
|
|
|
|
|
2015-10-07 04:40:25 +02:00
|
|
|
helpers.each(this._data.datasets, function(dataset, datasetIndex) {
|
2015-10-12 22:51:00 +02:00
|
|
|
|
2015-10-07 04:40:25 +02:00
|
|
|
xLabel.push(element._xScale.getLabelForIndex(element._index, datasetIndex));
|
|
|
|
yLabel.push(element._yScale.getLabelForIndex(element._index, datasetIndex));
|
|
|
|
});
|
2015-10-12 22:51:00 +02:00
|
|
|
|
|
|
|
helpers.each(this._active, function(active, i) {
|
|
|
|
labelColors.push({
|
|
|
|
borderColor: active._view.borderColor,
|
|
|
|
backgroundColor: active._view.backgroundColor
|
|
|
|
});
|
|
|
|
}, this);
|
|
|
|
|
2015-10-07 04:40:25 +02:00
|
|
|
tooltipPosition = this._active[0].tooltipPosition();
|
2015-10-12 22:51:00 +02:00
|
|
|
tooltipPosition.y = this._active[0]._yScale.getPixelForDecimal(0.5);
|
2015-10-07 04:40:25 +02:00
|
|
|
|
|
|
|
}
|
|
|
|
|
2015-10-12 22:51:00 +02:00
|
|
|
|
2015-10-07 04:40:25 +02:00
|
|
|
// Build the Text Lines
|
|
|
|
helpers.extend(this._model, {
|
|
|
|
title: this.getTitle(xLabel, yLabel, element._index, element._datasetIndex, this._data),
|
2015-10-12 22:51:00 +02:00
|
|
|
beforeBody: this._options.tooltips.callbacks.beforeBody.call(this, xLabel, yLabel, element._index, element._datasetIndex, this._data),
|
2015-10-07 04:40:25 +02:00
|
|
|
body: this.getBody(xLabel, yLabel, element._index, element._datasetIndex, this._data),
|
2015-10-12 22:51:00 +02:00
|
|
|
afterBody: this._options.tooltips.callbacks.afterBody.call(this, xLabel, yLabel, element._index, element._datasetIndex, this._data),
|
2015-10-07 04:40:25 +02:00
|
|
|
footer: this.getFooter(xLabel, yLabel, element._index, element._datasetIndex, this._data),
|
|
|
|
});
|
|
|
|
|
|
|
|
helpers.extend(this._model, {
|
|
|
|
x: Math.round(tooltipPosition.x),
|
|
|
|
y: Math.round(tooltipPosition.y),
|
2015-10-12 22:51:00 +02:00
|
|
|
caretPadding: tooltipPosition.padding,
|
|
|
|
labelColors: labelColors,
|
2015-10-07 04:40:25 +02:00
|
|
|
});
|
|
|
|
|
2015-06-13 16:15:21 +02:00
|
|
|
return this;
|
|
|
|
},
|
|
|
|
draw: function() {
|
|
|
|
|
|
|
|
var ctx = this._chart.ctx;
|
|
|
|
var vm = this._view;
|
|
|
|
|
2015-10-07 04:40:25 +02:00
|
|
|
// Get Dimensions
|
2015-06-13 16:15:21 +02:00
|
|
|
|
2015-10-07 04:40:25 +02:00
|
|
|
vm.position = "top";
|
2015-06-13 16:15:21 +02:00
|
|
|
|
2015-10-07 04:40:25 +02:00
|
|
|
var caretPadding = vm.caretPadding || 2;
|
2015-10-12 22:51:00 +02:00
|
|
|
var combinedBodyLength = vm.body.length + (vm.beforeBody ? 1 : 0) + (vm.afterBody ? 1 : 0);
|
2015-06-13 16:15:21 +02:00
|
|
|
|
2015-10-07 04:40:25 +02:00
|
|
|
// Height
|
2015-10-12 22:51:00 +02:00
|
|
|
var tooltipHeight = vm.yPadding * 2; // Tooltip Padding
|
|
|
|
|
|
|
|
tooltipHeight += vm.title.length * vm.titleFontSize; // Title Lines
|
|
|
|
tooltipHeight += (vm.title.length - 1) * vm.titleSpacing; // Title Line Spacing
|
|
|
|
tooltipHeight += vm.title.length ? vm.titleMarginBottom : 0; // Title's bottom Margin
|
|
|
|
|
|
|
|
tooltipHeight += combinedBodyLength * vm.bodyFontSize; // Body Lines
|
|
|
|
tooltipHeight += (combinedBodyLength - 1) * vm.bodySpacing; // Body Line Spacing
|
2015-06-13 16:15:21 +02:00
|
|
|
|
2015-10-12 22:51:00 +02:00
|
|
|
tooltipHeight += vm.footer.length ? vm.footerMarginTop : 0; // Footer Margin
|
|
|
|
tooltipHeight += vm.footer.length * (vm.footerFontSize); // Footer Lines
|
|
|
|
tooltipHeight += (vm.footer.length - 1) * vm.footerSpacing; // Footer Line Spacing
|
2015-06-13 16:15:21 +02:00
|
|
|
|
2015-10-07 04:40:25 +02:00
|
|
|
// Width
|
|
|
|
var tooltipWidth = 0;
|
|
|
|
helpers.each(vm.title, function(line, i) {
|
|
|
|
ctx.font = helpers.fontString(vm.titleFontSize, vm._titleFontStyle, vm._titleFontFamily);
|
|
|
|
tooltipWidth = Math.max(tooltipWidth, ctx.measureText(line).width);
|
|
|
|
});
|
|
|
|
helpers.each(vm.body, function(line, i) {
|
|
|
|
ctx.font = helpers.fontString(vm.bodyFontSize, vm._bodyFontStyle, vm._bodyFontFamily);
|
2015-10-12 22:51:00 +02:00
|
|
|
tooltipWidth = Math.max(tooltipWidth, ctx.measureText(line).width + (this._options.tooltips.mode != 'single' ? (vm.bodyFontSize + 2) : 0));
|
|
|
|
}, this);
|
2015-10-07 04:40:25 +02:00
|
|
|
helpers.each(vm.footer, function(line, i) {
|
|
|
|
ctx.font = helpers.fontString(vm.footerFontSize, vm._footerFontStyle, vm._footerFontFamily);
|
|
|
|
tooltipWidth = Math.max(tooltipWidth, ctx.measureText(line).width);
|
|
|
|
});
|
|
|
|
tooltipWidth += 2 * vm.xPadding;
|
|
|
|
var tooltipTotalWidth = tooltipWidth + vm.caretSize + caretPadding;
|
2015-06-13 16:15:21 +02:00
|
|
|
|
|
|
|
|
|
|
|
|
2015-10-12 22:51:00 +02:00
|
|
|
// Smart Tooltip placement to stay on the canvas
|
2015-10-07 04:40:25 +02:00
|
|
|
// Top, center, or bottom
|
|
|
|
vm.yAlign = "center";
|
|
|
|
if (vm.y - (tooltipHeight / 2) < 0) {
|
|
|
|
vm.yAlign = "top";
|
|
|
|
} else if (vm.y + (tooltipHeight / 2) > this._chart.height) {
|
|
|
|
vm.yAlign = "bottom";
|
|
|
|
}
|
2015-06-13 16:15:21 +02:00
|
|
|
|
|
|
|
|
2015-10-07 04:40:25 +02:00
|
|
|
// Left or Right
|
|
|
|
vm.xAlign = "right";
|
|
|
|
if (vm.x + tooltipTotalWidth > this._chart.width) {
|
|
|
|
vm.xAlign = "left";
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// Background Position
|
|
|
|
var tooltipX = vm.x,
|
|
|
|
tooltipY = vm.y;
|
|
|
|
|
|
|
|
if (vm.yAlign == 'top') {
|
|
|
|
tooltipY = vm.y - vm.caretSize - vm.cornerRadius;
|
|
|
|
} else if (vm.yAlign == 'bottom') {
|
|
|
|
tooltipY = vm.y - tooltipHeight + vm.caretSize + vm.cornerRadius;
|
|
|
|
} else {
|
|
|
|
tooltipY = vm.y - (tooltipHeight / 2);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (vm.xAlign == 'left') {
|
|
|
|
tooltipX = vm.x - tooltipTotalWidth;
|
|
|
|
} else if (vm.xAlign == 'right') {
|
|
|
|
tooltipX = vm.x + caretPadding + vm.caretSize;
|
|
|
|
} else {
|
|
|
|
tooltipX = vm.x + (tooltipTotalWidth / 2);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Draw Background
|
|
|
|
|
|
|
|
if (this._options.tooltips.enabled) {
|
|
|
|
ctx.fillStyle = helpers.color(vm.backgroundColor).alpha(vm.opacity).rgbString();
|
|
|
|
helpers.drawRoundedRectangle(ctx, tooltipX, tooltipY, tooltipWidth, tooltipHeight, vm.cornerRadius);
|
|
|
|
ctx.fill();
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// Draw Caret
|
|
|
|
if (this._options.tooltips.enabled) {
|
|
|
|
ctx.fillStyle = helpers.color(vm.backgroundColor).alpha(vm.opacity).rgbString();
|
|
|
|
|
|
|
|
if (vm.xAlign == 'left') {
|
|
|
|
|
|
|
|
ctx.beginPath();
|
|
|
|
ctx.moveTo(vm.x - caretPadding, vm.y);
|
|
|
|
ctx.lineTo(vm.x - caretPadding - vm.caretSize, vm.y - vm.caretSize);
|
|
|
|
ctx.lineTo(vm.x - caretPadding - vm.caretSize, vm.y + vm.caretSize);
|
|
|
|
ctx.closePath();
|
|
|
|
ctx.fill();
|
|
|
|
} else {
|
|
|
|
ctx.beginPath();
|
|
|
|
ctx.moveTo(vm.x + caretPadding, vm.y);
|
|
|
|
ctx.lineTo(vm.x + caretPadding + vm.caretSize, vm.y - vm.caretSize);
|
|
|
|
ctx.lineTo(vm.x + caretPadding + vm.caretSize, vm.y + vm.caretSize);
|
|
|
|
ctx.closePath();
|
|
|
|
ctx.fill();
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// Draw Title, Body, and Footer
|
|
|
|
|
|
|
|
if (this._options.tooltips.enabled) {
|
|
|
|
|
2015-10-12 22:51:00 +02:00
|
|
|
var yBase = tooltipY + vm.yPadding;
|
|
|
|
var xBase = tooltipX + vm.xPadding;
|
2015-10-07 04:40:25 +02:00
|
|
|
|
|
|
|
// Titles
|
|
|
|
|
2015-10-12 22:51:00 +02:00
|
|
|
if (vm.title.length) {
|
|
|
|
ctx.textAlign = vm._titleAlign;
|
|
|
|
ctx.textBaseline = "top";
|
|
|
|
ctx.fillStyle = helpers.color(vm.titleColor).alpha(vm.opacity).rgbString();
|
|
|
|
ctx.font = helpers.fontString(vm.titleFontSize, vm._titleFontStyle, vm._titleFontFamily);
|
|
|
|
|
|
|
|
helpers.each(vm.title, function(title, i) {
|
|
|
|
ctx.fillText(title, xBase, yBase);
|
|
|
|
yBase += vm.titleFontSize + vm.titleSpacing; // Line Height and spacing
|
|
|
|
if (i + 1 == vm.title.length) {
|
|
|
|
yBase += vm.titleMarginBottom - vm.titleSpacing; // If Last, add margin, remove spacing
|
|
|
|
}
|
|
|
|
}, this);
|
|
|
|
}
|
2015-10-07 04:40:25 +02:00
|
|
|
|
|
|
|
|
|
|
|
// Body
|
|
|
|
ctx.textAlign = vm._bodyAlign;
|
|
|
|
ctx.textBaseline = "top";
|
|
|
|
ctx.fillStyle = helpers.color(vm.bodyColor).alpha(vm.opacity).rgbString();
|
|
|
|
ctx.font = helpers.fontString(vm.bodyFontSize, vm._bodyFontStyle, vm._bodyFontFamily);
|
2015-06-13 16:15:21 +02:00
|
|
|
|
2015-10-12 22:51:00 +02:00
|
|
|
// Before Body
|
|
|
|
if (vm.beforeBody) {
|
|
|
|
ctx.fillText(vm.beforeBody, xBase, yBase);
|
|
|
|
yBase += vm.bodyFontSize + vm.bodySpacing;
|
|
|
|
}
|
2015-10-07 04:40:25 +02:00
|
|
|
|
|
|
|
helpers.each(vm.body, function(body, i) {
|
|
|
|
|
2015-10-12 22:51:00 +02:00
|
|
|
// Draw Legend-like boxes if needed
|
|
|
|
if (this._options.tooltips.mode != 'single') {
|
|
|
|
ctx.fillStyle = helpers.color(vm.labelColors[i].borderColor).alpha(vm.opacity).rgbString();
|
|
|
|
ctx.fillRect(xBase, yBase, vm.bodyFontSize, vm.bodyFontSize);
|
2015-10-07 04:40:25 +02:00
|
|
|
|
2015-10-12 22:51:00 +02:00
|
|
|
ctx.fillStyle = helpers.color(vm.labelColors[i].backgroundColor).alpha(vm.opacity).rgbString();
|
|
|
|
ctx.fillRect(xBase + 1, yBase + 1, vm.bodyFontSize - 2, vm.bodyFontSize - 2);
|
2015-10-07 04:40:25 +02:00
|
|
|
|
2015-10-12 22:51:00 +02:00
|
|
|
ctx.fillStyle = helpers.color(vm.bodyColor).alpha(vm.opacity).rgbString(); // Return fill style for text
|
|
|
|
}
|
2015-06-13 16:15:21 +02:00
|
|
|
|
2015-10-12 22:51:00 +02:00
|
|
|
// Body Line
|
|
|
|
ctx.fillText(body, xBase + (this._options.tooltips.mode != 'single' ? (vm.bodyFontSize + 2) : 0), yBase);
|
2015-06-22 21:10:49 +02:00
|
|
|
|
2015-10-12 22:51:00 +02:00
|
|
|
yBase += vm.bodyFontSize + vm.bodySpacing;
|
2015-06-13 16:15:21 +02:00
|
|
|
|
2015-10-12 22:51:00 +02:00
|
|
|
}, this);
|
2015-06-13 16:15:21 +02:00
|
|
|
|
2015-10-12 22:51:00 +02:00
|
|
|
// After Body
|
|
|
|
if (vm.afterBody) {
|
|
|
|
ctx.fillText(vm.afterBody, xBase, yBase);
|
|
|
|
yBase += vm.bodyFontSize;
|
|
|
|
} else {
|
|
|
|
yBase -= vm.bodySpacing; // Remove last body spacing
|
|
|
|
}
|
2015-06-13 16:15:21 +02:00
|
|
|
|
|
|
|
|
2015-10-12 22:51:00 +02:00
|
|
|
// Footer
|
|
|
|
if (vm.footer.length) {
|
2015-06-13 16:15:21 +02:00
|
|
|
|
2015-10-12 22:51:00 +02:00
|
|
|
yBase += vm.footerMarginTop;
|
2015-06-13 16:15:21 +02:00
|
|
|
|
2015-10-12 22:51:00 +02:00
|
|
|
ctx.textAlign = vm._footerAlign;
|
|
|
|
ctx.textBaseline = "top";
|
|
|
|
ctx.fillStyle = helpers.color(vm.footerColor).alpha(vm.opacity).rgbString();
|
|
|
|
ctx.font = helpers.fontString(vm.footerFontSize, vm._footerFontStyle, vm._footerFontFamily);
|
2015-06-13 16:15:21 +02:00
|
|
|
|
2015-10-12 22:51:00 +02:00
|
|
|
helpers.each(vm.footer, function(footer, i) {
|
|
|
|
ctx.fillText(footer, xBase, yBase);
|
|
|
|
yBase += vm.footerFontSize + vm.footerSpacing;
|
2015-06-13 16:15:21 +02:00
|
|
|
}, this);
|
2015-10-12 22:51:00 +02:00
|
|
|
}
|
|
|
|
|
2015-06-13 16:15:21 +02:00
|
|
|
}
|
|
|
|
},
|
|
|
|
});
|
2015-06-12 22:00:48 +02:00
|
|
|
|
|
|
|
}).call(this);
|