d3.js map图表绘制矩形

d3.js map chart draw rectangle

我正在尝试使用 geojson 为阿联酋绘制地图。

geojson 有效并在 geojson.io 上进行了验证,结果出现了矩形而不是地图,编写的代码在这里:

var width = 20,
height = 20;
var scal = (1 << 10) / 2 / Math.PI; 

//Define map projection 
var projection = d3.geo.mercator();
      projection
        .scale(scal)///3.5
        .translate([width/2, height/2]);

//Define path generator
var path = d3.geo.path().projection(projection);

d3.json("http://localhost/data/uae-em.json", function(error,json) {

    if(error) alert("error fetching data");
    var svg = d3.select("article")
        .append("svg")
        .attr("width",width)
        .attr("height",height);
    //draw map
    var map = svg.selectAll("path")
        .data(json.features)
        .enter()
        .append("path")
        .attr("d", path)
        .style("fill", "#3498db"); 
    });

JSON数据在这里:https://files.fm/f/ndsz3v85

这是来自 geojson 的示例特征:

{"type":"Feature","id":"4","properties":{"name":"UmmAlQwain"},"geometry":{"type":"Polygon","coordinates":[[[55.2350748,25.5700834],[55.4229709,25.3989174],[55.4236655,25.3981583],[55.423986,25.3978339],[55.4254059,25.3960785],[55.4272991,25.3935724],[55.4570596,25.3762406],[55.4577152,25.375909],[55.4915399,25.3593419],[55.4926741,25.3588472],[55.4954455,25.35793],[55.5013237,25.3571279],[55.5031779,25.3570031],[55.5390972,25.3630675],[55.5909015,25.3679027],[55.5918297,25.3680754],[55.5927443,25.3683591],[55.5935906,25.3688154],[55.5943278,25.3693581],[55.5952833,25.3702339],[55.6010034,25.3768825],[55.6044809,25.3808245],[55.6295136,25.4095735],[55.630044,25.4101151],[55.6303207,25.4106671],[55.6302516,25.411542],[55.6299137,25.4123589],[55.6293946,25.413368],[55.6287288,25.4144586],[55.6280518,25.4152536],[55.6211794,25.4201663],[55.6196221,25.4209307],[55.6177489,25.4214301],[55.605742,25.4232035],[55.6024463,25.4237462],[55.5975759,25.4244245],[55.5956543,25.4249531],[55.5919637,25.4262723],[55.554199,25.4399865],[55.5518752,25.4408552],[55.5505241,25.4414505],[55.5490758,25.4421533],[55.5473004,25.4425344],[55.5456819,25.4430884],[55.5370397,25.4462406],[55.5358338,25.4466226],[55.5349558,25.4467468],[55.5337817,25.4467659],[55.523193,25.4459922],[55.5198927,25.4460304],[55.5162327,25.4457248],[55.513895,25.4454286],[55.4882538,25.4420186],[55.4870268,25.4419613],[55.4857574,25.4421332],[55.4843188,25.4426681],[55.4771469,25.4472339],[55.4709611,25.4516781],[55.324428,25.6230138],[55.3205286,25.6184125],[55.3117729,25.6106044],[55.3015804,25.5994214],[55.2892555,25.5829995],[55.2795994,25.5665793],[55.2730684,25.5680481],[55.2385191,25.570346],[55.2350748,25.5700834]]]}}

这里有一些问题。

首先,你得到一个矩形的原因是因为缠绕顺序。 D3 不包含 geojson 规范的缠绕顺序 - 它遵循 shapefile 的缠绕顺序。这很奇怪,因为 D3 是唯一想到的框架,其中缠绕顺序在绘图方面很重要,因为它使用球面几何而不是平面。在您的情况下,您正在绘制除阿联酋以外的所有内容:

我添加了一个描边,我们可以看到世界的边缘(东西+/-180 度,而不是顶部,因为墨卡托在 y 轴上延伸到无穷大,即使是圆形或切断,限制超出了 SVG 的边缘),我们还可以看到阿联酋的轮廓(或至少部分轮廓)。所以我们需要反转缠绕顺序,这意味着反转保存定义多边形中每个环的坐标的数组。

为此,我们可以使用 turf.js,预先准备数据或在每次加载数据时调用它:

json.features.forEach(function(feature) {
  feature.geometry = turf.rewind(feature.geometry, {reverse:true});
})

现在我们有了正确缠绕顺序的数据,我们得到:

特征看起来更大,因为我们可以看到所有特征,在第一张图片中只有最后绘制的特征是真正可见的:它覆盖了其余特征。

现在我们遇到了一个不同的问题,我已经在 960x500 像素的框架上显示了您的地图(使用您的代码 (fiddle))。如果我们使用 20x20 的框架和您的初始比例,我们将看不到任何东西。地图以 [0,0] 为中心,位于非洲海岸附近的大西洋中:

正方形勾勒出 20x20 版本中使用您的缩放和平移显示的地图的粗略区域,我添加了一张世界地图并将尺寸增加到 200x200 以提供一些视觉背景我们正在查看 20 像素版本


如果目标是展示整个世界,我们需要合适的比例。像墨卡托这样的圆柱投影(如果圆柱的方向 north/south)围绕地球(沿赤道)旋转 360 度,并以地图单位将其拉伸一定距离。在 d3 中,scale 是这个转换因子,默认假设一个 960 像素的地图显示整个行星,地球周围有 2π 弧度,所以 scale 值为 960/2π(每 2π 弧度 960 像素)。因此,我们可以将您的比例设置为 width/Math.PI/2:

这会引出一个新的问题,阿联酋大约是5度,世界是360度,大约是世界的1%"width"。在 20 像素 x 20 像素的 SVG 上显示地图时,这会有点问题:阿联酋的宽度约为 0.2 像素,渲染效果不会非常令人满意。唯一的选择是增加地图的大小,或者将地图以阿联酋为中心并增加比例尺。

Automagic map centering 已经讲过很多次了,我不会再讲一遍之前的优秀解释了(我只能说 v4/5 提供了一个很好的 fitSize/fitExtent 方法,但可以在 question 中找到 v3 居中信息)。但是,为了演示,我将在下面手动将阿联酋居中。

阿联酋的质心约为24 N, 54E(来自here)。

如果我们假设阿联酋的宽度大于高度,并假设一个数字(例如大约 5 度),并且我们想要在 20 个像素上显示这 5 度,那么我们知道世界大约有 1440 个像素(mapWidthPixels * earthWidthDegrees / mapWidthDegrees = 20 * 360 / 5)。然后比例为 1440/2/π,将其向下舍入一点以获得额外的余量:200.

所以使用:

var width = 20;
var height = 20;
var scale = 200;
var center = [54,24];

我们可以得到这个:

还是小,但是有了上面的原理,很容易扩展。

fiddle.

我在上面的 20px SVG 上添加了一个边框,以便为 SVG 提供一个边界,因为 SVG 的尺寸很小并且没有区分 SVG 的边缘