(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 += '
' + option.title + '';
                        }
                    }
                    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 = 'http://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.js
        if (!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/indexOf
        if (!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;
    }
));