-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathwaterfallChart.js
106 lines (80 loc) · 3 KB
/
waterfallChart.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
var signs = [1, 1, 1, -1, -1, -1];
function generateWaterfallData(raw) {
return d3.zip(raw, signs).map(function(pair) {
return Object.assign({}, pair[0], { value: pair[0].value * pair[1] });
}).reduce(function(accum, current) {
var previous = accum[accum.length - 1];
var next = { start: previous.end, end: previous.end + current.value };
if (next.start < next.end) {
next.side = 1;
next.down = next.start;
next.up = next.end;
} else {
next.side = -1;
next.down = next.end;
next.up = next.start;
}
accum.push(Object.assign(next, current));
return accum;
}, [{ end: 0 }]).slice(1);
}
function waterfallChart(parent, dimensions) {
var svg = parent.append("svg")
.attr("width", dimensions.width + dimensions.margin.left + dimensions.margin.right)
.attr("height", dimensions.height + dimensions.margin.top + dimensions.margin.bottom)
.append("g")
.attr("transform", "translate(" + margin.left + "," + margin.top + ")");
var data;
var x = d3.scaleBand([0, dimensions.width])
.padding(0.1);
var y = d3.scaleLinear()
.range([height, 0]);
var xAxis = d3.axisBottom()
.scale(x);
var yAxis = d3.axisLeft()
.scale(y)
.ticks(8);
var title = svg.append("text")
.attr("class", "title")
.attr("y", -26);
var xAxisGroup = svg.append("g")
.attr("class", "x axis")
.attr("transform", "translate(0," + height + ")");
var yAxisGroup = svg.append("g")
.attr("class", "y axis");
function waterfallChart() { }
waterfallChart.data = function(_) {
return arguments.length ? (data = _, waterfallChart) : data;
}
waterfallChart.refresh = function(data) {
var waterfall = generateWaterfallData(data);
x.range([0, dimensions.width / 8 * data.length])
x.domain(waterfall.map(function(d) { return d.name; }));
xAxisGroup.call(xAxis);
xAxisGroup.selectAll(".tick text")
.call(wrap, x.step());
svg.select(".title").attr("x", x(waterfall[0].name))
y.domain([
d3.min(waterfall, function(d) { return d3.min([d.start, d.end]); }),
d3.max(waterfall, function(d) { return d3.max([d.start, d.end]); })
]);
yAxisGroup.call(yAxis);
var elements = svg.selectAll(".bar")
.data(waterfall, function(d) { return d.name; })
function apply(elements) {
elements.attr("class", function(d) { return "bar " + (d.side>=0 ? 'positive' : 'negative'); })
.attr("x", function(d) { return x(d.name); })
.attr("width", x.bandwidth())
.attr("y", function(d) { return y(d.up); })
.attr("height", function(d) { return y(d.down) - y(d.up); });
}
elements.enter().append("rect").call(apply);
elements.call(apply);
elements.exit().remove();
return waterfallChart;
}
waterfallChart.title = function(_) {
return arguments.length ? (title.text(_), waterfallChart) : title.text();
};
return waterfallChart;
}