/**
 * @name polylineEdit for Google Maps V3 API
 * @version 1.0.1 [January 29, 2011]
 * @author: ryshkin@gmail.com
 * @fileoverview <b>Author:</b> ryshkin@gmail.com<br/> <b>Licence:</b>
 *               Licensed under <a
 *               href="http://opensource.org/licenses/mit-license.php">MIT</a>
 *               license.<br/> This library Extends the functionality of a
 *               class google.maps.Polyline by methods runEdit() and stopEdit()<br/>
 *               Enjoy guys:)<br/>
 *               Special thanks <code>Jan Pieter Waagmeester jieter@jpwaag.com</code> for the idea of using the library google.maps.geometry , which performs spherical linear interpolation between the two locations.
 */
/**
 * @name google
 * @class The fundamental namespace for Google APIs 
 */
/**
 * @name google.maps
 * @class The fundamental namespace for Google Maps V3 API 
 */
/**
 * @name google.maps.Polyline
 * @class Extends standart class google.maps.Polyline by methods runEdit() and
 *        stopEdit()
 */
if (typeof(google.maps.Polyline.prototype.runEdit) === "undefined") {
  /**
   * Starts editing the polyline. Optional parameter <code>flag</code>
   * indicates the use of ghost markers in the middle of each segment. By
   * default, the <code>flag</code> is true.
   * 
   * @param {}
   *            flag - (true) include additional points in the middle of each
   *            segment
   */
  google.maps.Polyline.prototype.runEdit = function (flag) {
    if (!flag) {
      flag = true;
    }
    var self = this;
    if (flag) {
      var imgGhostVertex = new google.maps.MarkerImage(
                'http://digitizer.avmap.gr/DesktopModules/AVMap.MapDigitizer/Scripts/css/ghostVertex.png', new google.maps.Size(11, 11),
                new google.maps.Point(0, 0), new google.maps.Point(6, 6));
      var imgGhostVertexOver = new google.maps.MarkerImage(
                'http://digitizer.avmap.gr/DesktopModules/AVMap.MapDigitizer/Scripts/css/ghostVertexOver.png', new google.maps.Size(11, 11),
                new google.maps.Point(0, 0), new google.maps.Point(6, 6));
      var ghostPath = new google.maps.Polyline({
        map : this.getMap(),
        strokeColor : this.strokeColor,
        strokeOpacity : 0.2,
        strokeWeight : this.strokeWeight
      });
      var vertexGhostMouseOver = function () {
        this.setIcon(imgGhostVertexOver);
      };
      var vertexGhostMouseOut = function () {
        this.setIcon(imgGhostVertex);
      };
      var vertexGhostDrag = function () {
        editMode=true;  //pkedit
        if (ghostPath.getPath().getLength() === 0) {
          ghostPath.setPath([this.marker.getPosition(), this.getPosition(), self.getPath().getAt(this.marker.inex + 1)]);
        }  
        ghostPath.getPath().setAt(1, this.getPosition());
      };
      var moveGhostMarkers = function (marker) {
        var Vertex = self.getPath().getAt(marker.inex);
        var prevVertex = self.getPath().getAt(marker.inex - 1);
        if ((typeof(Vertex) !== "undefined") && (typeof(Vertex.ghostMarker) !== "undefined")) {
          if (typeof(google.maps.geometry) === "undefined") {
            Vertex.ghostMarker.setPosition(new google.maps.LatLng(Vertex.lat() + 0.5 * (self.getPath().getAt(marker.inex + 1).lat() - Vertex.lat()), Vertex.lng() + 0.5 * (self.getPath().getAt(marker.inex + 1).lng() - Vertex.lng())));
          } else {
            Vertex.ghostMarker.setPosition(google.maps.geometry.spherical.interpolate(Vertex, self.getPath().getAt(marker.inex + 1), 0.5));
          }
        }
        if ((typeof(prevVertex) !== "undefined") && (typeof(prevVertex.ghostMarker) !== "undefined")) {
          if (typeof(google.maps.geometry) === "undefined") {
            prevVertex.ghostMarker.setPosition(new google.maps.LatLng(prevVertex.lat() + 0.5 * (marker.getPosition().lat() - prevVertex.lat()), prevVertex.lng() + 0.5 * (marker.getPosition().lng() - prevVertex.lng())));
          } else {
            prevVertex.ghostMarker.setPosition(google.maps.geometry.spherical.interpolate(prevVertex, marker.getPosition(), 0.5));
          }
        }
      };
      var vertexGhostDragEnd = function () {
        ghostPath.getPath().forEach(function () {
          ghostPath.getPath().pop();
        });
        self.getPath().insertAt(this.marker.inex + 1, this.getPosition());
        createMarkerVertex(self.getPath().getAt(this.marker.inex + 1)).inex = this.marker.inex + 1;
        moveGhostMarkers(this.marker);
        createGhostMarkerVertex(self.getPath().getAt(this.marker.inex + 1));
        self.getPath().forEach(function (vertex, inex) {
          if (vertex.marker) {
            vertex.marker.inex = inex;
          }
        });
        editMode=false; //pkedit
      };
      var createGhostMarkerVertex = function (point) {
        if (point.marker.inex < self.getPath().getLength() - 1) {
          var markerGhostVertex = new google.maps.Marker({
            position : (typeof(google.maps.geometry) === "undefined") ? new google.maps.LatLng(
                                                                          point.lat() + 0.5 * (self.getPath().getAt(point.marker.inex + 1).lat() - point.lat()),
                                                                          point.lng() + 0.5 * (self.getPath().getAt(point.marker.inex + 1).lng() - point.lng()))
                       :google.maps.geometry.spherical.interpolate(point, self.getPath().getAt(point.marker.inex + 1), 0.5),
            map : self.getMap(),
            icon : imgGhostVertex,
            draggable : true,
            raiseOnDrag : false
          });
          google.maps.event.addListener(markerGhostVertex, "mouseover", vertexGhostMouseOver);
          google.maps.event.addListener(markerGhostVertex, "mouseout", vertexGhostMouseOut);
          google.maps.event.addListener(markerGhostVertex, "drag", vertexGhostDrag);
          google.maps.event.addListener(markerGhostVertex, "dragend", vertexGhostDragEnd);
          point.ghostMarker = markerGhostVertex;
          markerGhostVertex.marker = point.marker;
          return markerGhostVertex;
        }
        return null;
      };
    }
    var imgVertex = new google.maps.MarkerImage('http://digitizer.avmap.gr/DesktopModules/AVMap.MapDigitizer/Scripts/css/vertex.png',
      new google.maps.Size(11, 11), new google.maps.Point(0, 0),
      new google.maps.Point(6, 6));
    var imgVertexOver = new google.maps.MarkerImage('http://digitizer.avmap.gr/DesktopModules/AVMap.MapDigitizer/Scripts/css/vertexOver.png',
      new google.maps.Size(11, 11), new google.maps.Point(0, 0),
      new google.maps.Point(6, 6));
    var vertexMouseOver = function () {
      this.setIcon(imgVertexOver);
    };
    var vertexMouseOut = function () {
      this.setIcon(imgVertex);
    };
    var vertexDrag = function () {
               editMode = true;    //pkedit
      var movedVertex = this.getPosition();
      movedVertex.marker = this;
      movedVertex.ghostMarker = self.getPath().getAt(this.inex).ghostMarker;
      self.getPath().setAt(this.inex, movedVertex);
      if (flag) {
        moveGhostMarkers(this);
      }
              editMode = false;    //pkedit
    };
    var vertexRightClick = function () {
      if (flag) {
        var Vertex = self.getPath().getAt(this.inex);
        var prevVertex = self.getPath().getAt(this.inex - 1);
        if (typeof(Vertex.ghostMarker) !== "undefined") {
          Vertex.ghostMarker.setMap(null);
        }
        self.getPath().removeAt(this.inex);
        if (typeof(prevVertex) !== "undefined") {
          if (this.inex < self.getPath().getLength()) {
            moveGhostMarkers(prevVertex.marker);
          }
          else {
            prevVertex.ghostMarker.setMap(null);
            prevVertex.ghostMarker = undefined;
          }
        }
      } 
      else {
        self.getPath().removeAt(this.inex);
      }
      this.setMap(null);
      self.getPath().forEach(function (vertex, inex) {
        if (vertex.marker) {
          vertex.marker.inex = inex;
        }
      });
      if (self.getPath().getLength() === 1) {
        self.getPath().pop().marker.setMap(null);
      }
    };
    var createMarkerVertex = function (point) {
      var markerVertex = new google.maps.Marker({
        position : point,
        map : self.getMap(),
        icon : imgVertex,
        draggable : true,
        raiseOnDrag : false
      });
      google.maps.event.addListener(markerVertex, "mouseover", vertexMouseOver);
      google.maps.event.addListener(markerVertex, "mouseout", vertexMouseOut);
      google.maps.event.addListener(markerVertex, "drag", vertexDrag);
      google.maps.event.addListener(markerVertex, "rightclick", vertexRightClick);
      point.marker = markerVertex;
      return markerVertex;
    };
    this.getPath().forEach(function (vertex, inex) {
      createMarkerVertex(vertex).inex = inex;
      if (flag) {
        createGhostMarkerVertex(vertex);
      }
    });
  };
}
if (typeof(google.maps.Polyline.prototype.stopEdit) === "undefined") {
  /**
   * Stops editing polyline
  */
  google.maps.Polyline.prototype.stopEdit = function () {
    this.getPath().forEach(function (vertex, inex) {
      if (vertex.marker) {
        vertex.marker.setMap(null);
        vertex.marker = undefined;
      }
      if (vertex.ghostMarker) {
        vertex.ghostMarker.setMap(null);
        vertex.ghostMarker = undefined;
      }
    });
  };
}
