diff --git a/js/gmap_shapes.js b/js/gmap_shapes.js
index 0de3594f0eafc5fe6be11fada016f7e722fe415c..d8dd8b211099c84a1b855d7db9a8d6333537bee1 100644
--- a/js/gmap_shapes.js
+++ b/js/gmap_shapes.js
@@ -3,157 +3,194 @@
  * @file
  * GMap Shapes
  * GMap API version / Base case
+ *
+ * @NOTE This code now depends in various places, on the google.maps.geometry library.
+ * 1. points encoding/decoding
+ * 2. spherical distance calculations
+ * If this library isn't loaded then YOU WILL GET SILENT FAILURE.  There should be some kind
+ * of user warning here, if the geometry library hasn't been loaded. I am not sure what the
+ * gmaps convention for such reporting is?  is a console.debug acceptable?
+ * Note that you can specify to add the geometry library at the php side of things using hook_gmap()
  */
 
-/*global jQuery, Drupal, GEvent, GLatLng, GPolygon, GPolyline */
+/*global $, Drupal, google.maps, !google.maps.geometry! */
 
 Drupal.gmap.addHandler('gmap', function (elem) {
   var obj = this;
-/*
-  obj.bind('init',function() {
-    if (obj.vars.behavior.autozoom) {
-      obj.bounds = new GLatLngBounds(new GLatLng(obj.vars.latitude,obj.vars.longitude),new GLatLng(obj.vars.latitude,obj.vars.longitude));
-    }
-  });
-*/
+
+  // prepare and build a google.maps shape object
   obj.bind('prepareshape', function (shape) {
-    var pa, cargs, style;
-    //var m = new GMarker(new GLatLng(marker.latitude,marker.longitude),marker.opts);
-    pa = []; // point array (array of GLatLng-objects)
-    var fillstyle = true;
-    if (shape.type === 'circle') {
-      pa = obj.poly.calcPolyPoints(new GLatLng(shape.center[0], shape.center[1]), shape.radius * 1000, shape.numpoints);
-    }
-    else if (shape.type === 'rpolygon') {
-      shape.center = new GLatLng(shape.center[0], shape.center[1]);
-      shape.point2 = new GLatLng(shape.point2[0], shape.point2[1]);
-      var radius = shape.center.distanceFrom(shape.point2);
-      pa = obj.poly.calcPolyPoints(shape.center, radius, shape.numpoints);
-    }
-    else if (shape.type === 'polygon') {
-      var coords = new Array();
-      jQuery.each(shape.points, function (i, n) {
-        coords.push(new google.maps.LatLng(n[0], n[1]));
-      });
-    }
-    else if (shape.type === 'line') {
-      var coords = new Array();
-      jQuery.each(shape.points, function (i, n) {
-        coords.push(new google.maps.LatLng(n[0], n[1]));
-      });
-    }
-    cargs = [pa];
+    var pa, cargs, style fillstyle;
+    fillstyle = false; // does this shape have a fill option?
+    cargs = {};
+    pa = []; // point array (array of LatLng-objects)
 
-    // Style normalization
-    if (fillstyle) {
-      style = obj.vars.styles.poly_default.slice();
-    }
-    else {
-      style = obj.vars.styles.line_default.slice();
+    // positioning determination
+    switch (shape.type) {
+      case 'circle':
+        if ( typeof shape.center === 'array') { shape.center = new google.maps.LatLng(shape.center[0], shape.center[1]); } // otherwise it should be a LatLng already
+        if ( shape.point2 ) {
+          if ( typeof shape.point2 === 'array') { shape.point2 = new google.maps.LatLng(shape.point2[0], shape.point2[1]); } // otherwise it should be a LatLng already
+          shape.radius = (google.maps.geometry) ? google.maps.geometry.spherical.computeDistanceBetween( shape.center, shape.point2 ) : 250;
+        } // if you didn't pass a shape.point2, then you should have passed a shape.radius in meters
+        break;
+
+      case 'rpolygon': /* this is deprecated as we have circle now.  It is left for backwards compatibility */
+        if ( typeof shape.center === 'array') { shape.center = new google.maps.LatLng(shape.center[0], shape.center[1]); } // otherwise it should be a LatLng already
+        if ( typeof shape.point2 === 'array') { shape.point2 = new google.maps.LatLng(shape.point2[0], shape.point2[1]); } // otherwise it should be a LatLng already
+        shape.radius = (google.maps.geometry) ? google.maps.geometry.spherical.computeDistanceBetween( shape.center, shape.point2 ) : 250;
+        if ( !shape.numpoints ) { shape.numpoints = 20; }
+        pa = obj.poly.calcPolyPoints(shape.center, radius, shape.numpoints);
+        break;
+
+      case 'polygon':
+        fillstyle = true;
+      case 'line':
+        $.each(shape.points, function (i, n) {
+          if ( typeof n === 'array') { n = new google.maps.LatLng(n[0], n[1]); } // otherwise it should be a LatLng already
+          pa.push( n );
+        });
+        break;
+
+      case 'encoded_polygon':
+        fillstyle = true;
+      case 'encoded_line':
+        pa = ( google.maps.geometry ) ? google.maps.geometry.encoding.decodePath(shape.path) : []; // this trinary prevents errors if the google.maps gemoetry library wasn't loaded
+        break;
     }
+
+    /**
+     * the shapes.style processing is a leftover from the gmaps v2
+     * code, and the shapes configuration system, from the php side
+     * of things.
+     */
     if (shape.style) {
-      if (typeof shape.style === 'string') {
+      if (typeof shape.style === 'string') { // the style is an index for one of our vars.styles
         if (obj.vars.styles[shape.style]) {
-          style = obj.vars.styles[shape.style].slice();
+          style = obj.vars.styles[shape.style].slice(); // copy the array
         }
       }
       else {
-        style = shape.style.slice();
+        style = shape.style.slice(); // copy that array
+      }
+      style[0] = '#' + style[0]; // color
+      style[1] = Number(style[1]); // strokewidth
+      style[2] = style[2] / 100; // strokeOpacity
+      if (fillstyle) {
+        style[3] = '#' + style[3]; // fill colour
+        style[4] = style[4] / 100; // fill opacity
       }
-    }
-    style[0] = '#' + style[0];
-    style[1] = Number(style[1]);
-    style[2] = style[2] / 100;
-    if (fillstyle) {
-      style[3] = '#' + style[3];
-      style[4] = style[4] / 100;
-    }
 
-    if (shape.type == 'encoded_line') {
-      shape.color = style[0];
-      shape.weight = style[1];
-      shape.opacity = style[2];
-    }
-    else if (shape.type == 'encoded_polygon') {
-      jQuery.each(shape.polylines, function(i, polyline) {
-        polyline.color = style[0];
-        polyline.weight = style[1];
-        polyline.opacity = style[2];
-      });
-      shape.fill = true;
-      shape.color = style[3];
-      shape.opacity = style[4];
-      shape.outline = true;
+      if (shape.type == 'encoded_line') {
+        shape.color = style[0];
+        shape.weight = style[1];
+        shape.opacity = style[2];
+      }
+      else if (shape.type == 'encoded_polygon') {
+        if ( shape.polylines ) {
+          $.each(shape.polylines, function(i, polyline) {
+            polyline.color = style[0];
+            polyline.weight = style[1];
+            polyline.opacity = style[2];
+          });
+        }
+        shape.fill = true;
+        shape.color = style[3];
+        shape.opacity = style[4];
+        shape.outline = true;
+      }
     }
 
-    jQuery.each(style, function (i, n) {
-      cargs.push(n);
-    });
+    // add any options to the configuration
     if (shape.opts) {
-      cargs.push(shape.opts);
-    }
-    var Pg = function (args) {
-      GPolygon.apply(this, args);
-    };
-    Pg.prototype = new GPolygon();
-    var Pl = function (args) {
-      GPolyline.apply(this, args);
-    };
-    Pl.prototype = new GPolyline();
-
-    var polyObject = {
-      path: coords,
-      strokeColor: style[0],
-      strokeWeight: style[1],
-      strokeOpacity: style[2],
-      fillColor: style[3],
-      fillOpacity: style[4],
+      $.extend(cargs, shape.opts);
     }
 
+    /**
+     * In general: (inherited concepts.  If you change these, make sure you check them on the php side of things)
+     *  shape.color : color used for line and fill
+     *  shape.weight : stroke weight in pixels
+     *  shape.opacity : fill/stroke opacity in decimal value (0< opactity <1)
+     *  shape.fill : boolean direction to fill the shape
+     */
+    // build the shape with options
     switch (shape.type) {
       case 'circle':
-      case 'polygon':
+        cargs = {center:shape.center, radius: shape.radius, strokeColor: shape.color }; // required arges
+        if ( shape.color ) { cargs.strokeColor = shape.color; } // outline color
+        if ( shape.weight ) { cargs.strokeWeight = shape.weight; } // boundary line weight
+        if ( shape.fill ) { cargs.fillColor = shape.color; } // shape fill color
+        if ( shape.opacity ) { cargs.strokeOpacity = shape.opacity; cargs.fillOpacity = shape.opacity; } // shape opacity
+        shape.shape = new google.maps.Circle(cargs);
+        break;
       case 'rpolygon':
-        shape.shape = new google.maps.Polygon(polyObject);
+      case 'encoded_polygon':
+      case 'polygon':
+        cargs = { path: pa }; // required args
+        if ( shape.outline ) { cargs.strokeColor = shape.color; }
+        if ( shape.weight ) { cargs.strokeWeight = shape.weight; }
+        if ( shape.fill ) { cargs.fillColor = shape.color; }
+        if ( shape.opacity ) { cargs.strokeOpacity = shape.opacity; cargs.fillOpacity = shape.opacity; }
+        shape.shape = new google.maps.Polygon(cargs);
         break;
       case 'line':
-        shape.shape = new google.maps.Polyline(polyObject);
-        break;
       case 'encoded_line':
-        shape.shape = GPolyline.fromEncoded(shape);
-        break;
-      case 'encoded_polygon':
-        shape.shape = GPolygon.fromEncoded(shape);
+        cargs = { path: pa }; // required args
+        if ( shape.color ) { cargs.strokeColor = shape.color; }
+        if ( shape.weight ) { cargs.strokeWeight = shape.weight; }
+        if ( shape.opacity ) { cargs.strokeOpacity = shape.opacity; }
+        shape.shape = new google.maps.Polyline(cargs);
         break;
     }
   });
 
+  // add a prepared shape to the map
   obj.bind('addshape', function (shape) {
-    if ( !obj.map.shapes ) obj.map.shapes = new Array();
+    if (!obj.vars.shapes) {
+      obj.vars.shapes = [];
+    }
+    obj.vars.shapes.push(shape);
     shape.shape.setMap(obj.map);
-    //obj.map.shapes.push( shape.shape );
 
-    /*if (obj.vars.behavior.clickableshapes) {
-      GEvent.addListener(shape.shape, 'click', function () {
+    if (obj.vars.behavior.clickableshapes) {
+      google.maps.event.addListener(shape.shape, 'click', function () {
         obj.change('clickshape', -1, shape);
       });
-    }*/
-
+    }
+    if (obj.vars.behavior.shapesextraevents) {
+      google.maps.event.addListener(shape.shape, 'dblclick', function () {
+        obj.change('dblclickshape', -1, shape);
+      });
+      google.maps.event.addListener(shape.shape, 'mousedown', function () {
+        obj.change('mousedownshape', -1, shape);
+      });
+      google.maps.event.addListener(shape.shape, 'mouseout', function () {
+        obj.change('mouseoutshape', -1, shape);
+      });
+      google.maps.event.addListener(shape.shape, 'mouseover', function () {
+        obj.change('mouseovershape', -1, shape);
+      });
+      google.maps.event.addListener(shape.shape, 'mouseup', function () {
+        obj.change('mouseupshape', -1, shape);
+      });
+      google.maps.event.addListener(shape.shape, 'mousemove', function () {
+        obj.change('mousemoveshape', -1, shape);
+      });
+      google.maps.event.addListener(shape.shape, 'rightclick', function () {
+        obj.change('rightclickshape', -1, shape);
+      });
+    }
   });
 
   obj.bind('delshape', function (shape) {
-    obj.map.removeOverlay(shape.shape);
+    shape.shape.setMap( null );
   });
 
   obj.bind('clearshapes', function () {
     if (obj.vars.shapes) {
-      jQuery.each(obj.vars.shapes, function (i, n) {
+      $.each(obj.vars.shapes, function (i, n) {
         obj.change('delshape', -1, n);
       });
     }
   });
 });
-
-/**
-This is a comment
-*/