本文作者:字节跳动数据平台 早千 这是一张再熟悉不过的地图,但右上角黑龙江与内蒙古之间的那片飞地是什么呢? 查一下,摘自互联网:
哦豁,原来如此。知道了,但是我突然更好奇这个地图是怎么画在网页上的呢? 将地图画在页面上,概括一下分三步。1. 拿到地图信息数据 2. 投影 3. 画 地图数据这里介绍两种可以用于表示地理信息的数据类型
GeoJSON
如下表所示,坐标表示点。点连成线,线连成面,从而表现出各种各样的形状。 一段GeoJSON示例 { "type": "FeatureCollection", "features": [ { "type": "Feature", "geometry": { "type": "Point", "coordinates": [102.0, 0.5] }, "properties": { "prop0": "value0" } }, { "type": "Feature", "geometry": { "type": "LineString", "coordinates": [ [102.0, 0.0], [103.0, 1.0], [104.0, 0.0], [105.0, 1.0] ] }, "properties": { "prop0": "value0", "prop1": 0.0 } }, { "type": "Feature", "geometry": { "type": "Polygon", "coordinates": [ [ [100.0, 0.0], [101.0, 0.0], [101.0, 1.0], [100.0, 1.0], [100.0, 0.0] ] ] }, "properties": { "prop0": "value0", "prop1": { "this": "that" } } } ] } TopoJSONTopoJSON 是 GeoJSON 按拓扑学编码后的扩展形式,是由 D3 的作者 Mike Bostock 制定的。相比 GeoJSON 直接使用 Polygon、Point 之类的几何体来表示图形的方法,TopoJSON 中的每一个几何体都是通过将共享边(被称为arcs)整合后组成的。 简单说,它的线并不是由点构成,而是有自己的id。 边在一起组成几何形状时,最后一个坐标必须与后续边的第一个坐标(如果有)相同。例如,如果边6代表点序列A→B→C,边7代表点序列C→D→E,则[6,7]代表点序列A→B→C→D→E。 由于边是共享的所以在表示不同方向时需要反转。使用补码表示。-1(〜0)为反0,-2(〜1)为反1,依此类推。 以上图举例说明。绿色点与红色点坐标分别代表两个点。紫色边为3。如果定义蓝色边顺时针方向为1,黑色边顺时针方向为0,则蓝色边与黑色边组成几何图形为[[0, 1]]。在此基础上红色顺时针为2,则红色边与逆黑色边(~0 = -1)组成图形为[[2, -1]]。 获取数据
投影地图投影,是指按照一定的数学法则将地球椭球面上的经纬网转换到平面上,使地面的地理坐标与平面直角坐标建立起函数关系,是绘制地图的数学基础之一。由于地球是一个不可展的球体,使用物理方法将其展平会引起褶皱、拉伸和断裂,因此要使用地图投影实现由曲面向平面的转化。 下表是一些地图投影与经纬线形状特征 示例的中国地图就是采用了墨卡托投影。 画经过上面的介绍,接下来只需要将二维平面内容绘制出来即可。 这里使用d3与d3-geo举例绘制上面提到的中国地图。为了美观,使用d3-scale-chromatic的一个主题随意上了颜色。 import * as d3 from 'd3'; import * as geo from 'd3-geo'; import * as d3Color from 'd3-scale-chromatic'; 在页面中创建一个svg元素 <svg id="svg" width="1080" height="600"></svg> 创建投影,设置参数 const projection = geo.geoMercator() // 墨卡托投影 .scale(450) // 投影的比例因子,可以按比例放大投影。 .center([105, 38]) // 中心点设置为中国的中心位置 大约经度105,纬度38 利用得到的投影创建path,补充颜色与边线 const svg = d3.select('#svg'); const path = geo.geoPath(projection); const colors = d3.scaleOrdinal(d3Color.schemeBlues[9]); const area = svg.selectAll('path') .data(geoData.features) // 这里的geoData是下载好的中国地图GeoJSON格式数据 .enter() .append('path') .attr('d', path) .attr('fill', (_, i) => { return colors(i); }) .attr('stroke', '#fff') .attr('stroke-width', 1); 到此为止,中国地图就已经出现在了我们面前。 如果在这个基础上将北京上海两座城市标注出来呢?定义经纬度 const places = [ { 'name': '北京', 'log': '116.3', 'lat': '39.9' }, { 'name': '上海', 'log': '121.4', 'lat': '31.2' } ]; 故技重施,用svg画出两个点 const location = svg.selectAll('g') .data(places) .enter() .append('g') .attr('transform', (d) => { const coor = projection([d.log, d.lat]); // 经纬度转换为绘制坐标 return 'translate(' + coor[0] + ',' + coor[1] + ')'; }); // 绿色圆点 location.append('circle') .attr('r', 4) .attr('fill', '#a0d911') 现在地图上的北京与上海两地就被标注出来咯 看来要绘制一幅地图也并不难呀。 |
|