【问题标题】:Getting coordinates of rectangle/polygon when drawn on google maps with drawing manager使用绘图管理器在谷歌地图上绘制时获取矩形/多边形的坐标
【发布时间】:2015-10-02 01:26:38
【问题描述】:

我发现了一个要点,它可以启用绘图工具并能够从几种颜色中进行选择以在谷歌地图上绘图:https://gist.github.com/Hagith/5765919

我正在尝试将它与 socket.io 一起使用,以便多人可以查看地图,并且当一个人在地图上绘制某些东西时,所有人都可以看到绘制的内容。

我已经用标记记下了基本的想法

socket.emit("marker", e.overlay.position);

但是,当放置带有矩形、多边形和圆形的标记时,它似乎有点困难。当我用这些形状中的任何一个注销地图上的点击事件时,它返回的数据似乎比它用标记返回的数据复杂得多,而且我找不到我需要的点的坐标能够广播给其他用户。有谁知道在上述要点的背景下在哪里可以找到这些?

编辑:我已经能够通过e.overlay.j.center找到中心

【问题讨论】:

    标签: javascript google-maps google-maps-api-3 socket.io


    【解决方案1】:

    不建议使用这些属性 (e.overlay.j),因为它们不是供公众访问的,并且无法保证它们不会在 Google Maps JavaScript API 的下一版本中发生变化时间>。

    对于google.maps.drawing.OverlayType.RECTANGLEgoogle.maps.drawing.OverlayType.CIRCLE 类型,您可以使用getBounds() function 来确定当前形状的纬度/经度范围,如下所示:

    //get lat/lng bounds of the current shape
    var bounds = e.overlay.getBounds();
    var start = bounds.getNorthEast();
    var end = bounds.getSouthWest();
    var center = bounds.getCenter();
    

    对于google.maps.drawing.OverlayType.POLYLINEgoogle.maps.drawing.OverlayType.POLYGON 类型,您可以使用getPath() function

    //get lat/lng array of the current shape
    var locations = e.overlay.getPath().getArray()
    

    修改示例

    var drawingManager;
    var selectedShape;
    var colors = ['#1E90FF', '#FF1493', '#32CD32', '#FF8C00', '#4B0082'];
    var selectedColor;
    var colorButtons = {};
    function clearSelection() {
        if (selectedShape) {
            selectedShape.setEditable(false);
            selectedShape = null;
        }
    }
    function setSelection(shape) {
        clearSelection();
        selectedShape = shape;
        shape.setEditable(true);
        selectColor(shape.get('fillColor') || shape.get('strokeColor'));
    }
    function deleteSelectedShape() {
        if (selectedShape) {
            selectedShape.setMap(null);
        }
    }
    function selectColor(color) {
        selectedColor = color;
        for (var i = 0; i < colors.length; ++i) {
            var currColor = colors[i];
            colorButtons[currColor].style.border = currColor == color ? '2px solid #789' : '2px solid #fff';
        }
        // Retrieves the current options from the drawing manager and replaces the
        // stroke or fill color as appropriate.
        var polylineOptions = drawingManager.get('polylineOptions');
        polylineOptions.strokeColor = color;
        drawingManager.set('polylineOptions', polylineOptions);
        var rectangleOptions = drawingManager.get('rectangleOptions');
        rectangleOptions.fillColor = color;
        drawingManager.set('rectangleOptions', rectangleOptions);
        var circleOptions = drawingManager.get('circleOptions');
        circleOptions.fillColor = color;
        drawingManager.set('circleOptions', circleOptions);
        var polygonOptions = drawingManager.get('polygonOptions');
        polygonOptions.fillColor = color;
        drawingManager.set('polygonOptions', polygonOptions);
    }
    function setSelectedShapeColor(color) {
        if (selectedShape) {
            if (selectedShape.type == google.maps.drawing.OverlayType.POLYLINE) {
                selectedShape.set('strokeColor', color);
            } else {
                selectedShape.set('fillColor', color);
            }
        }
    }
    function makeColorButton(color) {
        var button = document.createElement('span');
        button.className = 'color-button';
        button.style.backgroundColor = color;
        google.maps.event.addDomListener(button, 'click', function () {
            selectColor(color);
            setSelectedShapeColor(color);
        });
        return button;
    }
    function buildColorPalette() {
        var colorPalette = document.getElementById('color-palette');
        for (var i = 0; i < colors.length; ++i) {
            var currColor = colors[i];
            var colorButton = makeColorButton(currColor);
            colorPalette.appendChild(colorButton);
            colorButtons[currColor] = colorButton;
        }
        selectColor(colors[0]);
    }
    function initialize() {
        var map = new google.maps.Map(document.getElementById('map'), {
            zoom: 16,
            center: new google.maps.LatLng(52.25097, 20.97114),
            mapTypeId: google.maps.MapTypeId.SATELLITE,
            disableDefaultUI: true,
            zoomControl: true
        });
        var polyOptions = {
            strokeWeight: 0,
            fillOpacity: 0.45,
            editable: true,
            draggable: true
        };
        // Creates a drawing manager attached to the map that allows the user to draw
        // markers, lines, and shapes.
        drawingManager = new google.maps.drawing.DrawingManager({
            drawingMode: google.maps.drawing.OverlayType.POLYGON,
            markerOptions: {
                draggable: true
            },
            polylineOptions: {
                editable: true,
                draggable: true
            },
            rectangleOptions: polyOptions,
            circleOptions: polyOptions,
            polygonOptions: polyOptions,
            map: map
        });
        google.maps.event.addListener(drawingManager, 'overlaycomplete', function (e) {
            if (e.type !== google.maps.drawing.OverlayType.MARKER) {
                // Switch back to non-drawing mode after drawing a shape.
                drawingManager.setDrawingMode(null);
                // Add an event listener that selects the newly-drawn shape when the user
                // mouses down on it.
                var newShape = e.overlay;
                newShape.type = e.type;
                google.maps.event.addListener(newShape, 'click', function (e) {
                    if (e.vertex !== undefined) {
                        if (newShape.type === google.maps.drawing.OverlayType.POLYGON) {
                            var path = newShape.getPaths().getAt(e.path);
                            path.removeAt(e.vertex);
                            if (path.length < 3) {
                                newShape.setMap(null);
                            }
                        }
                        if (newShape.type === google.maps.drawing.OverlayType.POLYLINE) {
                            var path = newShape.getPath();
                            path.removeAt(e.vertex);
                            if (path.length < 2) {
                                newShape.setMap(null);
                            }
                        }
                    }
                    setSelection(newShape);
                });
                setSelection(newShape);
    
                if (e.type == google.maps.drawing.OverlayType.POLYLINE || google.maps.drawing.OverlayType.POLYGON) {
                    var locations = e.overlay.getPath().getArray()
                    //console.log(bounds.toString());    
                    document.getElementById('output').innerHTML = locations.toString();
                }
                else {
                    //get lat/lng bounds of the current shape
                    var bounds = e.overlay.getBounds();
                    var start = bounds.getNorthEast();
                    var end = bounds.getSouthWest();
                    var center = bounds.getCenter();
                    //console.log(bounds.toString());    
                    document.getElementById('output').innerHTML = bounds.toString();
                }
    
    
            }
        });
        // Clear the current selection when the drawing mode is changed, or when the
        // map is clicked.
        google.maps.event.addListener(drawingManager, 'drawingmode_changed', clearSelection);
        google.maps.event.addListener(map, 'click', clearSelection);
        google.maps.event.addDomListener(document.getElementById('delete-button'), 'click', deleteSelectedShape);
        buildColorPalette();
    }
    google.maps.event.addDomListener(window, 'load', initialize);
    #map, html, body {
                padding: 0;
                margin: 0;
                width: 960px;
                height: 300px;
            }
    
            #panel {
                width: 200px;
                font-family: Arial, sans-serif;
                font-size: 13px;
                float: right;
                margin: 10px;
            }
    
            #color-palette {
                clear: both;
            }
    
            .color-button {
                width: 14px;
                height: 14px;
                font-size: 0;
                margin: 2px;
                float: left;
                cursor: pointer;
            }
    
            #delete-button {
                margin-top: 5px;
            }
    <script type="text/javascript"
                src="http://maps.google.com/maps/api/js?sensor=false&libraries=drawing"></script> 
    <div id="panel">
            <div id="color-palette"></div>
            <div>
                <button id="delete-button">Delete Selected Shape</button>
            </div>
     </div>
     <div id="map"></div>
     <div id="output"></div>

    【讨论】:

      【解决方案2】:
      getCoords(someShape){
          const paths = someShape.getPath().getArray();
          const coords = paths.map((a) => [a.lat(), a.lng()]);
      }
      

      给出 [lat, lng] 的列表

      【讨论】:

        【解决方案3】:

        编辑:我已经能够通过 e.overlay.j.center 找到中心

        您希望非常小心地使用在浏览器调试工具中可以看到的单字母属性。它们没有文档记录,或者是静态的,并且会在没有警告的情况下更改。


        要回答实际问题 - e.overlay 的类型取决于您初始化 DrawingManager 的内容,请参阅文档 here。因此,如果您正在绘制多边形,e.overlay 的类型将是 Polygon。然后,您可以使用e.overlay.getPath(0).getArray() 获取构成该多边形的点,这将为您提供LatLng 对象的数组。 (显然循环遍历所有可用路径,而不仅仅是0)。

        有一个很好的例子here,它显示了基于叠加层返回的几何类型的切换行为。

        【讨论】:

          猜你喜欢
          • 1970-01-01
          • 2011-01-20
          • 1970-01-01
          • 1970-01-01
          • 1970-01-01
          • 1970-01-01
          • 1970-01-01
          • 2012-05-31
          • 1970-01-01
          相关资源
          最近更新 更多