Javascript 使用d3.js在分组条形图中直观显示JSON数据
我需要关于使用D3.JS可视化显示JSON数据的帮助。我可以让图形显示出来,但我的问题是,这些条彼此堆叠在一起,而不是根据它们显示的类别转换成组。在下面,您可以找到我的代码以及指向我当前输出截图的链接,以及我使用的JSON文件 链接到屏幕抓图: 这是我的代码:Javascript 使用d3.js在分组条形图中直观显示JSON数据,javascript,json,d3.js,Javascript,Json,D3.js,我需要关于使用D3.JS可视化显示JSON数据的帮助。我可以让图形显示出来,但我的问题是,这些条彼此堆叠在一起,而不是根据它们显示的类别转换成组。在下面,您可以找到我的代码以及指向我当前输出截图的链接,以及我使用的JSON文件 链接到屏幕抓图: 这是我的代码: $(document).ready(function(){ var margin = {top: 20, right: 20, bottom: 30, left: 40}, width = 960 - margin.left - ma
$(document).ready(function(){
var margin = {top: 20, right: 20, bottom: 30, left: 40},
width = 960 - margin.left - margin.right,
height = 500 - margin.top - margin.bottom;
var x0 = d3.scale.ordinal()
.rangeRoundBands([0, width], .1);
var x1 = d3.scale.ordinal();
var y = d3.scale.linear()
.range([height, 0]);
var color = d3.scale.ordinal()
.range(["#98abc5", "#8a89a6", "#7b6888", "#6b486b", "#a05d56", "#d0743c", "#ff8c00"]);
var xAxis = d3.svg.axis()
.scale(x0)
.orient("bottom");
var yAxis = d3.svg.axis()
.scale(y)
.orient("left")
.tickFormat(d3.format(".2s"));
var svg = d3.select("body").append("svg")
.attr("width", width + margin.left + margin.right)
.attr("height", height + margin.top + margin.bottom)
.append("g")
.attr("transform", "translate(" + margin.left + "," + margin.top + ")");
d3.json("ronaldo.json", function(error, data) {
console.log(data);
var playerNames = []
data.forEach(function(d){playerNames.push(d.name)})
var attr = [];
data[0]['statistics']['2005'].forEach(function(d){return attr.push(d.attr)})
console.log(attr)
x0.domain(attr.map(function(d){return d}));
//x0.domain(data.map(function(d){return d['statistics']['2005']['attr']}));
x1.domain(playerNames).rangeRoundBands([0, x0.rangeBand()]);
y.domain([0, d3.max(data, function(d) { return d3.max(d["statistics"]["2005"], function(d) { return d.value; }); })]);
svg.append("g")
.attr("class", "x axis")
.attr("transform", "translate(0," + height + ")")
.call(xAxis);
svg.append("g")
.attr("class", "y axis")
.call(yAxis)
.append("text")
.attr("transform", "rotate(-90)")
.attr("y", 6)
.attr("dy", ".71em")
.style("text-anchor", "end")
.text("Units");
var state = svg.selectAll(".state")
.data(data)
.enter().append("g")
.attr("class", "g")
.attr("transform", function(d) { return "translate(" + x0(+d['statistics']['2005']['attr']) + ",0)"; });
state.selectAll("rect")
.data(function(d) { return d['statistics']['2005']; })
.enter().append("rect")
.attr("class","bars")
.attr("width", x1.rangeBand())
.attr("x", function(d) { return x1(d['attr']); })
.attr("y", function(d) { return y(d.value); })
.attr("height", function(d) { return height - y(d['value']); })
.style("fill", function(d) { return color(d.attr);});
var legend = svg.selectAll(".legend")
.data(playerNames.slice())
.enter().append("g")
.attr("class", "legend")
.attr("transform", function(d, i) { return "translate(0," + i * 20 + ")"; });
legend.append("rect")
.attr("x", width - 18)
.attr("width", 18)
.attr("height", 18)
.style("fill", color);
legend.append("text")
.attr("x", width - 24)
.attr("y", 9)
.attr("dy", ".35em")
.style("text-anchor", "end")
.text(function(d) { return d; });
});
});
JSON数据:
[
{
"name": "Cristiano Ronaldo",
"age" : 28,
"team": "Real Madrid",
"statistics" : {
"2005" : [
{"attr" :"Appearances",
"value": 100},
{"attr" :"Goals",
"value": 92},
{"attr" :"Yellow Cards",
"value": 10},
{"attr" :"Red Cards",
"value": 1}
]
}
},
{
"name": "Lionel Messi",
"age" : 29,
"team": "Barcelona",
"statistics" : {
"2005" : [
{"attr" :"Appearances",
"value": 90},
{"attr" :"Goals",
"value": 87},
{"attr" :"Yellow Cards",
"value": 13},
{"attr" :"Red Cards",
"value": 43}
]
}
}
]
我不明白为什么你有一个
x0
和一个x1
刻度,但你的问题就在这里
您应该只有一个在中使用的刻度:
- xAxis(定义
后,请小心覆盖xAxis的比例)x1
.attr(“x”,函数(d){returnx1(d['attr']);})
.attr(“x”,…)
行更改为:
.attr("x", function(d) { return x0(d['attr']); })
谢谢你的回复。我之所以使用x0和x1刻度,是因为x0刻度处理x轴上每个分组的间距(因此,当比较多个玩家时,他们各自的图形根据“外观”、“目标”等进行分割)。x1标尺处理各自分组内的钢筋间距。因此,最终显示应在x轴上每个类别显示多个条形图(在本例中为两个)。所以现在唯一缺少的是另一个球员(梅西)的酒吧,这是一个完全不同的任务,但我希望我的回答为你指明了正确的方向:)你现在必须为每个现有酒吧腾出2个酒吧的空间,而不是现在的1个。如果我正确理解了您的解释,那么使用x属性和宽度应该可以达到目的。