import * as d3 from 'd3'; import _ from 'lodash'; import { svgOneWaySegments, svgPath, svgRelationMemberTags, svgTagClasses } from './index'; import { osmEntity, osmSimpleMultipolygonOuterMember } from '../osm/index'; import { utilDetect } from '../util/detect'; export function svgLines(projection) { var detected = utilDetect(); var highway_stack = { motorway: 0, motorway_link: 1, trunk: 2, trunk_link: 3, primary: 4, primary_link: 5, secondary: 6, tertiary: 7, unclassified: 8, residential: 9, service: 10, footway: 11 }; function waystack(a, b) { var as = 0, bs = 0; if (a.tags.highway) { as -= highway_stack[a.tags.highway]; } if (b.tags.highway) { bs -= highway_stack[b.tags.highway]; } return as - bs; } return function drawLines(selection, graph, entities, filter) { var ways = [], pathdata = {}, onewaydata = {}, getPath = svgPath(projection, graph); for (var i = 0; i < entities.length; i++) { var entity = entities[i], outer = osmSimpleMultipolygonOuterMember(entity, graph); if (outer) { ways.push(entity.mergeTags(outer.tags)); } else if (entity.geometry(graph) === 'line') { ways.push(entity); } } ways = ways.filter(getPath); pathdata = _.groupBy(ways, function(way) { return way.layer(); }); _.forOwn(pathdata, function(v, k) { onewaydata[k] = _(v) .filter(function(d) { return d.isOneWay(); }) .map(svgOneWaySegments(projection, graph, 35)) .flatten() .valueOf(); }); var layer = selection.selectAll('.layer-lines'); var layergroup = layer .selectAll('g.layergroup') .data(d3.range(-10, 11)); layergroup = layergroup.enter() .append('g') .attr('class', function(d) { return 'layergroup layer' + String(d); }) .merge(layergroup); var linegroup = layergroup .selectAll('g.linegroup') .data(['shadow', 'casing', 'stroke']); linegroup = linegroup.enter() .append('g') .attr('class', function(d) { return 'linegroup line-' + d; }) .merge(linegroup); var lines = linegroup .selectAll('path') .filter(filter) .data( function() { return pathdata[this.parentNode.__data__] || []; }, osmEntity.key ); lines.exit() .remove(); // Optimization: call simple TagClasses only on enter selection. This // works because osmEntity.key is defined to include the entity v attribute. lines.enter() .append('path') .attr('class', function(d) { return 'way line ' + this.parentNode.__data__ + ' ' + d.id; }) .call(svgTagClasses()) .merge(lines) .sort(waystack) .attr('d', getPath) .call(svgTagClasses().tags(svgRelationMemberTags(graph))); var onewaygroup = layergroup .selectAll('g.onewaygroup') .data(['oneway']); onewaygroup = onewaygroup.enter() .append('g') .attr('class', 'onewaygroup') .merge(onewaygroup); var oneways = onewaygroup .selectAll('path') .filter(filter) .data( function() { return onewaydata[this.parentNode.__data__] || []; }, function(d) { return [d.id, d.index]; } ); oneways.exit() .remove(); oneways = oneways.enter() .append('path') .attr('class', 'oneway') .attr('marker-mid', 'url(#oneway-marker)') .merge(oneways) .attr('d', function(d) { return d.d; }); if (detected.ie) { oneways.each(function() { this.parentNode.insertBefore(this, this); }); } }; }