【问题标题】:Utilize two objects for a single topojson merge. Multiple datum in d3.select("svg")利用两个对象进行单个 topojson 合并。 d3.select("svg") 中的多个数据
【发布时间】:2017-02-15 04:13:28
【问题描述】:

我正在尝试将俄克拉荷马州与德克萨斯州的一个县合并。

我有两个变量,一个使用整个俄克拉荷马州的州 ID,另一个包含德克萨斯州县的县 ID。如何组合 state.geometries.filter 和 County.geometries.filter?德克萨斯州实际上有许多县将被合并,但为了举例,这个县是随机选择的。

我的代码,只返回最底部的数据,如下:

.county-borders {
  fill: none;
  stroke: #fff;
  stroke-width: 0.5px;
  stroke-linejoin: round;
  stroke-linecap: round;
  pointer-events: none;
}

.state-borders {
  fill: none;
  stroke: #fff;
  stroke-width: 0.7px;
  stroke-linejoin: round;
  stroke-linecap: round;
  pointer-events: none;
}

.state.southcentral {
  fill: steelblue;
  stroke: #fff;
}

<svg width="960" height="600"></svg>
<script src="https://d3js.org/d3.v4.min.js"></script>
<script src="https://d3js.org/topojson.v2.min.js"></script>

var svg = d3.select("svg");

var path = d3.geoPath();

var southcentral = {
  "40": 1
};

var southcentral_c = {
  "48043": 1
};

d3.json("https://d3js.org/us-10m.v1.json", function(error, us) {
  if (error) throw error;

  svg.append("g")
      .attr("class", "counties")
      .selectAll("path")
      .data(topojson.feature(us, us.objects.counties).features)
      .enter().append("path")
      .attr("d", path);

  svg.append("path")
      .attr("class", "state-borders")
      .attr("d", path(topojson.mesh(us, us.objects.states, function(a, b) { return a !== b; })));

     //multiple datum, only shows the bottom most one. Need to combine into one merge but they use different objects in the JSON.

     svg.append("path")
      .datum(topojson.merge(us, us.objects.states.geometries.filter(function(d) { return d.id in southcentral; })))
      .datum(topojson.merge(us, us.objects.counties.geometries.filter(function(d) { return d.id in southcentral_c; })))
      .attr("class", "state southcentral")
      .attr("d", path);

});

【问题讨论】:

    标签: javascript d3.js topojson


    【解决方案1】:

    尚未对其进行测试,但merge 的 api 声明它接受一个多面体对象数组。所以我认为你可以只连接 2 个过滤对象并传入 us 拓扑。

    var arr = [];
    arr[0] = us.objects.states.geometries.filter(function(d) { return d.id in southcentral; })
    arr[1] = us.objects.counties.geometries.filter(function(d) { return d.id in southcentral_c; })
    
    svg.append("path")
          .datum(topojson.merge(us, arr))
          .attr("class", "state southcentral")
          .attr("d", path);
    

    编辑::

    var arr = [];
    Array.prototype.push.apply(arr,us.objects.states.geometries.filter(function(d) { return d.id in southcentral; }))
    Array.prototype.push.apply(arr,us.objects.counties.geometries.filter(function(d) { return d.id in southcentral_c; }))
    

    【讨论】:

    • 不幸的是它没有用。如果我从数组 .datum(topojson.merge(us, arr[1])) 中调用一个值,那么它会返回县,但如果我只使用 arr 它将不起作用。一些使用 arr[i] 的函数可以工作吗?
    • 哦,我认为这是因为我忽略了filter 返回一个数组的事实。作为测试,尝试将 [0] 添加到 2 个过滤语句的末尾,以便 arr 包含 2 个对象,而不是 2 个数组。当您尝试arr[1] 时它会起作用,这是有道理的,因为我的答案不正确,arr[1] 返回一个包含单个对象的数组。
    • 成功了!我不完全确定我理解发生了什么变化。不过谢谢!
    • 看到我的目标是让 arr 包含 2 个多面体对象。喜欢arr[obj,obj]filter 函数返回一个数组,即使您只找到 1 个对象,所以您的两个过滤函数都返回 arr[obj]。当我将这两个结果都放入arr 时,我们得到的是arr[arr[obj],arr[obj]] 而不是arr[obj,obj]。希望这是有道理的,只是我的一个愚蠢的疏忽。
    • 随着我的进展,我意识到通过将[0] 添加到每个过滤语句的末尾,它只会使每个数组中的第一个值出现,在某些情况下我会有数组中有多达 50 个值。它适用于我发布的示例,因为我只包括 1 个州和 1 个县。我忽略了这里的修复吗?
    猜你喜欢
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    • 2021-08-17
    • 2021-04-17
    • 1970-01-01
    • 1970-01-01
    • 2022-11-08
    相关资源
    最近更新 更多