mirror of
https://github.com/gosticks/DefinitelyTyped.git
synced 2025-10-16 12:05:41 +00:00
* Added empty() and size() methods to interface Enter<Datum>. As per D3 API these methods are defined on the enter selection. * Added test function testEnterSizeEmpty() for .enter().empty() and .enter.size() methods.
2722 lines
89 KiB
TypeScript
2722 lines
89 KiB
TypeScript
/// <reference path="d3.d.ts" />
|
|
|
|
//Example from http://bl.ocks.org/3887235
|
|
interface TestPieChartData {
|
|
population: number;
|
|
age: string;
|
|
}
|
|
function testPieChart() {
|
|
var width = 960,
|
|
height = 500,
|
|
radius = Math.min(width, height) / 2;
|
|
|
|
var color = d3.scale.ordinal<string>()
|
|
.range(["#98abc5", "#8a89a6", "#7b6888", "#6b486b", "#a05d56", "#d0743c", "#ff8c00"]);
|
|
|
|
var arc = d3.svg.arc<d3.layout.pie.Arc<TestPieChartData>>()
|
|
.outerRadius(radius - 10)
|
|
.innerRadius(0);
|
|
|
|
var pie = d3.layout.pie<TestPieChartData>()
|
|
.sort(null)
|
|
.value(function (d) { return d.population; });
|
|
|
|
var svg = d3.select("body").append("svg")
|
|
.attr("width", width)
|
|
.attr("height", height)
|
|
.append("g")
|
|
.attr("transform", "translate(" + width / 2 + "," + height / 2 + ")");
|
|
|
|
d3.csv("data.csv", d => ({ population: +d['population'], age: d['age'] }), function (error, data) {
|
|
var g = svg.selectAll(".arc")
|
|
.data(pie(data))
|
|
.enter().append("g")
|
|
.attr("class", "arc");
|
|
|
|
g.append("path")
|
|
.attr("d", arc)
|
|
.style("fill", function (d) { return color(d.data.age); });
|
|
|
|
g.append("text")
|
|
.attr("transform", function (d) { return "translate(" + arc.centroid(d) + ")"; })
|
|
.attr("dy", ".35em")
|
|
.style("text-anchor", "middle")
|
|
.text(function (d) { return d.data.age; });
|
|
|
|
});
|
|
}
|
|
|
|
function testMapConstructor() {
|
|
//No arg constructor
|
|
var emptyMap: d3.Map<any> = d3.map();
|
|
|
|
//Object constructor
|
|
var object:{[key: string]: number } = {a: 1, b: 2, c: 3};
|
|
var objectMap: d3.Map<number> = d3.map<number>(object);
|
|
|
|
//Array constructor
|
|
var numberArray: number[] = [1, 2, 3]
|
|
var numberArrayMap: d3.Map<number> = d3.map<number>(numberArray);
|
|
|
|
//Array with keyFn constructor
|
|
var objectArray: {key: string}[] = [{key: "v1"}, {key: "v2"}, {key: "v3"}];
|
|
var indexes: number[] = [];
|
|
//keyFn with index
|
|
var objectArrayMap1: d3.Map<{key: string}>
|
|
= d3.map<{key: string}>(objectArray, (o: {key: string}, index: number) => {
|
|
indexes.push(index);
|
|
return o.key;
|
|
});
|
|
//keyFn without index
|
|
var objectArrayMap2: d3.Map<{key: string}>
|
|
= d3.map<{key: string}>(objectArray, (o: {key: string}) => {
|
|
return o.key;
|
|
});
|
|
|
|
//Map constructor
|
|
var duplicateMap: d3.Map<number> = d3.map(numberArrayMap);
|
|
}
|
|
|
|
//Example from http://bl.ocks.org/3887051
|
|
interface GroupedData {
|
|
State: string;
|
|
ages: Array<{ name: string; value: number;}>;
|
|
}
|
|
function groupedBarChart() {
|
|
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<number>();
|
|
|
|
var y = d3.scale.linear()
|
|
.range([height, 0]);
|
|
|
|
var color = d3.scale.ordinal<string>()
|
|
.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.csv("data.csv", <any> function (error: any, data: Array<GroupedData>) {
|
|
var ageNames = d3.keys(data[0]).filter(function (key) { return key !== "State"; });
|
|
|
|
data.forEach(function (d) {
|
|
d.ages = ageNames.map(function (name) { return { name: name, value: +(<any> d)[name] }; });
|
|
});
|
|
|
|
x0.domain(data.map(function (d) { return d.State; }));
|
|
x1.domain(ageNames).rangeRoundBands([0, x0.rangeBand()]);
|
|
y.domain([0, d3.max(data, function (d) { return d3.max(d.ages, 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("Population");
|
|
|
|
var state = svg.selectAll(".state")
|
|
.data(data)
|
|
.enter().append("g")
|
|
.attr("class", "g")
|
|
.attr("transform", function (d) { return "translate(" + x0(d.State) + ",0)"; });
|
|
|
|
state.selectAll<GroupedData>("rect")
|
|
.data(function (d) { return d.ages; })
|
|
.enter().append("rect")
|
|
.attr("width", x1.rangeBand())
|
|
.attr("x", function (d) { return x1(d.name); })
|
|
.attr("y", function (d) { return y(d.value); })
|
|
.attr("height", function (d) { return height - y(d.value); })
|
|
.style("fill", function (d) { return color(d.name); });
|
|
|
|
var legend = svg.selectAll(".legend")
|
|
.data(ageNames.reverse())
|
|
.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; });
|
|
|
|
});
|
|
}
|
|
|
|
//Example from http://bl.ocks.org/3886208
|
|
function stackedBarChart() {
|
|
var margin = { top: 20, right: 20, bottom: 30, left: 40 },
|
|
width = 960 - margin.left - margin.right,
|
|
height = 500 - margin.top - margin.bottom;
|
|
|
|
var x = d3.scale.ordinal()
|
|
.rangeRoundBands([0, width], .1);
|
|
|
|
var y = d3.scale.linear()
|
|
.rangeRound([height, 0]);
|
|
|
|
var color = d3.scale.ordinal<string>()
|
|
.range(["#98abc5", "#8a89a6", "#7b6888", "#6b486b", "#a05d56", "#d0743c", "#ff8c00"]);
|
|
|
|
var xAxis = d3.svg.axis()
|
|
.scale(x)
|
|
.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.csv("data.csv", function (error: any, data: any) {
|
|
color.domain(d3.keys(data[0]).filter(function (key) { return key !== "State"; }));
|
|
|
|
data.forEach(function (d: any) {
|
|
var y0 = 0;
|
|
d.ages = color.domain().map(function (name) { return { name: name, y0: y0, y1: y0 += +d[name] }; });
|
|
d.total = d.ages[d.ages.length - 1].y1;
|
|
});
|
|
|
|
data.sort(function (a: any, b: any) { return b.total - a.total; });
|
|
|
|
x.domain(data.map(function (d: any) { return d.State; }));
|
|
y.domain([0, d3.max(data, function (d: { total: number }) { return d.total; })]);
|
|
|
|
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("Population");
|
|
|
|
var state = svg.selectAll(".state")
|
|
.data(data)
|
|
.enter().append("g")
|
|
.attr("class", "g")
|
|
.attr("transform", function (d: any) { return "translate(" + x(d.State) + ",0)"; });
|
|
|
|
state.selectAll("rect")
|
|
.data(function (d: any) { return d.ages; })
|
|
.enter().append("rect")
|
|
.attr("width", x.rangeBand())
|
|
.attr("y", function (d: any) { return y(d.y1); })
|
|
.attr("height", function (d: any) { return y(d.y0) - y(d.y1); })
|
|
.style("fill", function (d: any) { return color(d.name); });
|
|
|
|
var legend = svg.selectAll(".legend")
|
|
.data(color.domain().reverse())
|
|
.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; });
|
|
|
|
});
|
|
}
|
|
|
|
// example from http://bl.ocks.org/3886394
|
|
function normalizedBarChart() {
|
|
var margin = { top: 20, right: 100, bottom: 30, left: 40 },
|
|
width = 960 - margin.left - margin.right,
|
|
height = 500 - margin.top - margin.bottom;
|
|
|
|
var x = d3.scale.ordinal()
|
|
.rangeRoundBands([0, width], .1);
|
|
|
|
var y = d3.scale.linear()
|
|
.rangeRound([height, 0]);
|
|
|
|
var color = d3.scale.ordinal<string>()
|
|
.range(["#98abc5", "#8a89a6", "#7b6888", "#6b486b", "#a05d56", "#d0743c", "#ff8c00"]);
|
|
|
|
var xAxis = d3.svg.axis()
|
|
.scale(x)
|
|
.orient("bottom");
|
|
|
|
var yAxis = d3.svg.axis()
|
|
.scale(y)
|
|
.orient("left")
|
|
.tickFormat(d3.format(".0%"));
|
|
|
|
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.csv("data.csv", function (error, data) {
|
|
color.domain(d3.keys(data[0]).filter(function (key) { return key !== "State"; }));
|
|
|
|
data.forEach(function (d: any) {
|
|
var y0 = 0;
|
|
d.ages = color.domain().map(function (name) { return { name: name, y0: y0, y1: y0 += +d[name] }; });
|
|
d.ages.forEach(function (d: any) { d.y0 /= y0; d.y1 /= y0; });
|
|
});
|
|
|
|
data.sort(function (a: any, b: any) { return b.ages[0].y1 - a.ages[0].y1; });
|
|
|
|
x.domain(data.map(function (d: any) { return d.State; }));
|
|
|
|
svg.append("g")
|
|
.attr("class", "x axis")
|
|
.attr("transform", "translate(0," + height + ")")
|
|
.call(xAxis);
|
|
|
|
svg.append("g")
|
|
.attr("class", "y axis")
|
|
.call(yAxis);
|
|
|
|
var state = svg.selectAll(".state")
|
|
.data(data)
|
|
.enter().append("g")
|
|
.attr("class", "state")
|
|
.attr("transform", function (d: any) { return "translate(" + x(d.State) + ",0)"; });
|
|
|
|
state.selectAll("rect")
|
|
.data(function (d) { return d.ages; })
|
|
.enter().append("rect")
|
|
.attr("width", x.rangeBand())
|
|
.attr("y", function (d: any) { return y(d.y1); })
|
|
.attr("height", function (d: any) { return y(d.y0) - y(d.y1); })
|
|
.style("fill", function (d: any) { return color(d.name); });
|
|
|
|
var legend = svg.select(".state:last-child").selectAll(".legend")
|
|
.data(function (d: any) { return d.ages; })
|
|
.enter().append("g")
|
|
.attr("class", "legend")
|
|
.attr("transform", function (d: any) { return "translate(" + x.rangeBand() / 2 + "," + y((d.y0 + d.y1) / 2) + ")"; });
|
|
|
|
legend.append("line")
|
|
.attr("x2", 10);
|
|
|
|
legend.append("text")
|
|
.attr("x", 13)
|
|
.attr("dy", ".35em")
|
|
.text(function (d: any) { return d.name; });
|
|
|
|
});
|
|
}
|
|
|
|
// example from http://bl.ocks.org/3885705
|
|
function sortablebarChart() {
|
|
var margin = { top: 20, right: 20, bottom: 30, left: 40 },
|
|
width = 960 - margin.left - margin.right,
|
|
height = 500 - margin.top - margin.bottom;
|
|
|
|
var formatPercent = d3.format(".0%");
|
|
|
|
var x = d3.scale.ordinal()
|
|
.rangeRoundBands([0, width], .1, 1);
|
|
|
|
var y = d3.scale.linear()
|
|
.range([height, 0]);
|
|
|
|
var xAxis = d3.svg.axis()
|
|
.scale(x)
|
|
.orient("bottom");
|
|
|
|
var yAxis = d3.svg.axis()
|
|
.scale(y)
|
|
.orient("left")
|
|
.tickFormat(formatPercent);
|
|
|
|
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.tsv("data.tsv", function (error: any, data: any) {
|
|
|
|
data.forEach(function (d: any) {
|
|
d.frequency = +d.frequency;
|
|
});
|
|
|
|
x.domain(data.map(function (d: any) { return d.letter; }));
|
|
y.domain([0, d3.max(data, function (d: any) { return d.frequency; })]);
|
|
|
|
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("Frequency");
|
|
|
|
svg.selectAll(".bar")
|
|
.data(data)
|
|
.enter().append("rect")
|
|
.attr("class", "bar")
|
|
.attr("x", function (d: any) { return x(d.letter); })
|
|
.attr("width", x.rangeBand())
|
|
.attr("y", function (d: any) { return y(d.frequency); })
|
|
.attr("height", function (d: any) { return height - y(d.frequency); });
|
|
|
|
d3.select("input").on("change", change);
|
|
|
|
var sortTimeout = setTimeout(function () {
|
|
d3.select("input").property("checked", true).each(change);
|
|
}, 2000);
|
|
|
|
function change() {
|
|
clearTimeout(sortTimeout);
|
|
|
|
var x0 = x.domain(data.sort(this.checked
|
|
? function (a: any, b: any) { return b.frequency - a.frequency; }
|
|
: function (a: any, b: any) { return d3.ascending(a.letter, b.letter); })
|
|
.map(function (d: any) { return d.letter; }))
|
|
.copy();
|
|
|
|
var transition = svg.transition().duration(750),
|
|
delay = function (d: any, i: number) { return i * 50; };
|
|
|
|
transition.selectAll(".bar")
|
|
.delay(delay)
|
|
.attr("x", function (d: any) { return x0(d.letter); });
|
|
|
|
transition.select(".x.axis")
|
|
.call(xAxis)
|
|
.selectAll("g")
|
|
.delay(delay);
|
|
}
|
|
});
|
|
}
|
|
|
|
//example from http://bl.ocks.org/4063318
|
|
function callenderView() {
|
|
var width = 960,
|
|
height = 136,
|
|
cellSize = 17; // cell size
|
|
|
|
var day = d3.time.format("%w"),
|
|
week = d3.time.format("%U"),
|
|
percent = d3.format(".1%"),
|
|
format = d3.time.format("%Y-%m-%d");
|
|
|
|
var color = d3.scale.quantize()
|
|
.domain([-.05, .05])
|
|
.range(d3.range(11).map(function (d) { return "q" + d + "-11"; }));
|
|
|
|
var svg = d3.select("body").selectAll("svg")
|
|
.data(d3.range(1990, 2011))
|
|
.enter().append("svg")
|
|
.attr("width", width)
|
|
.attr("height", height)
|
|
.attr("class", "RdYlGn")
|
|
.append("g")
|
|
.attr("transform", "translate(" + ((width - cellSize * 53) / 2) + "," + (height - cellSize * 7 - 1) + ")");
|
|
|
|
svg.append("text")
|
|
.attr("transform", "translate(-6," + cellSize * 3.5 + ")rotate(-90)")
|
|
.style("text-anchor", "middle")
|
|
.text(function (d) { return d; });
|
|
|
|
var rect = svg.selectAll(".day")
|
|
.data(function (d) { return d3.time.days(new Date(d, 0, 1), new Date(d + 1, 0, 1)); })
|
|
.enter().append("rect")
|
|
.attr("class", "day")
|
|
.attr("width", cellSize)
|
|
.attr("height", cellSize)
|
|
.attr("x", function (d) { return parseInt(week(d)) * cellSize; })
|
|
.attr("y", function (d) { return parseInt(day(d)) * cellSize; })
|
|
.datum(format);
|
|
|
|
rect.append("title")
|
|
.text(function (d) { return d; });
|
|
|
|
svg.selectAll(".month")
|
|
.data(function (d) { return d3.time.months(new Date(d, 0, 1), new Date(d + 1, 0, 1)); })
|
|
.enter().append("path")
|
|
.attr("class", "month")
|
|
.attr("d", monthPath);
|
|
|
|
d3.csv("dji.csv", function (error, csv) {
|
|
var data = d3.nest()
|
|
.key(function (d: any) { return d.Date; })
|
|
.rollup(function (d: any) { return (d[0].Close - d[0].Open) / d[0].Open; })
|
|
.map(csv);
|
|
|
|
rect.filter(function (d) { return d in data; })
|
|
.attr("class", function (d) { return "day " + color(data[d]); })
|
|
.select("title")
|
|
.text(function (d) { return d + ": " + percent(data[d]); });
|
|
});
|
|
|
|
function monthPath(t0: Date) {
|
|
var t1 = new Date(t0.getFullYear(), t0.getMonth() + 1, 0),
|
|
d0 = +day(t0), w0 = +week(t0),
|
|
d1 = +day(t1), w1 = +week(t1);
|
|
return "M" + (w0 + 1) * cellSize + "," + d0 * cellSize
|
|
+ "H" + w0 * cellSize + "V" + 7 * cellSize
|
|
+ "H" + w1 * cellSize + "V" + (d1 + 1) * cellSize
|
|
+ "H" + (w1 + 1) * cellSize + "V" + 0
|
|
+ "H" + (w0 + 1) * cellSize + "Z";
|
|
}
|
|
|
|
d3.select(self.frameElement).style("height", "2910px");
|
|
}
|
|
|
|
// example from http://bl.ocks.org/3883245
|
|
function lineChart() {
|
|
var margin = { top: 20, right: 20, bottom: 30, left: 50 },
|
|
width = 960 - margin.left - margin.right,
|
|
height = 500 - margin.top - margin.bottom;
|
|
|
|
var parseDate = d3.time.format("%d-%b-%y").parse;
|
|
|
|
var x = d3.time.scale<number>()
|
|
.range([0, width]);
|
|
|
|
var y = d3.scale.linear()
|
|
.range([height, 0]);
|
|
|
|
var xAxis = d3.svg.axis()
|
|
.scale(x)
|
|
.orient("bottom");
|
|
|
|
var yAxis = d3.svg.axis()
|
|
.scale(y)
|
|
.orient("left");
|
|
|
|
var line = d3.svg.line<{ date: Date; close: number }>()
|
|
.x(function (d) { return x(d.date); })
|
|
.y(function (d) { return y(d.close); });
|
|
|
|
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.tsv("data.tsv", function (error: any, data: any) {
|
|
data.forEach(function (d: any) {
|
|
d.date = parseDate(d.date);
|
|
d.close = +d.close;
|
|
});
|
|
|
|
x.domain(d3.extent(data, function (d: any) { return d.date; }));
|
|
y.domain(d3.extent(data, function (d: any) { return d.close; }));
|
|
|
|
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("Price ($)");
|
|
|
|
svg.append("path")
|
|
.datum(data)
|
|
.attr("class", "line")
|
|
.attr("d", line);
|
|
});
|
|
}
|
|
|
|
//example from http://bl.ocks.org/3884914
|
|
function bivariateAreaChart() {
|
|
var margin = { top: 20, right: 20, bottom: 30, left: 50 },
|
|
width = 960 - margin.left - margin.right,
|
|
height = 500 - margin.top - margin.bottom;
|
|
|
|
var parseDate = d3.time.format("%Y%m%d").parse;
|
|
|
|
var x = d3.time.scale()
|
|
.range([0, width]);
|
|
|
|
var y = d3.scale.linear()
|
|
.range([height, 0]);
|
|
|
|
var xAxis = d3.svg.axis()
|
|
.scale(x)
|
|
.orient("bottom");
|
|
|
|
var yAxis = d3.svg.axis()
|
|
.scale(y)
|
|
.orient("left");
|
|
|
|
var area = d3.svg.area<{ date: Date; low: number; high: number }>()
|
|
.x(function (d) { return x(d.date); })
|
|
.y0(function (d) { return y(d.low); })
|
|
.y1(function (d) { return y(d.high); });
|
|
|
|
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.tsv("data.tsv", function (error: any, data: any) {
|
|
data.forEach(function (d: any) {
|
|
d.date = parseDate(d.date);
|
|
d.low = +d.low;
|
|
d.high = +d.high;
|
|
});
|
|
|
|
x.domain(d3.extent(data, function (d: any) { return d.date; }));
|
|
y.domain([d3.min(data, function (d: any) { return d.low; }), d3.max(data, function (d: any) { return d.high; })]);
|
|
|
|
svg.append("path")
|
|
.datum(data)
|
|
.attr("class", "area")
|
|
.attr("d", area);
|
|
|
|
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("Temperature (ºF)");
|
|
});
|
|
}
|
|
|
|
//Example from http://bl.ocks.org/mbostock/1557377
|
|
function dragMultiples() {
|
|
var width = 238,
|
|
height = 123,
|
|
radius = 20;
|
|
|
|
var drag = d3.behavior.drag()
|
|
.origin(Object)
|
|
.on("drag", dragmove);
|
|
|
|
var svg = d3.select("body").selectAll("svg")
|
|
.data(d3.range(16).map(function () { return { x: width / 2, y: height / 2 }; }))
|
|
.enter().append("svg")
|
|
.attr("width", width)
|
|
.attr("height", height);
|
|
|
|
svg.append("circle")
|
|
.attr("r", radius)
|
|
.attr("cx", function (d) { return d.x; })
|
|
.attr("cy", function (d) { return d.y; })
|
|
.call(drag);
|
|
|
|
function dragmove(d: { x: number; y: number }) {
|
|
d3.select(this)
|
|
.attr("cx", d.x = Math.max(radius, Math.min(width - radius, (<d3.DragEvent> d3.event).x)))
|
|
.attr("cy", d.y = Math.max(radius, Math.min(height - radius, (<d3.DragEvent> d3.event).y)));
|
|
}
|
|
}
|
|
|
|
//Example from http://bl.ocks.org/mbostock/3892919
|
|
function panAndZoom() {
|
|
var margin = { top: 20, right: 20, bottom: 30, left: 40 },
|
|
width = 960 - margin.left - margin.right,
|
|
height = 500 - margin.top - margin.bottom;
|
|
|
|
var x = d3.scale.linear()
|
|
.domain([-width / 2, width / 2])
|
|
.range([0, width]);
|
|
|
|
var y = d3.scale.linear()
|
|
.domain([-height / 2, height / 2])
|
|
.range([height, 0]);
|
|
|
|
var xAxis = d3.svg.axis()
|
|
.scale(x)
|
|
.orient("bottom")
|
|
.tickSize(-height);
|
|
|
|
var yAxis = d3.svg.axis()
|
|
.scale(y)
|
|
.orient("left")
|
|
.ticks(5)
|
|
.tickSize(-width);
|
|
|
|
var zoom = d3.behavior.zoom()
|
|
.x(x)
|
|
.y(y)
|
|
.scaleExtent([1, 10])
|
|
.on("zoom", zoomed);
|
|
|
|
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 + ")")
|
|
.call(zoom);
|
|
|
|
svg.append("rect")
|
|
.attr("width", width)
|
|
.attr("height", height);
|
|
|
|
svg.append("g")
|
|
.attr("class", "x axis")
|
|
.attr("transform", "translate(0," + height + ")")
|
|
.call(xAxis);
|
|
|
|
svg.append("g")
|
|
.attr("class", "y axis")
|
|
.call(yAxis);
|
|
|
|
function zoomed() {
|
|
svg.select(".x.axis").call(xAxis);
|
|
svg.select(".y.axis").call(yAxis);
|
|
}
|
|
}
|
|
|
|
//Example from http://bl.ocks.org/mbostock/1125997
|
|
function chainedTransitions() {
|
|
var w = 960,
|
|
h = 500,
|
|
y = d3.scale.ordinal<number, number>().domain(d3.range(50)).rangePoints([20, h - 20]),
|
|
t = Date.now();
|
|
|
|
var svg = d3.select("body").append("svg:svg")
|
|
.attr("width", w)
|
|
.attr("height", h);
|
|
|
|
var circle = svg.selectAll("circle")
|
|
.data(y.domain())
|
|
.enter().append("svg:circle")
|
|
.attr("r", 16)
|
|
.attr("cx", 20)
|
|
.attr("cy", y)
|
|
.each(slide(20, w - 20));
|
|
|
|
function slide(x0: number, x1: number) {
|
|
t += 50;
|
|
return function() {
|
|
d3.select(this).transition()
|
|
.duration(t - Date.now())
|
|
.attr("cx", x1)
|
|
.each("end", slide(x1, x0));
|
|
};
|
|
}
|
|
}
|
|
|
|
//Example from http://bl.ocks.org/mbostock/4062085
|
|
interface PyramidData {
|
|
people: number;
|
|
year: number;
|
|
age: number;
|
|
}
|
|
function populationPyramid() {
|
|
var margin = { top: 20, right: 40, bottom: 30, left: 20 },
|
|
width = 960 - margin.left - margin.right,
|
|
height = 500 - margin.top - margin.bottom,
|
|
barWidth = Math.floor(width / 19) - 1;
|
|
|
|
var x = d3.scale.linear()
|
|
.range([barWidth / 2, width - barWidth / 2]);
|
|
|
|
var y = d3.scale.linear()
|
|
.range([height, 0]);
|
|
|
|
var yAxis = d3.svg.axis()
|
|
.scale(y)
|
|
.orient("right")
|
|
.tickSize(-width)
|
|
.tickFormat(function (d) { return Math.round(d / 1e6) + "M"; } );
|
|
|
|
// An SVG element with a bottom-right origin.
|
|
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 + ")");
|
|
|
|
// A sliding container to hold the bars by birthyear.
|
|
var birthyears = svg.append("g")
|
|
.attr("class", "birthyears");
|
|
|
|
// A label for the current year.
|
|
var title = svg.append("text")
|
|
.attr("class", "title")
|
|
.attr("dy", ".71em")
|
|
.text(2000);
|
|
|
|
d3.csv("population.csv", <any> function (error: any, rows: Array<PyramidData>) {
|
|
|
|
// Convert strings to numbers.
|
|
rows.forEach(function (d) {
|
|
d.people = +d.people;
|
|
d.year = +d.year;
|
|
d.age = +d.age;
|
|
} );
|
|
|
|
// Compute the extent of the data set in age and years.
|
|
var age1 = d3.max(rows, function (d) { return d.age; } ),
|
|
year0 = d3.min(rows, function (d) { return d.year; } ),
|
|
year1 = d3.max(rows, function (d) { return d.year; } ),
|
|
year = year1;
|
|
|
|
// Update the scale domains.
|
|
x.domain([year1 - age1, year1]);
|
|
y.domain([0, d3.max(rows, function (d) { return d.people; } )]);
|
|
|
|
// Produce a map from year and birthyear to [male, female].
|
|
var data = d3.nest<PyramidData>()
|
|
.key(function (d) { return '' + d.year; } )
|
|
.key(function (d) { return '' + (d.year - d.age); } )
|
|
.rollup(function (v) { return v.map(function (d) { return d.people; } ); } )
|
|
.map(rows);
|
|
|
|
// Add an axis to show the population values.
|
|
svg.append("g")
|
|
.attr("class", "y axis")
|
|
.attr("transform", "translate(" + width + ",0)")
|
|
.call(yAxis)
|
|
.selectAll("g")
|
|
.filter(function (value) { return !value; } )
|
|
.classed("zero", true);
|
|
|
|
// Add labeled rects for each birthyear (so that no enter or exit is required).
|
|
var birthyear = birthyears.selectAll(".birthyear")
|
|
.data(d3.range(year0 - age1, year1 + 1, 5))
|
|
.enter().append("g")
|
|
.attr("class", "birthyear")
|
|
.attr("transform", function (birthyear) { return "translate(" + x(birthyear) + ",0)"; } );
|
|
|
|
birthyear.selectAll("rect")
|
|
.data(function (birthyear): number[] { return data[year][birthyear] || [0, 0]; } )
|
|
.enter().append("rect")
|
|
.attr("x", -barWidth / 2)
|
|
.attr("width", barWidth)
|
|
.attr("y", y)
|
|
.attr("height", function (value) { return height - y(value); } );
|
|
|
|
// Add labels to show birthyear.
|
|
birthyear.append("text")
|
|
.attr("y", height - 4)
|
|
.text(function (birthyear) { return birthyear; } );
|
|
|
|
// Add labels to show age (separate; not animated).
|
|
svg.selectAll(".age")
|
|
.data(d3.range(0, age1 + 1, 5))
|
|
.enter().append("text")
|
|
.attr("class", "age")
|
|
.attr("x", function (age) { return x(year - age); } )
|
|
.attr("y", height + 4)
|
|
.attr("dy", ".71em")
|
|
.text(function (age) { return age; } );
|
|
|
|
// Allow the arrow keys to change the displayed year.
|
|
window.focus();
|
|
d3.select(window).on("keydown", function () {
|
|
switch ((<KeyboardEvent> d3.event).keyCode) {
|
|
case 37: year = Math.max(year0, year - 10); break;
|
|
case 39: year = Math.min(year1, year + 10); break;
|
|
}
|
|
update();
|
|
} );
|
|
|
|
function update() {
|
|
if (!(<any>year in data)) return;
|
|
title.text(year);
|
|
|
|
birthyears.transition()
|
|
.duration(750)
|
|
.attr("transform", "translate(" + (x(year1) - x(year)) + ",0)");
|
|
|
|
birthyear.selectAll("rect")
|
|
.data(function (birthyear): number[] { return data[year][birthyear] || [0, 0]; } )
|
|
.transition()
|
|
.duration(750)
|
|
.attr("y", y)
|
|
.attr("height", function (value) { return height - y(value); } );
|
|
}
|
|
} );
|
|
}
|
|
|
|
//Example from http://bl.ocks.org/MoritzStefaner/1377729
|
|
namespace forcedBasedLabelPlacemant {
|
|
interface Node extends d3.layout.force.Node {
|
|
label: string;
|
|
}
|
|
|
|
interface LabelAnchor extends d3.layout.force.Node {
|
|
node: Node;
|
|
}
|
|
|
|
interface LabelAnchorLink extends d3.layout.force.Link<Node> {
|
|
source: Node;
|
|
target: Node;
|
|
weight: number;
|
|
}
|
|
|
|
var w = 960, h = 500;
|
|
|
|
var labelDistance = 0;
|
|
|
|
var vis = d3.select("body").append("svg:svg").attr("width", w).attr("height", h);
|
|
|
|
var nodes: Node[] = [];
|
|
var labelAnchors: LabelAnchor[] = [];
|
|
var labelAnchorLinks: { source: number; target: number; weight: number }[] = [];
|
|
var links: typeof labelAnchorLinks = [];
|
|
|
|
for (var i = 0; i < 30; i++) {
|
|
var nodeLabel = {
|
|
label: "node " + i
|
|
};
|
|
nodes.push(nodeLabel);
|
|
labelAnchors.push({
|
|
node: nodeLabel
|
|
});
|
|
labelAnchors.push({
|
|
node: nodeLabel
|
|
});
|
|
};
|
|
|
|
for (var i = 0; i < nodes.length; i++) {
|
|
for (var j = 0; j < i; j++) {
|
|
if (Math.random() > .95)
|
|
links.push({
|
|
source: i,
|
|
target: j,
|
|
weight: Math.random()
|
|
});
|
|
}
|
|
labelAnchorLinks.push({
|
|
source: i * 2,
|
|
target: i * 2 + 1,
|
|
weight: 1
|
|
});
|
|
};
|
|
|
|
var force = d3.layout.force<LabelAnchorLink, Node>().size([w, h]).nodes(nodes).links(links).gravity(1).linkDistance(50).charge(-3000).linkStrength(function (x) {
|
|
return x.weight * 10
|
|
} );
|
|
|
|
|
|
force.start();
|
|
|
|
var force2 = d3.layout.force<LabelAnchorLink, LabelAnchor>().nodes(labelAnchors).links(labelAnchorLinks).gravity(0).linkDistance(0).linkStrength(8).charge(-100).size([w, h]);
|
|
force2.start();
|
|
|
|
var link = vis.selectAll("line.link").data(links).enter().append("svg:line").attr("class", "link").style("stroke", "#CCC");
|
|
|
|
var node = vis.selectAll("g.node").data(force.nodes()).enter().append("svg:g").attr("class", "node");
|
|
node.append("svg:circle").attr("r", 5).style("fill", "#555").style("stroke", "#FFF").style("stroke-width", 3);
|
|
node.call(force.drag);
|
|
|
|
|
|
var anchorLink = vis.selectAll("line.anchorLink").data(labelAnchorLinks)//.enter().append("svg:line").attr("class", "anchorLink").style("stroke", "#999");
|
|
|
|
var anchorNode = vis.selectAll("g.anchorNode").data(force2.nodes()).enter().append("svg:g").attr("class", "anchorNode");
|
|
anchorNode.append("svg:circle").attr("r", 0).style("fill", "#FFF");
|
|
anchorNode.append("svg:text").text(function (d, i) {
|
|
return i % 2 == 0 ? "" : d.node.label
|
|
} ).style("fill", "#555").style("font-family", "Arial").style("font-size", 12);
|
|
|
|
var updateLink = function () {
|
|
(<d3.Selection<any>> this).attr("x1", function (d) {
|
|
return d.source.x;
|
|
} ).attr("y1", function (d) {
|
|
return d.source.y;
|
|
} ).attr("x2", function (d) {
|
|
return d.target.x;
|
|
} ).attr("y2", function (d) {
|
|
return d.target.y;
|
|
} );
|
|
|
|
}
|
|
|
|
var updateNode = function () {
|
|
(<d3.Selection<any>> this).attr("transform", function (d) {
|
|
return "translate(" + d.x + "," + d.y + ")";
|
|
} );
|
|
|
|
}
|
|
|
|
force.on("tick", function () {
|
|
|
|
force2.start();
|
|
|
|
node.call(updateNode);
|
|
|
|
anchorNode.each(function (d, i) {
|
|
if (i % 2 == 0) {
|
|
d.x = d.node.x;
|
|
d.y = d.node.y;
|
|
} else {
|
|
var b = this.childNodes[1].getBBox();
|
|
|
|
var diffX = d.x - d.node.x;
|
|
var diffY = d.y - d.node.y;
|
|
|
|
var dist = Math.sqrt(diffX * diffX + diffY * diffY);
|
|
|
|
var shiftX = b.width * (diffX - dist) / (dist * 2);
|
|
shiftX = Math.max(-b.width, Math.min(0, shiftX));
|
|
var shiftY = 5;
|
|
this.childNodes[1].setAttribute("transform", "translate(" + shiftX + "," + shiftY + ")");
|
|
}
|
|
} );
|
|
|
|
|
|
anchorNode.call(updateNode);
|
|
|
|
link.call(updateLink);
|
|
anchorLink.call(updateLink);
|
|
|
|
} );
|
|
}
|
|
|
|
//Example from http://bl.ocks.org/mbostock/1125997
|
|
namespace forceCollapsable {
|
|
interface Node extends d3.layout.force.Node {
|
|
id: string;
|
|
_children: Node[];
|
|
children?: Node[];
|
|
size: number;
|
|
}
|
|
|
|
var w = 1280,
|
|
h = 800,
|
|
node: d3.selection.Update<Node>,
|
|
link: d3.selection.Update<d3.layout.force.Link<Node>>,
|
|
root: any;
|
|
|
|
var force = d3.layout.force<Node>()
|
|
.on("tick", tick)
|
|
.charge(function (d) { return d._children ? -d.size / 100 : -30; } )
|
|
.linkDistance(function (d) { return d.target._children ? 80 : 30; } )
|
|
.size([w, h - 160]);
|
|
|
|
var vis = d3.select("body").append("svg:svg")
|
|
.attr("width", w)
|
|
.attr("height", h);
|
|
|
|
d3.json("flare.json", function (json) {
|
|
root = json;
|
|
root.fixed = true;
|
|
root.x = w / 2;
|
|
root.y = h / 2 - 80;
|
|
update();
|
|
} );
|
|
|
|
function update() {
|
|
var nodes = flatten(root),
|
|
links = d3.layout.tree<Node>().links(nodes);
|
|
|
|
// Restart the force layout.
|
|
force
|
|
.nodes(nodes)
|
|
.links(links)
|
|
.start();
|
|
|
|
// Update the links…
|
|
link = vis.selectAll("line.link")
|
|
.data(links, function (d) { return d.target.id; } );
|
|
|
|
// Enter any new links.
|
|
link.enter().insert("svg:line", ".node")
|
|
.attr("class", "link")
|
|
.attr("x1", function (d) { return d.source.x; } )
|
|
.attr("y1", function (d) { return d.source.y; } )
|
|
.attr("x2", function (d) { return d.target.x; } )
|
|
.attr("y2", function (d) { return d.target.y; } );
|
|
|
|
// Exit any old links.
|
|
link.exit().remove();
|
|
|
|
// Update the nodes…
|
|
node = vis.selectAll("circle.node")
|
|
.data(nodes, function (d) { return d.id; } )
|
|
.style("fill", color);
|
|
|
|
node.transition()
|
|
.attr("r", function (d) { return d.children ? 4.5 : Math.sqrt(d.size) / 10; } );
|
|
|
|
// Enter any new nodes.
|
|
node.enter().append("svg:circle")
|
|
.attr("class", "node")
|
|
.attr("cx", function (d) { return d.x; } )
|
|
.attr("cy", function (d) { return d.y; } )
|
|
.attr("r", function (d) { return d.children ? 4.5 : Math.sqrt(d.size) / 10; } )
|
|
.style("fill", color)
|
|
.on("click", click)
|
|
.call(force.drag);
|
|
|
|
// Exit any old nodes.
|
|
node.exit().remove();
|
|
}
|
|
|
|
function tick() {
|
|
link.attr("x1", function (d) { return d.source.x; } )
|
|
.attr("y1", function (d) { return d.source.y; } )
|
|
.attr("x2", function (d) { return d.target.x; } )
|
|
.attr("y2", function (d) { return d.target.y; } );
|
|
|
|
node.attr("cx", function (d) { return d.x; } )
|
|
.attr("cy", function (d) { return d.y; } );
|
|
}
|
|
|
|
// Color leaf nodes orange, and packages white or blue.
|
|
function color(d: Node) {
|
|
return d._children ? "#3182bd" : d.children ? "#c6dbef" : "#fd8d3c";
|
|
}
|
|
|
|
// Toggle children on click.
|
|
function click(d: Node) {
|
|
if (d.children) {
|
|
d._children = d.children;
|
|
d.children = null;
|
|
} else {
|
|
d.children = d._children;
|
|
d._children = null;
|
|
}
|
|
update();
|
|
}
|
|
|
|
// Returns a list of all nodes under the root.
|
|
function flatten(root: Node) {
|
|
var nodes: Node[] = [], i = 0;
|
|
|
|
function recurse(node: Node) {
|
|
if (node.children) node.size = node.children.reduce(function (p, v) { return p + recurse(v); } , 0);
|
|
if (!node.id) node.id = String(++i);
|
|
nodes.push(node);
|
|
return node.size;
|
|
}
|
|
|
|
root.size = recurse(root);
|
|
return nodes;
|
|
}
|
|
}
|
|
|
|
//Example from http://bl.ocks.org/mbostock/3757110
|
|
function azimuthalEquidistant() {
|
|
var width = 960,
|
|
height = 960;
|
|
var topojson: any;
|
|
|
|
var projection = d3.geo.azimuthalEquidistant()
|
|
.scale(150)
|
|
.translate([width / 2, height / 2])
|
|
.clipAngle(180 - 1e-3)
|
|
.precision(.1);
|
|
|
|
var path = d3.geo.path()
|
|
.projection(projection);
|
|
|
|
var graticule = d3.geo.graticule();
|
|
|
|
var svg = d3.select("body").append("svg")
|
|
.attr("width", width)
|
|
.attr("height", height);
|
|
|
|
svg.append("defs").append("path")
|
|
.datum({ type: "Sphere" })
|
|
.attr("id", "sphere")
|
|
.attr("d", path);
|
|
|
|
svg.append("use")
|
|
.attr("class", "stroke")
|
|
.attr("xlink:href", "#sphere");
|
|
|
|
svg.append("use")
|
|
.attr("class", "fill")
|
|
.attr("xlink:href", "#sphere");
|
|
|
|
svg.append("path")
|
|
.datum(graticule)
|
|
.attr("class", "graticule")
|
|
.attr("d", path);
|
|
|
|
d3.json("/mbostock/raw/4090846/world-50m.json", function (error, world) {
|
|
svg.insert("path", ".graticule")
|
|
.datum(topojson.feature(world, world.objects.land))
|
|
.attr("class", "land")
|
|
.attr("d", path);
|
|
|
|
svg.insert("path", ".graticule")
|
|
.datum(topojson.mesh(world, world.objects.countries, function (a: any, b: any) { return a !== b; } ))
|
|
.attr("class", "boundary")
|
|
.attr("d", path);
|
|
} );
|
|
|
|
d3.select(self.frameElement).style("height", height + "px");
|
|
}
|
|
|
|
//Example from http://bl.ocks.org/mbostock/4060366
|
|
function voronoiTesselation() {
|
|
var width = 960,
|
|
height = 500;
|
|
|
|
var vertices = d3.range(100).map(function (d): [number, number] {
|
|
return [Math.random() * width, Math.random() * height];
|
|
} );
|
|
|
|
var voronoi = d3.geom.voronoi()
|
|
.clipExtent([[0, 0], [width, height]]);
|
|
|
|
var svg = d3.select("body").append("svg")
|
|
.attr("width", width)
|
|
.attr("height", height)
|
|
.attr("class", "PiYG")
|
|
.on("mousemove", function () { vertices[0] = d3.mouse(this); redraw(); } );
|
|
|
|
var path = <d3.selection.Update<string>>svg.append("g").selectAll("path");
|
|
|
|
svg.selectAll("circle")
|
|
.data(vertices.slice(1))
|
|
.enter().append("circle")
|
|
.attr("transform", function (d) { return "translate(" + d + ")"; } )
|
|
.attr("r", 2);
|
|
|
|
redraw();
|
|
|
|
function redraw() {
|
|
path = path.data(voronoi(vertices).map(function (d) { return "M" + d.join("L") + "Z"; } ), String);
|
|
path.exit().remove();
|
|
path.enter().append("path").attr("class", function (d, i) { return "q" + (i % 9) + "-9"; } ).attr("d", String);
|
|
path.order();
|
|
}
|
|
}
|
|
|
|
// Example from https://gist.github.com/christophermanning/1734663
|
|
function forceDirectedVoronoi() {
|
|
var w = window.innerWidth > 960 ? 960 : (window.innerWidth || 960),
|
|
h = window.innerHeight > 500 ? 500 : (window.innerHeight || 500),
|
|
radius = 5.25,
|
|
simulate = true,
|
|
zoomToAdd = true,
|
|
color = d3.scale.quantize<string>().domain([10000, 7250]).range(["#dadaeb","#bcbddc","#9e9ac8","#807dba","#6a51a3","#54278f","#3f007d"])
|
|
|
|
var numVertices = (w*h) / 3000;
|
|
var vertices = d3.range(numVertices).map(function(i) {
|
|
var angle = radius * (i+10);
|
|
return {x: angle*Math.cos(angle)+(w/2), y: angle*Math.sin(angle)+(h/2)};
|
|
});
|
|
var d3_geom_voronoi = d3.geom.voronoi<{ x: number; y: number }>()
|
|
.x(function(d) { return d.x; })
|
|
.y(function(d) { return d.y; })
|
|
var prevEventScale = 1;
|
|
var zoom = d3.behavior.zoom().on("zoom", function(d,i) {
|
|
if (zoomToAdd){
|
|
if ((<d3.ZoomEvent> d3.event).scale > prevEventScale) {
|
|
var angle = radius * vertices.length;
|
|
vertices.push({x: angle*Math.cos(angle)+(w/2), y: angle*Math.sin(angle)+(h/2)})
|
|
} else if (vertices.length > 2 && (<d3.ZoomEvent> d3.event).scale != prevEventScale) {
|
|
vertices.pop();
|
|
}
|
|
force.nodes(vertices).start()
|
|
} else {
|
|
if ((<d3.ZoomEvent> d3.event).scale > prevEventScale) {
|
|
radius+= .01
|
|
} else {
|
|
radius -= .01
|
|
}
|
|
vertices.forEach(function(d, i) {
|
|
var angle = radius * (i+10);
|
|
vertices[i] = {x: angle*Math.cos(angle)+(w/2), y: angle*Math.sin(angle)+(h/2)};
|
|
});
|
|
force.nodes(vertices).start()
|
|
}
|
|
prevEventScale = (<d3.ZoomEvent> d3.event).scale;
|
|
});
|
|
|
|
d3.select(window)
|
|
.on("keydown", function() {
|
|
// shift
|
|
if((<KeyboardEvent> d3.event).keyCode == 16) {
|
|
zoomToAdd = false
|
|
}
|
|
|
|
// s
|
|
if((<KeyboardEvent> d3.event).keyCode == 83) {
|
|
simulate = !simulate
|
|
if(simulate) {
|
|
force.start()
|
|
} else {
|
|
force.stop()
|
|
}
|
|
}
|
|
})
|
|
.on("keyup", function() {
|
|
zoomToAdd = true
|
|
})
|
|
|
|
var svg = d3.select("#chart")
|
|
.append("svg")
|
|
.attr("width", w)
|
|
.attr("height", h)
|
|
.call(zoom)
|
|
|
|
var force = d3.layout.force()
|
|
.charge(-300)
|
|
.size([w, h])
|
|
.on("tick", update);
|
|
|
|
force.nodes(vertices).start();
|
|
|
|
var circle = <d3.selection.Update<any>> svg.selectAll("circle");
|
|
var path = <d3.selection.Update<any>> svg.selectAll("path");
|
|
var link = <d3.selection.Update<any>> svg.selectAll("line");
|
|
|
|
function update() {
|
|
path = path.data(d3_geom_voronoi(vertices));
|
|
path.enter().append("path")
|
|
// drag node by dragging cell
|
|
.call(d3.behavior.drag()
|
|
.on("drag", function(d, i) {
|
|
vertices[i] = {x: vertices[i].x + (<d3.DragEvent> d3.event).dx, y: vertices[i].y + (<d3.DragEvent> d3.event).dy}
|
|
})
|
|
)
|
|
.style("fill", function(d, i) { return color(0) })
|
|
path.attr("d", function(d) { return "M" + d.join("L") + "Z"; })
|
|
.transition().duration(150).style("fill", function(d, i) { return color(d3.geom.polygon(d).area()) })
|
|
path.exit().remove();
|
|
|
|
circle = circle.data(vertices)
|
|
circle.enter().append("circle")
|
|
.attr("r", 0)
|
|
.transition().duration(1000).attr("r", 5);
|
|
circle.attr("cx", function(d) { return d.x; })
|
|
.attr("cy", function(d) { return d.y; });
|
|
circle.exit().transition().attr("r", 0).remove();
|
|
|
|
link = link.data(d3_geom_voronoi.links(vertices))
|
|
link.enter().append("line")
|
|
link.attr("x1", function(d) { return d.source.x; })
|
|
.attr("y1", function(d) { return d.source.y; })
|
|
.attr("x2", function(d) { return d.target.x; })
|
|
.attr("y2", function(d) { return d.target.y; })
|
|
|
|
link.exit().remove()
|
|
|
|
if(!simulate) force.stop()
|
|
}
|
|
}
|
|
|
|
//Example from http://bl.ocks.org/mbostock/4341156
|
|
function delaunayTesselation() {
|
|
var width = 960,
|
|
height = 500;
|
|
|
|
var vertices = d3.range(100).map(function (d): [number, number] {
|
|
return [Math.random() * width, Math.random() * height];
|
|
} );
|
|
|
|
var svg = d3.select("body").append("svg")
|
|
.attr("width", width)
|
|
.attr("height", height)
|
|
.attr("class", "PiYG")
|
|
.on("mousemove", function () { vertices[0] = d3.mouse(this); redraw(); } );
|
|
|
|
var path = <d3.selection.Update<any>>svg.append("g").selectAll("path");
|
|
|
|
svg.selectAll("circle")
|
|
.data(vertices.slice(1))
|
|
.enter().append("circle")
|
|
.attr("transform", function (d) { return "translate(" + d + ")"; } )
|
|
.attr("r", 2);
|
|
|
|
redraw();
|
|
|
|
function redraw() {
|
|
path = path.data(d3.geom.delaunay(vertices).map(function (d) { return "M" + d.join("L") + "Z"; } ), String);
|
|
path.exit().remove();
|
|
path.enter().append("path").attr("class", function (d, i) { return "q" + (i % 9) + "-9"; } ).attr("d", String);
|
|
}
|
|
}
|
|
|
|
//Example from http://bl.ocks.org/mbostock/4343214
|
|
function quadtree() {
|
|
var width = 960,
|
|
height = 500;
|
|
|
|
var data = d3.range(5000).map(function(): [number, number] {
|
|
return [Math.random() * width, Math.random() * width];
|
|
});
|
|
|
|
var quadtree = d3.geom.quadtree()
|
|
.extent([[-1, -1], [width + 1, height + 1]])
|
|
(data);
|
|
|
|
var brush = d3.svg.brush()
|
|
.x(d3.scale.identity().domain([0, width]))
|
|
.y(d3.scale.identity().domain([0, height]))
|
|
.on("brush", brushed)
|
|
.extent([[100, 100], [200, 200]]);
|
|
|
|
var svg = d3.select("body").append("svg")
|
|
.attr("width", width)
|
|
.attr("height", height);
|
|
|
|
svg.selectAll(".node")
|
|
.data(nodes(quadtree))
|
|
.enter().append("rect")
|
|
.attr("class", "node")
|
|
.attr("x", function (d) { return d.x; } )
|
|
.attr("y", function (d) { return d.y; } )
|
|
.attr("width", function (d) { return d.width; } )
|
|
.attr("height", function (d) { return d.height; } );
|
|
|
|
var point = svg.selectAll(".point")
|
|
.data(<{ scanned?: boolean; selected?: boolean; 0: number; 1: number }[]> data)
|
|
.enter().append("circle")
|
|
.attr("class", "point")
|
|
.attr("cx", function (d) { return d[0]; } )
|
|
.attr("cy", function (d) { return d[1]; } )
|
|
.attr("r", 4);
|
|
|
|
svg.append("g")
|
|
.attr("class", "brush")
|
|
.call(brush);
|
|
|
|
brushed();
|
|
|
|
function brushed() {
|
|
var extent = <[[number, number], [number, number]]> brush.extent();
|
|
point.each(function (d) { d.scanned = d.selected = false; } );
|
|
search(quadtree, extent[0][0], extent[0][1], extent[1][0], extent[1][1]);
|
|
point.classed("scanned", function (d) { return d.scanned; } );
|
|
point.classed("selected", function (d) { return d.selected; } );
|
|
}
|
|
|
|
// Collapse the quadtree into an array of rectangles.
|
|
function nodes(quadtree: d3.geom.quadtree.Quadtree<[number, number]>) {
|
|
var nodes: Array<{x: number; y: number; width: number; height: number}> = [];
|
|
quadtree.visit(function (node: d3.geom.quadtree.Node<[number, number]>, x1: number, y1: number, x2:number, y2: number) {
|
|
nodes.push({ x: x1, y: y1, width: x2 - x1, height: y2 - y1 });
|
|
} );
|
|
return nodes;
|
|
}
|
|
|
|
// Find the nodes within the specified rectangle.
|
|
function search(quadtree: d3.geom.quadtree.Quadtree<{ scanned?: boolean; selected?: boolean; 0: number; 1: number }>, x0: number, y0: number, x3: number, y3: number) {
|
|
quadtree.visit(function (node: d3.geom.quadtree.Node<{ scanned?: boolean; selected?: boolean; 0: number; 1: number }>, x1: number, y1: number, x2:number, y2: number) {
|
|
var p = node.point;
|
|
if (p) {
|
|
p.scanned = true;
|
|
p.selected = (p[0] >= x0) && (p[0] < x3) && (p[1] >= y0) && (p[1] < y3);
|
|
}
|
|
return x1 >= x3 || y1 >= y3 || x2 < x0 || y2 < y0;
|
|
} );
|
|
}
|
|
}
|
|
|
|
//Example from http://bl.ocks.org/mbostock/4341699
|
|
function convexHull() {
|
|
var width = 960,
|
|
height = 500;
|
|
|
|
var randomX = d3.random.normal(width / 2, 60),
|
|
randomY = d3.random.normal(height / 2, 60),
|
|
vertices = d3.range(100).map(function (): [number, number] { return [randomX(), randomY()]; } );
|
|
|
|
var svg = d3.select("body").append("svg")
|
|
.attr("width", width)
|
|
.attr("height", height)
|
|
.on("mousemove", function () { vertices[0] = d3.mouse(this); redraw(); } )
|
|
.on("click", function () { vertices.push(d3.mouse(this)); redraw(); } );
|
|
|
|
svg.append("rect")
|
|
.attr("width", width)
|
|
.attr("height", height);
|
|
|
|
var hull = svg.append("path")
|
|
.attr("class", "hull");
|
|
|
|
var circle = <d3.selection.Update<[number, number]>> svg.selectAll("circle");
|
|
|
|
redraw();
|
|
|
|
function redraw() {
|
|
hull.datum(d3.geom.hull(vertices)).attr("d", function (d) { return "M" + d.join("L") + "Z"; } );
|
|
circle = circle.data(vertices);
|
|
circle.enter().append("circle").attr("r", 3);
|
|
circle.attr("transform", function (d) { return "translate(" + d + ")"; } );
|
|
}
|
|
}
|
|
|
|
// example from http://bl.ocks.org/mbostock/1044242
|
|
namespace hierarchicalEdgeBundling {
|
|
interface Result extends d3.layout.cluster.Result {
|
|
parent: Result;
|
|
size: number;
|
|
key: string;
|
|
}
|
|
|
|
var diameter = 960,
|
|
radius = diameter / 2,
|
|
innerRadius = radius - 120;
|
|
|
|
var cluster = d3.layout.cluster<Result>()
|
|
.size([360, innerRadius])
|
|
.sort(null)
|
|
.value(function (d) { return d.size; } );
|
|
|
|
var bundle = d3.layout.bundle<Result>();
|
|
|
|
var line = d3.svg.line.radial<Result>()
|
|
.interpolate("bundle")
|
|
.tension(.85)
|
|
.radius(function (d) { return d.y; } )
|
|
.angle(function (d) { return d.x / 180 * Math.PI; } );
|
|
|
|
var svg = d3.select("body").append("svg")
|
|
.attr("width", diameter)
|
|
.attr("height", diameter)
|
|
.append("g")
|
|
.attr("transform", "translate(" + radius + "," + radius + ")");
|
|
|
|
d3.json("readme-flare-imports.json", function (error, classes) {
|
|
var nodes = cluster.nodes(packages.root(classes)),
|
|
links = packages.imports(nodes);
|
|
|
|
svg.selectAll(".link")
|
|
.data(bundle(links))
|
|
.enter().append("path")
|
|
.attr("class", "link")
|
|
.attr("d", line);
|
|
|
|
svg.selectAll(".node")
|
|
.data(nodes.filter(function (n) { return !n.children; } ))
|
|
.enter().append("g")
|
|
.attr("class", "node")
|
|
.attr("transform", function (d) { return "rotate(" + (d.x - 90) + ")translate(" + d.y + ")"; } )
|
|
.append("text")
|
|
.attr("dx", function (d) { return d.x < 180 ? 8 : -8; } )
|
|
.attr("dy", ".31em")
|
|
.attr("text-anchor", function (d) { return d.x < 180 ? "start" : "end"; } )
|
|
.attr("transform", function (d) { return d.x < 180 ? null : "rotate(180)"; } )
|
|
.text(function (d) { return d.key; } );
|
|
} );
|
|
|
|
d3.select(self.frameElement).style("height", diameter + "px");
|
|
|
|
var packages = {
|
|
|
|
// Lazily construct the package hierarchy from class names.
|
|
root: function (classes: any[]) {
|
|
var map: {[key: string]: Result} = {};
|
|
|
|
function find(name: string, data?: any) {
|
|
var node: Result = map[name], i: number;
|
|
if (!node) {
|
|
node = map[name] = data || { name: name, children: [] };
|
|
if (name.length) {
|
|
node.parent = find(name.substring(0, i = name.lastIndexOf(".")));
|
|
node.parent.children.push(node);
|
|
node.key = name.substring(i + 1);
|
|
}
|
|
}
|
|
return node;
|
|
}
|
|
|
|
classes.forEach(function (d) {
|
|
find(d.name, d);
|
|
} );
|
|
|
|
return map[""];
|
|
} ,
|
|
|
|
// Return a list of imports for the given array of nodes.
|
|
imports: function (nodes: any[]) {
|
|
var map: {[key: string]: Result} = {},
|
|
imports: d3.layout.cluster.Link<Result>[] = [];
|
|
|
|
// Compute a map from name to node.
|
|
nodes.forEach(function (d) {
|
|
map[d.name] = d;
|
|
} );
|
|
|
|
// For each import, construct a link from the source to target node.
|
|
nodes.forEach(function (d) {
|
|
if (d.imports) d.imports.forEach(function (i: string) {
|
|
imports.push({ source: map[d.name], target: map[i] });
|
|
} );
|
|
} );
|
|
|
|
return imports;
|
|
}
|
|
};
|
|
}
|
|
|
|
// example from http://bl.ocks.org/mbostock/1123639
|
|
function roundedRectangles() {
|
|
var mouse = [480, 250],
|
|
count = 0;
|
|
|
|
var svg = d3.select("body").append("svg:svg")
|
|
.attr("width", 960)
|
|
.attr("height", 500);
|
|
|
|
var g = svg.selectAll("g")
|
|
.data(d3.range(25))
|
|
.enter().append("svg:g")
|
|
.attr("transform", "translate(" + mouse + ")");
|
|
|
|
g.append("svg:rect")
|
|
.attr("rx", 6)
|
|
.attr("ry", 6)
|
|
.attr("x", -12.5)
|
|
.attr("y", -12.5)
|
|
.attr("width", 25)
|
|
.attr("height", 25)
|
|
.attr("transform", function (d, i) { return "scale(" + (1 - d / 25) * 20 + ")"; } )
|
|
.style("fill", d3.scale.category20c<number>());
|
|
|
|
var g0 = g.datum(function (d) {
|
|
return { center: [0, 0], angle: 0 };
|
|
} );
|
|
|
|
svg.on("mousemove", function () {
|
|
mouse = d3.mouse(this);
|
|
} );
|
|
|
|
d3.timer(function () {
|
|
count++;
|
|
g0.attr("transform", function (d, i) {
|
|
d.center[0] += (mouse[0] - d.center[0]) / (i + 5);
|
|
d.center[1] += (mouse[1] - d.center[1]) / (i + 5);
|
|
d.angle += Math.sin((count + i) / 10) * 7;
|
|
return "translate(" + d.center + ")rotate(" + d.angle + ")";
|
|
} );
|
|
return true;
|
|
} );
|
|
}
|
|
|
|
// example from http://bl.ocks.org/mbostock/4060954
|
|
function streamGraph() {
|
|
var n = 20, // number of layers
|
|
m = 200, // number of samples per layer
|
|
stack = d3.layout.stack().offset("wiggle"),
|
|
layers0 = stack(d3.range(n).map(function () { return bumpLayer(m); })),
|
|
layers1 = stack(d3.range(n).map(function () { return bumpLayer(m); }));
|
|
|
|
var width = 960,
|
|
height = 500;
|
|
|
|
var x = d3.scale.linear()
|
|
.domain([0, m - 1])
|
|
.range([0, width]);
|
|
|
|
var y = d3.scale.linear()
|
|
.domain([0, d3.max(layers0.concat(layers1), function (layer) { return d3.max(layer, function (d) { return d.y0 + d.y; }); })])
|
|
.range([height, 0]);
|
|
|
|
var color = d3.scale.linear<string>()
|
|
.range(["#aad", "#556"]);
|
|
|
|
var area = d3.svg.area<{ x: number; y: number; y0: number }>()
|
|
.x(function (d) { return x(d.x); })
|
|
.y0(function (d) { return y(d.y0); })
|
|
.y1(function (d) { return y(d.y0 + d.y); });
|
|
|
|
var svg = d3.select("body").append("svg")
|
|
.attr("width", width)
|
|
.attr("height", height);
|
|
|
|
svg.selectAll("path")
|
|
.data(layers0)
|
|
.enter().append("path")
|
|
.attr("d", area)
|
|
.style("fill", function () { return color(Math.random()); });
|
|
|
|
function transition() {
|
|
d3.selectAll("path")
|
|
.data(function () {
|
|
var d = layers1;
|
|
layers1 = layers0;
|
|
return layers0 = d;
|
|
})
|
|
.transition()
|
|
.duration(2500)
|
|
.attr("d", area);
|
|
}
|
|
|
|
// Inspired by Lee Byron's test data generator.
|
|
function bumpLayer(n: number): Array<{x: number; y: number;y0?:number;}> {
|
|
|
|
function bump(a: number[]) {
|
|
var x = 1 / (.1 + Math.random()),
|
|
y = 2 * Math.random() - .5,
|
|
z = 10 / (.1 + Math.random());
|
|
for (var i = 0; i < n; i++) {
|
|
var w = (i / n - y) * z;
|
|
a[i] += x * Math.exp(-w * w);
|
|
}
|
|
}
|
|
|
|
var a: number[] = [], i: number;
|
|
for (i = 0; i < n; ++i) a[i] = 0;
|
|
for (i = 0; i < 5; ++i) bump(a);
|
|
return a.map(function (d, i) { return { x: i, y: Math.max(0, d) }; } );
|
|
}
|
|
}
|
|
|
|
// example from http://mbostock.github.io/d3/talk/20111116/force-collapsible.html
|
|
namespace forceCollapsable2 {
|
|
interface Node extends d3.layout.force.Node {
|
|
children: Node[];
|
|
_children: Node[];
|
|
size: number;
|
|
id: string;
|
|
}
|
|
|
|
var w = 1280,
|
|
h = 800,
|
|
node: d3.selection.Update<Node>,
|
|
link: d3.selection.Update<d3.layout.force.Link<Node>>,
|
|
root: Node;
|
|
|
|
var force = d3.layout.force<Node>()
|
|
.on("tick", tick)
|
|
.charge(function (d) { return d._children ? -d.size / 100 : -30; } )
|
|
.linkDistance(function (d) { return d.target._children ? 80 : 30; } )
|
|
.size([w, h - 160]);
|
|
|
|
var vis = d3.select("body").append("svg:svg")
|
|
.attr("width", w)
|
|
.attr("height", h);
|
|
|
|
d3.json("flare.json", function (json) {
|
|
root = json;
|
|
root.fixed = true;
|
|
root.x = w / 2;
|
|
root.y = h / 2 - 80;
|
|
update();
|
|
} );
|
|
|
|
function update() {
|
|
var nodes = flatten(root),
|
|
links = d3.layout.tree<Node>().links(nodes);
|
|
|
|
// Restart the force layout.
|
|
force
|
|
.nodes(nodes)
|
|
.links(links)
|
|
.start();
|
|
|
|
// Update the links…
|
|
link = vis.selectAll("line.link")
|
|
.data(links, function (d) { return d.target.id; } );
|
|
|
|
// Enter any new links.
|
|
link.enter().insert("svg:line", ".node")
|
|
.attr("class", "link")
|
|
.attr("x1", function (d) { return d.source.x; } )
|
|
.attr("y1", function (d) { return d.source.y; } )
|
|
.attr("x2", function (d) { return d.target.x; } )
|
|
.attr("y2", function (d) { return d.target.y; } );
|
|
|
|
// Exit any old links.
|
|
link.exit().remove();
|
|
|
|
// Update the nodes…
|
|
node = vis.selectAll("circle.node")
|
|
.data(nodes, function (d) { return d.id; } )
|
|
.style("fill", color);
|
|
|
|
node.transition()
|
|
.attr("r", function (d) { return d.children ? 4.5 : Math.sqrt(d.size) / 10; } );
|
|
|
|
// Enter any new nodes.
|
|
node.enter().append("svg:circle")
|
|
.attr("class", "node")
|
|
.attr("cx", function (d) { return d.x; } )
|
|
.attr("cy", function (d) { return d.y; } )
|
|
.attr("r", function (d) { return d.children ? 4.5 : Math.sqrt(d.size) / 10; } )
|
|
.style("fill", color)
|
|
.on("click", click)
|
|
.call(force.drag);
|
|
|
|
// Exit any old nodes.
|
|
node.exit().remove();
|
|
}
|
|
|
|
function tick() {
|
|
link.attr("x1", function (d) { return d.source.x; } )
|
|
.attr("y1", function (d) { return d.source.y; } )
|
|
.attr("x2", function (d) { return d.target.x; } )
|
|
.attr("y2", function (d) { return d.target.y; } );
|
|
|
|
node.attr("cx", function (d) { return d.x; } )
|
|
.attr("cy", function (d) { return d.y; } );
|
|
}
|
|
|
|
// Color leaf nodes orange, and packages white or blue.
|
|
function color(d: Node) {
|
|
return d._children ? "#3182bd" : d.children ? "#c6dbef" : "#fd8d3c";
|
|
}
|
|
|
|
// Toggle children on click.
|
|
function click(d: Node) {
|
|
if (d.children) {
|
|
d._children = d.children;
|
|
d.children = null;
|
|
} else {
|
|
d.children = d._children;
|
|
d._children = null;
|
|
}
|
|
update();
|
|
}
|
|
|
|
// Returns a list of all nodes under the root.
|
|
function flatten(root: Node) {
|
|
var nodes: Node[] = [], i = 0;
|
|
|
|
function recurse(node: Node) {
|
|
if (node.children) node.size = node.children.reduce(function (p, v) { return p + recurse(v); } , 0);
|
|
if (!node.id) node.id = String(++i);
|
|
nodes.push(node);
|
|
return node.size;
|
|
}
|
|
|
|
root.size = recurse(root);
|
|
return nodes;
|
|
}
|
|
}
|
|
|
|
//exapmle from http://bl.ocks.org/mbostock/4062006
|
|
function chordDiagram() {
|
|
var matrix = [
|
|
[11975, 5871, 8916, 2868],
|
|
[1951, 10048, 2060, 6171],
|
|
[8010, 16145, 8090, 8045],
|
|
[1013, 990, 940, 6907]
|
|
];
|
|
|
|
var chord = d3.layout.chord()
|
|
.padding(.05)
|
|
.sortSubgroups(d3.descending)
|
|
.matrix(matrix);
|
|
|
|
var width = 960,
|
|
height = 500,
|
|
innerRadius = Math.min(width, height) * .41,
|
|
outerRadius = innerRadius * 1.1;
|
|
|
|
var fill = d3.scale.ordinal<number, string>()
|
|
.domain(d3.range(4))
|
|
.range(["#000000", "#FFDD89", "#957244", "#F26223"]);
|
|
|
|
var svg = d3.select("body").append("svg")
|
|
.attr("width", width)
|
|
.attr("height", height)
|
|
.append("g")
|
|
.attr("transform", "translate(" + width / 2 + "," + height / 2 + ")");
|
|
|
|
svg.append("g").selectAll("path")
|
|
.data(chord.groups)
|
|
.enter().append("path")
|
|
.style("fill", function (d) { return fill(d.index); } )
|
|
.style("stroke", function (d) { return fill(d.index); } )
|
|
.attr("d", d3.svg.arc<d3.layout.chord.Node>().innerRadius(innerRadius).outerRadius(outerRadius))
|
|
.on("mouseover", fade(.1))
|
|
.on("mouseout", fade(1));
|
|
|
|
var ticks = svg.append("g").selectAll("g")
|
|
.data(chord.groups)
|
|
.enter().append("g").selectAll("g")
|
|
.data(groupTicks)
|
|
.enter().append("g")
|
|
.attr("transform", function (d) {
|
|
return "rotate(" + (d.angle * 180 / Math.PI - 90) + ")"
|
|
+ "translate(" + outerRadius + ",0)";
|
|
} );
|
|
|
|
ticks.append("line")
|
|
.attr("x1", 1)
|
|
.attr("y1", 0)
|
|
.attr("x2", 5)
|
|
.attr("y2", 0)
|
|
.style("stroke", "#000");
|
|
|
|
ticks.append("text")
|
|
.attr("x", 8)
|
|
.attr("dy", ".35em")
|
|
.attr("transform", function (d) { return d.angle > Math.PI ? "rotate(180)translate(-16)" : null; } )
|
|
.style("text-anchor", function (d) { return d.angle > Math.PI ? "end" : null; } )
|
|
.text(function (d) { return d.label; } );
|
|
|
|
svg.append("g")
|
|
.attr("class", "chord")
|
|
.selectAll("path")
|
|
.data(chord.chords)
|
|
.enter().append("path")
|
|
.attr("d", d3.svg.chord<d3.layout.chord.Node>().radius(innerRadius))
|
|
.style("fill", function (d) { return fill(d.target.index); } )
|
|
.style("opacity", 1);
|
|
|
|
// Returns an array of tick angles and labels, given a group.
|
|
function groupTicks(d: d3.layout.chord.Node) {
|
|
var k = (d.endAngle - d.startAngle) / d.value;
|
|
return d3.range(0, d.value, 1000).map(function (v, i) {
|
|
return {
|
|
angle: v * k + d.startAngle,
|
|
label: i % 5 ? null : v / 1000 + "k"
|
|
};
|
|
} );
|
|
}
|
|
|
|
// Returns an event handler for fading a given chord group.
|
|
function fade(opacity: number) {
|
|
return function (g: {}, i: number) {
|
|
svg.selectAll(".chord path")
|
|
.filter(function (d) { return d.source.index != i && d.target.index != i; } )
|
|
.transition()
|
|
.style("opacity", opacity);
|
|
};
|
|
}
|
|
}
|
|
|
|
//example from http://mbostock.github.io/d3/talk/20111116/iris-parallel.html
|
|
function irisParallel() {
|
|
var species = ["setosa", "versicolor", "virginica"],
|
|
traits = ["sepal length", "petal length", "sepal width", "petal width"];
|
|
|
|
var m = [80, 160, 200, 160],
|
|
w = 1280 - m[1] - m[3],
|
|
h = 800 - m[0] - m[2];
|
|
|
|
var x = d3.scale.ordinal().domain(traits).rangePoints([0, w]),
|
|
y: {[key: string]: any} = {};
|
|
|
|
var line = d3.svg.line(),
|
|
axis = d3.svg.axis().orient("left"),
|
|
foreground: d3.Selection<{ [key: string]: string}>;
|
|
|
|
var svg = d3.select("body").append("svg:svg")
|
|
.attr("width", w + m[1] + m[3])
|
|
.attr("height", h + m[0] + m[2])
|
|
.append("svg:g")
|
|
.attr("transform", "translate(" + m[3] + "," + m[0] + ")");
|
|
|
|
d3.csv("iris.csv", function (flowers) {
|
|
var i: number;
|
|
|
|
// Create a scale and brush for each trait.
|
|
traits.forEach(function (d) {
|
|
y[d] = d3.scale.linear<number>()
|
|
.domain(d3.extent(flowers, function (p) { return +p[d]; } ))
|
|
.range([h, 0]);
|
|
|
|
y[d].brush = d3.svg.brush()
|
|
.y(y[d])
|
|
.on("brush", brush);
|
|
} );
|
|
|
|
// Add a legend.
|
|
var legend = svg.selectAll("g.legend")
|
|
.data(species)
|
|
.enter().append("svg:g")
|
|
.attr("class", "legend")
|
|
.attr("transform", function (d, i) { return "translate(0," + (i * 20 + 584) + ")"; } );
|
|
|
|
legend.append("svg:line")
|
|
.attr("class", String)
|
|
.attr("x2", 8);
|
|
|
|
legend.append("svg:text")
|
|
.attr("x", 12)
|
|
.attr("dy", ".31em")
|
|
.text(function (d) { return "Iris " + d; } );
|
|
|
|
// Add foreground lines.
|
|
foreground = svg.append("svg:g")
|
|
.attr("class", "foreground")
|
|
.selectAll("path")
|
|
.data(flowers)
|
|
.enter().append("svg:path")
|
|
.attr("d", path)
|
|
.attr("class", function (d) { return d['species']; } );
|
|
|
|
// Add a group element for each trait.
|
|
var g = svg.selectAll(".trait")
|
|
.data(traits)
|
|
.enter().append("svg:g")
|
|
.attr("class", "trait")
|
|
.attr("transform", function (d) { return "translate(" + x(d) + ")"; } )
|
|
.call(d3.behavior.drag<string>()
|
|
.origin(function (d) { return { x: x(d), y: NaN }; } )
|
|
.on("dragstart", dragstart)
|
|
.on("drag", drag)
|
|
.on("dragend", dragend));
|
|
|
|
// Add an axis and title.
|
|
g.append("svg:g")
|
|
.attr("class", "axis")
|
|
.each(function (d) { d3.select(this).call(axis.scale(y[d])); } )
|
|
.append("svg:text")
|
|
.attr("text-anchor", "middle")
|
|
.attr("y", -9)
|
|
.text(String);
|
|
|
|
// Add a brush for each axis.
|
|
g.append("svg:g")
|
|
.attr("class", "brush")
|
|
.each(function (d) { d3.select(this).call(y[d].brush); } )
|
|
.selectAll("rect")
|
|
.attr("x", -8)
|
|
.attr("width", 16);
|
|
|
|
function dragstart(d: string) {
|
|
i = traits.indexOf(d);
|
|
}
|
|
|
|
function drag(d: string) {
|
|
x.range()[i] = (<d3.DragEvent> d3.event).x;
|
|
traits.sort(function (a, b) { return x(a) - x(b); } );
|
|
g.attr("transform", function (d) { return "translate(" + x(d) + ")"; } );
|
|
foreground.attr("d", path);
|
|
}
|
|
|
|
function dragend(d: string) {
|
|
x.domain(traits).rangePoints([0, w]);
|
|
var t = d3.transition().duration(500);
|
|
t.selectAll(".trait").attr("transform", function (d) { return "translate(" + x(d) + ")"; } );
|
|
t.selectAll(".foreground path").attr("d", path);
|
|
}
|
|
} );
|
|
|
|
// Returns the path for a given data point.
|
|
function path(d: any): string {
|
|
return line(traits.map(function (p): [number, number] { return [x(p), y[p](d[p])]; } ));
|
|
}
|
|
|
|
// Handles a brush event, toggling the display of foreground lines.
|
|
function brush() {
|
|
var actives = traits.filter(function (p) { return !y[p].brush.empty(); } ),
|
|
extents = actives.map(function (p) { return y[p].brush.extent(); } );
|
|
foreground.classed("fade", function (d) {
|
|
return !actives.every(function (p, i) {
|
|
return extents[i][0] <= d[p] && d[p] <= extents[i][1];
|
|
} );
|
|
} );
|
|
}
|
|
}
|
|
|
|
//example from
|
|
function healthAndWealth() {
|
|
// Various accessors that specify the four dimensions of data to visualize.
|
|
function x(d: any) { return d.income; }
|
|
function y(d: any) { return d.lifeExpectancy; }
|
|
function radius(d: any) { return d.population; }
|
|
function color(d: any) { return d.region; }
|
|
function key(d: any) { return d.name; }
|
|
|
|
// Chart dimensions.
|
|
var margin = { top: 19.5, right: 19.5, bottom: 19.5, left: 39.5 },
|
|
width = 960 - margin.right,
|
|
height = 500 - margin.top - margin.bottom;
|
|
|
|
// Various scales. These domains make assumptions of data, naturally.
|
|
var xScale = d3.scale.log().domain([300, 1e5]).range([0, width]),
|
|
yScale = d3.scale.linear().domain([10, 85]).range([height, 0]),
|
|
radiusScale = d3.scale.sqrt().domain([0, 5e8]).range([0, 40]),
|
|
colorScale = d3.scale.category10();
|
|
|
|
// The x & y axes.
|
|
var xAxis = d3.svg.axis().orient("bottom").scale(xScale).ticks(12, d3.format(",d")),
|
|
yAxis = d3.svg.axis().scale(yScale).orient("left");
|
|
|
|
// Create the SVG container and set the origin.
|
|
var svg = d3.select("#chart").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 + ")");
|
|
|
|
// Add the x-axis.
|
|
svg.append("g")
|
|
.attr("class", "x axis")
|
|
.attr("transform", "translate(0," + height + ")")
|
|
.call(xAxis);
|
|
|
|
// Add the y-axis.
|
|
svg.append("g")
|
|
.attr("class", "y axis")
|
|
.call(yAxis);
|
|
|
|
// Add an x-axis label.
|
|
svg.append("text")
|
|
.attr("class", "x label")
|
|
.attr("text-anchor", "end")
|
|
.attr("x", width)
|
|
.attr("y", height - 6)
|
|
.text("income per capita, inflation-adjusted (dollars)");
|
|
|
|
// Add a y-axis label.
|
|
svg.append("text")
|
|
.attr("class", "y label")
|
|
.attr("text-anchor", "end")
|
|
.attr("y", 6)
|
|
.attr("dy", ".75em")
|
|
.attr("transform", "rotate(-90)")
|
|
.text("life expectancy (years)");
|
|
|
|
// Add the year label; the value is set on transition.
|
|
var label = svg.append("text")
|
|
.attr("class", "year label")
|
|
.attr("text-anchor", "end")
|
|
.attr("y", height - 24)
|
|
.attr("x", width)
|
|
.text(1800);
|
|
|
|
// Load the data.
|
|
d3.json("nations.json", function (nations: any[]) {
|
|
|
|
// A bisector since many nation's data is sparsely-defined.
|
|
var bisect = d3.bisector(function (d: any) { return d[0]; } );
|
|
|
|
// Add a dot per nation. Initialize the data at 1800, and set the colors.
|
|
var dot = svg.append("g")
|
|
.attr("class", "dots")
|
|
.selectAll(".dot")
|
|
.data(interpolateData(1800))
|
|
.enter().append("circle")
|
|
.attr("class", "dot")
|
|
.style("fill", function (d) { return colorScale(color(d)); } )
|
|
.call(position)
|
|
.sort(order);
|
|
|
|
// Add a title.
|
|
dot.append("title")
|
|
.text(function (d) { return d.name; } );
|
|
|
|
// Add an overlay for the year label.
|
|
var box = (<SVGTextElement>label.node()).getBBox();
|
|
|
|
var overlay = svg.append("rect")
|
|
.attr("class", "overlay")
|
|
.attr("x", box.x)
|
|
.attr("y", box.y)
|
|
.attr("width", box.width)
|
|
.attr("height", box.height)
|
|
.on("mouseover", enableInteraction);
|
|
|
|
// Start a transition that interpolates the data based on year.
|
|
svg.transition()
|
|
.duration(30000)
|
|
.ease("linear")
|
|
.tween("year", tweenYear)
|
|
.each("end", enableInteraction);
|
|
|
|
// Positions the dots based on data.
|
|
function position(dot: d3.Selection<any>) {
|
|
dot.attr("cx", function (d) { return xScale(x(d)); } )
|
|
.attr("cy", function (d) { return yScale(y(d)); } )
|
|
.attr("r", function (d) { return radiusScale(radius(d)); } );
|
|
}
|
|
|
|
// Defines a sort order so that the smallest dots are drawn on top.
|
|
function order(a: any, b: any) {
|
|
return radius(b) - radius(a);
|
|
}
|
|
|
|
// After the transition finishes, you can mouseover to change the year.
|
|
function enableInteraction() {
|
|
var yearScale = d3.scale.linear()
|
|
.domain([1800, 2009])
|
|
.range([box.x + 10, box.x + box.width - 10])
|
|
.clamp(true);
|
|
|
|
// Cancel the current transition, if any.
|
|
svg.transition().duration(0);
|
|
|
|
overlay
|
|
.on("mouseover", mouseover)
|
|
.on("mouseout", mouseout)
|
|
.on("mousemove", mousemove)
|
|
.on("touchmove", mousemove);
|
|
|
|
function mouseover() {
|
|
label.classed("active", true);
|
|
}
|
|
|
|
function mouseout() {
|
|
label.classed("active", false);
|
|
}
|
|
|
|
function mousemove() {
|
|
displayYear(yearScale.invert(d3.mouse(this)[0]));
|
|
}
|
|
}
|
|
|
|
// Tweens the entire chart by first tweening the year, and then the data.
|
|
// For the interpolated data, the dots and label are redrawn.
|
|
function tweenYear() {
|
|
var year = d3.interpolateNumber(1800, 2009);
|
|
return function (t: number) { displayYear(year(t)); };
|
|
}
|
|
|
|
// Updates the display to show the specified year.
|
|
function displayYear(year: number) {
|
|
dot.data(interpolateData(year), key).call(position).sort(order);
|
|
label.text(Math.round(year));
|
|
}
|
|
|
|
// Interpolates the dataset for the given (fractional) year.
|
|
function interpolateData(year: number) {
|
|
return nations.map(function (d) {
|
|
return {
|
|
name: d.name,
|
|
region: d.region,
|
|
income: interpolateValues(d.income, year),
|
|
population: interpolateValues(d.population, year),
|
|
lifeExpectancy: interpolateValues(d.lifeExpectancy, year)
|
|
};
|
|
} );
|
|
}
|
|
|
|
// Finds (and possibly interpolates) the value for the specified year.
|
|
function interpolateValues(values: any[], year: number) {
|
|
var i = bisect.left(values, year, 0, values.length - 1),
|
|
a = values[i];
|
|
if (i > 0) {
|
|
var b = values[i - 1],
|
|
t = (year - a[0]) / (b[0] - a[0]);
|
|
return a[1] * (1 - t) + b[1] * t;
|
|
}
|
|
return a[1];
|
|
}
|
|
} );
|
|
}
|
|
|
|
// Test for d3.functor
|
|
function functorTest () {
|
|
var f: (n: number) => number = d3.functor(10);
|
|
var g = d3.functor(function (v: number) { return v; });
|
|
|
|
return f(10) === g(10);
|
|
}
|
|
|
|
// Test for d3.Nest
|
|
// Most test adopted from: https://github.com/mbostock/d3/blob/master/test/arrays/nest-test.js
|
|
function nestTest () {
|
|
var data = [
|
|
{
|
|
a: 10,
|
|
b: [ 1, 2 ]
|
|
},
|
|
{
|
|
a: 20,
|
|
b: [ 2, 3 ]
|
|
},
|
|
{
|
|
a: 30,
|
|
b: [ 1, 2 ]
|
|
}
|
|
];
|
|
|
|
var n1 = d3.nest<{a: number; b: number[] }>()
|
|
.key(function (d) { return String(d.a); })
|
|
.sortKeys(d3.descending)
|
|
.rollup(function (vals) {
|
|
return d3.sum(vals[0].b);
|
|
});
|
|
n1.map(data);
|
|
n1.entries(data);
|
|
|
|
var n2 = d3.nest<{ a: number; b: number[] }>()
|
|
.key(function (d) { return String(d.a); })
|
|
.sortValues(function (x1, x2) {
|
|
return x1.b[0] < x1.b[1] ? -1 : (x1.b[0] > x1.b[0] ? 1 : 0); });
|
|
n2.map(data);
|
|
n2.entries(data);
|
|
|
|
// Tests adopted from d3's tests.
|
|
var keys = d3.nest<{ foo: number; }>()
|
|
.key(function(d) { return String(d.foo); })
|
|
.entries([{foo: 1}, {foo: 1}, {foo: 2}])
|
|
.map(function(d) { return d.key; })
|
|
.sort(d3.ascending);
|
|
|
|
var entries = d3.nest<{foo: number; bar?: number}>()
|
|
.key(function(d) { return String(d.foo); })
|
|
.entries([{foo: 1, bar: 0}, {foo: 2}, {foo: 1, bar: 1}]);
|
|
|
|
keys = d3.nest<{foo: number}>()
|
|
.key(function(d) { return String(d.foo); }).sortKeys(d3.descending)
|
|
.entries([{foo: 1}, {foo: 1}, {foo: 2}])
|
|
.map(function(d) { return d.key; });
|
|
|
|
entries = d3.nest<{ foo: number; bar?: number }>()
|
|
.key(function(d) { return String(d.foo); })
|
|
.sortValues(function(a, b) { return a.bar - b.bar; })
|
|
.entries([{foo: 1, bar: 2}, {foo: 1, bar: 0}, {foo: 1, bar: 1}, {foo: 2}]);
|
|
|
|
entries = d3.nest<{ foo: number; bar?: number }>()
|
|
.key(function(d) { return String(d.foo); })
|
|
.rollup(function(values) { return d3.sum<any>(values, function(d) { return d.bar; }); })
|
|
.entries([{foo: 1, bar: 2}, {foo: 1, bar: 0}, {foo: 1, bar: 1}, {foo: 2}]);
|
|
|
|
entries = d3.nest<[number, number]>()
|
|
.key(function(d) { return String(d[0]); }).sortKeys(d3.ascending)
|
|
.key(function(d) { return String(d[1]); }).sortKeys(d3.ascending)
|
|
.entries([[0, 1], [0, 2], [1, 1], [1, 2], [0, 2]]);
|
|
|
|
entries = d3.nest<[number, number]>()
|
|
.key(function(d) { return String(d[0]); }).sortKeys(d3.ascending)
|
|
.key(function(d) { return String(d[1]); }).sortKeys(d3.ascending)
|
|
.rollup(function(values) { return values.length; })
|
|
.entries([[0, 1], [0, 2], [1, 1], [1, 2], [0, 2]]);
|
|
|
|
entries = d3.nest<{ 0: number; 1: number; 2?: number }>()
|
|
.key(function(d) { return String(d[0]); }).sortKeys(d3.ascending)
|
|
.key(function(d) { return String(d[1]); }).sortKeys(d3.ascending)
|
|
.sortValues(function(a, b) { return a[2] - b[2]; })
|
|
.entries([[0, 1], [0, 2, 1], [1, 1], [1, 2], [0, 2, 0]]);
|
|
|
|
var map = d3.nest<{ 0: number; 1: number; 2?: number }>()
|
|
.key(function(d) { return String(d[0]); }).sortKeys(d3.ascending)
|
|
.key(function(d) { return String(d[1]); }).sortKeys(d3.ascending)
|
|
.sortValues(function(a, b) { return a[2] - b[2]; })
|
|
.map([[0, 1], [0, 2, 1], [1, 1], [1, 2], [0, 2, 0]]);
|
|
}
|
|
|
|
// Test for setting attributes as an object
|
|
// From https://github.com/mbostock/d3/blob/master/test/selection/attr-test.js
|
|
function attrObjTest () {
|
|
d3.select('body')
|
|
.data(["orange"])
|
|
.attr({"xlink:href": function(d, i) { return d + "-" + i + ".png"; }});
|
|
}
|
|
|
|
// Test for setting styles as an object
|
|
// From https://github.com/mbostock/d3/blob/master/test/selection/style-test.js
|
|
function styleObjTest () {
|
|
d3.select('body')
|
|
.style({"background-color": "white", opacity: .42});
|
|
}
|
|
|
|
// Test for setting styles as an object
|
|
// From https://github.com/mbostock/d3/blob/master/test/selection/property-test.js
|
|
function propertyObjTest () {
|
|
d3.select('body')
|
|
.property({bgcolor: "purple", opacity: .41});
|
|
}
|
|
|
|
|
|
// Test for brushes
|
|
function brushTest() {
|
|
var xScale = d3.scale.linear(),
|
|
yScale = d3.scale.linear();
|
|
|
|
var xMin = 0, xMax = 1,
|
|
yMin = 0, yMax = 1;
|
|
|
|
// Setting only x scale.
|
|
var brush1 = d3.svg.brush()
|
|
.x(xScale)
|
|
.on('brush', function () {
|
|
var extent = <[number, number]> brush1.extent();
|
|
xMin = Math.max(extent[0], 0);
|
|
xMax = Math.min(extent[1], 1);
|
|
brush1.extent([xMin, xMax]);
|
|
});
|
|
|
|
// Setting both the x and y scale
|
|
var brush2 = d3.svg.brush()
|
|
.x(xScale)
|
|
.y(yScale)
|
|
.on('brush', function () {
|
|
var extent = <[[number, number], [number, number]]> brush2.extent();
|
|
var xExtent = extent[0],
|
|
yExtent = extent[1];
|
|
|
|
xMin = Math.max(xExtent[0], 0);
|
|
xMax = Math.min(xExtent[1], 1);
|
|
|
|
yMin = Math.max(yExtent[0], 0);
|
|
yMax = Math.min(yExtent[1], 1);
|
|
|
|
brush1.extent([[xMin, xMax], [yMin, yMax]]);
|
|
});
|
|
}
|
|
|
|
|
|
// Tests for area
|
|
// Adopted from: https://github.com/mbostock/d3/blob/master/test/svg/area-test.js
|
|
function svgAreaTest () {
|
|
var a = d3.svg.area(),
|
|
f: () => number,
|
|
n: number;
|
|
|
|
a.x(f).x() === f;
|
|
a.x(n).x() === n;
|
|
a.x(0);
|
|
a.x(function (d) { return d[0] * 10; });
|
|
a.x(function (d, i) { return i * 10; });
|
|
|
|
a.x(f).x0() === f;
|
|
a.x(n).x0() === n;
|
|
a.x0(0);
|
|
a.x0(function (d) { return d[0] * 10; });
|
|
a.x0(function (d, i) { return i * 10; });
|
|
|
|
a.x(f).x1() === f;
|
|
a.x(n).x1() === n;
|
|
a.x1(0);
|
|
a.x1(function (d) { return d[0] * 10; });
|
|
a.x1(function (d, i) { return i * 10; });
|
|
|
|
a.y(f).y() === f;
|
|
a.y(n).y() === n;
|
|
a.y(0);
|
|
a.y(function (d) { return d[0] * 10; });
|
|
a.y(function (d, i) { return i * 10; });
|
|
|
|
a.y(f).y0() === f;
|
|
a.y(n).y0() === n;
|
|
a.y0(0);
|
|
a.y0(function (d) { return d[0] * 10; });
|
|
a.y0(function (d, i) { return i * 10; });
|
|
|
|
a.y(f).y1() === f;
|
|
a.y(n).y1() === n;
|
|
a.y1(0);
|
|
a.y1(function (d) { return d[0] * 10; });
|
|
a.y1(function (d, i) { return i * 10; });
|
|
}
|
|
|
|
// Tests for areaRadial
|
|
// Adopted from: https://github.com/mbostock/d3/blob/master/test/svg/area-radial-test.js
|
|
function svgAreaRadialTest () {
|
|
var a = d3.svg.area.radial(),
|
|
f: () => number,
|
|
n: number;
|
|
|
|
a.radius(f).radius() === f;
|
|
a.radius(n).radius() === n;
|
|
a.radius(0);
|
|
a.radius(function (d) { return d[0] * 10; });
|
|
a.radius(function (d, i) { return i * 10; });
|
|
|
|
a.radius(f).innerRadius() === f;
|
|
a.radius(n).innerRadius() === n;
|
|
a.innerRadius(0);
|
|
a.innerRadius(function (d) { return d[0] * 10; });
|
|
a.innerRadius(function (d, i) { return i * 10; });
|
|
|
|
a.radius(f).outerRadius() === f;
|
|
a.radius(n).outerRadius() === n;
|
|
a.outerRadius(0);
|
|
a.outerRadius(function (d) { return d[1] * 10; });
|
|
a.outerRadius(function (d, i) { return i * 10; });
|
|
|
|
a.angle(f).angle() === f;
|
|
a.angle(n).angle() === n;
|
|
a.angle(0);
|
|
a.angle(function (d) { return d[0] * 10; });
|
|
a.angle(function (d, i) { return i * 10; });
|
|
|
|
a.angle(f).startAngle() === f;
|
|
a.angle(n).startAngle() === n;
|
|
a.startAngle(0);
|
|
a.startAngle(function (d) { return d[0] * 10; });
|
|
a.startAngle(function (d, i) { return i * 10; });
|
|
|
|
a.angle(f).endAngle() === f;
|
|
a.angle(n).endAngle() === n;
|
|
a.endAngle(0);
|
|
a.endAngle(function (d) { return d[1] * 10; });
|
|
a.endAngle(function (d, i) { return i * 10; });
|
|
}
|
|
|
|
// Tests for d3.svg.line
|
|
// Adopted from: https://github.com/mbostock/d3/blob/master/test/svg/line-test.js
|
|
function svgLineTest () {
|
|
var l = d3.svg.line(),
|
|
f: () => number,
|
|
n: number;
|
|
|
|
l.x(f).x() === f;
|
|
l.x(n).x() === n;
|
|
l.x(0);
|
|
l.x(function (d) { return d[0]; });
|
|
l.x(function (d, i) { return i; });
|
|
|
|
l.y(f).y() === f;
|
|
l.y(n).y() === n;
|
|
l.y(0);
|
|
l.y(function (d) { return d[1]; });
|
|
l.y(function (d, i) { return i; });
|
|
}
|
|
|
|
// Tests for d3.svg.line.radial
|
|
// Adopted from: https://github.com/mbostock/d3/blob/master/test/svg/line-radial-test.js
|
|
function svgLineRadialTest () {
|
|
var l = d3.svg.line.radial(),
|
|
f: () => number,
|
|
n: number;
|
|
|
|
l.radius(f).radius() === f;
|
|
l.radius(n).radius() === n;
|
|
l.radius(0);
|
|
l.radius(function (d) { return d[0]; });
|
|
l.radius(function (d, i) { return i; });
|
|
|
|
l.angle(f).angle() === f;
|
|
l.angle(n).angle() === n;
|
|
l.angle(0);
|
|
l.angle(function (d) { return d[1]; });
|
|
l.angle(function (d, i) { return i; });
|
|
}
|
|
|
|
// Tests for d3.svg.arc
|
|
// Adopted from: https://github.com/mbostock/d3/blob/master/test/svg/arc-test.js
|
|
function svgArcTest () {
|
|
var l = d3.svg.arc(),
|
|
f: () => number;
|
|
|
|
l.innerRadius(f).innerRadius() === f;
|
|
l.innerRadius(0);
|
|
l.innerRadius(function (d) { return d.innerRadius; });
|
|
l.innerRadius(function (d, i) { return i; });
|
|
|
|
l.outerRadius(f).outerRadius() === f;
|
|
l.outerRadius(0);
|
|
l.outerRadius(function (d) { return d.outerRadius; });
|
|
l.outerRadius(function (d, i) { return i; });
|
|
|
|
l.startAngle(f).startAngle() === f;
|
|
l.startAngle(0);
|
|
l.startAngle(function (d) { return d.innerRadius; });
|
|
l.startAngle(function (d, i) { return i; });
|
|
|
|
l.endAngle(f).endAngle() === f;
|
|
l.endAngle(0);
|
|
l.endAngle(function (d) { return d.outerRadius; });
|
|
l.endAngle(function (d, i) { return i; });
|
|
}
|
|
|
|
function svgArcTest2 () {
|
|
var l = d3.svg.arc<[number, number]>();
|
|
|
|
l.innerRadius(d => d[0]);
|
|
l.outerRadius(d => d[1]);
|
|
}
|
|
|
|
// Tests for d3.svg.diagonal
|
|
// Adopted from: https://github.com/mbostock/d3/blob/master/test/svg/diagonal-test.js
|
|
function svgDiagonalTest () {
|
|
var d = d3.svg.diagonal();
|
|
|
|
d.projection()({ x: 0, y: 1}, 0);
|
|
d.projection(function (d) { return [d.x, d.y]; });
|
|
d.projection(function (d, i) { return [i, i + 1]; });
|
|
|
|
d.source()({ source: {x: 0, y: 1}, target: null }, 0);
|
|
d.source({x: 0, y: 1});
|
|
d.source(function (d) { return {x: d.source.x, y: d.source.y}; });
|
|
d.source(function (d, i) { return {x: d.source.x * i, y: d.source.y * i}; });
|
|
|
|
d.target()({ target: {x: 0, y: 1}, source: null }, 0);
|
|
d.target({x: 0, y: 1});
|
|
d.target(function (d) { return {x: d.target.x, y: d.target.y}; });
|
|
d.target(function (d, i) { return {x: d.target.x * i, y: d.target.y * i}; });
|
|
}
|
|
|
|
// Tests for d3.extent
|
|
// Adopted from: https://github.com/mbostock/d3/blob/master/test/arrays/extent-test.js
|
|
function extentTest() {
|
|
// usages of `o' suppressed as well as mixed-type comparisons
|
|
// see https://github.com/Microsoft/TypeScript/commit/c0db7ffe8f55b6ec335880482ca43b93b066689d
|
|
var o = { valueOf: function () { return NaN; } };
|
|
|
|
d3.extent([1]);
|
|
d3.extent([5, 1, 2, 3, 4]);
|
|
d3.extent([20, 3]);
|
|
d3.extent([3, 20]);
|
|
d3.extent(["c", "a", "b"]);
|
|
d3.extent(["20", "3"]);
|
|
d3.extent(["3", "20"]);
|
|
d3.extent([NaN, 1, 2, 3, 4, 5]);
|
|
// d3.extent([o, 1, 2, 3, 4, 5]);
|
|
d3.extent([1, 2, 3, 4, 5, NaN]);
|
|
// d3.extent([1, 2, 3, 4, 5, o]);
|
|
d3.extent([10, null, 3, undefined, 5, NaN]);
|
|
d3.extent([-1, null, -3, undefined, -5, NaN]);
|
|
// d3.extent([20, "3"]);
|
|
// d3.extent(["20", 3]);
|
|
// d3.extent([3, "20"]);
|
|
// d3.extent(["3", 20]);
|
|
|
|
d3.extent([1], (d) => { return d; });
|
|
d3.extent([5, 1, 2, 3, 4], (d) => { return d; });
|
|
d3.extent([20, 3], (d) => { return d; });
|
|
d3.extent([3, 20], (d) => { return d; });
|
|
d3.extent(["c", "a", "b"], (d) => { return d; });
|
|
d3.extent(["20", "3"], (d) => { return d; });
|
|
d3.extent(["3", "20"], (d) => { return d; });
|
|
d3.extent([NaN, 1, 2, 3, 4, 5], (d) => { return d; });
|
|
// d3.extent([o, 1, 2, 3, 4, 5], (d) => { return d; });
|
|
d3.extent([1, 2, 3, 4, 5, NaN], (d) => { return d; });
|
|
// d3.extent([1, 2, 3, 4, 5, o], (d) => { return d; });
|
|
d3.extent([10, null, 3, undefined, 5, NaN], (d) => { return d; });
|
|
d3.extent([-1, null, -3, undefined, -5, NaN], (d) => { return d; });
|
|
// d3.extent([20, "3"], (d) => { return d; });
|
|
// d3.extent(["20", 3], (d) => { return d; });
|
|
// d3.extent([3, "20"], (d) => { return d; });
|
|
// d3.extent(["3", 20], (d) => { return d; });
|
|
}
|
|
|
|
// Tests for d3.time.format.multi
|
|
// Adopted from http://bl.ocks.org/mbostock/4149176
|
|
function multiTest() {
|
|
var customTimeFormat = d3.time.format.multi([
|
|
[".%L", function(d) { return !!d.getMilliseconds(); }],
|
|
[":%S", function(d) { return !!d.getSeconds(); }],
|
|
["%I:%M", function(d) { return !!d.getMinutes(); }],
|
|
["%I %p", function(d) { return !!d.getHours(); }],
|
|
["%a %d", function(d) { return !!d.getDay() && d.getDate() != 1; }],
|
|
["%b %d", function(d) { return d.getDate() != 1; }],
|
|
["%B", function(d) { return !!d.getMonth(); }],
|
|
["%Y", function() { return true; }]
|
|
]);
|
|
|
|
var margin = {top: 250, right: 40, bottom: 250, left: 40},
|
|
width = 960 - margin.left - margin.right,
|
|
height = 500 - margin.top - margin.bottom;
|
|
|
|
var x = d3.time.scale()
|
|
.domain([new Date(2012, 0, 1), new Date(2013, 0, 1)])
|
|
.range([0, width]);
|
|
|
|
var xAxis = d3.svg.axis()
|
|
.scale(x)
|
|
.tickFormat(customTimeFormat);
|
|
|
|
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 + ")");
|
|
|
|
svg.append("g")
|
|
.attr("class", "x axis")
|
|
.attr("transform", "translate(0," + height + ")")
|
|
.call(xAxis);
|
|
}
|
|
|
|
function testD3Events () {
|
|
d3.select('svg')
|
|
.on('click', () => {
|
|
let e = <MouseEvent>d3.event;
|
|
var coords = [e.pageX, e.pageY];
|
|
console.log("clicked", e.target, "at " + coords);
|
|
})
|
|
.on('keypress', () => {
|
|
let e = <KeyboardEvent>d3.event;
|
|
if (e.shiftKey) {
|
|
console.log('shift + ' + e.which);
|
|
}
|
|
});
|
|
}
|
|
|
|
function testD3MutlieTimeFormat() {
|
|
var format = d3.time.format.multi([
|
|
[".%L", function(d) { return d.getMilliseconds(); }],
|
|
[":%S", function(d) { return d.getSeconds(); }],
|
|
["%I:%M", function(d) { return d.getMinutes(); }],
|
|
["%I %p", function(d) { return d.getHours(); }],
|
|
["%a %d", function(d) { return d.getDay() && d.getDate() != 1; }],
|
|
["%b %d", function(d) { return d.getDate() != 1; }],
|
|
["%B", function(d) { return d.getMonth(); }],
|
|
["%Y", function() { return true; }]
|
|
]);
|
|
}
|
|
|
|
function testMultiUtcFormat() {
|
|
var format = d3.time.format.utc.multi([
|
|
[".%L", function(d) { return d.getMilliseconds(); }],
|
|
[":%S", function(d) { return d.getSeconds(); }],
|
|
["%I:%M", function(d) { return d.getMinutes(); }],
|
|
["%I %p", function(d) { return d.getHours(); }],
|
|
["%a %d", function(d) { return d.getDay() && d.getDate() != 1; }],
|
|
["%b %d", function(d) { return d.getDate() != 1; }],
|
|
["%B", function(d) { return d.getMonth(); }],
|
|
["%Y", function() { return true; }]
|
|
]);
|
|
}
|
|
|
|
function testEnterSizeEmpty() {
|
|
|
|
var selectionSize: number,
|
|
emptyStatus: boolean;
|
|
|
|
var newNodes = d3.selectAll('.test')
|
|
.data(['1', '2', '3'])
|
|
.enter();
|
|
|
|
emptyStatus = newNodes.empty();
|
|
selectionSize = newNodes.size();
|
|
|
|
} |