D3.js - Gráfico de barras empilhado de JSON?
Tenho tentado recriar este violino: https://jsfiddle.net/1c5eqrp3/3/
No contexto de um gráfico de barras empilhadas horizontais que estou construindo usando meus dados.
var bar1Data = [{
position: 1,
label: '70k',
value: 1708026
},
{
position: 2,
label: '71K - 149K',
value: 1915059
},
];
//sort bars based on value
bar1Data = bar1Data.sort(function(a, b) {
return d3.ascending(a.position, b.position);
})
var colors = ['green', 'blue'];
var bars = d3.select('#bars')
.append('svg')
.attr('width', 800)
.attr('height', 200);
bars.selectAll('rect')
.data(bar1Data)
.enter()
.append('rect')
.attr('width', function(d) {
return d;
})
.attr('x', function(d, i) {
return sum(bar1Data.value, 0, i);
})
.attr('fill', function(d, i) {
return colors[i];
})
.attr('y', 0)
.attr('height', 100);
function sum(array, start, end) {
var total = 0;
for (var i = start; i < end; i++) total += array[i];
return total;
}
<script src="https://cdnjs.cloudflare.com/ajax/libs/d3/5.7.0/d3.min.js"></script>
<div id="bar"></div>
Tudo estava indo bem, até que tentei fazer com que ele lesse o bar1Data que criei.
Agora, estou recebendo os seguintes erros:
Error: <rect> attribute width: Expected length, "[object Object]".
(anonymous) @ d3.js:1211
selection_each @ d3.js:1176
selection_attr @ d3.js:1233
(anonymous) @ bars3.js:23
Promise.then (async)
(anonymous) @ bars3.js:1
bars3.js:33 Uncaught (in promise) TypeError: Cannot read property '0' of undefined
at sum (bars3.js:33)
at SVGRectElement.<anonymous> (bars3.js:26)
at SVGRectElement.<anonymous> (d3.js:1209)
at Selection.selection_each [as each] (d3.js:1176)
at Selection.selection_attr [as attr] (d3.js:1233)
at bars3.js:25
Tenho me esforçado muito para criar um gráfico de barras horizontais empilhadas usando d3.v5. Tenho tentado vários exemplos usando d3.v4, mas não consegui fazê-los funcionar. Este foi o exemplo mais simples e leve que encontrei. Mas ainda não inclui escalas (que espero colocar um pouco mais tarde).
Você poderia me ajudar a resolver esse problema com uma explicação?
Respostas
Em .attr('width', function(d){ return d;})
, você está definindo a largura da barra para todo o objeto d
. Você precisa entender que isso .data(bar1Data)
cria uma entrada por valor de bar1Data
, e que esse valor é acessível como d
em todas essas funções. Isso significa que, assim que você passar .data(bar1Data)
, você não deve mais tocar diretamente naquele objeto.
Agora, a partir de seus dados, parece que você deseja usar os valores de todas as entradas anteriores para calcular a soma cumulativa até aquele ponto. Vou implementar isso na resposta, mas também gostaria de apontar para o stack = d3.stack()
que faz isso por você.
Observe também que seus valores são muito grandes. Você precisa transformá-los usando um domínio, caso contrário, você obtém um pixel por dólar (ou outra unidade de dinheiro), o que significa que as barras são desenhadas 100x a largura do seu monitor à direita.
var bar1Data = [{
position: 1,
label: '70k',
value: 50
},
{
position: 2,
label: '71K - 149K',
value: 40
},
];
//sort bars based on value
bar1Data = bar1Data.sort(function(a, b) {
return d3.ascending(a.position, b.position);
})
var colors = ['green', 'blue'];
var bars = d3.select('#bars')
.append('svg')
.attr('width', 800)
.attr('height', 200);
bars.selectAll('rect')
.data(bar1Data)
.enter()
.append('rect')
.attr('width', function(d) {
return d.value;
})
.attr('x', function(d, i) {
// Note that I use `.map` here to take only the `.value` attribute
// from the data and create an array of numbers.
return sum(bar1Data.map((e) => e.value), 0, i);
})
.attr('fill', function(d, i) {
return colors[i];
})
.attr('y', 0)
.attr('height', 100);
function sum(array, start, end) {
var total = 0;
for (var i = start; i < end; i++) total += array[i];
return total;
}
<script src="https://cdnjs.cloudflare.com/ajax/libs/d3/5.7.0/d3.min.js"></script>
<div id='bars'>
</div>