В D3 4.0, callback function for the .on()
method is passed 3 arguments: текущая точка привязки (D), то текущий индекс (i) и текущую группу (узлы).
В обратном вызове мыши можно указать selectAll("rect")
и отфильтровать элементы, находящиеся в текущей группе (node
). С помощью этого выбора вы установите непрозрачность на 0,5. На выводе вам просто нужно установить все непрозрачность на 1.0. Соответствующий код:
...
.on('mouseover', function(d, i, node) {
d3.selectAll("rect")
.filter(function (x) { return !isInArray(this, node)})
.attr('opacity', 0.5);
}
)
.on('mouseout', function() {
d3.selectAll("rect").attr('opacity', 1.0);
});
с небольшой вспомогательной функцией, чтобы проверить, если значение присутствует в массиве (массив элементов DOM в нашем случае):
function isInArray(value, array) {
return array.indexOf(value) > -1;
}
Полный код в контексте (с учетом вашего связанного примера):
g.append("g")
.selectAll("g")
.data(data)
.enter().append("g")
.attr("transform", function(d) { return "translate(" + x0(d.State) + ",0)"; })
.selectAll("rect")
.data(function(d) { return keys.map(function(key) { return {key: key, value: d[key]}; }); })
.enter().append("rect")
.attr("x", function(d) { return x1(d.key); })
.attr("y", function(d) { return y(d.value); })
.attr("width", x1.bandwidth())
.attr("height", function(d) { return height - y(d.value); })
.attr("fill", function(d) { return z(d.key); })
.on('mouseover', function(d, i, node) {
d3.selectAll("rect")
.filter(function (x) { return !isInArray(this, node)})
.attr('opacity', 0.5);
}
)
.on('mouseout', function() {
d3.selectAll("rect").attr('opacity', 1.0);
});
Удивительное решение, спасибо! – TommyF