Проблема при преобразовании векторной карты мозаики SVG для использования Canvas - PullRequest
0 голосов
/ 19 декабря 2018

Я работаю над визуализацией карты избирательных участков канадских выборов.Мне удалось получить рабочую версию SVG здесь.

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

class VectorMap {

constructor(map_file, element_id) {
    this.map_file = map_file;
    this.root = $(`#${element_id}`)[0];
    this.init_map();
}

init_map() {
    var root_rect = this.root.getBoundingClientRect();
    this.width = root_rect.width;
    this.height = 720; 

    this.zoom = d3.zoom()
        .on('zoom', _ => this.onZoom(d3.event.transform));

    var canvas = d3.select(this.root).append('canvas')
        .attr('width', this.width)
        .attr('height', this.height)
        .call(this.zoom);
    this.context = canvas.node().getContext('2d');

    this.projection = d3.geoMercator()
        .scale(1 / 2 / Math.PI)
        .translate([0, 0]);

    this.path = d3.geoPath()
        .projection(this.projection)
        .context(this.context);

    this.onZoom(d3.zoomIdentity);
}

onZoom(transform) {       
    var tiler = d3.tile()
        .size([this.width, this.height])
        .scale(transform.k)
        .translate([transform.x, transform.y])

    //init drawing parameters
    this.context.save();
    this.context.fillStyle = 'gray';
    this.context.strokeStyle = 'red';
    //this.context.lineWidth = 1 / transform.k;
    this.context.setTransform(1, 0, 0, 1, 0, 0);
    this.context.clearRect(0, 0, this.width, this.height);
    this.context.scale(transform.k, transform.k);
    this.context.translate(transform.x, transform.y);

    var tiles = tiler();
    tiles.forEach(tile => {
        console.log(tile);
        var zoom = tile[2];
        var x = tile[0];
        var y = tile[1];
        if (zoom > 10) {
            // since these are vector tiles we can just overzoom the z10 ones
            x = Math.floor(x / Math.pow(2, zoom - 10));
            y = Math.floor(y / Math.pow(2, zoom - 10));
            zoom = 10;
        }
        d3.json(`/maps/${zoom}/${x}/${y}/`)
            .then(function(json) {
                json.features.forEach(feature => {
                    this.context.beginPath();
                    path(feature);
                    this.context.closePath();
                    this.context.stroke();
                    this.context.fill();
                });
            });
    });
    this.context.restore();
}

, но это ничего не рисует.Очевидно, что что-то изменилось между SVG и Canvas, но я не уверен, что.Кажется, что просто рисование прямоугольников работает, поэтому я предполагаю, что это проблема с генератором пути холста.

...