|
function timeSeriesChart() { |
|
var margin = {top: 20, right: 20, bottom: 20, left: 20}, |
|
width = 760, |
|
height = 120, |
|
xValue = function(d) { return d[0]; }, |
|
yValue = function(d) { return d[1]; }, |
|
xScale = d3.scaleTime(), |
|
yScale = d3.scaleLinear(), |
|
xAxis = d3.axisBottom().scale(xScale).tickSize(6, 0), |
|
area = d3.area().x(X).y1(Y), |
|
line = d3.line().x(X).y(Y), |
|
values = function (d){ return d; }, |
|
title = null, |
|
titleX = -30, |
|
titleY = 25, |
|
xExtent = null; |
|
|
|
function chart(selection) { |
|
selection.each(function(d) { |
|
|
|
// Convert data to standard representation greedily; |
|
// this is needed for nondeterministic accessors. |
|
var data = values(d).map(function(d, i) { |
|
return [xValue.call(data, d, i), yValue.call(data, d, i)]; |
|
}); |
|
|
|
// Update the x-scale. |
|
xScale |
|
.domain(xExtent || d3.extent(data, function(d) { return d[0]; })) |
|
.range([0, width - margin.left - margin.right]); |
|
|
|
// Update the y-scale. |
|
yScale |
|
.domain([0, d3.max(data, function(d) { return d[1]; })]) |
|
.range([height - margin.top - margin.bottom, 0]); |
|
|
|
// Select the svg element, if it exists. |
|
var svg = d3.select(this).selectAll("svg").data([data]); |
|
var svgEnter = svg.enter().append("svg"); |
|
|
|
// Otherwise, create the skeletal chart. |
|
var gEnter = svgEnter.append("g"); |
|
gEnter.append("text").attr("class", "title"); |
|
gEnter.append("path").attr("class", "area"); |
|
gEnter.append("path").attr("class", "line"); |
|
gEnter.append("g").attr("class", "x axis"); |
|
|
|
// Update the outer dimensions. |
|
svg = svg.merge(svgEnter); |
|
svg .attr("width", width) |
|
.attr("height", height); |
|
|
|
// Update the inner dimensions. |
|
var g = svg.select("g") |
|
.attr("transform", "translate(" + margin.left + "," + margin.top + ")"); |
|
|
|
// Update the area path. |
|
g.select(".area") |
|
.attr("d", area.y0(yScale.range()[0])); |
|
|
|
// Update the line path. |
|
g.select(".line") |
|
.attr("d", line); |
|
|
|
// Update the x-axis. |
|
g.select(".x.axis") |
|
.attr("transform", "translate(0," + yScale.range()[0] + ")") |
|
.call(xAxis); |
|
|
|
// Update the title. |
|
if(title){ |
|
g.select(".title") |
|
.attr("x", titleX) |
|
.attr("y", titleY) |
|
.attr("text-anchor", "end") |
|
.text(title(d)); |
|
} |
|
}); |
|
} |
|
|
|
// The x-accessor for the path generator; xScale ∘ xValue. |
|
function X(d) { |
|
return xScale(d[0]); |
|
} |
|
|
|
// The x-accessor for the path generator; yScale ∘ yValue. |
|
function Y(d) { |
|
return yScale(d[1]); |
|
} |
|
|
|
chart.margin = function(_) { |
|
if (!arguments.length) return margin; |
|
margin = _; |
|
return chart; |
|
}; |
|
|
|
chart.width = function(_) { |
|
if (!arguments.length) return width; |
|
width = _; |
|
return chart; |
|
}; |
|
|
|
chart.height = function(_) { |
|
if (!arguments.length) return height; |
|
height = _; |
|
return chart; |
|
}; |
|
|
|
chart.x = function(_) { |
|
if (!arguments.length) return xValue; |
|
xValue = _; |
|
return chart; |
|
}; |
|
|
|
chart.y = function(_) { |
|
if (!arguments.length) return yValue; |
|
yValue = _; |
|
return chart; |
|
}; |
|
|
|
chart.values = function(_) { |
|
if (!arguments.length) return values; |
|
values = _; |
|
return chart; |
|
}; |
|
|
|
chart.title = function(_) { |
|
if (!arguments.length) return title; |
|
title = _; |
|
return chart; |
|
}; |
|
|
|
chart.titleX = function(_) { |
|
if (!arguments.length) return titleX; |
|
titleX = _; |
|
return chart; |
|
}; |
|
|
|
chart.titleY = function(_) { |
|
if (!arguments.length) return titleY; |
|
titleY = _; |
|
return chart; |
|
}; |
|
|
|
chart.xExtent = function(_) { |
|
if (!arguments.length) return xExtent; |
|
xExtent = _; |
|
return chart; |
|
}; |
|
|
|
return chart; |
|
} |