Для вас было бы проще вместо рендеринга ваших прямоугольников и последующего удаления их просто отфильтровать данные перед рукой.Проблема, с которой вы сталкиваетесь, заключается в том, что, хотя вы используете одни и те же данные для текста и прямоугольников, они на самом деле не имеют связей между ними.Как указал @ rioV8, правильным способом было бы использовать элемент g
для группировки связанных элементов.
Также я хочу, чтобы мои прямоугольники имели непрозрачность, но когда я вставляю эту строку кода.attr ("class", function (d) {if ((dx + d.width)> width || (dy + d.height)> height) return "delete"});они теряют свою непрозрачность
Это происходит потому, что attr
переопределяет предыдущие значения, и поскольку вы возвращаете либо "delete", либо undefined, у него больше нет класса "rect".При работе с классами вместо attr
используйте classed
, например
.classed('delete', function(d){
if (shouldBeDeleted) return true;
return false;
})
Таким образом, вы переключаете только класс delete
, и все другие классы для элемента сохраняются.
Здесь демонстрируется, как фильтровать данные и как группировать связанные прямоугольники и метки в элемент g
<!DOCTYPE html>
<head>
<meta charset="utf-8">
<script src="https://d3js.org/d3.v4.min.js"></script>
<style>
body { margin:0;position:fixed;top:0;right:0;bottom:0;left:0; }
svg { margin-left: auto; margin-right: auto; border: solid 1px; }
g.item rect {
fill-opacity: 0.2;
}
</style>
</head>
<body>
<script>
var data = [{
x: -50,
y: 150,
width: 100,
height: 100,
text: 'A',
color: '#f00'
}, {
x: 100,
y: 150,
width: 100,
height: 100,
text: 'B',
color: '#0f0'
}, {
x: 350,
y: 150,
width: 100,
height: 100,
text: 'C',
color: '#00f'
}];
var width = 400,
height = 400;
var svg = d3.select("body").append("svg")
.attr("width", width)
.attr("height", height);
var items = svg.selectAll('g.item')
.data(data.filter(function(i){
return i.x > 0 && i.x + i.width < width && i.y > 0 && i.y + i.height < height;
}));
items = items.enter()
.append('g')
.classed('item', true)
.merge(items)
.attr('transform', function(d){
return "translate("+d.x +"," +d.y+")";
});
items.append('rect')
.attr('width', (d) => d.width)
.attr('height', (d) => d.height)
.style('fill', (d) => d.color);
items.append('text')
.text((d) => d.text);
</script>
</body>