d3-гео-вороной проблема заполнения полигона d3-тайлом - PullRequest
0 голосов
/ 25 марта 2020

Я пытаюсь использовать d3-geo-voronoi для отображения векторных данных мозаики, используя d3-плитку. Моя первоначальная попытка отображения данных с заполнением, установленным на «none», сработала, что было очень интересно! Карта тайлов Вороного без цветовой заливки

Однако, когда я попытался заполнить полигоны, некоторые из плиток были искажены. Карта плиток Вороного с цветной заливкой

Я не смог понять, почему это происходит. Я проверил svg в дом, и все выглядит правильно. Svg верны там, где нет полигонов, они просто не отображаются должным образом, возможно, они скрыты. Ниже приведен код, который я использовал:

const d3 = require('d3');
const d3tile = require('d3-tile');
const d3geovoronoi = require('d3-geo-voronoi');
const vt2geojson = require('@mapbox/vt2geojson');

const pi = Math.PI,
      tau = 2 * pi;

const width = Math.max(960, window.innerWidth),
      height = Math.max(500, window.innerHeight);

const map = d3.select("body").append("div")
  .attr("class", "map")
  .style("width", width + "px")
  .style("height", height + "px")
  .on("mousemove", mousemoved);

let projection = d3.geoMercator()
  .scale(1 / tau)
  .translate([0, 0]);

let center = projection([-76.3, 38.794745]);

const tile = d3tile.tile()
  .size([width, height]);

const zoom = d3.zoom()
  .scaleExtent([1 << 15, 1 << 24])
  .on("zoom", zoomed);

const svg = map.append("g")
  .attr("pointer-events", "none")
  .attr("class", "svg");

const info = map.append("g")
  .attr("class", "info");

const ramp = d3.scaleLinear().domain([0.05,0.07]).interpolate(d3.interpolateHcl).range(['#34d8eb','#3a34eb']).unknown("#5c5752")

map.call(zoom)
  .call(zoom.transform, d3.zoomIdentity
    .translate(width / 2, height / 2)
    .scale(1 << 21)
    .translate(-center[0], -center[1]));

function zoomed() {
  let transform = d3.event.transform;

  let tiles = tile(transform);

  let image = svg 
    .style("transform", stringify(tiles.scale, tiles.translate))
    .selectAll(".tile")
    .data(tiles, function(d) { return d; })
    .enter().append("svg")
    .attr("class", "tile")
    .attr("fill", "none")
    .attr("stroke", "#000")
    .attr("stroke-width", "0.5")
    .attr("stroke-linejoin", "round")
    .attr("stroke-linecap", "round")
    .style("left", function(d) { return d[0] * 256 + "px"; })
    .style("top", function(d) { return d[1] * 256 + "px"; })
    .each(function(d) { this._xhr = render(d, this); });

  projection
    .scale(transform.k / tau)
    .translate([transform.x, transform.y]);

}

function render(d, xnode) {
  let k = Math.pow(2, d[2]) * 256;
  vt2geojson({
    uri: 'http://localhost:7800/public.r3sim_fort_temp/'+d[2]+'/'+d[0]+'/'+d[1]+'.pbf?properties=node,zeta,mask,bathymetry'
  }, function (err, json) {
    if (err) throw err;
    d3.select(xnode)
      .selectAll("path")
      .data(d3geovoronoi.geoVoronoi().polygons(json).features)
      .enter().append("path")
    //.attr('fill', 'none')
        .attr("fill", function(d) {return ramp(d.properties.site.properties.zeta)})
        .attr("stroke", "#fff")
        .attr("stroke-width", "0.5")
        .attr("stroke-linejoin", "round")
        .attr("stroke-linecap", "round")
        .attr("d", d3.geoPath()
            .projection(d3.geoMercator()
                .scale(k / tau)
                .translate([k / 2 - d[0] * 256, k / 2 - d[1] * 256])
                .precision(0)));
  })
}

function stringify(scale, translate) {
  const k = scale / 256, r = scale % 1 ? Number : Math.round;
  return "matrix3d(" + [k, 0, 0, 0, 0, k, 0, 0, 0, 0, k, 0, r(translate[0] * scale), r(translate[1] * scale), 0, 1 ] + ")";
}

function mousemoved() {
  info.text(formatLocation(projection.invert(d3.mouse(this)), d3.zoomTransform(this).k));
}

function formatLocation(p, k) {
  const format = d3.format("." + Math.floor(Math.log(k) / 2 - 2) + "f");
  return (p[1] < 0 ? format(-p[1]) + "°S" : format(p[1]) + "°N") + " "
       + (p[0] < 0 ? format(-p[0]) + "°W" : format(p[0]) + "°E");
}

<!doctype html>
<head>
  <meta charset="utf-8">
  <title>D3 V5 Vector Tile Example</title>
  <style>
    body {
      margin: 0;
    }
    .map {
      background: #5c5752;
      position: relative;
      overflow: hidden;
    }
    .svg {
      position: absolute;
      will-change: transform;
    }
    .tile {
      position: absolute;
      width: 256px;
      height: 256px;
    }
    .info {
      position: absolute;
      bottom: 10px;
      left: 10px;
    }
  </style>
</head>
<body>
  <script src="bundle.js"></script>
</body>

В этом примере я залил полигоны с различными значениями цвета. Тем не менее, точно такие же искажения происходят, если я использую одно значение цвета. Искажения также всегда остаются в одном и том же месте, если я перезагружаю все данные.

1 Ответ

0 голосов
/ 27 марта 2020

Я сделал более глубокое погружение в данные, нашел неверный путь SVG, а затем нашел данные, связанные с ним. Похоже, d3.geo.voronoi выдает неправильные координаты, но входные данные выглядят нормально. Ниже представлены две распечатки узла 1192. Первая - это входные данные geo json, показывающие координаты, а вторая - геометрия вороной. Геометрия вороной содержит значения долготы в восточном полушарии (103,86 ...), что находится далеко за пределами диапазона данных. Я все еще пытаюсь определить, почему эти плохие ценности производятся. Опять же, входные координаты выглядят правильно, но, возможно, это другие данные, которые входят в расчет voronoi?

1192 
{…}
  geometry: {…}
    coordinates: (2) […]
      0: -76.12801194190979
      1: 38.78622954627738
      length: 2
      <prototype>: Array []
    type: "Point"
    <prototype>: Object { … }
  properties: Object { node: 180407, zeta: "NaN", mask: "True", … }
  type: "Feature"
  <prototype>: Object { … }

1192 (11) […]
  0: Array [ 103.86695733932268, -44.964779133003304 ]
  1: Array [ -76.13308210176842, 38.75793814039401 ]
  2: Array [ -76.13020999558496, 38.782688154120585 ]
  3: Array [ -76.12890669699081, 38.78647064351637 ]
  4: Array [ -76.12807302385534, 38.786723650244355 ]
  5: Array [ -76.12754554182737, 38.78651000385868 ]
  6: Array [ -76.12640847594942, 38.78408839960177 ]
  7: Array [ -76.11435851540921, 38.636536130021334 ]
  8: Array [ 103.858096036925, -39.00570100251519 ]
  9: Array [ 103.860092112702, -39.367933188411186 ]
  10: Array [ 103.86695733932268, -44.964779133003304 ]
  length: 11
  <prototype>: []
...