Drag (move) a polygon using Google Maps v3

前端 未结 5 879
清歌不尽
清歌不尽 2021-01-02 15:49

The Google Maps API for a Polygon does not offer a drag method.

What would be an efficient way of implementing such a feature (i.e., sufficiently optimised so that i

相关标签:
5条回答
  • 2021-01-02 16:23

    Here's how I do it. Find the approximate center of the polygon, and add a marker, then add a drag listener to the marker. On lat/lng change, subtract the difference from the original marker lat/lng, subtract the difference to each of the paths, then, set the original position to the new position. Make sure that in your javascript api call that you have library=geometry,drawing

    google.maps.event.addListener(draw, 'overlaycomplete', function(shape) {
    // POLYGON
          if (shape.type == 'polygon') {
            var bounds = new google.maps.LatLngBounds(); var i;  
            var path = shape.overlay.getPath();
            for (i = 0; i < path.length; i++) { bounds.extend(path.getAt(i)); }
            shape.latLng = bounds.getCenter();
            marker = getMarker(map,shape);
            shape.overlay.marker = marker;
            markers.push(marker); 
          }
          google.maps.event.addListener(marker, 'drag', function(event) {
             shape.overlay.move(event.latLng, shape, path);
          });
    
              google.maps.event.addListener(shape.overlay, 'rightclick', function() {
                this.setMap(null);
                this.marker.setMap(null);
                draw.setDrawingMode('polygon');
              });
    
      });
    }
    google.maps.Polygon.prototype.move = function(latLng, shape, p) {
        var lat = latLng.lat();
        var lng = latLng.lng();
    
        latDiff = shape.latLng.lat()-lat;
        lngDiff = shape.latLng.lng()-lng;
    
       for (i = 0; i < p.length; i++) {
        pLat = p.getAt(i).lat();
        pLng = p.getAt(i).lng();
        p.setAt(i,new google.maps.LatLng(pLat-latDiff,pLng-lngDiff));
       }
       shape.latLng = latLng; 
    }
    function getMarker(map,shape){
      var infowindow = new google.maps.InfoWindow();
      if(shape.type=='polygon'){ latLng = shape.latLng; }
      marker = new google.maps.Marker({
                  position: latLng,
                  map:map,
                  draggable:true,
                  clickable: true,
                  animation: google.maps.Animation.DROP
                });
               shape.overlay.marker = marker;
               shape.overlay.bindTo('center',marker,'position');
               google.maps.event.addListener(marker, 'click', (function(marker) {
                return function() {
                  infowindow.setContent('polygon');
                  infowindow.open(map, marker);
                  toggleBounce(marker);
                }
              })(marker));
              google.maps.event.addListener(infowindow,'closeclick', (function(marker) {      
                return function() {
                marker.setAnimation(null);
                }
              })(marker));
     return marker;
    }
    

    If you have any questions, feel free to contact me.

    0 讨论(0)
  • 2021-01-02 16:23

    You could have markers for each point on the polygon, these markers could have drag and at the end of each drag, the polygon could be redrawn.

    You could also have a marker in the center of the polygon representing the polygon as a whole, when you move that marker, every marker could be moved by the same amount to maintain the shape.

    0 讨论(0)
  • 2021-01-02 16:30

    I found the Google Maps V2 Polygon Implementation to be very limiting for the needs I have had and solved it by creating a custom overlay. My group is currently stuck on IE6 so I have yet to migrate over to Google Maps V3 - but taking a quick look at the API shows that you could probably do a similar thing that I did in V2 with V3.

    Essentially the idea is:

    1. Create a Custom Overlay
    2. Populate it with your own SVG/VML Polygons and attach a drag event to this custom polygon object

    Custom Overlays:

    Here is some information to get you started on making your own custom overlay:

    http://code.google.com/apis/maps/documentation/javascript/overlays.html#CustomOverlays


    Creating your own "Dragable" Polygon Object:

    Once you get that down you'll want to add your own polygons to the custom overlay instead of using GPolygons. I went through the painful process of learning SVG/VML and writing a library to bridge SVG/VML together - you could do that, but I would recommend starting by trying to use another library such as Raphaël.

    http://raphaeljs.com/

    Using Raphaël will save you a whole lot of time trying to figure out how to get cross-browser Vector Graphic (Polygon) functionality - and best of all it supports drag events already, here is an example from their library:

    http://raphaeljs.com/graffle.html

    Once you have a custom overlay and you are able to throw some Raphaël objects onto it the last step is to translate the coordinates you want from a Lat/Lng value to a Pixel value. This is available in the MapCanvasProjection of V3:

    http://code.google.com/apis/maps/documentation/javascript/reference.html#MapCanvasProjection

    You can use fromLatLngToDivPixel to figure out what the actual pixel values are for the points on your Raphael polygon, draw it, then add it to the overlay with a drag event.

    0 讨论(0)
  • 2021-01-02 16:43

    Since version 3.11 (dated Jan 22, 2013) it's possible to just set the draggable property onto the google.maps.Polygon instance; see this example.

    If you want to programmatically move a polygon, you'll need a custom Google Maps Extension which I wrote, as the API does not provide such a method.

    0 讨论(0)
  • 2021-01-02 16:48

    Okay - so after seeing the website you are trying to implement I started to feel like Raphael may not be necessary because it is a pretty heavy JS Library - and if you are only trying to draw a rectangle polygon I thought, why not just do it with a single lightweight DIV instead?

    However I think the Raphael solution would still hold water for many other cases - so I think I'll just post another possible answer.

    Here is a working example I threw together:

    http://www.johnmick.net/drag-div-v3/

    Feel free to take a look at the source:

    http://www.johnmick.net/drag-div-v3/js/main.js

    Essentially we do the following

    1. Create the Custom Overlay
    2. Create the draggable div polygon and, using jQuery UI, make it draggable
    3. Tie an event that listens to when the dragging has stopped that updates the LatLng position of the rectangle
    4. Add the object to the Custom Overlay
    5. Implement the draw function to redraw the rectangle during zooms and pans

    Currently I am only storing one LatLng value for the Rectangle (being the top left corner) - you could easily extend this example to store all 4 points of the rectangle and have the shape dynamically resize itself on zooms. You may want to do that, otherwise as users zoom out they will get a climate report for a larger and larger area.

    0 讨论(0)
提交回复
热议问题