UNPKG

google-react-maps

Version:

A more powerfully custom version of the Google Maps Javascript API built for React. Multiple Datalayer support. GEOJSON Enabled.

11 lines (10 loc) 152 kB
<html> <head> </head> <body style="background: transparent;"> <script src="scripts/docstrap.lib.js"></script> <script src="scripts/lunr.min.js"></script> <script src="scripts/fulltext-search.js"></script> <script type="text/x-docstrap-searchdb"> {"components_circle.js.html":{"id":"components_circle.js.html","title":"Source: components/circle.js","body":" Google React Maps Namespaces google.mapsUtils Classes CircleClusterClusterIconInfoClusterIconStyleFeaturegoogle.maps.LatLngLiteralMapMap.InfoWindowMap.MarkerMap.MarkerClusterMapControlMarkerClustererMarkerClustererOptions Events MarkerClusterer#event:clickMarkerClusterer#event:clusteringbeginMarkerClusterer#event:clusteringendMarkerClusterer#event:mouseoutMarkerClusterer#event:mouseover Global processPoints Source: components/circle.js import React from 'react'; import PropTypes from 'prop-types'; import {mapChildren} from '../utils/utils'; /** Defines a circle shape in the map. Relates to google.maps.Circle class. * @property {object} props * @property {number} props.radius * @property {object} props.center * @property {number} props.center.lat * @property {number} props.center.lng * @property {function} props.onRadiusChange * @property {function} props.onCenterChange */ class Circle extends React.Component { constructor(props) { super(props); this.displayName = 'Circle'; this.setupListeners = this.setupListeners.bind(this); this.state = {circle : null}; this.focus = this.focus.bind(this); } /** Focus the map on this circle. */ focus() { var {circle} = this.state; var {map} = this.state; if(circle &amp;&amp; map) { map.fitBounds(circle.getBounds()); } } /** Setup the listeners for this circle */ setupListeners() { var {circle} = this.state; var {maps, map} = this.props; if(maps &amp;&amp; circle) { maps.event.addListener(circle, 'radius_changed',e =&gt; { if(typeof this.props.onRadiusChange === 'function') this.props.onRadiusChange(circle.getRadius()); }); maps.event.addListener(circle, 'center_changed', e =&gt; { if(typeof this.props.onCenterChange === 'function') this.props.onCenterChange(circle.getCenter().toJSON()); }); maps.event.addListener(circle, 'click', e =&gt; { if(typeof this.props.onClick === 'function') this.props.onClick(Object.assign({coords : circle.getCenter().toJSON()},e)); }); maps.event.addListener(circle, 'rightclick', ({latLng}) =&gt; typeof this.props.onRightClick === 'function' ? this.props.onRightClick({coords:latLng.toJSON()}) : f =&gt; f); } else console.error(new Error(&quot;You must pass maps and map to this component. Otherwise, run it inside a map component.&quot;)); } componentDidUpdate(prevProps, prevState) { var currCenter = new this.props.maps.LatLng(this.props.center); var prevCenter = this.state.circle.getCenter(); if(!currCenter.equals(prevCenter)) this.state.circle.setCenter(currCenter); if(this.props.radius != this.state.circle.getRadius()) this.state.circle.setRadius(this.props.radius); } componentWillMount() { var { map, maps, } = this.props; if(map &amp;&amp; maps) { var CircleOptions = Object.assign({}, this.props, {maps : undefined}); var circle = new maps.Circle(CircleOptions); this.setState({circle}, this.setupListeners); } } componentWillUnmount() { if(this.state.circle) { this.props.maps.event.clearListeners(this.state.circle); this.state.circle.setMap(null); } } render() { var children = []; if(this.props.children &amp;&amp; this.props.maps &amp;&amp; this.props.map) children = mapChildren({ coords : this.state.circle.getCenter().toJSON() }, this); return &lt;div&gt;{children}&lt;/div&gt;; } } var {number, shape} = PropTypes; Circle.propTypes = { radius : number.isRequired, center : shape({ lat : number, lng : number }).isRequired } export default Circle; × Search results Close "},"utils_v3-utility-library-master_markerclustererplus_src_markerclusterer.js.html":{"id":"utils_v3-utility-library-master_markerclustererplus_src_markerclusterer.js.html","title":"Source: utils/v3-utility-library-master/markerclustererplus/src/markerclusterer.js","body":" Google React Maps Namespaces google.mapsUtils Classes CircleClusterClusterIconInfoClusterIconStyleFeaturegoogle.maps.LatLngLiteralMapMap.InfoWindowMap.MarkerMap.MarkerClusterMapControlMarkerClustererMarkerClustererOptions Events MarkerClusterer#event:clickMarkerClusterer#event:clusteringbeginMarkerClusterer#event:clusteringendMarkerClusterer#event:mouseoutMarkerClusterer#event:mouseover Global processPoints Source: utils/v3-utility-library-master/markerclustererplus/src/markerclusterer.js /** * @name ClusterIconStyle * @class This class represents the object for values in the &lt;code&gt;styles&lt;/code&gt; array passed * to the {@link MarkerClusterer} constructor. The element in this array that is used to * style the cluster icon is determined by calling the &lt;code&gt;calculator&lt;/code&gt; function. * * @property {string} url The URL of the cluster icon image file. Required. * @property {number} height The display height (in pixels) of the cluster icon. Required. * @property {number} width The display width (in pixels) of the cluster icon. Required. * @property {Array} [anchorText] The position (in pixels) from the center of the cluster icon to * where the text label is to be centered and drawn. The format is &lt;code&gt;[yoffset, xoffset]&lt;/code&gt; * where &lt;code&gt;yoffset&lt;/code&gt; increases as you go down from center and &lt;code&gt;xoffset&lt;/code&gt; * increases to the right of center. The default is &lt;code&gt;[0, 0]&lt;/code&gt;. * @property {Array} [anchorIcon] The anchor position (in pixels) of the cluster icon. This is the * spot on the cluster icon that is to be aligned with the cluster position. The format is * &lt;code&gt;[yoffset, xoffset]&lt;/code&gt; where &lt;code&gt;yoffset&lt;/code&gt; increases as you go down and * &lt;code&gt;xoffset&lt;/code&gt; increases to the right of the top-left corner of the icon. The default * anchor position is the center of the cluster icon. * @property {string} [textColor=&quot;black&quot;] The color of the label text shown on the * cluster icon. * @property {number} [textSize=11] The size (in pixels) of the label text shown on the * cluster icon. * @property {string} [textDecoration=&quot;none&quot;] The value of the CSS &lt;code&gt;text-decoration&lt;/code&gt; * property for the label text shown on the cluster icon. * @property {string} [fontWeight=&quot;bold&quot;] The value of the CSS &lt;code&gt;font-weight&lt;/code&gt; * property for the label text shown on the cluster icon. * @property {string} [fontStyle=&quot;normal&quot;] The value of the CSS &lt;code&gt;font-style&lt;/code&gt; * property for the label text shown on the cluster icon. * @property {string} [fontFamily=&quot;Arial,sans-serif&quot;] The value of the CSS &lt;code&gt;font-family&lt;/code&gt; * property for the label text shown on the cluster icon. * @property {string} [backgroundPosition=&quot;0 0&quot;] The position of the cluster icon image * within the image defined by &lt;code&gt;url&lt;/code&gt;. The format is &lt;code&gt;&quot;xpos ypos&quot;&lt;/code&gt; * (the same format as for the CSS &lt;code&gt;background-position&lt;/code&gt; property). You must set * this property appropriately when the image defined by &lt;code&gt;url&lt;/code&gt; represents a sprite * containing multiple images. Note that the position &lt;i&gt;must&lt;/i&gt; be specified in px units. */ /** * @name ClusterIconInfo * @class This class is an object containing general information about a cluster icon. This is * the object that a &lt;code&gt;calculator&lt;/code&gt; function returns. * * @property {string} text The text of the label to be shown on the cluster icon. * @property {number} index The index plus 1 of the element in the &lt;code&gt;styles&lt;/code&gt; * array to be used to style the cluster icon. * @property {string} title The tooltip to display when the mouse moves over the cluster icon. * If this value is &lt;code&gt;undefined&lt;/code&gt; or &lt;code&gt;&quot;&quot;&lt;/code&gt;, &lt;code&gt;title&lt;/code&gt; is set to the * value of the &lt;code&gt;title&lt;/code&gt; property passed to the MarkerClusterer. */ /** * A cluster icon. * * @constructor * @extends google.maps.OverlayView * @param {Cluster} cluster The cluster with which the icon is to be associated. * @param {Array} [styles] An array of {@link ClusterIconStyle} defining the cluster icons * to use for various cluster sizes. * @private */ function ClusterIcon(cluster, styles) { cluster.getMarkerClusterer().extend(ClusterIcon, google.maps.OverlayView); this.cluster_ = cluster; this.className_ = cluster.getMarkerClusterer().getClusterClass(); this.styles_ = styles; this.center_ = null; this.div_ = null; this.sums_ = null; this.visible_ = false; this.setMap(cluster.getMap()); // Note: this causes onAdd to be called } /** * Adds the icon to the DOM. */ ClusterIcon.prototype.onAdd = function () { var cClusterIcon = this; var cMouseDownInCluster; var cDraggingMapByCluster; this.div_ = document.createElement(&quot;div&quot;); this.div_.className = this.className_; if (this.visible_) { this.show(); } this.getPanes().overlayMouseTarget.appendChild(this.div_); // Fix for Issue 157 this.boundsChangedListener_ = google.maps.event.addListener(this.getMap(), &quot;bounds_changed&quot;, function () { cDraggingMapByCluster = cMouseDownInCluster; }); google.maps.event.addDomListener(this.div_, &quot;mousedown&quot;, function () { cMouseDownInCluster = true; cDraggingMapByCluster = false; }); google.maps.event.addDomListener(this.div_, &quot;click&quot;, function (e) { cMouseDownInCluster = false; if (!cDraggingMapByCluster) { var theBounds; var mz; var mc = cClusterIcon.cluster_.getMarkerClusterer(); /** * This event is fired when a cluster marker is clicked. * @name MarkerClusterer#click * @param {Cluster} c The cluster that was clicked. * @event */ google.maps.event.trigger(mc, &quot;click&quot;, cClusterIcon.cluster_); google.maps.event.trigger(mc, &quot;clusterclick&quot;, cClusterIcon.cluster_); // deprecated name // The default click handler follows. Disable it by setting // the zoomOnClick property to false. if (mc.getZoomOnClick()) { // Zoom into the cluster. mz = mc.getMaxZoom(); theBounds = cClusterIcon.cluster_.getBounds(); mc.getMap().fitBounds(theBounds); // There is a fix for Issue 170 here: setTimeout(function () { mc.getMap().fitBounds(theBounds); // Don't zoom beyond the max zoom level if (mz !== null &amp;&amp; (mc.getMap().getZoom() &gt; mz)) { mc.getMap().setZoom(mz + 1); } }, 100); } // Prevent event propagation to the map: e.cancelBubble = true; if (e.stopPropagation) { e.stopPropagation(); } } }); google.maps.event.addDomListener(this.div_, &quot;mouseover&quot;, function () { var mc = cClusterIcon.cluster_.getMarkerClusterer(); /** * This event is fired when the mouse moves over a cluster marker. * @name MarkerClusterer#mouseover * @param {Cluster} c The cluster that the mouse moved over. * @event */ google.maps.event.trigger(mc, &quot;mouseover&quot;, cClusterIcon.cluster_); }); google.maps.event.addDomListener(this.div_, &quot;mouseout&quot;, function () { var mc = cClusterIcon.cluster_.getMarkerClusterer(); /** * This event is fired when the mouse moves out of a cluster marker. * @name MarkerClusterer#mouseout * @param {Cluster} c The cluster that the mouse moved out of. * @event */ google.maps.event.trigger(mc, &quot;mouseout&quot;, cClusterIcon.cluster_); }); }; /** * Removes the icon from the DOM. */ ClusterIcon.prototype.onRemove = function () { if (this.div_ &amp;&amp; this.div_.parentNode) { this.hide(); google.maps.event.removeListener(this.boundsChangedListener_); google.maps.event.clearInstanceListeners(this.div_); this.div_.parentNode.removeChild(this.div_); this.div_ = null; } }; /** * Draws the icon. */ ClusterIcon.prototype.draw = function () { if (this.visible_) { var pos = this.getPosFromLatLng_(this.center_); this.div_.style.top = pos.y + &quot;px&quot;; this.div_.style.left = pos.x + &quot;px&quot;; } }; /** * Hides the icon. */ ClusterIcon.prototype.hide = function () { if (this.div_) { this.div_.style.display = &quot;none&quot;; } this.visible_ = false; }; /** * Positions and shows the icon. */ ClusterIcon.prototype.show = function () { if (this.div_) { var img = &quot;&quot;; // NOTE: values must be specified in px units var bp = this.backgroundPosition_.split(&quot; &quot;); var spriteH = parseInt(bp[0].replace(/^\\s+|\\s+$/g, &quot;&quot;), 10); var spriteV = parseInt(bp[1].replace(/^\\s+|\\s+$/g, &quot;&quot;), 10); var pos = this.getPosFromLatLng_(this.center_); this.div_.style.cssText = this.createCss(pos); img = &quot;&lt;img src='&quot; + this.url_ + &quot;' style='position: absolute; top: &quot; + spriteV + &quot;px; left: &quot; + spriteH + &quot;px; &quot;; if (!this.cluster_.getMarkerClusterer().enableRetinaIcons_) { img += &quot;clip: rect(&quot; + (-1 * spriteV) + &quot;px, &quot; + ((-1 * spriteH) + this.width_) + &quot;px, &quot; + ((-1 * spriteV) + this.height_) + &quot;px, &quot; + (-1 * spriteH) + &quot;px);&quot;; } img += &quot;'&gt;&quot;; this.div_.innerHTML = img + &quot;&lt;div style='&quot; + &quot;position: absolute;&quot; + &quot;top: &quot; + this.anchorText_[0] + &quot;px;&quot; + &quot;left: &quot; + this.anchorText_[1] + &quot;px;&quot; + &quot;color: &quot; + this.textColor_ + &quot;;&quot; + &quot;font-size: &quot; + this.textSize_ + &quot;px;&quot; + &quot;font-family: &quot; + this.fontFamily_ + &quot;;&quot; + &quot;font-weight: &quot; + this.fontWeight_ + &quot;;&quot; + &quot;font-style: &quot; + this.fontStyle_ + &quot;;&quot; + &quot;text-decoration: &quot; + this.textDecoration_ + &quot;;&quot; + &quot;text-align: center;&quot; + &quot;width: &quot; + this.width_ + &quot;px;&quot; + &quot;line-height:&quot; + this.height_ + &quot;px;&quot; + &quot;'&gt;&quot; + this.sums_.text + &quot;&lt;/div&gt;&quot;; if (typeof this.sums_.title === &quot;undefined&quot; || this.sums_.title === &quot;&quot;) { this.div_.title = this.cluster_.getMarkerClusterer().getTitle(); } else { this.div_.title = this.sums_.title; } this.div_.style.display = &quot;&quot;; } this.visible_ = true; }; /** * Sets the icon styles to the appropriate element in the styles array. * * @param {ClusterIconInfo} sums The icon label text and styles index. */ ClusterIcon.prototype.useStyle = function (sums) { this.sums_ = sums; var index = Math.max(0, sums.index - 1); index = Math.min(this.styles_.length - 1, index); var style = this.styles_[index]; this.url_ = style.url; this.height_ = style.height; this.width_ = style.width; this.anchorText_ = style.anchorText || [0, 0]; this.anchorIcon_ = style.anchorIcon || [parseInt(this.height_ / 2, 10), parseInt(this.width_ / 2, 10)]; this.textColor_ = style.textColor || &quot;black&quot;; this.textSize_ = style.textSize || 11; this.textDecoration_ = style.textDecoration || &quot;none&quot;; this.fontWeight_ = style.fontWeight || &quot;bold&quot;; this.fontStyle_ = style.fontStyle || &quot;normal&quot;; this.fontFamily_ = style.fontFamily || &quot;Arial,sans-serif&quot;; this.backgroundPosition_ = style.backgroundPosition || &quot;0 0&quot;; }; /** * Sets the position at which to center the icon. * * @param {google.maps.LatLng} center The latlng to set as the center. */ ClusterIcon.prototype.setCenter = function (center) { this.center_ = center; }; /** * Creates the cssText style parameter based on the position of the icon. * * @param {google.maps.Point} pos The position of the icon. * @return {string} The CSS style text. */ ClusterIcon.prototype.createCss = function (pos) { var style = []; style.push(&quot;cursor: pointer;&quot;); style.push(&quot;position: absolute; top: &quot; + pos.y + &quot;px; left: &quot; + pos.x + &quot;px;&quot;); style.push(&quot;width: &quot; + this.width_ + &quot;px; height: &quot; + this.height_ + &quot;px;&quot;); return style.join(&quot;&quot;); }; /** * Returns the position at which to place the DIV depending on the latlng. * * @param {google.maps.LatLng} latlng The position in latlng. * @return {google.maps.Point} The position in pixels. */ ClusterIcon.prototype.getPosFromLatLng_ = function (latlng) { var pos = this.getProjection().fromLatLngToDivPixel(latlng); pos.x -= this.anchorIcon_[1]; pos.y -= this.anchorIcon_[0]; pos.x = parseInt(pos.x, 10); pos.y = parseInt(pos.y, 10); return pos; }; /** * Creates a single cluster that manages a group of proximate markers. * Used internally, do not call this constructor directly. * @constructor * @param {MarkerClusterer} mc The &lt;code&gt;MarkerClusterer&lt;/code&gt; object with which this * cluster is associated. */ function Cluster(mc) { this.markerClusterer_ = mc; this.map_ = mc.getMap(); this.gridSize_ = mc.getGridSize(); this.minClusterSize_ = mc.getMinimumClusterSize(); this.averageCenter_ = mc.getAverageCenter(); this.markers_ = []; this.center_ = null; this.bounds_ = null; this.clusterIcon_ = new ClusterIcon(this, mc.getStyles()); } /** * Returns the number of markers managed by the cluster. You can call this from * a &lt;code&gt;click&lt;/code&gt;, &lt;code&gt;mouseover&lt;/code&gt;, or &lt;code&gt;mouseout&lt;/code&gt; event handler * for the &lt;code&gt;MarkerClusterer&lt;/code&gt; object. * * @return {number} The number of markers in the cluster. */ Cluster.prototype.getSize = function () { return this.markers_.length; }; /** * Returns the array of markers managed by the cluster. You can call this from * a &lt;code&gt;click&lt;/code&gt;, &lt;code&gt;mouseover&lt;/code&gt;, or &lt;code&gt;mouseout&lt;/code&gt; event handler * for the &lt;code&gt;MarkerClusterer&lt;/code&gt; object. * * @return {Array} The array of markers in the cluster. */ Cluster.prototype.getMarkers = function () { return this.markers_; }; /** * Returns the center of the cluster. You can call this from * a &lt;code&gt;click&lt;/code&gt;, &lt;code&gt;mouseover&lt;/code&gt;, or &lt;code&gt;mouseout&lt;/code&gt; event handler * for the &lt;code&gt;MarkerClusterer&lt;/code&gt; object. * * @return {google.maps.LatLng} The center of the cluster. */ Cluster.prototype.getCenter = function () { return this.center_; }; /** * Returns the map with which the cluster is associated. * * @return {google.maps.Map} The map. * @ignore */ Cluster.prototype.getMap = function () { return this.map_; }; /** * Returns the &lt;code&gt;MarkerClusterer&lt;/code&gt; object with which the cluster is associated. * * @return {MarkerClusterer} The associated marker clusterer. * @ignore */ Cluster.prototype.getMarkerClusterer = function () { return this.markerClusterer_; }; /** * Returns the bounds of the cluster. * * @return {google.maps.LatLngBounds} the cluster bounds. * @ignore */ Cluster.prototype.getBounds = function () { var i; var bounds = new google.maps.LatLngBounds(this.center_, this.center_); var markers = this.getMarkers(); for (i = 0; i &lt; markers.length; i++) { bounds.extend(markers[i].getPosition()); } return bounds; }; /** * Removes the cluster from the map. * * @ignore */ Cluster.prototype.remove = function () { this.clusterIcon_.setMap(null); this.markers_ = []; delete this.markers_; }; /** * Adds a marker to the cluster. * * @param {google.maps.Marker} marker The marker to be added. * @return {boolean} True if the marker was added. * @ignore */ Cluster.prototype.addMarker = function (marker) { var i; var mCount; var mz; if (this.isMarkerAlreadyAdded_(marker)) { return false; } if (!this.center_) { this.center_ = marker.getPosition(); this.calculateBounds_(); } else { if (this.averageCenter_) { var l = this.markers_.length + 1; var lat = (this.center_.lat() * (l - 1) + marker.getPosition().lat()) / l; var lng = (this.center_.lng() * (l - 1) + marker.getPosition().lng()) / l; this.center_ = new google.maps.LatLng(lat, lng); this.calculateBounds_(); } } marker.isAdded = true; this.markers_.push(marker); mCount = this.markers_.length; mz = this.markerClusterer_.getMaxZoom(); if (mz !== null &amp;&amp; this.map_.getZoom() &gt; mz) { // Zoomed in past max zoom, so show the marker. if (marker.getMap() !== this.map_) { marker.setMap(this.map_); } } else if (mCount &lt; this.minClusterSize_) { // Min cluster size not reached so show the marker. if (marker.getMap() !== this.map_) { marker.setMap(this.map_); } } else if (mCount === this.minClusterSize_) { // Hide the markers that were showing. for (i = 0; i &lt; mCount; i++) { this.markers_[i].setMap(null); } } else { marker.setMap(null); } this.updateIcon_(); return true; }; /** * Determines if a marker lies within the cluster's bounds. * * @param {google.maps.Marker} marker The marker to check. * @return {boolean} True if the marker lies in the bounds. * @ignore */ Cluster.prototype.isMarkerInClusterBounds = function (marker) { return this.bounds_.contains(marker.getPosition()); }; /** * Calculates the extended bounds of the cluster with the grid. */ Cluster.prototype.calculateBounds_ = function () { var bounds = new google.maps.LatLngBounds(this.center_, this.center_); this.bounds_ = this.markerClusterer_.getExtendedBounds(bounds); }; /** * Updates the cluster icon. */ Cluster.prototype.updateIcon_ = function () { var mCount = this.markers_.length; var mz = this.markerClusterer_.getMaxZoom(); if (mz !== null &amp;&amp; this.map_.getZoom() &gt; mz) { this.clusterIcon_.hide(); return; } if (mCount &lt; this.minClusterSize_) { // Min cluster size not yet reached. this.clusterIcon_.hide(); return; } var numStyles = this.markerClusterer_.getStyles().length; var sums = this.markerClusterer_.getCalculator()(this.markers_, numStyles); this.clusterIcon_.setCenter(this.center_); this.clusterIcon_.useStyle(sums); this.clusterIcon_.show(); }; /** * Determines if a marker has already been added to the cluster. * * @param {google.maps.Marker} marker The marker to check. * @return {boolean} True if the marker has already been added. */ Cluster.prototype.isMarkerAlreadyAdded_ = function (marker) { var i; if (this.markers_.indexOf) { return this.markers_.indexOf(marker) !== -1; } else { for (i = 0; i &lt; this.markers_.length; i++) { if (marker === this.markers_[i]) { return true; } } } return false; }; /** * @name MarkerClustererOptions * @class This class represents the optional parameter passed to * the {@link MarkerClusterer} constructor. * @property {number} [gridSize=60] The grid size of a cluster in pixels. The grid is a square. * @property {number} [maxZoom=null] The maximum zoom level at which clustering is enabled or * &lt;code&gt;null&lt;/code&gt; if clustering is to be enabled at all zoom levels. * @property {boolean} [zoomOnClick=true] Whether to zoom the map when a cluster marker is * clicked. You may want to set this to &lt;code&gt;false&lt;/code&gt; if you have installed a handler * for the &lt;code&gt;click&lt;/code&gt; event and it deals with zooming on its own. * @property {boolean} [averageCenter=false] Whether the position of a cluster marker should be * the average position of all markers in the cluster. If set to &lt;code&gt;false&lt;/code&gt;, the * cluster marker is positioned at the location of the first marker added to the cluster. * @property {number} [minimumClusterSize=2] The minimum number of markers needed in a cluster * before the markers are hidden and a cluster marker appears. * @property {boolean} [ignoreHidden=false] Whether to ignore hidden markers in clusters. You * may want to set this to &lt;code&gt;true&lt;/code&gt; to ensure that hidden markers are not included * in the marker count that appears on a cluster marker (this count is the value of the * &lt;code&gt;text&lt;/code&gt; property of the result returned by the default &lt;code&gt;calculator&lt;/code&gt;). * If set to &lt;code&gt;true&lt;/code&gt; and you change the visibility of a marker being clustered, be * sure to also call &lt;code&gt;MarkerClusterer.repaint()&lt;/code&gt;. * @property {string} [title=&quot;&quot;] The tooltip to display when the mouse moves over a cluster * marker. (Alternatively, you can use a custom &lt;code&gt;calculator&lt;/code&gt; function to specify a * different tooltip for each cluster marker.) * @property {function} [calculator=MarkerClusterer.CALCULATOR] The function used to determine * the text to be displayed on a cluster marker and the index indicating which style to use * for the cluster marker. The input parameters for the function are (1) the array of markers * represented by a cluster marker and (2) the number of cluster icon styles. It returns a * {@link ClusterIconInfo} object. The default &lt;code&gt;calculator&lt;/code&gt; returns a * &lt;code&gt;text&lt;/code&gt; property which is the number of markers in the cluster and an * &lt;code&gt;index&lt;/code&gt; property which is one higher than the lowest integer such that * &lt;code&gt;10^i&lt;/code&gt; exceeds the number of markers in the cluster, or the size of the styles * array, whichever is less. The &lt;code&gt;styles&lt;/code&gt; array element used has an index of * &lt;code&gt;index&lt;/code&gt; minus 1. For example, the default &lt;code&gt;calculator&lt;/code&gt; returns a * &lt;code&gt;text&lt;/code&gt; value of &lt;code&gt;&quot;125&quot;&lt;/code&gt; and an &lt;code&gt;index&lt;/code&gt; of &lt;code&gt;3&lt;/code&gt; * for a cluster icon representing 125 markers so the element used in the &lt;code&gt;styles&lt;/code&gt; * array is &lt;code&gt;2&lt;/code&gt;. A &lt;code&gt;calculator&lt;/code&gt; may also return a &lt;code&gt;title&lt;/code&gt; * property that contains the text of the tooltip to be used for the cluster marker. If * &lt;code&gt;title&lt;/code&gt; is not defined, the tooltip is set to the value of the &lt;code&gt;title&lt;/code&gt; * property for the MarkerClusterer. * @property {string} [clusterClass=&quot;cluster&quot;] The name of the CSS class defining general styles * for the cluster markers. Use this class to define CSS styles that are not set up by the code * that processes the &lt;code&gt;styles&lt;/code&gt; array. * @property {Array} [styles] An array of {@link ClusterIconStyle} elements defining the styles * of the cluster markers to be used. The element to be used to style a given cluster marker * is determined by the function defined by the &lt;code&gt;calculator&lt;/code&gt; property. * The default is an array of {@link ClusterIconStyle} elements whose properties are derived * from the values for &lt;code&gt;imagePath&lt;/code&gt;, &lt;code&gt;imageExtension&lt;/code&gt;, and * &lt;code&gt;imageSizes&lt;/code&gt;. * @property {boolean} [enableRetinaIcons=false] Whether to allow the use of cluster icons that * have sizes that are some multiple (typically double) of their actual display size. Icons such * as these look better when viewed on high-resolution monitors such as Apple's Retina displays. * Note: if this property is &lt;code&gt;true&lt;/code&gt;, sprites cannot be used as cluster icons. * @property {number} [batchSize=MarkerClusterer.BATCH_SIZE] Set this property to the * number of markers to be processed in a single batch when using a browser other than * Internet Explorer (for Internet Explorer, use the batchSizeIE property instead). * @property {number} [batchSizeIE=MarkerClusterer.BATCH_SIZE_IE] When Internet Explorer is * being used, markers are processed in several batches with a small delay inserted between * each batch in an attempt to avoid Javascript timeout errors. Set this property to the * number of markers to be processed in a single batch; select as high a number as you can * without causing a timeout error in the browser. This number might need to be as low as 100 * if 15,000 markers are being managed, for example. * @property {string} [imagePath=MarkerClusterer.IMAGE_PATH] * The full URL of the root name of the group of image files to use for cluster icons. * The complete file name is of the form &lt;code&gt;imagePath&lt;/code&gt;n.&lt;code&gt;imageExtension&lt;/code&gt; * where n is the image file number (1, 2, etc.). * @property {string} [imageExtension=MarkerClusterer.IMAGE_EXTENSION] * The extension name for the cluster icon image files (e.g., &lt;code&gt;&quot;png&quot;&lt;/code&gt; or * &lt;code&gt;&quot;jpg&quot;&lt;/code&gt;). * @property {Array} [imageSizes=MarkerClusterer.IMAGE_SIZES] * An array of numbers containing the widths of the group of * &lt;code&gt;imagePath&lt;/code&gt;n.&lt;code&gt;imageExtension&lt;/code&gt; image files. * (The images are assumed to be square.) */ /** * Creates a MarkerClusterer object with the options specified in {@link MarkerClustererOptions}. * @constructor * @extends google.maps.OverlayView * @param {google.maps.Map} map The Google map to attach to. * @param {Array.&lt;google.maps.Marker&gt;} [opt_markers] The markers to be added to the cluster. * @param {MarkerClustererOptions} [opt_options] The optional parameters. */ function MarkerClusterer(map, opt_markers, opt_options) { // MarkerClusterer implements google.maps.OverlayView interface. We use the // extend function to extend MarkerClusterer with google.maps.OverlayView // because it might not always be available when the code is defined so we // look for it at the last possible moment. If it doesn't exist now then // there is no point going ahead :) this.extend(MarkerClusterer, google.maps.OverlayView); opt_markers = opt_markers || []; opt_options = opt_options || {}; this.markers_ = []; this.clusters_ = []; this.listeners_ = []; this.activeMap_ = null; this.ready_ = false; this.gridSize_ = opt_options.gridSize || 60; this.minClusterSize_ = opt_options.minimumClusterSize || 2; this.maxZoom_ = opt_options.maxZoom || null; this.styles_ = opt_options.styles || []; this.title_ = opt_options.title || &quot;&quot;; this.zoomOnClick_ = true; if (opt_options.zoomOnClick !== undefined) { this.zoomOnClick_ = opt_options.zoomOnClick; } this.averageCenter_ = false; if (opt_options.averageCenter !== undefined) { this.averageCenter_ = opt_options.averageCenter; } this.ignoreHidden_ = false; if (opt_options.ignoreHidden !== undefined) { this.ignoreHidden_ = opt_options.ignoreHidden; } this.enableRetinaIcons_ = false; if (opt_options.enableRetinaIcons !== undefined) { this.enableRetinaIcons_ = opt_options.enableRetinaIcons; } this.imagePath_ = opt_options.imagePath || MarkerClusterer.IMAGE_PATH; this.imageExtension_ = opt_options.imageExtension || MarkerClusterer.IMAGE_EXTENSION; this.imageSizes_ = opt_options.imageSizes || MarkerClusterer.IMAGE_SIZES; this.calculator_ = opt_options.calculator || MarkerClusterer.CALCULATOR; this.batchSize_ = opt_options.batchSize || MarkerClusterer.BATCH_SIZE; this.batchSizeIE_ = opt_options.batchSizeIE || MarkerClusterer.BATCH_SIZE_IE; this.clusterClass_ = opt_options.clusterClass || &quot;cluster&quot;; if (navigator.userAgent.toLowerCase().indexOf(&quot;msie&quot;) !== -1) { // Try to avoid IE timeout when processing a huge number of markers: this.batchSize_ = this.batchSizeIE_; } this.setupStyles_(); this.addMarkers(opt_markers, true); this.setMap(map); // Note: this causes onAdd to be called } /** * Implementation of the onAdd interface method. * @ignore */ MarkerClusterer.prototype.onAdd = function () { var cMarkerClusterer = this; this.activeMap_ = this.getMap(); this.ready_ = true; this.repaint(); // Add the map event listeners this.listeners_ = [ google.maps.event.addListener(this.getMap(), &quot;zoom_changed&quot;, function () { cMarkerClusterer.resetViewport_(false); // Workaround for this Google bug: when map is at level 0 and &quot;-&quot; of // zoom slider is clicked, a &quot;zoom_changed&quot; event is fired even though // the map doesn't zoom out any further. In this situation, no &quot;idle&quot; // event is triggered so the cluster markers that have been removed // do not get redrawn. Same goes for a zoom in at maxZoom. if (this.getZoom() === (this.get(&quot;minZoom&quot;) || 0) || this.getZoom() === this.get(&quot;maxZoom&quot;)) { google.maps.event.trigger(this, &quot;idle&quot;); } }), google.maps.event.addListener(this.getMap(), &quot;idle&quot;, function () { cMarkerClusterer.redraw_(); }) ]; }; /** * Implementation of the onRemove interface method. * Removes map event listeners and all cluster icons from the DOM. * All managed markers are also put back on the map. * @ignore */ MarkerClusterer.prototype.onRemove = function () { var i; // Put all the managed markers back on the map: for (i = 0; i &lt; this.markers_.length; i++) { if (this.markers_[i].getMap() !== this.activeMap_) { this.markers_[i].setMap(this.activeMap_); } } // Remove all clusters: for (i = 0; i &lt; this.clusters_.length; i++) { this.clusters_[i].remove(); } this.clusters_ = []; // Remove map event listeners: for (i = 0; i &lt; this.listeners_.length; i++) { google.maps.event.removeListener(this.listeners_[i]); } this.listeners_ = []; this.activeMap_ = null; this.ready_ = false; }; /** * Implementation of the draw interface method. * @ignore */ MarkerClusterer.prototype.draw = function () {}; /** * Sets up the styles object. */ MarkerClusterer.prototype.setupStyles_ = function () { var i, size; if (this.styles_.length &gt; 0) { return; } for (i = 0; i &lt; this.imageSizes_.length; i++) { size = this.imageSizes_[i]; this.styles_.push({ url: this.imagePath_ + (i + 1) + &quot;.&quot; + this.imageExtension_, height: size, width: size }); } }; /** * Fits the map to the bounds of the markers managed by the clusterer. */ MarkerClusterer.prototype.fitMapToMarkers = function () { var i; var markers = this.getMarkers(); var bounds = new google.maps.LatLngBounds(); for (i = 0; i &lt; markers.length; i++) { bounds.extend(markers[i].getPosition()); } this.getMap().fitBounds(bounds); }; /** * Returns the value of the &lt;code&gt;gridSize&lt;/code&gt; property. * * @return {number} The grid size. */ MarkerClusterer.prototype.getGridSize = function () { return this.gridSize_; }; /** * Sets the value of the &lt;code&gt;gridSize&lt;/code&gt; property. * * @param {number} gridSize The grid size. */ MarkerClusterer.prototype.setGridSize = function (gridSize) { this.gridSize_ = gridSize; }; /** * Returns the value of the &lt;code&gt;minimumClusterSize&lt;/code&gt; property. * * @return {number} The minimum cluster size. */ MarkerClusterer.prototype.getMinimumClusterSize = function () { return this.minClusterSize_; }; /** * Sets the value of the &lt;code&gt;minimumClusterSize&lt;/code&gt; property. * * @param {number} minimumClusterSize The minimum cluster size. */ MarkerClusterer.prototype.setMinimumClusterSize = function (minimumClusterSize) { this.minClusterSize_ = minimumClusterSize; }; /** * Returns the value of the &lt;code&gt;maxZoom&lt;/code&gt; property. * * @return {number} The maximum zoom level. */ MarkerClusterer.prototype.getMaxZoom = function () { return this.maxZoom_; }; /** * Sets the value of the &lt;code&gt;maxZoom&lt;/code&gt; property. * * @param {number} maxZoom The maximum zoom level. */ MarkerClusterer.prototype.setMaxZoom = function (maxZoom) { this.maxZoom_ = maxZoom; }; /** * Returns the value of the &lt;code&gt;styles&lt;/code&gt; property. * * @return {Array} The array of styles defining the cluster markers to be used. */ MarkerClusterer.prototype.getStyles = function () { return this.styles_; }; /** * Sets the value of the &lt;code&gt;styles&lt;/code&gt; property. * * @param {Array.&lt;ClusterIconStyle&gt;} styles The array of styles to use. */ MarkerClusterer.prototype.setStyles = function (styles) { this.styles_ = styles; }; /** * Returns the value of the &lt;code&gt;title&lt;/code&gt; property. * * @return {string} The content of the title text. */ MarkerClusterer.prototype.getTitle = function () { return this.title_; }; /** * Sets the value of the &lt;code&gt;title&lt;/code&gt; property. * * @param {string} title The value of the title property. */ MarkerClusterer.prototype.setTitle = function (title) { this.title_ = title; }; /** * Returns the value of the &lt;code&gt;zoomOnClick&lt;/code&gt; property. * * @return {boolean} True if zoomOnClick property is set. */ MarkerClusterer.prototype.getZoomOnClick = function () { return this.zoomOnClick_; }; /** * Sets the value of the &lt;code&gt;zoomOnClick&lt;/code&gt; property. * * @param {boolean} zoomOnClick The value of the zoomOnClick property. */ MarkerClusterer.prototype.setZoomOnClick = function (zoomOnClick) { this.zoomOnClick_ = zoomOnClick; }; /** * Returns the value of the &lt;code&gt;averageCenter&lt;/code&gt; property. * * @return {boolean} True if averageCenter property is set. */ MarkerClusterer.prototype.getAverageCenter = function () { return this.averageCenter_; }; /** * Sets the value of the &lt;code&gt;averageCenter&lt;/code&gt; property. * * @param {boolean} averageCenter The value of the averageCenter property. */ MarkerClusterer.prototype.setAverageCenter = function (averageCenter) { this.averageCenter_ = averageCenter; }; /** * Returns the value of the &lt;code&gt;ignoreHidden&lt;/code&gt; property. * * @return {boolean} True if ignoreHidden property is set. */ MarkerClusterer.prototype.getIgnoreHidden = function () { return this.ignoreHidden_; }; /** * Sets the value of the &lt;code&gt;ignoreHidden&lt;/code&gt; property. * * @param {boolean} ignoreHidden The value of the ignoreHidden property. */ MarkerClusterer.prototype.setIgnoreHidden = function (ignoreHidden) { this.ignoreHidden_ = ignoreHidden; }; /** * Returns the value of the &lt;code&gt;enableRetinaIcons&lt;/code&gt; property. * * @return {boolean} True if enableRetinaIcons property is set. */ MarkerClusterer.prototype.getEnableRetinaIcons = function () { return this.enableRetinaIcons_; }; /** * Sets the value of the &lt;code&gt;enableRetinaIcons&lt;/code&gt; property. * * @param {boolean} enableRetinaIcons The value of the enableRetinaIcons property. */ MarkerClusterer.prototype.setEnableRetinaIcons = function (enableRetinaIcons) { this.enableRetinaIcons_ = enableRetinaIcons; }; /** * Returns the value of the &lt;code&gt;imageExtension&lt;/code&gt; property. * * @return {string} The value of the imageExtension property. */ MarkerClusterer.prototype.getImageExtension = function () { return this.imageExtension_; }; /** * Sets the value of the &lt;code&gt;imageExtension&lt;/code&gt; property. * * @param {string} imageExtension The value of the imageExtension property. */ MarkerClusterer.prototype.setImageExtension = function (imageExtension) { this.imageExtension_ = imageExtension; }; /** * Returns the value of the &lt;code&gt;imagePath&lt;/code&gt; property. * * @return {string} The value of the imagePath property. */ MarkerClusterer.prototype.getImagePath = function () { return this.imagePath_; }; /** * Sets the value of the &lt;code&gt;imagePath&lt;/code&gt; property. * * @param {string} imagePath The value of the imagePath property. */ MarkerClusterer.prototype.setImagePath = function (imagePath) { this.imagePath_ = imagePath; }; /** * Returns the value of the &lt;code&gt;imageSizes&lt;/code&gt; property. * * @return {Array} The value of the imageSizes property. */ MarkerClusterer.prototype.getImageSizes = function () { return this.imageSizes_; }; /** * Sets the value of the &lt;code&gt;imageSizes&lt;/code&gt; property. * * @param {Array} imageSizes The value of the imageSizes property. */ MarkerClusterer.prototype.setImageSizes = function (imageSizes) { this.imageSizes_ = imageSizes; }; /** * Returns the value of the &lt;code&gt;calculator&lt;/code&gt; property. * * @return {function} the value of the calculator property. */ MarkerClusterer.prototype.getCalculator = function () { return this.calculator_; }; /** * Sets the value of the &lt;code&gt;calculator&lt;/code&gt; property. * * @param {function(Array.&lt;google.maps.Marker&gt;, number)} calculator The value * of the calculator property. */ MarkerClusterer.prototype.setCalculator = function (calculator) { this.calculator_ = calculator; }; /** * Returns the value of the &lt;code&gt;batchSizeIE&lt;/code&gt; property. * * @return {number} the value of the batchSizeIE property. */ MarkerClusterer.prototype.getBatchSizeIE = function () { return this.batchSizeIE_; }; /** * Sets the value of the &lt;code&gt;batchSizeIE&lt;/code&gt; property. * * @param {number} batchSizeIE The value of the batchSizeIE property. */ MarkerClusterer.prototype.setBatchSizeIE = function (batchSizeIE) { this.batchSizeIE_ = batchSizeIE; }; /** * Returns the value of the &lt;code&gt;clusterClass&lt;/code&gt; property. * * @return {string} the value of the clusterClass property. */ MarkerClusterer.prototype.getClusterClass = function () { return this.clusterClass_; }; /** * Sets the value of the &lt;code&gt;clusterClass&lt;/code&gt; property. * * @param {string} clusterClass The value of the clusterClass property. */ MarkerClusterer.prototype.setClusterClass = function (clusterClass) { this.clusterClass_ = clusterClass; }; /** * Returns the array of markers managed by the clusterer. * * @return {Array} The array of markers managed by the clusterer. */ MarkerClusterer.prototype.getMarkers = function () { return this.markers_; }; /** * Returns the number of markers managed by the clusterer. * * @return {number} The number of markers. */ MarkerClusterer.prototype.getTotalMarkers = function () { return this.markers_.length; }; /** * Returns the current array of clusters formed by the clusterer. * * @return {Array} The array of clusters formed by the clusterer. */ MarkerClusterer.prototype.getClusters = function () { return this.clusters_; }; /** * Returns the number of clusters formed by the clusterer. * * @return {number} The number of clusters formed by the clusterer. */ MarkerClusterer.prototype.getTotalClusters = function () { return this.clusters_.length; }; /** * Adds a marker to the clusterer. The clusters are redrawn unless * &lt;code&gt;opt_nodraw&lt;/code&gt; is set to &lt;code&gt;true&lt;/code&gt;. * * @param {google.maps.Marker} marker The marker to add. * @param {boolean} [opt_nodraw] Set to &lt;code&gt;true&lt;/code&gt; to prevent redrawing. */ MarkerClusterer.prototype.addMarker = function (marker, opt_nodraw) { this.pushMarkerTo_(marker); if (!opt_nodraw) { this.redraw_(); } }; /** * Adds an array of markers to the clusterer. The clusters are redrawn unless * &lt;code&gt;opt_nodraw&lt;/code&gt; is set to &lt;code&gt;true&lt;/code&gt;. * * @param {Array.&lt;google.maps.Marker&gt;} markers The markers to add. * @param {boolean} [opt_nodraw] Set to &lt;code&gt;true&lt;/code&gt; to prevent redrawing. */ MarkerClusterer.prototype.addMarkers = function (markers, opt_nodraw) { var key; for (key in markers) { if (markers.hasOwnProperty(key)) { this.pushMarkerTo_(markers[key]); } } if (!opt_nodraw) { this.redraw_(); } }; /** * Pushes a marker to the clusterer. * * @param {google.maps.Marker} marker The marker to add. */ MarkerClusterer.prototype.pushMarkerTo_ = function (marker) { // If the marker is draggable add a listener so we can update the clusters on the dragend: if (marker.getDraggable()) { var cMarkerClusterer = this; google.maps.event.addListener(marker, &quot;dragend&quot;, function () { if (cMarkerClusterer.ready_) { this.isAdded = false; cMarkerClusterer.repaint(); } }); } marker.isAdded = false; this.markers_.push(marker); }; /** * Removes a marker from the cluster. The clusters are redrawn unless * &lt;code&gt;opt_nodraw&lt;/code&gt; is set to &lt;code&gt;true&lt;/code&gt;. Returns &lt;code&gt;true&lt;/code&gt; if the * marker was removed from the clusterer. * * @param {google.maps.Marker} marker The marker to remove. * @param {boolean} [opt_nodraw] Set to &lt;code&gt;true&lt;/code&gt; to prevent redrawing. * @return {boolean} True if the marker was removed from the clusterer. */ MarkerClusterer.prototype.removeMarker = function (marker, opt_nodraw) { var removed = this.removeMarker_(marker); if (!opt_nodraw &amp;&amp; removed) { this.repaint(); } return removed; }; /** * Removes an array of markers from the cluster. The clusters are redrawn unless * &lt;code&gt;opt_nodraw&lt;/code&gt; is set to &lt;code&gt;true&lt;/code&gt;. Returns &lt;code&gt;true&lt;/code&gt; if markers * were removed from the clusterer. * * @param {Array.&lt;google.maps.Marker&gt;} markers The markers to remove. * @param {boolean} [opt_nodraw] Set to &lt;code&gt;true&lt;/code&gt; to prevent redrawing. * @return {boolean} True if markers were removed from the clusterer. */ MarkerClusterer.prototype.removeMarkers = function (markers, opt_nodraw) { var i, r; var removed = false; for (i = 0; i &lt; markers.length; i++) { r = this.removeMarker_(markers[i]); removed = removed || r; } if (!opt_nodraw &amp;&amp; removed) { this.repaint(); } return removed; }; /** * Removes a marker and returns true if removed, false if not. * * @param {google.maps.Marker} marker The marker to remove * @return {boolean} Whether the marker was removed or not */ MarkerClusterer.prototype.removeMarker_ = function (marker) { var i; var index = -1; if (this.markers_.indexOf) { index = this.markers_.indexOf(marker); } else { for (i = 0; i &lt; this.markers_.length; i++) { if (marker === this.markers_[i]) { index = i; break; } } } if (index === -1) { // Marker is not in our list of markers, so do nothing: return false; } marker.setMap(null); this.markers_.splice(index, 1); // Remove the marker from the list of managed markers return true; }; /** * Removes all clusters and markers from the map and also removes all markers * managed by the clusterer. */ MarkerClusterer.prototype.clearMarkers = function () { this.resetViewport_(true); this.markers_ = []; }; /** * Recalculates and redraws all the marker clusters from scratch. * Call this after changing any properties. */ MarkerClusterer.prototype.repaint = function () { var oldClusters = this.clusters_.slice(); this.clusters_ = []; this.resetViewport_(false); this.redraw_(); // Remove the old clusters. // Do it in a timeout to prevent blinking effect. setTimeout(function () { var i; for (i = 0; i &lt; oldClusters.length; i++) { oldClusters[i].remove(); } }, 0); }; /** * Returns the current bounds extended by the grid size. * * @param {google.maps.LatLngBounds} bounds The bounds to extend. * @return {google.maps.LatLngBounds} The extended bounds. * @ignore */ MarkerClusterer.prototype.getExtendedBounds = function (bounds) { var projection = this.getProjection(); // Turn the bounds into latlng. var tr = new google.maps.LatLng(bounds.getNorthEast().lat(), bounds.getNorthEast().lng()); var bl = new google.maps.LatLng(bounds.getSouthWest().lat(), bounds.getSouthWest().lng()); // Convert the points to pixels and the extend out by the grid size. var trPix = projection.fromLatLngToDivPixel(tr); trPix.x += this.gridSize_; trPix.y -= this.gridSize_; var blPix = projection.fromLatLngToDivPixel(bl); blPix.x -= this.gridSize_; blPix.y += this.gridSize_; // Convert the pixel points back to LatLng var ne = projection.fromDivPixelToLatLng(trPix); var sw = projection.fromDivPixelToLatLng(blPix); // Extend the bounds to contain the new bounds. bounds.extend(ne); bounds.extend(sw); return bounds; }; /** * Redraws all the clusters. */ MarkerClusterer.prototype.redraw_ = function () { this.createClusters_(0); }; /** * Removes all clusters from the map. The markers are also removed from the map * if &lt;code&gt;opt_hide&lt;/code&gt; is set to &lt;code&gt;true&lt;/code&gt;. * * @param {boolean} [opt_hide] Set to &lt;code&gt;true&lt;/code&gt; to also remove the markers * from the map. */ MarkerClusterer.prototype.resetViewport_ = function (opt_hide) { var i, marker; // Remove all the clusters for (i = 0; i &lt; this.clusters_.length; i++) { this.clusters_[i].remove(); } this.clusters_ = []; // Reset the markers to not be added and to be removed from the map. for (i = 0; i &lt; this.markers_.length; i++) { marker = this.markers_[i]; marker.isAdded = false; if (opt_hide) { marker.setMap(null); } } }; /** * Calculates the distance between two latlng locations in km. * * @param {google.maps.LatLng} p1 The first lat lng point. * @param {google.maps.LatLng} p2 The second lat lng point. * @return {number} The distance between the two points in km. * @see http://www.movable-type.co.uk/scripts/latlong.html */ MarkerClusterer.prototype.distanceBetweenPoints_ = function (p1, p2) { var R = 6371; // Radius of the Earth in km var dLat = (p2.lat() - p1.lat()) * Math.PI / 180; var dLon = (p2.lng() - p1.lng()) * Math.PI / 180; var a = Math.sin(dLat / 2) * Math.sin(dLat / 2) + Math.cos(p1.lat() * Math.PI / 180) * Math.cos(p2.lat() * Math.PI / 180) * Math.sin(dLon / 2) * Math.sin(dLon / 2); var c = 2 * Math.atan2(Math.sqrt(a), Math.sqrt(1 - a)); var d = R * c; return d; }; /** * Determines if a marker is contained in a bounds. * * @param {google.maps.Marker} marker The marker to check. * @param {google.maps.LatLngBounds} bounds The bounds to check against. * @return {boolean} True if the marker is in the bounds. */ MarkerClusterer.prototype.isMarkerInBounds_ = function (marker, bounds) { return bounds.contains(marker.getPosition()); }; /** * Adds a marker to a cluster, or creates a new cluster. * * @param {google.maps.Marker} marker The marker to add. */ MarkerClusterer.prototype.addToClosestCluster_ = function (marker) { var i, d, cluster, center; var distance = 40000; // Some large number var clusterToAddTo = null; for (i = 0; i &lt; this.clusters_.length; i++) { cluster = this.clusters_[i]; center = cluster.getCenter(); if (center) { d = this.distanceBetweenPoints_(center, marker.getPosition()); if (d &lt; distance) { distance = d; clusterToAddTo = cluster; } } } if (clusterToAddTo &amp;&amp; clusterToAddTo.isMarkerInClusterBounds(marker)) { clusterToAddTo.addMarker(marker); } else { cluster = new Cluster(this); cluster.addMarker(marker); this.clusters_.push(cluster); } }; /** * Creates the clusters. This is done in batches to avoid timeout errors * in some browsers when there is a huge number of markers. * * @param {number} iFirst The index of the first marker in the batch of * markers to be added to clusters. */ MarkerClusterer.prototype.createClusters_ = function (iFirst) { var i, marker; var mapBounds; var cMarkerClusterer = this; if (!this.ready_