Chart.js / Javascript如何覆盖Mouseout事件侦听器

时间:2019-05-06 18:50:20

标签: javascript jquery events chart.js

我正在尝试使用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以覆盖默认行为的方法吗?预先感谢。

1 个答案:

答案 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>