diff --git a/README.md b/README.md index dd938b1..7444b92 100644 --- a/README.md +++ b/README.md @@ -26,7 +26,7 @@ If you use NPM, `npm install d3-tube-map`. Otherwise, download the [latest relea var width = 1600; var height = 1024; - const map = tubeMap.tubeMap()() + const map = tubeMap.tubeMap() .width(width) .height(height) .margin({ diff --git a/example/index.html b/example/index.html index 11c9c54..1924ab8 100644 --- a/example/index.html +++ b/example/index.html @@ -21,7 +21,7 @@ var width = 1600; var height = 1024; - const map = tubeMap.tubeMap()() + const map = tubeMap.tubeMap() .width(width) .height(height) .margin({ diff --git a/package.json b/package.json index 5922039..3e215af 100644 --- a/package.json +++ b/package.json @@ -1,6 +1,6 @@ { "name": "d3-tube-map", - "version": "0.0.5", + "version": "0.0.6", "description": "Draw tube map.", "keywords": [ "d3", diff --git a/src/tubeMap.js b/src/tubeMap.js index 72feb05..1c21c94 100644 --- a/src/tubeMap.js +++ b/src/tubeMap.js @@ -1,187 +1,241 @@ import * as d3 from 'd3'; -export default function() { +export default function tubeMap() { + var margin = {top: 80, right: 80, bottom: 20, left: 80}; + var width = 760; + var height = 640; + var xScale = d3.scaleLinear(); + var yScale = d3.scaleLinear(); + var xGeoScale = d3.scaleLinear(); + var yGeoScale = d3.scaleLinear(); + var lineWidth; + var lineWidthMultiplier = 1.2; - function tubeMap() { - var margin = {top: 80, right: 80, bottom: 20, left: 80}; - var width = 760; - var height = 640; - var xScale = d3.scaleLinear(); - var yScale = d3.scaleLinear(); - var xGeoScale = d3.scaleLinear(); - var yGeoScale = d3.scaleLinear(); - var lineWidth; - var lineWidthMultiplier = 1.2; + var dispatch = d3.dispatch("click"); - var dispatch = d3.dispatch("click"); + var svg; - var svg; + var model; - var model; + var gEnter; + var zoom; - var gEnter; - var zoom; + var t; + var s; - var t; - var s; + function map(selection) { + selection.each(function(data) { + // Convert data to standard representation + data = mangleData(data); - function map(selection) { - selection.each(function(data) { - // Convert data to standard representation - data = mangleData(data); + model = data; - model = data; - - var minX = d3.min(data.raw, function(line) { - return d3.min(line.nodes, function(node) { return node.coords[0]; }); - }); - var maxX = d3.max(data.raw, function(line) { - return d3.max(line.nodes, function(node) { return node.coords[0]; }); + var minX = d3.min(data.raw, function(line) { + return d3.min(line.nodes, function(node) { return node.coords[0]; }); }); + var maxX = d3.max(data.raw, function(line) { + return d3.max(line.nodes, function(node) { return node.coords[0]; }); + }); - var minY = d3.min(data.raw, function(line) { - return d3.min(line.nodes, function(node) { return node.coords[1]; }); - }); - var maxY = d3.max(data.raw, function(line) { - return d3.max(line.nodes, function(node) { return node.coords[1]; }); + var minY = d3.min(data.raw, function(line) { + return d3.min(line.nodes, function(node) { return node.coords[1]; }); }); + var maxY = d3.max(data.raw, function(line) { + return d3.max(line.nodes, function(node) { return node.coords[1]; }); + }); - var desiredAspectRatio = (maxX - minX)/(maxY - minY); - var actualAspectRatio = (width - margin.left - margin.right)/(height - margin.top - margin.bottom); - var ratioRatio = actualAspectRatio/desiredAspectRatio; - - var maxXRange, maxYRange; - - // Note that we flip the sense of the y-axis here - if (desiredAspectRatio > actualAspectRatio) { - maxXRange = width - margin.left - margin.right; - maxYRange = (height - margin.top - margin.bottom)*ratioRatio; - } else { - maxXRange = (width - margin.left - margin.right)/ratioRatio; - maxYRange = height - margin.top - margin.bottom; - } - - // Update the x-scale - xScale - .domain([minX, maxX]) - .range([0, maxXRange]); - - // Update the y-scale - yScale - .domain([minY, maxY]) - .range([maxYRange, 0]); - - // Update the x-geo-scale - xGeoScale - .domain([d3.min(data.stations.toArray(), function(station) { if (station.position !== undefined) { return station.position.lon; }}), - d3.max(data.stations.toArray(), function(station) { if (station.position !== undefined) { return station.position.lon; }})]) - .range([0, maxXRange]); - - // Update the y--geo-scale - yGeoScale - .domain([d3.min(data.stations.toArray(), function(station) { if (station.position !== undefined) { return station.position.lat; }}), - d3.max(data.stations.toArray(), function(station) { if (station.position !== undefined) { return station.position.lat; }})]) - .range([maxYRange, 0]); - - // Update line width - lineWidth = lineWidthMultiplier*(xScale(1) - xScale(0)); - - // Select the svg element, if it exists - svg = d3.select(this).selectAll("svg").data([data]); - - var g = svg.enter().append("svg").append("g"); - - // Fill with white rectangle to capture zoom events - g.append("rect") - .attr("width", "100%") - .attr("height", "100%") - .attr('fill', 'white'); - - var zoomed = function() { - gEnter.attr("transform", d3.event.transform.toString()); - }; - - zoom = d3.zoom().scaleExtent([0.5, 6]).on("zoom", zoomed); - - gEnter = g.call(zoom).append("g"); - - var river = gEnter.append("g").attr("class", "river") - .selectAll("path").data(function(d) { return [d.river]; }); - - var lines = gEnter.append("g").attr("class", "lines") - .selectAll("path").data(function(d) { return d.lines.lines; }); - - var interchanges = gEnter.append("g").attr("class", "interchanges") - .selectAll("path").data(function(d) { return d.stations.interchanges(); }); - - var stations = gEnter.append("g").attr("class", "stations") - .selectAll("path").data(function(d) { return d.stations.normalStations(); }); - - var labels = gEnter.append("g").attr("class", "labels") - .selectAll("text").data(function(d) { return d.stations.toArray(); }); - - var geoStations = gEnter.append("g").attr("class", "geoStations").style("display", "none") - .selectAll("path").data(function(d) { return d.stations.toArray(); }); - - var discrepencies = gEnter.append("g").attr("class", "discrepencies").style("display", "none") - .selectAll("path").data(function(d) { return d.stations.toArray(); }); + var desiredAspectRatio = (maxX - minX)/(maxY - minY); + var actualAspectRatio = (width - margin.left - margin.right)/(height - margin.top - margin.bottom); + var ratioRatio = actualAspectRatio/desiredAspectRatio; - // Update the outer dimensions - svg.attr("width", '100%') - .attr("height", '100%'); + var maxXRange, maxYRange; - // Update the river - river.enter().append("path") - .attr("d", drawLine) - .attr("stroke", "#C4E8F8") - .attr("fill", "none") - .attr("stroke-width", 1.8*lineWidth); + // Note that we flip the sense of the y-axis here + if (desiredAspectRatio > actualAspectRatio) { + maxXRange = width - margin.left - margin.right; + maxYRange = (height - margin.top - margin.bottom)*ratioRatio; + } else { + maxXRange = (width - margin.left - margin.right)/ratioRatio; + maxYRange = height - margin.top - margin.bottom; + } - // Update the lines - lines.enter().append("path") - .attr("d", drawLine) - .attr("id", function(d) { return d.name; }) - .attr("stroke", function(d) { return d.color; }) - .attr("fill", "none") - .attr("stroke-width", function(d) { return d.highlighted ? lineWidth * 1.3 : lineWidth; }) - .classed("line", true); + // Update the x-scale + xScale + .domain([minX, maxX]) + .range([0, maxXRange]); - var fgColor = "#000000"; - var bgColor = "#ffffff"; + // Update the y-scale + yScale + .domain([minY, maxY]) + .range([maxYRange, 0]); - var markerFunction = d3.arc() - .innerRadius(0) - .outerRadius(lineWidth) - .startAngle(0) - .endAngle(2*Math.PI); + // Update the x-geo-scale + xGeoScale + .domain([d3.min(data.stations.toArray(), function(station) { if (station.position !== undefined) { return station.position.lon; }}), + d3.max(data.stations.toArray(), function(station) { if (station.position !== undefined) { return station.position.lon; }})]) + .range([0, maxXRange]); + + // Update the y--geo-scale + yGeoScale + .domain([d3.min(data.stations.toArray(), function(station) { if (station.position !== undefined) { return station.position.lat; }}), + d3.max(data.stations.toArray(), function(station) { if (station.position !== undefined) { return station.position.lat; }})]) + .range([maxYRange, 0]); + + // Update line width + lineWidth = lineWidthMultiplier*(xScale(1) - xScale(0)); + + // Select the svg element, if it exists + svg = d3.select(this).selectAll("svg").data([data]); + + var g = svg.enter().append("svg").append("g"); + + // Fill with white rectangle to capture zoom events + g.append("rect") + .attr("width", "100%") + .attr("height", "100%") + .attr('fill', 'white'); + + var zoomed = function() { + gEnter.attr("transform", d3.event.transform.toString()); + }; + + zoom = d3.zoom().scaleExtent([0.5, 6]).on("zoom", zoomed); + + gEnter = g.call(zoom).append("g"); + + var river = gEnter.append("g").attr("class", "river") + .selectAll("path").data(function(d) { return [d.river]; }); + + var lines = gEnter.append("g").attr("class", "lines") + .selectAll("path").data(function(d) { return d.lines.lines; }); + + var interchanges = gEnter.append("g").attr("class", "interchanges") + .selectAll("path").data(function(d) { return d.stations.interchanges(); }); + + var stations = gEnter.append("g").attr("class", "stations") + .selectAll("path").data(function(d) { return d.stations.normalStations(); }); + + var labels = gEnter.append("g").attr("class", "labels") + .selectAll("text").data(function(d) { return d.stations.toArray(); }); + + var geoStations = gEnter.append("g").attr("class", "geoStations").style("display", "none") + .selectAll("path").data(function(d) { return d.stations.toArray(); }); + + var discrepencies = gEnter.append("g").attr("class", "discrepencies").style("display", "none") + .selectAll("path").data(function(d) { return d.stations.toArray(); }); + + // Update the outer dimensions + svg.attr("width", '100%') + .attr("height", '100%'); + + // Update the river + river.enter().append("path") + .attr("d", drawLine) + .attr("stroke", "#C4E8F8") + .attr("fill", "none") + .attr("stroke-width", 1.8*lineWidth); + + // Update the lines + lines.enter().append("path") + .attr("d", drawLine) + .attr("id", function(d) { return d.name; }) + .attr("stroke", function(d) { return d.color; }) + .attr("fill", "none") + .attr("stroke-width", function(d) { return d.highlighted ? lineWidth * 1.3 : lineWidth; }) + .classed("line", true); + + var fgColor = "#000000"; + var bgColor = "#ffffff"; + + var markerFunction = d3.arc() + .innerRadius(0) + .outerRadius(lineWidth) + .startAngle(0) + .endAngle(2*Math.PI); + + // Update the interchanges + interchanges.enter().append("g") + .attr("id", function(d) { return d.name; }) + .on("click", function() { + var label = d3.select(this); + var name = label.attr("id"); + + selectStation(name); + + dispatch.call("click", this, name); + }) + .append("path") + .attr("d", markerFunction) + .attr("transform", function(d) { return "translate(" + xScale(d.x + d.marker[0].shiftX*lineWidthMultiplier) + "," + yScale(d.y + d.marker[0].shiftY*lineWidthMultiplier) + ")" }) + .attr("stroke-width", lineWidth/2) + .attr("fill", function(d) { return d.visited ? fgColor : bgColor; }) + .attr("stroke", function(d) { return d.visited ? bgColor : fgColor; }) + .classed("interchange", true) + .style("cursor", "pointer"); + + var lineFunction = d3.line() + .x(function(d) { return xScale(d[0]); }) + .y(function(d) { return yScale(d[1]); }); + + // Update the stations + stations.enter().append("g") + .attr("id", function(d) { return d.name; }) + .on("click", function() { + var label = d3.select(this); + var name = label.attr("id"); + + selectStation(name); + + dispatch.call("click", this, name); + }) + .append("path") + .attr("d", function(d) { + var dir; - // Update the interchanges - interchanges.enter().append("g") - .attr("id", function(d) { return d.name; }) - .on("click", function() { - var label = d3.select(this); - var name = label.attr("id"); + var sqrt2 = Math.sqrt(2); - selectStation(name); + switch (d.labelPos.toLowerCase()) { + case "n": + dir = [0, 1]; + break; + case "ne": + dir = [1/sqrt2, 1/sqrt2]; + break; + case "e": + dir = [1, 0]; + break; + case "se": + dir = [1/sqrt2, -1/sqrt2]; + break; + case "s": + dir = [0, -1]; + break; + case "sw": + dir = [-1/sqrt2, -1/sqrt2]; + break; + case "w": + dir = [-1, 0]; + break; + case "nw": + dir = [-1/sqrt2, 1/sqrt2]; + break; + default: + break; + } - dispatch.call("click", this, name); - }) - .append("path") - .attr("d", markerFunction) - .attr("transform", function(d) { return "translate(" + xScale(d.x + d.marker[0].shiftX*lineWidthMultiplier) + "," + yScale(d.y + d.marker[0].shiftY*lineWidthMultiplier) + ")" }) - .attr("stroke-width", lineWidth/2) - .attr("fill", function(d) { return d.visited ? fgColor : bgColor; }) - .attr("stroke", function(d) { return d.visited ? bgColor : fgColor; }) - .classed("interchange", true) - .style("cursor", "pointer"); - - var lineFunction = d3.line() - .x(function(d) { return xScale(d[0]); }) - .y(function(d) { return yScale(d[1]); }); - - // Update the stations - stations.enter().append("g") + return lineFunction([[d.x + (d.shiftX*lineWidthMultiplier) + lineWidthMultiplier/2.05*dir[0], d.y + (d.shiftY*lineWidthMultiplier) + lineWidthMultiplier/2.05*dir[1]], [d.x + (d.shiftX*lineWidthMultiplier) + lineWidthMultiplier*dir[0], d.y + (d.shiftY*lineWidthMultiplier) + lineWidthMultiplier*dir[1]]]); + }) + .attr("stroke", function(d) { return d.color; }) + .attr("stroke-width", lineWidth/2) + .attr("fill", "none") + .attr("class", function(d) { return d.line; }) + .attr("id", function(d) { return d.name; }) + .classed("station", true); + + // Update the label text + labels.enter().append("g") .attr("id", function(d) { return d.name; }) + .classed("label", true) .on("click", function() { var label = d3.select(this); var name = label.attr("id"); @@ -190,555 +244,495 @@ export default function() { dispatch.call("click", this, name); }) - .append("path") - .attr("d", function(d) { - var dir; - - var sqrt2 = Math.sqrt(2); - - switch (d.labelPos.toLowerCase()) { - case "n": - dir = [0, 1]; - break; - case "ne": - dir = [1/sqrt2, 1/sqrt2]; - break; - case "e": - dir = [1, 0]; - break; - case "se": - dir = [1/sqrt2, -1/sqrt2]; - break; - case "s": - dir = [0, -1]; - break; - case "sw": - dir = [-1/sqrt2, -1/sqrt2]; - break; - case "w": - dir = [-1, 0]; - break; - case "nw": - dir = [-1/sqrt2, 1/sqrt2]; - break; - default: - break; - } - - return lineFunction([[d.x + (d.shiftX*lineWidthMultiplier) + lineWidthMultiplier/2.05*dir[0], d.y + (d.shiftY*lineWidthMultiplier) + lineWidthMultiplier/2.05*dir[1]], [d.x + (d.shiftX*lineWidthMultiplier) + lineWidthMultiplier*dir[0], d.y + (d.shiftY*lineWidthMultiplier) + lineWidthMultiplier*dir[1]]]); + .append("text") + .text(function(d) { return d.label }) + .attr("dy", 0.1) + .attr("x", function(d) { return xScale(d.x + d.labelShiftX) + textPos(d).pos[0]; }) + .attr("y", function(d) { return yScale(d.y + d.labelShiftY) - textPos(d).pos[1]; }) // Flip y-axis + .attr("text-anchor", function(d) { return textPos(d).textAnchor }) + .style("display", function(d) { return d.hide !== true ? "block" : "none"; }) + .style("font-size", 1.2*lineWidth/lineWidthMultiplier + "px") + .style("-webkit-user-select", "none") + .attr("class", function(d) { + return d.marker.map(function(marker) { return marker.line; }).join(" "); }) - .attr("stroke", function(d) { return d.color; }) - .attr("stroke-width", lineWidth/2) - .attr("fill", "none") - .attr("class", function(d) { return d.line; }) - .attr("id", function(d) { return d.name; }) - .classed("station", true); - - // Update the label text - labels.enter().append("g") - .attr("id", function(d) { return d.name; }) - .classed("label", true) - .on("click", function() { - var label = d3.select(this); - var name = label.attr("id"); - - selectStation(name); - - dispatch.call("click", this, name); - }) - .append("text") - .text(function(d) { return d.label }) - .attr("dy", 0.1) - .attr("x", function(d) { return xScale(d.x + d.labelShiftX) + textPos(d).pos[0]; }) - .attr("y", function(d) { return yScale(d.y + d.labelShiftY) - textPos(d).pos[1]; }) // Flip y-axis - .attr("text-anchor", function(d) { return textPos(d).textAnchor }) - .style("display", function(d) { return d.hide !== true ? "block" : "none"; }) - .style("font-size", 1.2*lineWidth/lineWidthMultiplier + "px") - .style("-webkit-user-select", "none") - .attr("class", function(d) { - return d.marker.map(function(marker) { return marker.line; }).join(" "); - }) - .classed("highlighted", function(d) { return d.visited; }) - .call(wrap); - - var markerGeoFunction = d3.arc() - .innerRadius(0) - .outerRadius(lineWidth/4) - .startAngle(0) - .endAngle(2*Math.PI); - - // Update the geo stations - geoStations.enter().append("path") - .attr("d", markerGeoFunction) - .attr("transform", function(d) { return "translate(" + xGeoScale((d.position !== undefined) ? d.position.lon : NaN) + "," + yGeoScale(d.position !== undefined ? d.position.lat : NaN) + ")" }) - .attr("id", function(d) { return d.name; }) - .attr("fill", '#888888'); - - // Update the geo stations - discrepencies.enter().append("path") - .attr("d", function(d) { return d3.line()([[xScale(d.x), yScale(d.y)], [xGeoScale(d.position.lon), yGeoScale(d.position.lat)]]); }) - .attr("id", function(d) { return d.name; }) - .attr("stroke", '#AAAAAA') - .attr("stroke-width", lineWidth/4) - .style("stroke-dasharray", ("3, 3")); - }); - } - - map.width = function(w) { - if (!arguments.length) return width; - width = w; - return map; - } + .classed("highlighted", function(d) { return d.visited; }) + .call(wrap); - map.height = function(h) { - if (!arguments.length) return height; - height = h; - return map; - } - - map.margin = function(m) { - if (!arguments.length) return margin; - margin = m; - return map; - } - - map.highlightLine = function(name) { - var lines = d3.select("#map").selectAll(".line"); - var stations = d3.select("#map").selectAll(".station"); - var labels = d3.select("#map").selectAll(".label"); - - lines.classed("translucent", true); - stations.classed("translucent", true); - labels.classed("translucent", true); - - stations.filter("." + name).classed("translucent", false); - labels.filter("." + name).classed("translucent", false); - d3.select("#" + name).classed("translucent", false); + var markerGeoFunction = d3.arc() + .innerRadius(0) + .outerRadius(lineWidth/4) + .startAngle(0) + .endAngle(2*Math.PI); - } + // Update the geo stations + geoStations.enter().append("path") + .attr("d", markerGeoFunction) + .attr("transform", function(d) { return "translate(" + xGeoScale((d.position !== undefined) ? d.position.lon : NaN) + "," + yGeoScale(d.position !== undefined ? d.position.lat : NaN) + ")" }) + .attr("id", function(d) { return d.name; }) + .attr("fill", '#888888'); - map.unhighlightAll = function() { - var lines = d3.select("#map").selectAll(".line"); - var stations = d3.select("#map").selectAll(".station"); - var labels = d3.select("#map").selectAll(".label"); + // Update the geo stations + discrepencies.enter().append("path") + .attr("d", function(d) { return d3.line()([[xScale(d.x), yScale(d.y)], [xGeoScale(d.position.lon), yGeoScale(d.position.lat)]]); }) + .attr("id", function(d) { return d.name; }) + .attr("stroke", '#AAAAAA') + .attr("stroke-width", lineWidth/4) + .style("stroke-dasharray", ("3, 3")); + }); + } - lines.classed("translucent", false); - stations.classed("translucent", false); - labels.classed("translucent", false); - } + map.width = function(w) { + if (!arguments.length) return width; + width = w; + return map; + } - map.unhighlightLine = function() { - this.unhighlightAll(); - } + map.height = function(h) { + if (!arguments.length) return height; + height = h; + return map; + } - map.highlightNearestStation = function(name) { - var station = model.stations.stations[name]; - } + map.margin = function(m) { + if (!arguments.length) return margin; + margin = m; + return map; + } - map.centerOnPub = function(name) { - if (name === undefined) return; + map.highlightLine = function(name) { + var lines = d3.select("#map").selectAll(".line"); + var stations = d3.select("#map").selectAll(".station"); + var labels = d3.select("#map").selectAll(".label"); - var station = model.stations.stations[name]; + lines.classed("translucent", true); + stations.classed("translucent", true); + labels.classed("translucent", true); - var width = window.innerWidth; - var height = window.innerHeight; + stations.filter("." + name).classed("translucent", false); + labels.filter("." + name).classed("translucent", false); + d3.select("#" + name).classed("translucent", false); - var scale = 2; + } - t = [-scale*(xScale(station.x)) + width/2, -scale*(yScale(station.y)) + height/2] + map.unhighlightAll = function() { + var lines = d3.select("#map").selectAll(".line"); + var stations = d3.select("#map").selectAll(".station"); + var labels = d3.select("#map").selectAll(".label"); - zoom.translate(t).scaleTo(2); - gEnter.transition().duration(750).attr("transform", "translate(" + t[0] + "," + t[1] + ")scale(" + scale + ")"); - } + lines.classed("translucent", false); + stations.classed("translucent", false); + labels.classed("translucent", false); + } - map.addStation = function(name) { - visitStation(name, true); - } + map.unhighlightLine = function() { + this.unhighlightAll(); + } - map.removeStation = function(name) { - visitStation(name, false); - } + map.highlightNearestStation = function(name) { + var station = model.stations.stations[name]; + } - map.visitStations = function(visited) { - d3.selectAll(".labels").select("text").classed("highlighted", false); - visited.map((pub) => { - visitStation(pub, true); - }); - } + map.centerOnPub = function(name) { + if (name === undefined) return; - map.on = function(event, callback) { - dispatch.on(event, callback) - } + var station = model.stations.stations[name]; - map.selectStation = function(name) { - selectStation(name); - } + var width = window.innerWidth; + var height = window.innerHeight; - function selectStation(name) { - d3.select(".labels").selectAll(".label").classed("selected", false); - d3.select(".labels").select("#" + name).classed("selected", true); - } + var scale = 2; - function visitStation(name, highlighted) { - d3.select(".labels").select("#" + name).select("text").classed("highlighted", highlighted); - } + t = [-scale*(xScale(station.x)) + width/2, -scale*(yScale(station.y)) + height/2] - function drawLine(data) { - var path = ""; + zoom.translate(t).scaleTo(2); + gEnter.transition().duration(750).attr("transform", "translate(" + t[0] + "," + t[1] + ")scale(" + scale + ")"); + } - var lineNodes = data.nodes; + map.addStation = function(name) { + visitStation(name, true); + } - var unitLength = xScale(1) - xScale(0); + map.removeStation = function(name) { + visitStation(name, false); + } - var shiftCoords = [data.shiftCoords[0]*lineWidth/unitLength, data.shiftCoords[1]*lineWidth/unitLength]; + map.visitStations = function(visited) { + d3.selectAll(".labels").select("text").classed("highlighted", false); + visited.map((pub) => { + visitStation(pub, true); + }); + } - var lastSectionType = "diagonal"; // TODO: HACK + map.on = function(event, callback) { + dispatch.on(event, callback) + } - var nextNode, currNode, xDiff, yDiff; - var points; + map.selectStation = function(name) { + selectStation(name); + } - for (var lineNode = 0; lineNode < lineNodes.length; lineNode++) { - if (lineNode > 0) { - nextNode = lineNodes[lineNode]; - currNode = lineNodes[lineNode - 1]; + function selectStation(name) { + d3.select(".labels").selectAll(".label").classed("selected", false); + d3.select(".labels").select("#" + name).classed("selected", true); + } - var direction = ""; + function visitStation(name, highlighted) { + d3.select(".labels").select("#" + name).select("text").classed("highlighted", highlighted); + } - xDiff = Math.round(currNode.coords[0] - nextNode.coords[0]); - yDiff = Math.round(currNode.coords[1] - nextNode.coords[1]); + function drawLine(data) { + var path = ""; - var lineEndCorrection = [0, 0]; + var lineNodes = data.nodes; - if (lineNode === lineNodes.length - 1) { - if ((xDiff == 0) || (yDiff == 0)) { - if (xDiff > 0) - lineEndCorrection = [-lineWidth / (4 * unitLength), 0]; - if (xDiff < 0) - lineEndCorrection = [lineWidth / (4 * unitLength), 0]; - if (yDiff > 0) - lineEndCorrection = [0, -lineWidth / (4 * unitLength)]; - if (yDiff < 0) - lineEndCorrection = [0, lineWidth / (4 * unitLength)]; - } else { - var sqrt2 = Math.sqrt(2); + var unitLength = xScale(1) - xScale(0); - if ((xDiff > 0) && (yDiff > 0)) - lineEndCorrection = [-lineWidth / (4 * unitLength * sqrt2), -lineWidth / (4 * unitLength * sqrt2)]; - if ((xDiff > 0) && (yDiff < 0)) - lineEndCorrection = [-lineWidth / (4 * unitLength * sqrt2), lineWidth / (4 * unitLength * sqrt2)]; - if ((xDiff < 0) && (yDiff > 0)) - lineEndCorrection = [lineWidth / (4 * unitLength * sqrt2), -lineWidth / (4 * unitLength * sqrt2)]; - if ((xDiff < 0) && (yDiff < 0)) - lineEndCorrection = [lineWidth / (4 * unitLength * sqrt2), lineWidth / (4 * unitLength * sqrt2)]; - } - } + var shiftCoords = [data.shiftCoords[0]*lineWidth/unitLength, data.shiftCoords[1]*lineWidth/unitLength]; - points = [ - [ - xScale(currNode.coords[0] + shiftCoords[0]), - yScale(currNode.coords[1] + shiftCoords[1]) - ], - [ - xScale(nextNode.coords[0] + shiftCoords[0] + lineEndCorrection[0]), - yScale(nextNode.coords[1] + shiftCoords[1] + lineEndCorrection[1]) - ] - ]; + var lastSectionType = "diagonal"; // TODO: HACK - if ((xDiff == 0) || (yDiff == 0)) { - lastSectionType = "udlr"; - path += "L" + points[1][0] + "," + points[1][1]; - } else if ((Math.abs(xDiff) == Math.abs(yDiff)) && (Math.abs(xDiff) > 1)) { - lastSectionType = "diagonal"; - path += "L" + points[1][0] + "," + points[1][1]; - } - else if ((Math.abs(xDiff) == 1) && (Math.abs(yDiff) == 1)) { - direction = nextNode.dir.toLowerCase(); + var nextNode, currNode, xDiff, yDiff; + var points; - switch (direction) { - case "e": - path += "Q" + points[1][0] + "," + points[0][1] + "," + points[1][0] + "," + points[1][1]; - break; - case "s": - path += "Q" + points[0][0] + "," + points[1][1] + "," + points[1][0] + "," + points[1][1]; - break; - case "n": - path += "Q" + points[0][0] + "," + points[1][1] + "," + points[1][0] + "," + points[1][1]; - break; - case "w": - path += "Q" + points[1][0] + "," + points[0][1] + "," + points[1][0] + "," + points[1][1]; - break; - } - } - else if (((Math.abs(xDiff) == 1) && (Math.abs(yDiff) == 2)) || ((Math.abs(xDiff) == 2) && (Math.abs(yDiff) == 1))) { - var controlPoints; - if (xDiff == 1) { - if (lastSectionType == "udlr") { - controlPoints = [ - points[0][0], - points[0][1] + (points[1][1] - points[0][1]) / 2 - ]; - } else if (lastSectionType == "diagonal") { - controlPoints = [ - points[1][0], - points[0][1] + (points[1][1] - points[0][1]) / 2 - ]; - } - } else if (xDiff == -1) { - if (lastSectionType == "udlr") { - controlPoints = [ - points[0][0], - points[0][1] + (points[1][1] - points[0][1])/2 - ]; - } else if (lastSectionType == "diagonal") { - controlPoints = [ - points[1][0], - points[0][1] + (points[1][1] - points[0][1])/2 - ]; - } - } else if (xDiff == -2) { - if (lastSectionType == "udlr") { - controlPoints = [ - points[0][0] + (points[1][0] - points[0][0]) / 2, - points[0][1] - ]; - } else if (lastSectionType == "diagonal") { - controlPoints = [ - points[0][0] + (points[1][0] - points[0][0]) / 2, - points[1][1] - ]; - } - } else if (xDiff == 2) { - if (lastSectionType == "udlr") { - controlPoints = [ - points[0][0] + (points[1][0] - points[0][0]) / 2, - points[0][1] - ]; - } else if (lastSectionType == "diagonal") { - controlPoints = [ - points[0][0] + (points[1][0] - points[0][0]) / 2, - points[1][1] - ]; - } - } + for (var lineNode = 0; lineNode < lineNodes.length; lineNode++) { + if (lineNode > 0) { + nextNode = lineNodes[lineNode]; + currNode = lineNodes[lineNode - 1]; - path += "C" + controlPoints[0] + "," + controlPoints[1] + "," + controlPoints[0] + "," + controlPoints[1] + "," + points[1][0] + "," + points[1][1]; - } - } else { - nextNode = lineNodes[lineNode + 1]; - currNode = lineNodes[lineNode]; + var direction = ""; - xDiff = Math.round(currNode.coords[0] - nextNode.coords[0]); - yDiff = Math.round(currNode.coords[1] - nextNode.coords[1]); + xDiff = Math.round(currNode.coords[0] - nextNode.coords[0]); + yDiff = Math.round(currNode.coords[1] - nextNode.coords[1]); - var lineStartCorrection = [0, 0]; + var lineEndCorrection = [0, 0]; + if (lineNode === lineNodes.length - 1) { if ((xDiff == 0) || (yDiff == 0)) { if (xDiff > 0) - lineStartCorrection = [lineWidth / (4 * unitLength), 0]; + lineEndCorrection = [-lineWidth / (4 * unitLength), 0]; if (xDiff < 0) - lineStartCorrection = [-lineWidth / (4 * unitLength), 0]; + lineEndCorrection = [lineWidth / (4 * unitLength), 0]; if (yDiff > 0) - lineStartCorrection = [0, lineWidth / (4 * unitLength)]; + lineEndCorrection = [0, -lineWidth / (4 * unitLength)]; if (yDiff < 0) - lineStartCorrection = [0, -lineWidth / (4 * unitLength)]; + lineEndCorrection = [0, lineWidth / (4 * unitLength)]; } else { var sqrt2 = Math.sqrt(2); + if ((xDiff > 0) && (yDiff > 0)) - lineStartCorrection = [lineWidth / (4 * unitLength * sqrt2), lineWidth / (4 * unitLength * sqrt2)]; + lineEndCorrection = [-lineWidth / (4 * unitLength * sqrt2), -lineWidth / (4 * unitLength * sqrt2)]; if ((xDiff > 0) && (yDiff < 0)) - lineStartCorrection = [lineWidth / (4 * unitLength * sqrt2), -lineWidth / (4 * unitLength * sqrt2)]; + lineEndCorrection = [-lineWidth / (4 * unitLength * sqrt2), lineWidth / (4 * unitLength * sqrt2)]; if ((xDiff < 0) && (yDiff > 0)) - lineStartCorrection = [-lineWidth / (4 * unitLength * sqrt2), lineWidth / (4 * unitLength * sqrt2)]; + lineEndCorrection = [lineWidth / (4 * unitLength * sqrt2), -lineWidth / (4 * unitLength * sqrt2)]; if ((xDiff < 0) && (yDiff < 0)) - lineStartCorrection = [-lineWidth / (4 * unitLength * sqrt2), -lineWidth / (4 * unitLength * sqrt2)]; + lineEndCorrection = [lineWidth / (4 * unitLength * sqrt2), lineWidth / (4 * unitLength * sqrt2)]; } - - points = [ - xScale(currNode.coords[0] + shiftCoords[0] + lineStartCorrection[0]), - yScale(currNode.coords[1] + shiftCoords[1] + lineStartCorrection[1]) - ]; - - path += "M" + points[0] + "," + points[1]; } - } - return path; - } + points = [ + [ + xScale(currNode.coords[0] + shiftCoords[0]), + yScale(currNode.coords[1] + shiftCoords[1]) + ], + [ + xScale(nextNode.coords[0] + shiftCoords[0] + lineEndCorrection[0]), + yScale(nextNode.coords[1] + shiftCoords[1] + lineEndCorrection[1]) + ] + ]; + + if ((xDiff == 0) || (yDiff == 0)) { + lastSectionType = "udlr"; + path += "L" + points[1][0] + "," + points[1][1]; + } else if ((Math.abs(xDiff) == Math.abs(yDiff)) && (Math.abs(xDiff) > 1)) { + lastSectionType = "diagonal"; + path += "L" + points[1][0] + "," + points[1][1]; + } + else if ((Math.abs(xDiff) == 1) && (Math.abs(yDiff) == 1)) { + direction = nextNode.dir.toLowerCase(); + + switch (direction) { + case "e": + path += "Q" + points[1][0] + "," + points[0][1] + "," + points[1][0] + "," + points[1][1]; + break; + case "s": + path += "Q" + points[0][0] + "," + points[1][1] + "," + points[1][0] + "," + points[1][1]; + break; + case "n": + path += "Q" + points[0][0] + "," + points[1][1] + "," + points[1][0] + "," + points[1][1]; + break; + case "w": + path += "Q" + points[1][0] + "," + points[0][1] + "," + points[1][0] + "," + points[1][1]; + break; + } + } + else if (((Math.abs(xDiff) == 1) && (Math.abs(yDiff) == 2)) || ((Math.abs(xDiff) == 2) && (Math.abs(yDiff) == 1))) { + var controlPoints; + if (xDiff == 1) { + if (lastSectionType == "udlr") { + controlPoints = [ + points[0][0], + points[0][1] + (points[1][1] - points[0][1]) / 2 + ]; + } else if (lastSectionType == "diagonal") { + controlPoints = [ + points[1][0], + points[0][1] + (points[1][1] - points[0][1]) / 2 + ]; + } + } else if (xDiff == -1) { + if (lastSectionType == "udlr") { + controlPoints = [ + points[0][0], + points[0][1] + (points[1][1] - points[0][1])/2 + ]; + } else if (lastSectionType == "diagonal") { + controlPoints = [ + points[1][0], + points[0][1] + (points[1][1] - points[0][1])/2 + ]; + } + } else if (xDiff == -2) { + if (lastSectionType == "udlr") { + controlPoints = [ + points[0][0] + (points[1][0] - points[0][0]) / 2, + points[0][1] + ]; + } else if (lastSectionType == "diagonal") { + controlPoints = [ + points[0][0] + (points[1][0] - points[0][0]) / 2, + points[1][1] + ]; + } + } else if (xDiff == 2) { + if (lastSectionType == "udlr") { + controlPoints = [ + points[0][0] + (points[1][0] - points[0][0]) / 2, + points[0][1] + ]; + } else if (lastSectionType == "diagonal") { + controlPoints = [ + points[0][0] + (points[1][0] - points[0][0]) / 2, + points[1][1] + ]; + } + } - function mangleData(data) { - var mangledData = {}; + path += "C" + controlPoints[0] + "," + controlPoints[1] + "," + controlPoints[0] + "," + controlPoints[1] + "," + points[1][0] + "," + points[1][1]; + } + } else { + nextNode = lineNodes[lineNode + 1]; + currNode = lineNodes[lineNode]; + + xDiff = Math.round(currNode.coords[0] - nextNode.coords[0]); + yDiff = Math.round(currNode.coords[1] - nextNode.coords[1]); + + var lineStartCorrection = [0, 0]; + + if ((xDiff == 0) || (yDiff == 0)) { + if (xDiff > 0) + lineStartCorrection = [lineWidth / (4 * unitLength), 0]; + if (xDiff < 0) + lineStartCorrection = [-lineWidth / (4 * unitLength), 0]; + if (yDiff > 0) + lineStartCorrection = [0, lineWidth / (4 * unitLength)]; + if (yDiff < 0) + lineStartCorrection = [0, -lineWidth / (4 * unitLength)]; + } else { + var sqrt2 = Math.sqrt(2); + if ((xDiff > 0) && (yDiff > 0)) + lineStartCorrection = [lineWidth / (4 * unitLength * sqrt2), lineWidth / (4 * unitLength * sqrt2)]; + if ((xDiff > 0) && (yDiff < 0)) + lineStartCorrection = [lineWidth / (4 * unitLength * sqrt2), -lineWidth / (4 * unitLength * sqrt2)]; + if ((xDiff < 0) && (yDiff > 0)) + lineStartCorrection = [-lineWidth / (4 * unitLength * sqrt2), lineWidth / (4 * unitLength * sqrt2)]; + if ((xDiff < 0) && (yDiff < 0)) + lineStartCorrection = [-lineWidth / (4 * unitLength * sqrt2), -lineWidth / (4 * unitLength * sqrt2)]; + } - // Data manipulation - mangledData.raw = data.lines; - mangledData.river = data.river; - mangledData.stations = extractStations(data); - mangledData.lines = extractLines(data.lines); + points = [ + xScale(currNode.coords[0] + shiftCoords[0] + lineStartCorrection[0]), + yScale(currNode.coords[1] + shiftCoords[1] + lineStartCorrection[1]) + ]; - return mangledData; + path += "M" + points[0] + "," + points[1]; + } } - function extractStations(data) { + return path; + } - data.lines.forEach(function(line) { - for (var node = 0; node < line.nodes.length; node++) { - var d = line.nodes[node]; + function mangleData(data) { + var mangledData = {}; - if (!d.hasOwnProperty("name")) - continue; + // Data manipulation + mangledData.raw = data.lines; + mangledData.river = data.river; + mangledData.stations = extractStations(data); + mangledData.lines = extractLines(data.lines); - if (!data.stations.hasOwnProperty(d.name)) - throw new Error("Cannot find station with key: " + d.name); + return mangledData; + } - var station = data.stations[d.name]; + function extractStations(data) { - station.x = d.coords[0]; - station.y = d.coords[1]; + data.lines.forEach(function(line) { + for (var node = 0; node < line.nodes.length; node++) { + var d = line.nodes[node]; - if (station.labelPos===undefined) { - station.labelPos = d.labelPos; - station.labelShiftX = d.hasOwnProperty("shiftCoords") ? d.shiftCoords[0] : line.shiftCoords[0]; - station.labelShiftY = d.hasOwnProperty("shiftCoords") ? d.shiftCoords[1] : line.shiftCoords[1]; - } + if (!d.hasOwnProperty("name")) + continue; - if (d.hasOwnProperty("canonical")) { - station.labelShiftX = d.hasOwnProperty("shiftCoords") ? d.shiftCoords[0] : line.shiftCoords[0]; - station.labelShiftY = d.hasOwnProperty("shiftCoords") ? d.shiftCoords[1] : line.shiftCoords[1]; - station.labelPos = d.labelPos; - } + if (!data.stations.hasOwnProperty(d.name)) + throw new Error("Cannot find station with key: " + d.name); - station.label = data.stations[d.name].title; - station.position = data.stations[d.name].position; - station.visited = false; + var station = data.stations[d.name]; - if (!d.hide) { + station.x = d.coords[0]; + station.y = d.coords[1]; - station.marker = station.marker || []; + if (station.labelPos===undefined) { + station.labelPos = d.labelPos; + station.labelShiftX = d.hasOwnProperty("shiftCoords") ? d.shiftCoords[0] : line.shiftCoords[0]; + station.labelShiftY = d.hasOwnProperty("shiftCoords") ? d.shiftCoords[1] : line.shiftCoords[1]; + } - station.marker.push( - { - "line": line.name, - "color": line.color, - "labelPos": d.labelPos, - "marker": (d.hasOwnProperty("marker")) ? d.marker : "station", - "shiftX": d.hasOwnProperty("shiftCoords") ? d.shiftCoords[0] : line.shiftCoords[0], - "shiftY": d.hasOwnProperty("shiftCoords") ? d.shiftCoords[1] : line.shiftCoords[1] - } - ); - } + if (d.hasOwnProperty("canonical")) { + station.labelShiftX = d.hasOwnProperty("shiftCoords") ? d.shiftCoords[0] : line.shiftCoords[0]; + station.labelShiftY = d.hasOwnProperty("shiftCoords") ? d.shiftCoords[1] : line.shiftCoords[1]; + station.labelPos = d.labelPos; } - }); - return new Stations(data.stations); - } + station.label = data.stations[d.name].title; + station.position = data.stations[d.name].position; + station.visited = false; - function extractLines(data) { - var lines = []; + if (!d.hide) { - data.forEach(function(line) { + station.marker = station.marker || []; - var lineObj = { - "name": line.name, - "title": line.label, - "stations": [], - "color": line.color, - "shiftCoords": line.shiftCoords, - "nodes": line.nodes, - "highlighted": false - }; + station.marker.push( + { + "line": line.name, + "color": line.color, + "labelPos": d.labelPos, + "marker": (d.hasOwnProperty("marker")) ? d.marker : "station", + "shiftX": d.hasOwnProperty("shiftCoords") ? d.shiftCoords[0] : line.shiftCoords[0], + "shiftY": d.hasOwnProperty("shiftCoords") ? d.shiftCoords[1] : line.shiftCoords[1] + } + ); + } + } + }); - lines.push(lineObj); + return new Stations(data.stations); + } - for (var node = 0; node < line.nodes.length; node++) { - var data = line.nodes[node]; + function extractLines(data) { + var lines = []; - if (!data.hasOwnProperty("name")) - continue; + data.forEach(function(line) { - lineObj.stations.push(data.name); - } - }); + var lineObj = { + "name": line.name, + "title": line.label, + "stations": [], + "color": line.color, + "shiftCoords": line.shiftCoords, + "nodes": line.nodes, + "highlighted": false + }; - return new Lines(lines); - } + lines.push(lineObj); - function textPos(data) { - var pos; - var textAnchor; - var offset = lineWidth * 1.8; - - var numLines = data.label.split(/\n/).length; - - var sqrt2 = Math.sqrt(2); - - switch (data.labelPos.toLowerCase()) { - case "n": - pos = [0, lineWidth*(numLines - 1) + offset]; - textAnchor = "middle"; - break; - case "ne": - pos = [offset / sqrt2, (lineWidth*(numLines - 1) + offset) / sqrt2]; - textAnchor = "start"; - break; - case "e": - pos = [offset, 0]; - textAnchor = "start"; - break; - case "se": - pos = [offset / sqrt2, -offset / sqrt2]; - textAnchor = "start"; - break; - case "s": - pos = [0, -lineWidthMultiplier*offset]; - textAnchor = "middle"; - break; - case "sw": - pos = [-offset/sqrt2, -1.4*offset/sqrt2]; - textAnchor = "end"; - break; - case "w": - pos = [-offset, 0]; - textAnchor = "end"; - break; - case "nw": - pos = [-(lineWidth*(numLines - 1) + offset)/sqrt2, (lineWidth*(numLines - 1) + offset)/sqrt2]; - textAnchor = "end"; - break; - default: - break; - } + for (var node = 0; node < line.nodes.length; node++) { + var data = line.nodes[node]; + + if (!data.hasOwnProperty("name")) + continue; - return { - "pos": pos, - "textAnchor": textAnchor + lineObj.stations.push(data.name); } + }); + + return new Lines(lines); + } + + function textPos(data) { + var pos; + var textAnchor; + var offset = lineWidth * 1.8; + + var numLines = data.label.split(/\n/).length; + + var sqrt2 = Math.sqrt(2); + + switch (data.labelPos.toLowerCase()) { + case "n": + pos = [0, lineWidth*(numLines - 1) + offset]; + textAnchor = "middle"; + break; + case "ne": + pos = [offset / sqrt2, (lineWidth*(numLines - 1) + offset) / sqrt2]; + textAnchor = "start"; + break; + case "e": + pos = [offset, 0]; + textAnchor = "start"; + break; + case "se": + pos = [offset / sqrt2, -offset / sqrt2]; + textAnchor = "start"; + break; + case "s": + pos = [0, -lineWidthMultiplier*offset]; + textAnchor = "middle"; + break; + case "sw": + pos = [-offset/sqrt2, -1.4*offset/sqrt2]; + textAnchor = "end"; + break; + case "w": + pos = [-offset, 0]; + textAnchor = "end"; + break; + case "nw": + pos = [-(lineWidth*(numLines - 1) + offset)/sqrt2, (lineWidth*(numLines - 1) + offset)/sqrt2]; + textAnchor = "end"; + break; + default: + break; } - // Render line breaks for svg text - function wrap(text) { - text.each(function() { - var text = d3.select(this); - var lines = text.text().split(/\n/); + return { + "pos": pos, + "textAnchor": textAnchor + } + } - var y = text.attr("y"); - var x = text.attr("x"); - var dy = parseFloat(text.attr("dy")); + // Render line breaks for svg text + function wrap(text) { + text.each(function() { + var text = d3.select(this); + var lines = text.text().split(/\n/); - var tspan = text.text(null).append("tspan").attr("x", x).attr("y", y).attr("dy", dy + "em").text(lines[0]); + var y = text.attr("y"); + var x = text.attr("x"); + var dy = parseFloat(text.attr("dy")); - for (var lineNum = 1; lineNum < lines.length; lineNum++) { - tspan = text.append("tspan").attr("x", x).attr("y", y).attr("dy", lineNum * 1.1 + dy + "em").text(lines[lineNum]) - } + var tspan = text.text(null).append("tspan").attr("x", x).attr("y", y).attr("dy", dy + "em").text(lines[0]); - }); - } + for (var lineNum = 1; lineNum < lines.length; lineNum++) { + tspan = text.append("tspan").attr("x", x).attr("y", y).attr("dy", lineNum * 1.1 + dy + "em").text(lines[lineNum]) + } - return map; + }); } - return tubeMap; + return map; } - var Stations = function (stations) { this.stations = stations; }; diff --git a/test/tubeMap-test.js b/test/tubeMap-test.js index 986be24..c95ca15 100644 --- a/test/tubeMap-test.js +++ b/test/tubeMap-test.js @@ -2,6 +2,6 @@ var tape = require("tape"), tubeMap = require("../"); tape("tubeMap() does not throw", function(test) { - test.doesNotThrow(tubeMap.tubeMap(), ""); + test.doesNotThrow(tubeMap.tubeMap, ""); test.end(); });