Google Maps JS API v3 - Как проверить и отобразить расстояние между точками многоугольника - PullRequest
0 голосов
/ 29 апреля 2019

Я использую API карт Google, чтобы пользователь мог рисовать на карте собственные многоугольники. Мне нужно проверить и отобразить длину каждой границы.

Я уже использую Библиотеку геометрии и Библиотеку меток карты, чтобы получить и показать вычисленную площадь многоугольника (также изменяющуюся для событий 'insert_at' и 'set_at'), но, к сожалению, я не знаю, как получить длины границ. Любая помощь будет оценена.

Используемый пример кода:


var labels = [];
var allOverlays = [];

function setSelection(shape) {
  selectedShape = shape;
  shape.setEditable(true);
}

function initMap() {

 var options = {
    zoom: 14,
    center: {lat: 52.250618, lng: 20.9774}
}

var map = new google.maps.Map(document.getElementById('map'), options);

var drawingManager = new google.maps.drawing.DrawingManager({
 markerOption: {
 draggable: false
 },
 polygonOptions: {
 draggable: false,
 fillColor: '#5C6BC0',
 fillOpacity: 0.45,
 strokeWeight: 0,
 editable: true,
 zIndex: 1
 },
 drawingControl: true,
 drawingControlOptions: {
 position: google.maps.ControlPosition.TOP_CENTER,
 drawingModes: ['circle', 'polygon']
 },
 circleOptions: {
 fillColor: '#5C6BC0',
 fillOpacity: 0.45,
 strokeWeight: 0,
 editable: true,
 zIndex: 1
 },
 map: map
});

function attachPolygonInfoWindow(polygon) {
   var path = polygon.getPath();
   var points = path.getArray();
   var area = (google.maps.geometry.spherical.computeArea(path.getArray())).toFixed(0);
   var bounds = new google.maps.LatLngBounds();
   var i;

   for (i = 0; i < points.length; i++) {
      bounds.extend(points[i]);
   }

   var boundsCenter = bounds.getCenter();
   var mapLabel = new MapLabel({
      map: map,
      fontSize: 20,
      align: 'left'
   });

   if (!labels.length) {
      labels.push(mapLabel)
   }

   showPolygonInfoWindow(labels, boundsCenter, area);
}

function showPolygonInfoWindow(arr, position, text) {
   arr.forEach((el) => {
     el.set('position', position);
     el.set('text', text + 'm2')
   })
}

function removePolygonInfoWindow() {

   for (var i = 0; i < labels.length; i++) {
     labels[i].setMap(null);
   }

   labels = [];
}

google.maps.event.addListener(drawingManager, 'overlaycomplete', function(e) {
   allOverlays.push(e);

   if (e.type != google.maps.drawing.OverlayType.MARKER) {

   drawingManager.setDrawingMode(null);

   var newShape = e.overlay;
   newShape.type = e.type;

   google.maps.event.addListener(newShape, 'click', function() {
       setSelection(newShape);
   });


   if (newShape.type == "polygon") {
     var path = newShape.getPath();

     google.maps.event.addListener(path, 'insert_at', function() {
        attachPolygonInfoWindow(newShape);
     });

     google.maps.event.addListener(path, 'set_at', function() {
        attachPolygonInfoWindow(newShape);
     });

     attachPolygonInfoWindow(newShape);
   }

   setSelection(newShape);
  }
});
}

initMap();

<script src="https://maps.google.com/maps/api/js?sensor=false&libraries=drawing,geometry">
</script>
<script type="text/javascript" src="https://cdn.rawgit.com/googlemaps/js-map-label/gh-pages/src/maplabel.js"></script>

Рабочий пример Codepen

Я хотел бы отобразить длину каждой границы на стороне каждой границы.

1 Ответ

0 голосов
/ 06 мая 2019

Мое предложение будет, когда вы создаете метку центра, также обрабатываете пути многоугольника, вычисляете их длины и центры;затем создайте MapLabel объекты для каждого и поместите его в центр стороны.Что-то вроде:

