Создание многослойной карты с D3

Я пытаюсь загрузить карту D3 с помощью topjson и соединить с ней отдельный файл CSV с данными, которые я хотел бы использовать для раскраски карты. Я могу переключаться между несколькими слоями, но не перерисовывать карту каждый раз. Я искал доски для некоторой подсказки о том, как сделать это, но я не могу понять это. Может ли кто-нибудь помочь?

Я бы хотел разделить данные CSV и JSON на случай, если мы захотим динамически редактировать CSV по мере продвижения. Я хотел бы либо добавить соответствующие классы в каждую страну за один раз, либо добавить и удалить классы, когда пользователь переключается между тремя кнопками. Либо у меня работает, я просто не могу понять, как это сделать.

var width = 945

var height = 550

var svg = d3.select('#content').append('svg').attr('width', width).attr('height', height);

var projection = d3.geo.mercator().scale(175);

var path = d3.geo.path().projection(projection);

var all_data = {};

var tierById = d3.map();

var quantize = d3.scale.quantize()
    .domain([0, 1])
    .range(d3.range(2).map(function(i) { return "tier" + i; }));

queue()
  .defer(d3.json, "data/world.json")
  .defer(d3.csv, "data/wod.csv")
  .await(setUpChoropleth);

function setUpChoropleth(error, json) {

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

} 

function drawTierI() {
  queue()
      .defer(d3.json, "data/world.json")
      .defer(d3.csv, "data/wod.csv", function(d) { tierById.set(d.id, +d.tier_i); })
      .await(ready);

  function ready(error, json) {
    svg.append("g")
        .attr("class", "countries")
      .selectAll("path")
        .data(topojson.feature(json, json.objects.countries).features)
      .enter().append("path")
        .attr("class", function(d) { return quantize(tierById.get(d.id)); })
        .attr("d", path);

  }
}

function drawTierII() {
  queue()
      .defer(d3.json, "data/world.json")
      .defer(d3.csv, "data/wod.csv", function(d) { tierById.set(d.id, +d.tier_ii); })
      .await(ready);

  function ready(error, json) {
    svg.append("g")
        .attr("class", "countries")
      .selectAll("path")
        .data(topojson.feature(json, json.objects.countries).features)
      .enter().append("path")
        .attr("class", function(d) { return quantize(tierById.get(d.id)); })
        .attr("d", path);

  }
}

function drawTierIIPlus() {
  queue()
      .defer(d3.json, "data/world.json")
      .defer(d3.csv, "data/wod.csv", function(d) { tierById.set(d.id, +d.tier_ii_plus); })
      .await(ready);

  function ready(error, json) {
    svg.append("g")
        .attr("class", "countries")
      .selectAll("path")
        .data(topojson.feature(json, json.objects.countries).features)
      .enter().append("path")
        .attr("class", function(d) { return quantize(tierById.get(d.id)); })
        .attr("d", path);

  }
}

$('button#tier_i').click(function (e) {
  $( "svg" ).empty();
  drawTierI();
});

$('button#tier_ii').click(function (e) {
  $( "svg" ).empty();
  drawTierII();
});

$('button#tier_ii_plus').click(function (e) {
  $( "svg" ).empty();
  drawTierIIPlus();
});

Вот jfiddle моего кода: http://jsfiddle.net/2H7Pm/ меня, похоже, есть некоторые проблемы с синтаксисом JSON.

Однако пример карты работает на моем github: http://newamericafoundation.github.io/worldofdrones/

Спасибо за помощь!

1 ответ

Решение

Вам просто нужно выбрать существующие пути вместо добавления новых после первоначальной настройки хороплета, например

function drawTierI() {
  csv.forEach(function(d) { tierById.set(d.id, +d.tier_i); });

  function ready(error, json) {
    svg.selectAll("path")
        .attr("class", function(d) { return quantize(tierById.get(d.id)); })
        .attr("d", path);

  }
}

и аналогично для других функций. Вам также не нужно перезагружать JSON и CSV на каждом шаге - приведенный выше код предполагает, что ваши CSV-данные хранятся в переменной csv, который вы бы заполнили так:

var csv;
function setUpChoropleth(error, json, _csv) {
  csv = _csv;
  // rest of setup
}

Тогда вы можете просто сделать

$('button#tier_i').click(function (e) {
  drawTierI();
});

изменить отображаемую переменную без необходимости сначала очищать SVG.

Другие вопросы по тегам