我正在尝试使用chart.js使用自定义html工具提示创建条形图。我想在工具提示中包含一些互动内容。问题是默认情况下,当鼠标离开图表画布时,工具提示的不透明度变为0。我需要重写mouseout事件,以便用户可以与工具提示进行交互。
官方文档很好,但不全面...我找不到官方答案,因此我不得不审查开放源代码和我的开发人员工具。
使用Google Chrome的开发人员工具,我可以单击画布并查看事件侦听器选项卡。在mouseout下,我看到以下内容:
canvas#chart_0.dashboard-chart.chartjs-render-monitor chart.js 10969
在开发人员工具中删除此侦听器可解决此问题。当我单击链接时,我可以看到以下用于添加侦听器的代码,并且如果设置了断点,我可以看到它正在添加:
...
addEventListener: function(chart, type, listener) {
var canvas = chart.canvas;
if (type === 'resize') {
// Note: the resize event is not supported on all browsers.
addResizeListener(canvas, listener, chart);
return;
}
var expando = listener[EXPANDO_KEY] || (listener[EXPANDO_KEY] = {});
var proxies = expando.proxies || (expando.proxies = {});
var proxy = proxies[chart.id + '_' + type] = function(event) {
listener(fromNativeEvent(event, chart));
};
addEventListener(canvas, type, proxy);
},
...
悬停后,我从CDN获取chart.js,并且希望使用自己的JavaScript覆盖它,而不是滚动自定义版本的chart.js。不过,我不知道如何删除处理程序。在chart.js脚本加载后,我尝试了以下方法:
$('canvas').off('mouseout');
$('canvas#chart_0').off('mouseout');
$('canvas#chart_0.dashboard-chart.chartjs-render-monitor').off('mouseout');
$('#chart_0').off('mouseout');
$('.chartjs-render-monitor').off('mouseout');
这些选择器均未删除事件侦听器。我还尝试了一些$('canvas#chart_0')[0].removeEventListener('mouseout')
的作用,但这需要将侦听器作为参数,并且由于它是匿名传递的,因此我不确定如何获取对此的引用。任何人都可以推荐一种摆脱默认事件侦听器的方法,或者将配置选项传递给chart.js以覆盖默认行为的方法吗?预先感谢。
答案 0 :(得分:0)
对于以下示例,我使用了here上自定义工具提示文档中提供的示例。
传递给自定义函数的对象(在这种情况下为tooltipModel
)具有适当的属性,如果我们忽略了隐藏的行,则该属性将在光标离开悬停的项目时设置为0
工具提示将一直可见,直到您将鼠标悬停在另一个项目上为止。
因此必须更改的行如下:
// Hide if no tooltip
if (tooltipModel.opacity === 0) {
tooltipEl.style.opacity = 0; // remove this line
return;
}
完整的示例:
var ctx = document.getElementById('bar-chart').getContext('2d');
var chart = new Chart(ctx, {
type: 'line',
data: {
labels: ['A', ' B', ' C'],
datasets: [{
label: "Y",
data: [5, 2, 0]
},
{
label: "Z",
data: [7, 0, 3]
}
]
},
options: {
tooltips: {
enabled: false,
custom: function(tooltipModel) {
// Tooltip Element
var tooltipEl = document.getElementById('chartjs-tooltip');
// Create element on first render
if (!tooltipEl) {
tooltipEl = document.createElement('div');
tooltipEl.id = 'chartjs-tooltip';
tooltipEl.innerHTML = '<table></table>';
document.body.appendChild(tooltipEl);
}
// Hide if no tooltip
if (tooltipModel.opacity === 0) {
// remove the next line
//tooltipEl.style.opacity = 0;
return;
}
// Set caret Position
tooltipEl.classList.remove('above', 'below', 'no-transform');
if (tooltipModel.yAlign) {
tooltipEl.classList.add(tooltipModel.yAlign);
} else {
tooltipEl.classList.add('no-transform');
}
function getBody(bodyItem) {
return bodyItem.lines;
}
// Set Text
if (tooltipModel.body) {
var titleLines = tooltipModel.title || [];
var bodyLines = tooltipModel.body.map(getBody);
var innerHtml = '<thead>';
titleLines.forEach(function(title) {
innerHtml += '<tr><th>' + title + '</th></tr>';
});
innerHtml += '</thead><tbody>';
bodyLines.forEach(function(body, i) {
var colors = tooltipModel.labelColors[i];
var style = 'background:' + colors.backgroundColor;
style += '; border-color:' + colors.borderColor;
style += '; border-width: 2px';
var span = '<span style="' + style + '"></span>';
innerHtml += '<tr><td>' + span + body + '</td></tr>';
});
innerHtml += '</tbody>';
var tableRoot = tooltipEl.querySelector('table');
tableRoot.innerHTML = innerHtml;
}
// `this` will be the overall tooltip
var position = this._chart.canvas.getBoundingClientRect();
// Display, position, and set styles for font
tooltipEl.style.opacity = 1;
tooltipEl.style.position = 'absolute';
tooltipEl.style.left = position.left + window.pageXOffset + tooltipModel.caretX + 'px';
tooltipEl.style.top = position.top + window.pageYOffset + tooltipModel.caretY + 'px';
tooltipEl.style.fontFamily = tooltipModel._bodyFontFamily;
tooltipEl.style.fontSize = tooltipModel.bodyFontSize + 'px';
tooltipEl.style.fontStyle = tooltipModel._bodyFontStyle;
tooltipEl.style.padding = tooltipModel.yPadding + 'px ' + tooltipModel.xPadding + 'px';
tooltipEl.style.pointerEvents = 'none';
}
}
}
});
#chartjs-tooltip {
opacity: 1;
position: absolute;
background: rgba(0, 0, 0, .7);
color: white;
border-radius: 3px;
-webkit-transition: all .1s ease;
transition: all .1s ease;
pointer-events: none;
-webkit-transform: translate(-50%, 0);
transform: translate(-50%, 0);
}
<link rel="stylesheet" href="https://cdnjs.cloudflare.com/ajax/libs/Chart.js/2.8.0/Chart.min.css" />
<script src="https://cdnjs.cloudflare.com/ajax/libs/Chart.js/2.8.0/Chart.min.js"></script>
<canvas id="bar-chart" width="400" height="200"></canvas>