for (var i=0; i<polygon.getPath().getLength(); i++) {
  // for each side in path, compute center and length
  var start = polygon.getPath().getAt(i);
  var end = polygon.getPath().getAt(i<polygon.getPath().getLength()-1 ? i+1 : 0);
  var sideLength = google.maps.geometry.spherical.computeDistanceBetween(start,end);
  var sideCenter = google.maps.geometry.spherical.interpolate(start, end, 0.5);
  var sideLabel = new MapLabel({
    map: map,
    fontSize: 20,
    align: "center"
  });
  sideLabel.set("position", sideCenter);
  sideLabel.set("text", sideLength.toFixed(2)+"m");
  polygon.labels.push(sideLabel);
}

подтверждение концепции скрипта

screenshot of resulting map

фрагмент кода:

var labels = [];
var allOverlays = [];

function setSelection(shape) {
  selectedShape = shape;
  shape.setEditable(true);
}

function initMap() {
  var options = {
    zoom: 14,
    center: {
      lat: 52.250618,
      lng: 20.9774
    }
  };

  var map = new google.maps.Map(document.getElementById("map"), options);

  var drawingManager = new google.maps.drawing.DrawingManager({
    polygonOptions: {
      draggable: false,
      fillColor: "#5C6BC0",
      fillOpacity: 0.45,
      strokeWeight: 0,
      editable: true,
      zIndex: 1
    },
    drawingControl: true,
    drawingControlOptions: {
      position: google.maps.ControlPosition.TOP_CENTER,
      drawingModes: ["polygon"]
    },
    map: map,
    drawingMode: 'polygon'
  });

  function attachPolygonInfoWindow(polygon) {
    if (!polygon.labels) polygon.labels = [];
    for (var i = 0; i < polygon.labels.length; i++) {
      polygon.labels[i].setMap(null);
    }
    polygon.labels = [];
    var path = polygon.getPath();
    var points = path.getArray();
    var area = google.maps.geometry.spherical
      .computeArea(path.getArray())
      .toFixed(0);
    var bounds = new google.maps.LatLngBounds();
    var i;

    for (i = 0; i < points.length; i++) {
      bounds.extend(points[i]);
    }

    var boundsCenter = bounds.getCenter();
    var centerLabel = new MapLabel({
      map: map,
      fontSize: 20,
      align: "left"
    });

    polygon.labels.push(centerLabel);

    centerLabel.set("position", bounds.getCenter());
    centerLabel.set("text", area + "m2");
    if (path.getLength() < 2) return;
    for (var i = 0; i < polygon.getPath().getLength(); i++) {
      // for each side in path, compute center and length
      var start = polygon.getPath().getAt(i);
      var end = polygon.getPath().getAt(i < polygon.getPath().getLength() - 1 ? i + 1 : 0);
      var sideLength = google.maps.geometry.spherical.computeDistanceBetween(start, end);
      var sideCenter = google.maps.geometry.spherical.interpolate(start, end, 0.5);
      var sideLabel = new MapLabel({
        map: map,
        fontSize: 20,
        align: "center"
      });
      sideLabel.set("position", sideCenter);
      sideLabel.set("text", sideLength.toFixed(2) + "m");
      polygon.labels.push(sideLabel);
    }
  }

  function removePolygonInfoWindow() {
    for (var i = 0; i < labels.length; i++) {
      labels[i].setMap(null);
    }
    labels = [];
  }

  google.maps.event.addListener(drawingManager, "overlaycomplete", function(e) {
    allOverlays.push(e);

    if (e.type != google.maps.drawing.OverlayType.MARKER) {
      drawingManager.setDrawingMode(null);

      var newShape = e.overlay;
      newShape.type = e.type;

      google.maps.event.addListener(newShape, "click", function() {
        setSelection(newShape);
      });

      if (newShape.type == "polygon") {
        var path = newShape.getPath();

        google.maps.event.addListener(path, "insert_at", function() {
          attachPolygonInfoWindow(newShape);
        });

        google.maps.event.addListener(path, "set_at", function() {
          attachPolygonInfoWindow(newShape);
        });

        attachPolygonInfoWindow(newShape);
      }

      setSelection(newShape);
    }
  });
}

initMap();
html,
body,
#map {
  height: 100%;
  margin: 0;
  padding: 0;
}
<div id="map"></div>
<!-- Replace the value of the key parameter with your own API key. -->
<script src="https://maps.googleapis.com/maps/api/js?key=AIzaSyCkUOdZ5y7hMm0yrcCQoCvLwzdM6M8s5qk&libraries=drawing"></script>
<script src="https://cdn.jsdelivr.net/npm/js-map-label@1.0.1/src/maplabel.js"></script>
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...