如何使用 geoPath 居中和缩放
How to center and scale using geoPath
我正在使用 D3 和 topojson 过滤掉美国县的子集。我可以成功过滤掉县并将它们绘制到 SVG 元素,但如何更改地图边界以适应新选择的县?它正在渲染地图,就好像边界是所有美国县,而不是过滤后的县。
var width=960,
height=600,
centered;
const svgElement = d3.select(ref.current)
.append("svg")
.attr("preserveAspectRatio", "xMinYMin meet")
.attr("viewBox", [0,0,width,height]);
var path = d3.geoPath();
d3.json("https://d3js.org/us-10m.v2.json").then((d) => {
d.objects.counties.geometries = d.objects.counties.geometries.filter(function(item) {
return props.data.indexOf(item.id) > -1
})
var counties = topojson.feature(d, d.objects.counties).features;
svgElement.append("g")
.attr("class", "filled-territory")
.selectAll("path")
.data(counties)
.enter()
.append("path")
.attr("d", path)
});
其中 props.data
是一个县 ID 数组,例如 -
[
"45001",
"45003",
"45007",
"45083",
"45087",
"45091"
]
我可以让它像这样显示,但我不知道如何让路径填充 SVG:
这个特定文件使用像素坐标系(它是投影地理数据),它填充了 [0,0] 和 [960,600] 之间的范围。因为它使用这个坐标系,所以我们通常可以使用 d3.geoPath 的默认投影,一个将 json 中的坐标视为像素的空投影。但是,仅 geoPath 本身并不能提供一种将要素居中或缩放的方法。
D3 地理投影带有一种称为 fitSize(也称为 fitExtent 或其他)的方法,它采用 geojson object 并设置投影比例并将属性转换为中心在指定范围内的 geojson 对象。由于我们已经有了投影数据,我们可以使用 d3.geoIdentity() 它提供了一些方便的投影方法,如 fitSize 但不会投影我们的数据,因为它已经被投影了。
为了使用 fitSize,我们可以使用以下内容:
// get a geojson feature collection object:
var geojson = topojson.feature(d, d.objects.counties)
// create a geoidentity "projection" and center the geojson:
var projection = d3.geoIdentity()
.fitSize([width,height],geojson)
// set the path's projection:
path.projection(projection);
// Now access the array of features in the collection for use with .data():
var counties = geojson.features;
看起来像:
var data = [
"45001",
"45003",
"45007",
"45083",
"45087",
"45091"
]
var width=500,
height=500;
const svgElement = d3.select("body")
.append("svg")
.attr("width", width)
.attr("height", height);
var path = d3.geoPath();
d3.json("https://d3js.org/us-10m.v2.json").then((d) => {
d.objects.counties.geometries = d.objects.counties.geometries.filter(function(item) {
return data.indexOf(item.id) > -1
})
var geojson = topojson.feature(d, d.objects.counties)
var projection = d3.geoIdentity()
.fitSize([width,height],geojson)
path.projection(projection);
var counties = geojson.features;
svgElement.append("g")
.attr("class", "filled-territory")
.selectAll("path")
.data(counties)
.enter()
.append("path")
.attr("d", path)
});
<script src="https://cdnjs.cloudflare.com/ajax/libs/d3/5.7.0/d3.min.js"></script>
<script src="https://cdnjs.cloudflare.com/ajax/libs/topojson/3.0.2/topojson.min.js
"></script>
我正在使用 D3 和 topojson 过滤掉美国县的子集。我可以成功过滤掉县并将它们绘制到 SVG 元素,但如何更改地图边界以适应新选择的县?它正在渲染地图,就好像边界是所有美国县,而不是过滤后的县。
var width=960,
height=600,
centered;
const svgElement = d3.select(ref.current)
.append("svg")
.attr("preserveAspectRatio", "xMinYMin meet")
.attr("viewBox", [0,0,width,height]);
var path = d3.geoPath();
d3.json("https://d3js.org/us-10m.v2.json").then((d) => {
d.objects.counties.geometries = d.objects.counties.geometries.filter(function(item) {
return props.data.indexOf(item.id) > -1
})
var counties = topojson.feature(d, d.objects.counties).features;
svgElement.append("g")
.attr("class", "filled-territory")
.selectAll("path")
.data(counties)
.enter()
.append("path")
.attr("d", path)
});
其中 props.data
是一个县 ID 数组,例如 -
[
"45001",
"45003",
"45007",
"45083",
"45087",
"45091"
]
我可以让它像这样显示,但我不知道如何让路径填充 SVG:
这个特定文件使用像素坐标系(它是投影地理数据),它填充了 [0,0] 和 [960,600] 之间的范围。因为它使用这个坐标系,所以我们通常可以使用 d3.geoPath 的默认投影,一个将 json 中的坐标视为像素的空投影。但是,仅 geoPath 本身并不能提供一种将要素居中或缩放的方法。
D3 地理投影带有一种称为 fitSize(也称为 fitExtent 或其他)的方法,它采用 geojson object 并设置投影比例并将属性转换为中心在指定范围内的 geojson 对象。由于我们已经有了投影数据,我们可以使用 d3.geoIdentity() 它提供了一些方便的投影方法,如 fitSize 但不会投影我们的数据,因为它已经被投影了。
为了使用 fitSize,我们可以使用以下内容:
// get a geojson feature collection object:
var geojson = topojson.feature(d, d.objects.counties)
// create a geoidentity "projection" and center the geojson:
var projection = d3.geoIdentity()
.fitSize([width,height],geojson)
// set the path's projection:
path.projection(projection);
// Now access the array of features in the collection for use with .data():
var counties = geojson.features;
看起来像:
var data = [
"45001",
"45003",
"45007",
"45083",
"45087",
"45091"
]
var width=500,
height=500;
const svgElement = d3.select("body")
.append("svg")
.attr("width", width)
.attr("height", height);
var path = d3.geoPath();
d3.json("https://d3js.org/us-10m.v2.json").then((d) => {
d.objects.counties.geometries = d.objects.counties.geometries.filter(function(item) {
return data.indexOf(item.id) > -1
})
var geojson = topojson.feature(d, d.objects.counties)
var projection = d3.geoIdentity()
.fitSize([width,height],geojson)
path.projection(projection);
var counties = geojson.features;
svgElement.append("g")
.attr("class", "filled-territory")
.selectAll("path")
.data(counties)
.enter()
.append("path")
.attr("d", path)
});
<script src="https://cdnjs.cloudflare.com/ajax/libs/d3/5.7.0/d3.min.js"></script>
<script src="https://cdnjs.cloudflare.com/ajax/libs/topojson/3.0.2/topojson.min.js
"></script>