| 12345678910111213141516171819202122232425262728293031323334353637383940414243444546474849505152535455565758596061626364656667686970717273747576777879808182838485868788899091929394959697989910010110210310410510610710810911011111211311411511611711811912012112212312412512612712812913013113213313413513613713813914014114214314414514614714814915015115215315415515615715815916016116216316416516616716816917017117217317417517617717817918018118218318418518618718818919019119219319419519619719819920020120220320420520620720820921021121221321421521621721821922022122222322422522622722822923023123223323423523623723823924024124224324424524624724824925025125225325425525625725825926026126226326426526626726826927027127227327427527627727827928028128228328428528628728828929029129229329429529629729829930030130230330430530630730830931031131231331431531631731831932032132232332432532632732832933033133233333433533633733833934034134234334434534634734834935035135235335435535635735835936036136236336436536636736836937037137237337437537637737837938038138238338438538638738838939039139239339439539639739839940040140240340440540640740840941041141241341441541641741841942042142242342442542642742842943043143243343443543643743843944044144244344444544644744844945045145245345445545645745845946046146246346446546646746846947047147247347447547647747847948048148248348448548648748848949049149249349449549649749849950050150250350450550650750850951051151251351451551651751851952052152252352452552652752852953053153253353453553653753853954054154254354454554654754854955055155255355455555655755855956056156256356456556656756856957057157257357457557657757857958058158258358458558658758858959059159259359459559659759859960060160260360460560660760860961061161261361461561661761861962062162262362462562662762862963063163263363463563663763863964064164264364464564664764864965065165265365465565665765865966066166266366466566666766866967067167267367467567667767867968068168268368468568668768868969069169269369469569669769869970070170270370470570670770870971071171271371471571671771871972072172272372472572672772872973073173273373473573673773873974074174274374474574674774874975075175275375475575675775875976076176276376476576676776876977077177277377477577677777877978078178278378478578678778878979079179279379479579679779879980080180280380480580680780880981081181281381481581681781881982082182282382482582682782882983083183283383483583683783883984084184284384484584684784884985085185285385485585685785885986086186286386486586686786886987087187287387487587687787887988088188288388488588688788888989089189289389489589689789889990090190290390490590690790890991091191291391491591691791891992092192292392492592692792892993093193293393493593693793893994094194294394494594694794894995095195295395495595695795895996096196296396496596696796896997097197297397497597697797897998098198298398498598698798898999099199299399499599699799899910001001100210031004100510061007100810091010101110121013101410151016101710181019102010211022102310241025102610271028102910301031103210331034103510361037103810391040104110421043104410451046104710481049105010511052105310541055105610571058105910601061106210631064106510661067106810691070107110721073107410751076107710781079108010811082108310841085108610871088108910901091109210931094109510961097109810991100110111021103110411051106110711081109111011111112111311141115111611171118111911201121112211231124112511261127112811291130113111321133113411351136113711381139114011411142114311441145114611471148114911501151115211531154115511561157115811591160116111621163116411651166116711681169117011711172117311741175117611771178117911801181118211831184118511861187118811891190119111921193119411951196119711981199120012011202120312041205120612071208120912101211121212131214121512161217121812191220122112221223122412251226122712281229123012311232123312341235123612371238123912401241124212431244124512461247124812491250125112521253125412551256125712581259126012611262126312641265126612671268126912701271127212731274127512761277127812791280128112821283128412851286128712881289129012911292129312941295129612971298129913001301130213031304130513061307130813091310131113121313131413151316131713181319132013211322132313241325132613271328132913301331133213331334133513361337133813391340134113421343134413451346134713481349135013511352135313541355135613571358135913601361136213631364136513661367136813691370137113721373137413751376137713781379138013811382138313841385138613871388138913901391139213931394139513961397139813991400140114021403140414051406140714081409141014111412141314141415141614171418141914201421142214231424142514261427142814291430143114321433143414351436143714381439144014411442144314441445144614471448144914501451145214531454145514561457145814591460146114621463146414651466146714681469147014711472147314741475147614771478147914801481148214831484148514861487148814891490149114921493149414951496149714981499150015011502150315041505150615071508150915101511151215131514151515161517151815191520152115221523152415251526152715281529153015311532153315341535153615371538153915401541154215431544154515461547154815491550155115521553155415551556155715581559156015611562156315641565156615671568156915701571157215731574157515761577157815791580158115821583158415851586158715881589159015911592159315941595159615971598159916001601160216031604160516061607160816091610161116121613161416151616161716181619162016211622162316241625162616271628162916301631163216331634163516361637163816391640164116421643164416451646164716481649165016511652165316541655165616571658165916601661166216631664166516661667166816691670167116721673167416751676167716781679168016811682168316841685168616871688168916901691169216931694169516961697169816991700170117021703170417051706170717081709171017111712171317141715171617171718171917201721172217231724172517261727172817291730173117321733173417351736173717381739174017411742174317441745174617471748174917501751175217531754175517561757175817591760176117621763176417651766176717681769177017711772177317741775177617771778177917801781178217831784178517861787178817891790179117921793179417951796179717981799180018011802180318041805180618071808180918101811181218131814181518161817181818191820182118221823182418251826182718281829183018311832183318341835183618371838183918401841184218431844184518461847184818491850185118521853185418551856185718581859186018611862186318641865186618671868186918701871187218731874187518761877187818791880188118821883188418851886188718881889189018911892189318941895189618971898189919001901190219031904190519061907190819091910191119121913191419151916191719181919192019211922192319241925192619271928192919301931193219331934193519361937193819391940194119421943194419451946194719481949195019511952195319541955195619571958195919601961196219631964196519661967196819691970197119721973197419751976197719781979198019811982198319841985198619871988198919901991199219931994199519961997199819992000200120022003200420052006200720082009201020112012201320142015201620172018201920202021202220232024202520262027202820292030203120322033203420352036203720382039204020412042204320442045204620472048204920502051205220532054205520562057205820592060206120622063206420652066206720682069207020712072207320742075207620772078207920802081208220832084208520862087208820892090209120922093209420952096209720982099210021012102210321042105210621072108210921102111211221132114211521162117211821192120212121222123212421252126212721282129213021312132 | (function(root, factory) {  if(typeof exports === 'object') {    module.exports = factory();  }  else if(typeof define === 'function' && define.amd) {    define('GMaps', [], factory);  }  root.GMaps = factory();}(this, function() {/*! * GMaps.js v0.4.15 * http://hpneo.github.com/gmaps/ * * Copyright 2014, Gustavo Leon * Released under the MIT License. */if (!(typeof window.google === 'object' && window.google.maps)) {  throw 'Google Maps API is required. Please register the following JavaScript library http://maps.google.com/maps/api/js?sensor=true.'}var extend_object = function(obj, new_obj) {  var name;  if (obj === new_obj) {    return obj;  }  for (name in new_obj) {    obj[name] = new_obj[name];  }  return obj;};var replace_object = function(obj, replace) {  var name;  if (obj === replace) {    return obj;  }  for (name in replace) {    if (obj[name] != undefined) {      obj[name] = replace[name];    }  }  return obj;};var array_map = function(array, callback) {  var original_callback_params = Array.prototype.slice.call(arguments, 2),      array_return = [],      array_length = array.length,      i;  if (Array.prototype.map && array.map === Array.prototype.map) {    array_return = Array.prototype.map.call(array, function(item) {      callback_params = original_callback_params;      callback_params.splice(0, 0, item);      return callback.apply(this, callback_params);    });  }  else {    for (i = 0; i < array_length; i++) {      callback_params = original_callback_params;      callback_params.splice(0, 0, array[i]);      array_return.push(callback.apply(this, callback_params));    }  }  return array_return;};var array_flat = function(array) {  var new_array = [],      i;  for (i = 0; i < array.length; i++) {    new_array = new_array.concat(array[i]);  }  return new_array;};var coordsToLatLngs = function(coords, useGeoJSON) {  var first_coord = coords[0],      second_coord = coords[1];  if (useGeoJSON) {    first_coord = coords[1];    second_coord = coords[0];  }  return new google.maps.LatLng(first_coord, second_coord);};var arrayToLatLng = function(coords, useGeoJSON) {  var i;  for (i = 0; i < coords.length; i++) {    if (!(coords[i] instanceof google.maps.LatLng)) {      if (coords[i].length > 0 && typeof(coords[i][0]) == "object") {        coords[i] = arrayToLatLng(coords[i], useGeoJSON);      }      else {        coords[i] = coordsToLatLngs(coords[i], useGeoJSON);      }    }  }  return coords;};var getElementById = function(id, context) {  var element,  id = id.replace('#', '');  if ('jQuery' in this && context) {    element = $("#" + id, context)[0];  } else {    element = document.getElementById(id);  };  return element;};var findAbsolutePosition = function(obj)  {  var curleft = 0,      curtop = 0;  if (obj.offsetParent) {    do {      curleft += obj.offsetLeft;      curtop += obj.offsetTop;    } while (obj = obj.offsetParent);  }  return [curleft, curtop];};var GMaps = (function(global) {  "use strict";  var doc = document;  var GMaps = function(options) {    if (!this) return new GMaps(options);    options.zoom = options.zoom || 15;    options.mapType = options.mapType || 'roadmap';    var self = this,        i,        events_that_hide_context_menu = ['bounds_changed', 'center_changed', 'click', 'dblclick', 'drag', 'dragend', 'dragstart', 'idle', 'maptypeid_changed', 'projection_changed', 'resize', 'tilesloaded', 'zoom_changed'],        events_that_doesnt_hide_context_menu = ['mousemove', 'mouseout', 'mouseover'],        options_to_be_deleted = ['el', 'lat', 'lng', 'mapType', 'width', 'height', 'markerClusterer', 'enableNewStyle'],        container_id = options.el || options.div,        markerClustererFunction = options.markerClusterer,        mapType = google.maps.MapTypeId[options.mapType.toUpperCase()],        map_center = new google.maps.LatLng(options.lat, options.lng),        zoomControl = options.zoomControl || true,        zoomControlOpt = options.zoomControlOpt || {          style: 'DEFAULT',          position: 'TOP_LEFT'        },        zoomControlStyle = zoomControlOpt.style || 'DEFAULT',        zoomControlPosition = zoomControlOpt.position || 'TOP_LEFT',        panControl = options.panControl || true,        mapTypeControl = options.mapTypeControl || true,        scaleControl = options.scaleControl || true,        streetViewControl = options.streetViewControl || true,        overviewMapControl = overviewMapControl || true,        map_options = {},        map_base_options = {          zoom: this.zoom,          center: map_center,          mapTypeId: mapType        },        map_controls_options = {          panControl: panControl,          zoomControl: zoomControl,          zoomControlOptions: {            style: google.maps.ZoomControlStyle[zoomControlStyle],            position: google.maps.ControlPosition[zoomControlPosition]          },          mapTypeControl: mapTypeControl,          scaleControl: scaleControl,          streetViewControl: streetViewControl,          overviewMapControl: overviewMapControl        };    if (typeof(options.el) === 'string' || typeof(options.div) === 'string') {      this.el = getElementById(container_id, options.context);    } else {      this.el = container_id;    }    if (typeof(this.el) === 'undefined' || this.el === null) {      throw 'No element defined.';    }    window.context_menu = window.context_menu || {};    window.context_menu[self.el.id] = {};    this.controls = [];    this.overlays = [];    this.layers = []; // array with kml/georss and fusiontables layers, can be as many    this.singleLayers = {}; // object with the other layers, only one per layer    this.markers = [];    this.polylines = [];    this.routes = [];    this.polygons = [];    this.infoWindow = null;    this.overlay_el = null;    this.zoom = options.zoom;    this.registered_events = {};    this.el.style.width = options.width || this.el.scrollWidth || this.el.offsetWidth;    this.el.style.height = options.height || this.el.scrollHeight || this.el.offsetHeight;    google.maps.visualRefresh = options.enableNewStyle;    for (i = 0; i < options_to_be_deleted.length; i++) {      delete options[options_to_be_deleted[i]];    }    if(options.disableDefaultUI != true) {      map_base_options = extend_object(map_base_options, map_controls_options);    }    map_options = extend_object(map_base_options, options);    for (i = 0; i < events_that_hide_context_menu.length; i++) {      delete map_options[events_that_hide_context_menu[i]];    }    for (i = 0; i < events_that_doesnt_hide_context_menu.length; i++) {      delete map_options[events_that_doesnt_hide_context_menu[i]];    }    this.map = new google.maps.Map(this.el, map_options);    if (markerClustererFunction) {      this.markerClusterer = markerClustererFunction.apply(this, [this.map]);    }    var buildContextMenuHTML = function(control, e) {      var html = '',          options = window.context_menu[self.el.id][control];      for (var i in options){        if (options.hasOwnProperty(i)) {          var option = options[i];          html += '<li><a id="' + control + '_' + i + '" href="#">' + option.title + '</a></li>';        }      }      if (!getElementById('gmaps_context_menu')) return;      var context_menu_element = getElementById('gmaps_context_menu');            context_menu_element.innerHTML = html;      var context_menu_items = context_menu_element.getElementsByTagName('a'),          context_menu_items_count = context_menu_items.length,          i;      for (i = 0; i < context_menu_items_count; i++) {        var context_menu_item = context_menu_items[i];        var assign_menu_item_action = function(ev){          ev.preventDefault();          options[this.id.replace(control + '_', '')].action.apply(self, [e]);          self.hideContextMenu();        };        google.maps.event.clearListeners(context_menu_item, 'click');        google.maps.event.addDomListenerOnce(context_menu_item, 'click', assign_menu_item_action, false);      }      var position = findAbsolutePosition.apply(this, [self.el]),          left = position[0] + e.pixel.x - 15,          top = position[1] + e.pixel.y- 15;      context_menu_element.style.left = left + "px";      context_menu_element.style.top = top + "px";      context_menu_element.style.display = 'block';    };    this.buildContextMenu = function(control, e) {      if (control === 'marker') {        e.pixel = {};        var overlay = new google.maps.OverlayView();        overlay.setMap(self.map);                overlay.draw = function() {          var projection = overlay.getProjection(),              position = e.marker.getPosition();                    e.pixel = projection.fromLatLngToContainerPixel(position);          buildContextMenuHTML(control, e);        };      }      else {        buildContextMenuHTML(control, e);      }    };    this.setContextMenu = function(options) {      window.context_menu[self.el.id][options.control] = {};      var i,          ul = doc.createElement('ul');      for (i in options.options) {        if (options.options.hasOwnProperty(i)) {          var option = options.options[i];          window.context_menu[self.el.id][options.control][option.name] = {            title: option.title,            action: option.action          };        }      }      ul.id = 'gmaps_context_menu';      ul.style.display = 'none';      ul.style.position = 'absolute';      ul.style.minWidth = '100px';      ul.style.background = 'white';      ul.style.listStyle = 'none';      ul.style.padding = '8px';      ul.style.boxShadow = '2px 2px 6px #ccc';      doc.body.appendChild(ul);      var context_menu_element = getElementById('gmaps_context_menu')      google.maps.event.addDomListener(context_menu_element, 'mouseout', function(ev) {        if (!ev.relatedTarget || !this.contains(ev.relatedTarget)) {          window.setTimeout(function(){            context_menu_element.style.display = 'none';          }, 400);        }      }, false);    };    this.hideContextMenu = function() {      var context_menu_element = getElementById('gmaps_context_menu');      if (context_menu_element) {        context_menu_element.style.display = 'none';      }    };    var setupListener = function(object, name) {      google.maps.event.addListener(object, name, function(e){        if (e == undefined) {          e = this;        }        options[name].apply(this, [e]);        self.hideContextMenu();      });    };    //google.maps.event.addListener(this.map, 'idle', this.hideContextMenu);    google.maps.event.addListener(this.map, 'zoom_changed', this.hideContextMenu);    for (var ev = 0; ev < events_that_hide_context_menu.length; ev++) {      var name = events_that_hide_context_menu[ev];      if (name in options) {        setupListener(this.map, name);      }    }    for (var ev = 0; ev < events_that_doesnt_hide_context_menu.length; ev++) {      var name = events_that_doesnt_hide_context_menu[ev];      if (name in options) {        setupListener(this.map, name);      }    }    google.maps.event.addListener(this.map, 'rightclick', function(e) {      if (options.rightclick) {        options.rightclick.apply(this, [e]);      }      if(window.context_menu[self.el.id]['map'] != undefined) {        self.buildContextMenu('map', e);      }    });    this.refresh = function() {      google.maps.event.trigger(this.map, 'resize');    };    this.fitZoom = function() {      var latLngs = [],          markers_length = this.markers.length,          i;      for (i = 0; i < markers_length; i++) {        if(typeof(this.markers[i].visible) === 'boolean' && this.markers[i].visible) {          latLngs.push(this.markers[i].getPosition());        }      }      this.fitLatLngBounds(latLngs);    };    this.fitLatLngBounds = function(latLngs) {      var total = latLngs.length;      var bounds = new google.maps.LatLngBounds();      for(var i=0; i < total; i++) {        bounds.extend(latLngs[i]);      }      this.map.fitBounds(bounds);    };    this.setCenter = function(lat, lng, callback) {      this.map.panTo(new google.maps.LatLng(lat, lng));      if (callback) {        callback();      }    };    this.getElement = function() {      return this.el;    };    this.zoomIn = function(value) {      value = value || 1;      this.zoom = this.map.getZoom() + value;      this.map.setZoom(this.zoom);    };    this.zoomOut = function(value) {      value = value || 1;      this.zoom = this.map.getZoom() - value;      this.map.setZoom(this.zoom);    };    var native_methods = [],        method;    for (method in this.map) {      if (typeof(this.map[method]) == 'function' && !this[method]) {        native_methods.push(method);      }    }    for (i=0; i < native_methods.length; i++) {      (function(gmaps, scope, method_name) {        gmaps[method_name] = function(){          return scope[method_name].apply(scope, arguments);        };      })(this, this.map, native_methods[i]);    }  };  return GMaps;})(this);GMaps.prototype.createControl = function(options) {  var control = document.createElement('div');  control.style.cursor = 'pointer';    if (options.disableDefaultStyles !== true) {    control.style.fontFamily = 'Roboto, Arial, sans-serif';    control.style.fontSize = '11px';    control.style.boxShadow = 'rgba(0, 0, 0, 0.298039) 0px 1px 4px -1px';  }  for (var option in options.style) {    control.style[option] = options.style[option];  }  if (options.id) {    control.id = options.id;  }  if (options.classes) {    control.className = options.classes;  }  if (options.content) {    if (typeof options.content === 'string') {      control.innerHTML = options.content;    }    else if (options.content instanceof HTMLElement) {      control.appendChild(options.content);    }  }  if (options.position) {    control.position = google.maps.ControlPosition[options.position.toUpperCase()];  }  for (var ev in options.events) {    (function(object, name) {      google.maps.event.addDomListener(object, name, function(){        options.events[name].apply(this, [this]);      });    })(control, ev);  }  control.index = 1;  return control;};GMaps.prototype.addControl = function(options) {  var control = this.createControl(options);  this.controls.push(control);  this.map.controls[control.position].push(control);  return control;};GMaps.prototype.removeControl = function(control) {  var position = null;  for (var i = 0; i < this.controls.length; i++) {    if (this.controls[i] == control) {      position = this.controls[i].position;      this.controls.splice(i, 1);    }  }  if (position) {    for (i = 0; i < this.map.controls.length; i++) {      var controlsForPosition = this.map.controls[control.position]      if (controlsForPosition.getAt(i) == control) {        controlsForPosition.removeAt(i);        break;      }    }  }  return control;};GMaps.prototype.createMarker = function(options) {  if (options.lat == undefined && options.lng == undefined && options.position == undefined) {    throw 'No latitude or longitude defined.';  }  var self = this,      details = options.details,      fences = options.fences,      outside = options.outside,      base_options = {        position: new google.maps.LatLng(options.lat, options.lng),        map: null      },      marker_options = extend_object(base_options, options);  delete marker_options.lat;  delete marker_options.lng;  delete marker_options.fences;  delete marker_options.outside;  var marker = new google.maps.Marker(marker_options);  marker.fences = fences;  if (options.infoWindow) {    marker.infoWindow = new google.maps.InfoWindow(options.infoWindow);    var info_window_events = ['closeclick', 'content_changed', 'domready', 'position_changed', 'zindex_changed'];    for (var ev = 0; ev < info_window_events.length; ev++) {      (function(object, name) {        if (options.infoWindow[name]) {          google.maps.event.addListener(object, name, function(e){            options.infoWindow[name].apply(this, [e]);          });        }      })(marker.infoWindow, info_window_events[ev]);    }  }  var marker_events = ['animation_changed', 'clickable_changed', 'cursor_changed', 'draggable_changed', 'flat_changed', 'icon_changed', 'position_changed', 'shadow_changed', 'shape_changed', 'title_changed', 'visible_changed', 'zindex_changed'];  var marker_events_with_mouse = ['dblclick', 'drag', 'dragend', 'dragstart', 'mousedown', 'mouseout', 'mouseover', 'mouseup'];  for (var ev = 0; ev < marker_events.length; ev++) {    (function(object, name) {      if (options[name]) {        google.maps.event.addListener(object, name, function(){          options[name].apply(this, [this]);        });      }    })(marker, marker_events[ev]);  }  for (var ev = 0; ev < marker_events_with_mouse.length; ev++) {    (function(map, object, name) {      if (options[name]) {        google.maps.event.addListener(object, name, function(me){          if(!me.pixel){            me.pixel = map.getProjection().fromLatLngToPoint(me.latLng)          }                    options[name].apply(this, [me]);        });      }    })(this.map, marker, marker_events_with_mouse[ev]);  }  google.maps.event.addListener(marker, 'click', function() {    this.details = details;    if (options.click) {      options.click.apply(this, [this]);    }    if (marker.infoWindow) {      self.hideInfoWindows();      marker.infoWindow.open(self.map, marker);    }  });  google.maps.event.addListener(marker, 'rightclick', function(e) {    e.marker = this;    if (options.rightclick) {      options.rightclick.apply(this, [e]);    }    if (window.context_menu[self.el.id]['marker'] != undefined) {      self.buildContextMenu('marker', e);    }  });  if (marker.fences) {    google.maps.event.addListener(marker, 'dragend', function() {      self.checkMarkerGeofence(marker, function(m, f) {        outside(m, f);      });    });  }  return marker;};GMaps.prototype.addMarker = function(options) {  var marker;  if(options.hasOwnProperty('gm_accessors_')) {    // Native google.maps.Marker object    marker = options;  }  else {    if ((options.hasOwnProperty('lat') && options.hasOwnProperty('lng')) || options.position) {      marker = this.createMarker(options);    }    else {      throw 'No latitude or longitude defined.';    }  }  marker.setMap(this.map);  if(this.markerClusterer) {    this.markerClusterer.addMarker(marker);  }  this.markers.push(marker);  GMaps.fire('marker_added', marker, this);  return marker;};GMaps.prototype.addMarkers = function(array) {  for (var i = 0, marker; marker=array[i]; i++) {    this.addMarker(marker);  }  return this.markers;};GMaps.prototype.hideInfoWindows = function() {  for (var i = 0, marker; marker = this.markers[i]; i++){    if (marker.infoWindow) {      marker.infoWindow.close();    }  }};GMaps.prototype.removeMarker = function(marker) {  for (var i = 0; i < this.markers.length; i++) {    if (this.markers[i] === marker) {      this.markers[i].setMap(null);      this.markers.splice(i, 1);      if(this.markerClusterer) {        this.markerClusterer.removeMarker(marker);      }      GMaps.fire('marker_removed', marker, this);      break;    }  }  return marker;};GMaps.prototype.removeMarkers = function (collection) {  var new_markers = [];  if (typeof collection == 'undefined') {    for (var i = 0; i < this.markers.length; i++) {      this.markers[i].setMap(null);    }        this.markers = new_markers;  }  else {    for (var i = 0; i < collection.length; i++) {      if (this.markers.indexOf(collection[i]) > -1) {        this.markers[i].setMap(null);      }    }    for (var i = 0; i < this.markers.length; i++) {      if (this.markers[i].getMap() != null) {        new_markers.push(this.markers[i]);      }    }    this.markers = new_markers;  }};GMaps.prototype.drawOverlay = function(options) {  var overlay = new google.maps.OverlayView(),      auto_show = true;  overlay.setMap(this.map);  if (options.auto_show != null) {    auto_show = options.auto_show;  }  overlay.onAdd = function() {    var el = document.createElement('div');    el.style.borderStyle = "none";    el.style.borderWidth = "0px";    el.style.position = "absolute";    el.style.zIndex = 100;    el.innerHTML = options.content;    overlay.el = el;    if (!options.layer) {      options.layer = 'overlayLayer';    }        var panes = this.getPanes(),        overlayLayer = panes[options.layer],        stop_overlay_events = ['contextmenu', 'DOMMouseScroll', 'dblclick', 'mousedown'];    overlayLayer.appendChild(el);    for (var ev = 0; ev < stop_overlay_events.length; ev++) {      (function(object, name) {        google.maps.event.addDomListener(object, name, function(e){          if (navigator.userAgent.toLowerCase().indexOf('msie') != -1 && document.all) {            e.cancelBubble = true;            e.returnValue = false;          }          else {            e.stopPropagation();          }        });      })(el, stop_overlay_events[ev]);    }    if (options.click) {      panes.overlayMouseTarget.appendChild(overlay.el);      google.maps.event.addDomListener(overlay.el, 'click', function() {        options.click.apply(overlay, [overlay]);      });    }    google.maps.event.trigger(this, 'ready');  };  overlay.draw = function() {    var projection = this.getProjection(),        pixel = projection.fromLatLngToDivPixel(new google.maps.LatLng(options.lat, options.lng));    options.horizontalOffset = options.horizontalOffset || 0;    options.verticalOffset = options.verticalOffset || 0;    var el = overlay.el,        content = el.children[0],        content_height = content.clientHeight,        content_width = content.clientWidth;    switch (options.verticalAlign) {      case 'top':        el.style.top = (pixel.y - content_height + options.verticalOffset) + 'px';        break;      default:      case 'middle':        el.style.top = (pixel.y - (content_height / 2) + options.verticalOffset) + 'px';        break;      case 'bottom':        el.style.top = (pixel.y + options.verticalOffset) + 'px';        break;    }    switch (options.horizontalAlign) {      case 'left':        el.style.left = (pixel.x - content_width + options.horizontalOffset) + 'px';        break;      default:      case 'center':        el.style.left = (pixel.x - (content_width / 2) + options.horizontalOffset) + 'px';        break;      case 'right':        el.style.left = (pixel.x + options.horizontalOffset) + 'px';        break;    }    el.style.display = auto_show ? 'block' : 'none';    if (!auto_show) {      options.show.apply(this, [el]);    }  };  overlay.onRemove = function() {    var el = overlay.el;    if (options.remove) {      options.remove.apply(this, [el]);    }    else {      overlay.el.parentNode.removeChild(overlay.el);      overlay.el = null;    }  };  this.overlays.push(overlay);  return overlay;};GMaps.prototype.removeOverlay = function(overlay) {  for (var i = 0; i < this.overlays.length; i++) {    if (this.overlays[i] === overlay) {      this.overlays[i].setMap(null);      this.overlays.splice(i, 1);      break;    }  }};GMaps.prototype.removeOverlays = function() {  for (var i = 0, item; item = this.overlays[i]; i++) {    item.setMap(null);  }  this.overlays = [];};GMaps.prototype.drawPolyline = function(options) {  var path = [],      points = options.path;  if (points.length) {    if (points[0][0] === undefined) {      path = points;    }    else {      for (var i=0, latlng; latlng=points[i]; i++) {        path.push(new google.maps.LatLng(latlng[0], latlng[1]));      }    }  }  var polyline_options = {    map: this.map,    path: path,    strokeColor: options.strokeColor,    strokeOpacity: options.strokeOpacity,    strokeWeight: options.strokeWeight,    geodesic: options.geodesic,    clickable: true,    editable: false,    visible: true  };  if (options.hasOwnProperty("clickable")) {    polyline_options.clickable = options.clickable;  }  if (options.hasOwnProperty("editable")) {    polyline_options.editable = options.editable;  }  if (options.hasOwnProperty("icons")) {    polyline_options.icons = options.icons;  }  if (options.hasOwnProperty("zIndex")) {    polyline_options.zIndex = options.zIndex;  }  var polyline = new google.maps.Polyline(polyline_options);  var polyline_events = ['click', 'dblclick', 'mousedown', 'mousemove', 'mouseout', 'mouseover', 'mouseup', 'rightclick'];  for (var ev = 0; ev < polyline_events.length; ev++) {    (function(object, name) {      if (options[name]) {        google.maps.event.addListener(object, name, function(e){          options[name].apply(this, [e]);        });      }    })(polyline, polyline_events[ev]);  }  this.polylines.push(polyline);  GMaps.fire('polyline_added', polyline, this);  return polyline;};GMaps.prototype.removePolyline = function(polyline) {  for (var i = 0; i < this.polylines.length; i++) {    if (this.polylines[i] === polyline) {      this.polylines[i].setMap(null);      this.polylines.splice(i, 1);      GMaps.fire('polyline_removed', polyline, this);      break;    }  }};GMaps.prototype.removePolylines = function() {  for (var i = 0, item; item = this.polylines[i]; i++) {    item.setMap(null);  }  this.polylines = [];};GMaps.prototype.drawCircle = function(options) {  options =  extend_object({    map: this.map,    center: new google.maps.LatLng(options.lat, options.lng)  }, options);  delete options.lat;  delete options.lng;  var polygon = new google.maps.Circle(options),      polygon_events = ['click', 'dblclick', 'mousedown', 'mousemove', 'mouseout', 'mouseover', 'mouseup', 'rightclick'];  for (var ev = 0; ev < polygon_events.length; ev++) {    (function(object, name) {      if (options[name]) {        google.maps.event.addListener(object, name, function(e){          options[name].apply(this, [e]);        });      }    })(polygon, polygon_events[ev]);  }  this.polygons.push(polygon);  return polygon;};GMaps.prototype.drawRectangle = function(options) {  options = extend_object({    map: this.map  }, options);  var latLngBounds = new google.maps.LatLngBounds(    new google.maps.LatLng(options.bounds[0][0], options.bounds[0][1]),    new google.maps.LatLng(options.bounds[1][0], options.bounds[1][1])  );  options.bounds = latLngBounds;  var polygon = new google.maps.Rectangle(options),      polygon_events = ['click', 'dblclick', 'mousedown', 'mousemove', 'mouseout', 'mouseover', 'mouseup', 'rightclick'];  for (var ev = 0; ev < polygon_events.length; ev++) {    (function(object, name) {      if (options[name]) {        google.maps.event.addListener(object, name, function(e){          options[name].apply(this, [e]);        });      }    })(polygon, polygon_events[ev]);  }  this.polygons.push(polygon);  return polygon;};GMaps.prototype.drawPolygon = function(options) {  var useGeoJSON = false;  if(options.hasOwnProperty("useGeoJSON")) {    useGeoJSON = options.useGeoJSON;  }  delete options.useGeoJSON;  options = extend_object({    map: this.map  }, options);  if (useGeoJSON == false) {    options.paths = [options.paths.slice(0)];  }  if (options.paths.length > 0) {    if (options.paths[0].length > 0) {      options.paths = array_flat(array_map(options.paths, arrayToLatLng, useGeoJSON));    }  }  var polygon = new google.maps.Polygon(options),      polygon_events = ['click', 'dblclick', 'mousedown', 'mousemove', 'mouseout', 'mouseover', 'mouseup', 'rightclick'];  for (var ev = 0; ev < polygon_events.length; ev++) {    (function(object, name) {      if (options[name]) {        google.maps.event.addListener(object, name, function(e){          options[name].apply(this, [e]);        });      }    })(polygon, polygon_events[ev]);  }  this.polygons.push(polygon);  GMaps.fire('polygon_added', polygon, this);  return polygon;};GMaps.prototype.removePolygon = function(polygon) {  for (var i = 0; i < this.polygons.length; i++) {    if (this.polygons[i] === polygon) {      this.polygons[i].setMap(null);      this.polygons.splice(i, 1);      GMaps.fire('polygon_removed', polygon, this);      break;    }  }};GMaps.prototype.removePolygons = function() {  for (var i = 0, item; item = this.polygons[i]; i++) {    item.setMap(null);  }  this.polygons = [];};GMaps.prototype.getFromFusionTables = function(options) {  var events = options.events;  delete options.events;  var fusion_tables_options = options,      layer = new google.maps.FusionTablesLayer(fusion_tables_options);  for (var ev in events) {    (function(object, name) {      google.maps.event.addListener(object, name, function(e) {        events[name].apply(this, [e]);      });    })(layer, ev);  }  this.layers.push(layer);  return layer;};GMaps.prototype.loadFromFusionTables = function(options) {  var layer = this.getFromFusionTables(options);  layer.setMap(this.map);  return layer;};GMaps.prototype.getFromKML = function(options) {  var url = options.url,      events = options.events;  delete options.url;  delete options.events;  var kml_options = options,      layer = new google.maps.KmlLayer(url, kml_options);  for (var ev in events) {    (function(object, name) {      google.maps.event.addListener(object, name, function(e) {        events[name].apply(this, [e]);      });    })(layer, ev);  }  this.layers.push(layer);  return layer;};GMaps.prototype.loadFromKML = function(options) {  var layer = this.getFromKML(options);  layer.setMap(this.map);  return layer;};GMaps.prototype.addLayer = function(layerName, options) {  //var default_layers = ['weather', 'clouds', 'traffic', 'transit', 'bicycling', 'panoramio', 'places'];  options = options || {};  var layer;  switch(layerName) {    case 'weather': this.singleLayers.weather = layer = new google.maps.weather.WeatherLayer();      break;    case 'clouds': this.singleLayers.clouds = layer = new google.maps.weather.CloudLayer();      break;    case 'traffic': this.singleLayers.traffic = layer = new google.maps.TrafficLayer();      break;    case 'transit': this.singleLayers.transit = layer = new google.maps.TransitLayer();      break;    case 'bicycling': this.singleLayers.bicycling = layer = new google.maps.BicyclingLayer();      break;    case 'panoramio':        this.singleLayers.panoramio = layer = new google.maps.panoramio.PanoramioLayer();        layer.setTag(options.filter);        delete options.filter;        //click event        if (options.click) {          google.maps.event.addListener(layer, 'click', function(event) {            options.click(event);            delete options.click;          });        }      break;      case 'places':        this.singleLayers.places = layer = new google.maps.places.PlacesService(this.map);        //search, nearbySearch, radarSearch callback, Both are the same        if (options.search || options.nearbySearch || options.radarSearch) {          var placeSearchRequest  = {            bounds : options.bounds || null,            keyword : options.keyword || null,            location : options.location || null,            name : options.name || null,            radius : options.radius || null,            rankBy : options.rankBy || null,            types : options.types || null          };          if (options.radarSearch) {            layer.radarSearch(placeSearchRequest, options.radarSearch);          }          if (options.search) {            layer.search(placeSearchRequest, options.search);          }          if (options.nearbySearch) {            layer.nearbySearch(placeSearchRequest, options.nearbySearch);          }        }        //textSearch callback        if (options.textSearch) {          var textSearchRequest  = {            bounds : options.bounds || null,            location : options.location || null,            query : options.query || null,            radius : options.radius || null          };          layer.textSearch(textSearchRequest, options.textSearch);        }      break;  }  if (layer !== undefined) {    if (typeof layer.setOptions == 'function') {      layer.setOptions(options);    }    if (typeof layer.setMap == 'function') {      layer.setMap(this.map);    }    return layer;  }};GMaps.prototype.removeLayer = function(layer) {  if (typeof(layer) == "string" && this.singleLayers[layer] !== undefined) {     this.singleLayers[layer].setMap(null);     delete this.singleLayers[layer];  }  else {    for (var i = 0; i < this.layers.length; i++) {      if (this.layers[i] === layer) {        this.layers[i].setMap(null);        this.layers.splice(i, 1);        break;      }    }  }};var travelMode, unitSystem;GMaps.prototype.getRoutes = function(options) {  switch (options.travelMode) {    case 'bicycling':      travelMode = google.maps.TravelMode.BICYCLING;      break;    case 'transit':      travelMode = google.maps.TravelMode.TRANSIT;      break;    case 'driving':      travelMode = google.maps.TravelMode.DRIVING;      break;    default:      travelMode = google.maps.TravelMode.WALKING;      break;  }  if (options.unitSystem === 'imperial') {    unitSystem = google.maps.UnitSystem.IMPERIAL;  }  else {    unitSystem = google.maps.UnitSystem.METRIC;  }  var base_options = {        avoidHighways: false,        avoidTolls: false,        optimizeWaypoints: false,        waypoints: []      },      request_options =  extend_object(base_options, options);  request_options.origin = /string/.test(typeof options.origin) ? options.origin : new google.maps.LatLng(options.origin[0], options.origin[1]);  request_options.destination = /string/.test(typeof options.destination) ? options.destination : new google.maps.LatLng(options.destination[0], options.destination[1]);  request_options.travelMode = travelMode;  request_options.unitSystem = unitSystem;  delete request_options.callback;  delete request_options.error;  var self = this,      service = new google.maps.DirectionsService();  service.route(request_options, function(result, status) {    if (status === google.maps.DirectionsStatus.OK) {      for (var r in result.routes) {        if (result.routes.hasOwnProperty(r)) {          self.routes.push(result.routes[r]);        }      }      if (options.callback) {        options.callback(self.routes);      }    }    else {      if (options.error) {        options.error(result, status);      }    }  });};GMaps.prototype.removeRoutes = function() {  this.routes = [];};GMaps.prototype.getElevations = function(options) {  options = extend_object({    locations: [],    path : false,    samples : 256  }, options);  if (options.locations.length > 0) {    if (options.locations[0].length > 0) {      options.locations = array_flat(array_map([options.locations], arrayToLatLng,  false));    }  }  var callback = options.callback;  delete options.callback;  var service = new google.maps.ElevationService();  //location request  if (!options.path) {    delete options.path;    delete options.samples;    service.getElevationForLocations(options, function(result, status) {      if (callback && typeof(callback) === "function") {        callback(result, status);      }    });  //path request  } else {    var pathRequest = {      path : options.locations,      samples : options.samples    };    service.getElevationAlongPath(pathRequest, function(result, status) {     if (callback && typeof(callback) === "function") {        callback(result, status);      }    });  }};GMaps.prototype.cleanRoute = GMaps.prototype.removePolylines;GMaps.prototype.drawRoute = function(options) {  var self = this;  this.getRoutes({    origin: options.origin,    destination: options.destination,    travelMode: options.travelMode,    waypoints: options.waypoints,    unitSystem: options.unitSystem,    error: options.error,    callback: function(e) {      if (e.length > 0) {        self.drawPolyline({          path: e[e.length - 1].overview_path,          strokeColor: options.strokeColor,          strokeOpacity: options.strokeOpacity,          strokeWeight: options.strokeWeight        });                if (options.callback) {          options.callback(e[e.length - 1]);        }      }    }  });};GMaps.prototype.travelRoute = function(options) {  if (options.origin && options.destination) {    this.getRoutes({      origin: options.origin,      destination: options.destination,      travelMode: options.travelMode,      waypoints : options.waypoints,      unitSystem: options.unitSystem,      error: options.error,      callback: function(e) {        //start callback        if (e.length > 0 && options.start) {          options.start(e[e.length - 1]);        }        //step callback        if (e.length > 0 && options.step) {          var route = e[e.length - 1];          if (route.legs.length > 0) {            var steps = route.legs[0].steps;            for (var i=0, step; step=steps[i]; i++) {              step.step_number = i;              options.step(step, (route.legs[0].steps.length - 1));            }          }        }        //end callback        if (e.length > 0 && options.end) {           options.end(e[e.length - 1]);        }      }    });  }  else if (options.route) {    if (options.route.legs.length > 0) {      var steps = options.route.legs[0].steps;      for (var i=0, step; step=steps[i]; i++) {        step.step_number = i;        options.step(step);      }    }  }};GMaps.prototype.drawSteppedRoute = function(options) {  var self = this;    if (options.origin && options.destination) {    this.getRoutes({      origin: options.origin,      destination: options.destination,      travelMode: options.travelMode,      waypoints : options.waypoints,      error: options.error,      callback: function(e) {        //start callback        if (e.length > 0 && options.start) {          options.start(e[e.length - 1]);        }        //step callback        if (e.length > 0 && options.step) {          var route = e[e.length - 1];          if (route.legs.length > 0) {            var steps = route.legs[0].steps;            for (var i=0, step; step=steps[i]; i++) {              step.step_number = i;              self.drawPolyline({                path: step.path,                strokeColor: options.strokeColor,                strokeOpacity: options.strokeOpacity,                strokeWeight: options.strokeWeight              });              options.step(step, (route.legs[0].steps.length - 1));            }          }        }        //end callback        if (e.length > 0 && options.end) {           options.end(e[e.length - 1]);        }      }    });  }  else if (options.route) {    if (options.route.legs.length > 0) {      var steps = options.route.legs[0].steps;      for (var i=0, step; step=steps[i]; i++) {        step.step_number = i;        self.drawPolyline({          path: step.path,          strokeColor: options.strokeColor,          strokeOpacity: options.strokeOpacity,          strokeWeight: options.strokeWeight        });        options.step(step);      }    }  }};GMaps.Route = function(options) {  this.origin = options.origin;  this.destination = options.destination;  this.waypoints = options.waypoints;  this.map = options.map;  this.route = options.route;  this.step_count = 0;  this.steps = this.route.legs[0].steps;  this.steps_length = this.steps.length;  this.polyline = this.map.drawPolyline({    path: new google.maps.MVCArray(),    strokeColor: options.strokeColor,    strokeOpacity: options.strokeOpacity,    strokeWeight: options.strokeWeight  }).getPath();};GMaps.Route.prototype.getRoute = function(options) {  var self = this;  this.map.getRoutes({    origin : this.origin,    destination : this.destination,    travelMode : options.travelMode,    waypoints : this.waypoints || [],    error: options.error,    callback : function() {      self.route = e[0];      if (options.callback) {        options.callback.call(self);      }    }  });};GMaps.Route.prototype.back = function() {  if (this.step_count > 0) {    this.step_count--;    var path = this.route.legs[0].steps[this.step_count].path;    for (var p in path){      if (path.hasOwnProperty(p)){        this.polyline.pop();      }    }  }};GMaps.Route.prototype.forward = function() {  if (this.step_count < this.steps_length) {    var path = this.route.legs[0].steps[this.step_count].path;    for (var p in path){      if (path.hasOwnProperty(p)){        this.polyline.push(path[p]);      }    }    this.step_count++;  }};GMaps.prototype.checkGeofence = function(lat, lng, fence) {  return fence.containsLatLng(new google.maps.LatLng(lat, lng));};GMaps.prototype.checkMarkerGeofence = function(marker, outside_callback) {  if (marker.fences) {    for (var i = 0, fence; fence = marker.fences[i]; i++) {      var pos = marker.getPosition();      if (!this.checkGeofence(pos.lat(), pos.lng(), fence)) {        outside_callback(marker, fence);      }    }  }};GMaps.prototype.toImage = function(options) {  var options = options || {},      static_map_options = {};  static_map_options['size'] = options['size'] || [this.el.clientWidth, this.el.clientHeight];  static_map_options['lat'] = this.getCenter().lat();  static_map_options['lng'] = this.getCenter().lng();  if (this.markers.length > 0) {    static_map_options['markers'] = [];        for (var i = 0; i < this.markers.length; i++) {      static_map_options['markers'].push({        lat: this.markers[i].getPosition().lat(),        lng: this.markers[i].getPosition().lng()      });    }  }  if (this.polylines.length > 0) {    var polyline = this.polylines[0];        static_map_options['polyline'] = {};    static_map_options['polyline']['path'] = google.maps.geometry.encoding.encodePath(polyline.getPath());    static_map_options['polyline']['strokeColor'] = polyline.strokeColor    static_map_options['polyline']['strokeOpacity'] = polyline.strokeOpacity    static_map_options['polyline']['strokeWeight'] = polyline.strokeWeight  }  return GMaps.staticMapURL(static_map_options);};GMaps.staticMapURL = function(options){  var parameters = [],      data,      static_root = 'https://maps.googleapis.com/maps/api/staticmap';  if (options.url) {    static_root = options.url;    delete options.url;  }  static_root += '?';  var markers = options.markers;    delete options.markers;  if (!markers && options.marker) {    markers = [options.marker];    delete options.marker;  }  var styles = options.styles;  delete options.styles;  var polyline = options.polyline;  delete options.polyline;  /** Map options **/  if (options.center) {    parameters.push('center=' + options.center);    delete options.center;  }  else if (options.address) {    parameters.push('center=' + options.address);    delete options.address;  }  else if (options.lat) {    parameters.push(['center=', options.lat, ',', options.lng].join(''));    delete options.lat;    delete options.lng;  }  else if (options.visible) {    var visible = encodeURI(options.visible.join('|'));    parameters.push('visible=' + visible);  }  var size = options.size;  if (size) {    if (size.join) {      size = size.join('x');    }    delete options.size;  }  else {    size = '630x300';  }  parameters.push('size=' + size);  if (!options.zoom && options.zoom !== false) {    options.zoom = 15;  }  var sensor = options.hasOwnProperty('sensor') ? !!options.sensor : true;  delete options.sensor;  parameters.push('sensor=' + sensor);  for (var param in options) {    if (options.hasOwnProperty(param)) {      parameters.push(param + '=' + options[param]);    }  }  /** Markers **/  if (markers) {    var marker, loc;    for (var i=0; data=markers[i]; i++) {      marker = [];      if (data.size && data.size !== 'normal') {        marker.push('size:' + data.size);        delete data.size;      }      else if (data.icon) {        marker.push('icon:' + encodeURI(data.icon));        delete data.icon;      }      if (data.color) {        marker.push('color:' + data.color.replace('#', '0x'));        delete data.color;      }      if (data.label) {        marker.push('label:' + data.label[0].toUpperCase());        delete data.label;      }      loc = (data.address ? data.address : data.lat + ',' + data.lng);      delete data.address;      delete data.lat;      delete data.lng;      for(var param in data){        if (data.hasOwnProperty(param)) {          marker.push(param + ':' + data[param]);        }      }      if (marker.length || i === 0) {        marker.push(loc);        marker = marker.join('|');        parameters.push('markers=' + encodeURI(marker));      }      // New marker without styles      else {        marker = parameters.pop() + encodeURI('|' + loc);        parameters.push(marker);      }    }  }  /** Map Styles **/  if (styles) {    for (var i = 0; i < styles.length; i++) {      var styleRule = [];      if (styles[i].featureType){        styleRule.push('feature:' + styles[i].featureType.toLowerCase());      }      if (styles[i].elementType) {        styleRule.push('element:' + styles[i].elementType.toLowerCase());      }      for (var j = 0; j < styles[i].stylers.length; j++) {        for (var p in styles[i].stylers[j]) {          var ruleArg = styles[i].stylers[j][p];          if (p == 'hue' || p == 'color') {            ruleArg = '0x' + ruleArg.substring(1);          }          styleRule.push(p + ':' + ruleArg);        }      }      var rule = styleRule.join('|');      if (rule != '') {        parameters.push('style=' + rule);      }    }  }  /** Polylines **/  function parseColor(color, opacity) {    if (color[0] === '#'){      color = color.replace('#', '0x');      if (opacity) {        opacity = parseFloat(opacity);        opacity = Math.min(1, Math.max(opacity, 0));        if (opacity === 0) {          return '0x00000000';        }        opacity = (opacity * 255).toString(16);        if (opacity.length === 1) {          opacity += opacity;        }        color = color.slice(0,8) + opacity;      }    }    return color;  }  if (polyline) {    data = polyline;    polyline = [];    if (data.strokeWeight) {      polyline.push('weight:' + parseInt(data.strokeWeight, 10));    }    if (data.strokeColor) {      var color = parseColor(data.strokeColor, data.strokeOpacity);      polyline.push('color:' + color);    }    if (data.fillColor) {      var fillcolor = parseColor(data.fillColor, data.fillOpacity);      polyline.push('fillcolor:' + fillcolor);    }    var path = data.path;    if (path.join) {      for (var j=0, pos; pos=path[j]; j++) {        polyline.push(pos.join(','));      }    }    else {      polyline.push('enc:' + path);    }    polyline = polyline.join('|');    parameters.push('path=' + encodeURI(polyline));  }  /** Retina support **/  var dpi = window.devicePixelRatio || 1;  parameters.push('scale=' + dpi);  parameters = parameters.join('&');  return static_root + parameters;};GMaps.prototype.addMapType = function(mapTypeId, options) {  if (options.hasOwnProperty("getTileUrl") && typeof(options["getTileUrl"]) == "function") {    options.tileSize = options.tileSize || new google.maps.Size(256, 256);    var mapType = new google.maps.ImageMapType(options);    this.map.mapTypes.set(mapTypeId, mapType);  }  else {    throw "'getTileUrl' function required.";  }};GMaps.prototype.addOverlayMapType = function(options) {  if (options.hasOwnProperty("getTile") && typeof(options["getTile"]) == "function") {    var overlayMapTypeIndex = options.index;    delete options.index;    this.map.overlayMapTypes.insertAt(overlayMapTypeIndex, options);  }  else {    throw "'getTile' function required.";  }};GMaps.prototype.removeOverlayMapType = function(overlayMapTypeIndex) {  this.map.overlayMapTypes.removeAt(overlayMapTypeIndex);};GMaps.prototype.addStyle = function(options) {  var styledMapType = new google.maps.StyledMapType(options.styles, { name: options.styledMapName });  this.map.mapTypes.set(options.mapTypeId, styledMapType);};GMaps.prototype.setStyle = function(mapTypeId) {  this.map.setMapTypeId(mapTypeId);};GMaps.prototype.createPanorama = function(streetview_options) {  if (!streetview_options.hasOwnProperty('lat') || !streetview_options.hasOwnProperty('lng')) {    streetview_options.lat = this.getCenter().lat();    streetview_options.lng = this.getCenter().lng();  }  this.panorama = GMaps.createPanorama(streetview_options);  this.map.setStreetView(this.panorama);  return this.panorama;};GMaps.createPanorama = function(options) {  var el = getElementById(options.el, options.context);  options.position = new google.maps.LatLng(options.lat, options.lng);  delete options.el;  delete options.context;  delete options.lat;  delete options.lng;  var streetview_events = ['closeclick', 'links_changed', 'pano_changed', 'position_changed', 'pov_changed', 'resize', 'visible_changed'],      streetview_options = extend_object({visible : true}, options);  for (var i = 0; i < streetview_events.length; i++) {    delete streetview_options[streetview_events[i]];  }  var panorama = new google.maps.StreetViewPanorama(el, streetview_options);  for (var i = 0; i < streetview_events.length; i++) {    (function(object, name) {      if (options[name]) {        google.maps.event.addListener(object, name, function(){          options[name].apply(this);        });      }    })(panorama, streetview_events[i]);  }  return panorama;};GMaps.prototype.on = function(event_name, handler) {  return GMaps.on(event_name, this, handler);};GMaps.prototype.off = function(event_name) {  GMaps.off(event_name, this);};GMaps.custom_events = ['marker_added', 'marker_removed', 'polyline_added', 'polyline_removed', 'polygon_added', 'polygon_removed', 'geolocated', 'geolocation_failed'];GMaps.on = function(event_name, object, handler) {  if (GMaps.custom_events.indexOf(event_name) == -1) {    if(object instanceof GMaps) object = object.map;     return google.maps.event.addListener(object, event_name, handler);  }  else {    var registered_event = {      handler : handler,      eventName : event_name    };    object.registered_events[event_name] = object.registered_events[event_name] || [];    object.registered_events[event_name].push(registered_event);    return registered_event;  }};GMaps.off = function(event_name, object) {  if (GMaps.custom_events.indexOf(event_name) == -1) {    if(object instanceof GMaps) object = object.map;     google.maps.event.clearListeners(object, event_name);  }  else {    object.registered_events[event_name] = [];  }};GMaps.fire = function(event_name, object, scope) {  if (GMaps.custom_events.indexOf(event_name) == -1) {    google.maps.event.trigger(object, event_name, Array.prototype.slice.apply(arguments).slice(2));  }  else {    if(event_name in scope.registered_events) {      var firing_events = scope.registered_events[event_name];      for(var i = 0; i < firing_events.length; i++) {        (function(handler, scope, object) {          handler.apply(scope, [object]);        })(firing_events[i]['handler'], scope, object);      }    }  }};GMaps.geolocate = function(options) {  var complete_callback = options.always || options.complete;  if (navigator.geolocation) {    navigator.geolocation.getCurrentPosition(function(position) {      options.success(position);      if (complete_callback) {        complete_callback();      }    }, function(error) {      options.error(error);      if (complete_callback) {        complete_callback();      }    }, options.options);  }  else {    options.not_supported();    if (complete_callback) {      complete_callback();    }  }};GMaps.geocode = function(options) {  this.geocoder = new google.maps.Geocoder();  var callback = options.callback;  if (options.hasOwnProperty('lat') && options.hasOwnProperty('lng')) {    options.latLng = new google.maps.LatLng(options.lat, options.lng);  }  delete options.lat;  delete options.lng;  delete options.callback;    this.geocoder.geocode(options, function(results, status) {    callback(results, status);  });};//==========================// Polygon containsLatLng// https://github.com/tparkin/Google-Maps-Point-in-Polygon// Poygon getBounds extension - google-maps-extensions// http://code.google.com/p/google-maps-extensions/source/browse/google.maps.Polygon.getBounds.jsif (!google.maps.Polygon.prototype.getBounds) {  google.maps.Polygon.prototype.getBounds = function(latLng) {    var bounds = new google.maps.LatLngBounds();    var paths = this.getPaths();    var path;    for (var p = 0; p < paths.getLength(); p++) {      path = paths.getAt(p);      for (var i = 0; i < path.getLength(); i++) {        bounds.extend(path.getAt(i));      }    }    return bounds;  };}if (!google.maps.Polygon.prototype.containsLatLng) {  // Polygon containsLatLng - method to determine if a latLng is within a polygon  google.maps.Polygon.prototype.containsLatLng = function(latLng) {    // Exclude points outside of bounds as there is no way they are in the poly    var bounds = this.getBounds();    if (bounds !== null && !bounds.contains(latLng)) {      return false;    }    // Raycast point in polygon method    var inPoly = false;    var numPaths = this.getPaths().getLength();    for (var p = 0; p < numPaths; p++) {      var path = this.getPaths().getAt(p);      var numPoints = path.getLength();      var j = numPoints - 1;      for (var i = 0; i < numPoints; i++) {        var vertex1 = path.getAt(i);        var vertex2 = path.getAt(j);        if (vertex1.lng() < latLng.lng() && vertex2.lng() >= latLng.lng() || vertex2.lng() < latLng.lng() && vertex1.lng() >= latLng.lng()) {          if (vertex1.lat() + (latLng.lng() - vertex1.lng()) / (vertex2.lng() - vertex1.lng()) * (vertex2.lat() - vertex1.lat()) < latLng.lat()) {            inPoly = !inPoly;          }        }        j = i;      }    }    return inPoly;  };}if (!google.maps.Circle.prototype.containsLatLng) {  google.maps.Circle.prototype.containsLatLng = function(latLng) {    if (google.maps.geometry) {      return google.maps.geometry.spherical.computeDistanceBetween(this.getCenter(), latLng) <= this.getRadius();    }    else {      return true;    }  };}google.maps.LatLngBounds.prototype.containsLatLng = function(latLng) {  return this.contains(latLng);};google.maps.Marker.prototype.setFences = function(fences) {  this.fences = fences;};google.maps.Marker.prototype.addFence = function(fence) {  this.fences.push(fence);};google.maps.Marker.prototype.getId = function() {  return this['__gm_id'];};//==========================// Array indexOf// https://developer.mozilla.org/en-US/docs/JavaScript/Reference/Global_Objects/Array/indexOfif (!Array.prototype.indexOf) {  Array.prototype.indexOf = function (searchElement /*, fromIndex */ ) {      "use strict";      if (this == null) {          throw new TypeError();      }      var t = Object(this);      var len = t.length >>> 0;      if (len === 0) {          return -1;      }      var n = 0;      if (arguments.length > 1) {          n = Number(arguments[1]);          if (n != n) { // shortcut for verifying if it's NaN              n = 0;          } else if (n != 0 && n != Infinity && n != -Infinity) {              n = (n > 0 || -1) * Math.floor(Math.abs(n));          }      }      if (n >= len) {          return -1;      }      var k = n >= 0 ? n : Math.max(len - Math.abs(n), 0);      for (; k < len; k++) {          if (k in t && t[k] === searchElement) {              return k;          }      }      return -1;  }}  return GMaps;}));
 |