| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822 | // ==ClosureCompiler==// @compilation_level ADVANCED_OPTIMIZATIONS// @externs_url http://closure-compiler.googlecode.com/svn/trunk/contrib/externs/maps/google_maps_api_v3.js// @output_wrapper (function() {%output%})();// ==/ClosureCompiler==/** * @license * Copyright 2013 Google Inc. All Rights Reserved. * * Licensed under the Apache License, Version 2.0 (the "License"); * you may not use this file except in compliance with the License. * You may obtain a copy of the License at * *     http://www.apache.org/licenses/LICENSE-2.0 * * Unless required by applicable law or agreed to in writing, software * distributed under the License is distributed on an "AS IS" BASIS, * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. * See the License for the specific language governing permissions and * limitations under the License. *//** * A RichMarker that allows any HTML/DOM to be added to a map and be draggable. * * @param {Object.<string, *>=} opt_options Optional properties to set. * @extends {google.maps.OverlayView} * @constructor */function RichMarker(opt_options) {  var options = opt_options || {};  /**   * @type {boolean}   * @private   */  this.ready_ = false;  /**   * @type {boolean}   * @private   */  this.dragging_ = false;  if (opt_options['visible'] == undefined) {    opt_options['visible'] = true;  }  if (opt_options['shadow'] == undefined) {    // opt_options['shadow'] = '7px -3px 5px rgba(88,88,88,0.7)';  }  if (opt_options['anchor'] == undefined) {    opt_options['anchor'] = RichMarkerPosition['BOTTOM'];  }  this.setValues(options);}RichMarker.prototype = new google.maps.OverlayView();window['RichMarker'] = RichMarker;/** * Returns the current visibility state of the marker. * * @return {boolean} The visiblity of the marker. */RichMarker.prototype.getVisible = function() {  return /** @type {boolean} */ (this.get('visible'));};RichMarker.prototype['getVisible'] = RichMarker.prototype.getVisible;/** * Sets the visiblility state of the marker. * * @param {boolean} visible The visiblilty of the marker. */RichMarker.prototype.setVisible = function(visible) {  this.set('visible', visible);};RichMarker.prototype['setVisible'] = RichMarker.prototype.setVisible;/** *  The visible changed event. */RichMarker.prototype.visible_changed = function() {  if (this.ready_) {    this.markerWrapper_.style['display'] = this.getVisible() ? '' : 'none';    this.draw();  }};RichMarker.prototype['visible_changed'] = RichMarker.prototype.visible_changed;/** * Sets the marker to be flat. * * @param {boolean} flat If the marker is to be flat or not. */RichMarker.prototype.setFlat = function(flat) {  this.set('flat', !!flat);};RichMarker.prototype['setFlat'] = RichMarker.prototype.setFlat;/** * If the makrer is flat or not. * * @return {boolean} True the marker is flat. */RichMarker.prototype.getFlat = function() {  return /** @type {boolean} */ (this.get('flat'));};RichMarker.prototype['getFlat'] = RichMarker.prototype.getFlat;/** * Get the width of the marker. * * @return {Number} The width of the marker. */RichMarker.prototype.getWidth = function() {  return /** @type {Number} */ (this.get('width'));};RichMarker.prototype['getWidth'] = RichMarker.prototype.getWidth;/** * Get the height of the marker. * * @return {Number} The height of the marker. */RichMarker.prototype.getHeight = function() {  return /** @type {Number} */ (this.get('height'));};RichMarker.prototype['getHeight'] = RichMarker.prototype.getHeight;/** * Sets the marker's box shadow. * * @param {string} shadow The box shadow to set. */RichMarker.prototype.setShadow = function(shadow) {  this.set('shadow', shadow);  this.flat_changed();};RichMarker.prototype['setShadow'] = RichMarker.prototype.setShadow;/** * Gets the marker's box shadow. * * @return {string} The box shadow. */RichMarker.prototype.getShadow = function() {  return /** @type {string} */ (this.get('shadow'));};RichMarker.prototype['getShadow'] = RichMarker.prototype.getShadow;/** * Flat changed event. */RichMarker.prototype.flat_changed = function() {  if (!this.ready_) {    return;  }  this.markerWrapper_.style['boxShadow'] =      this.markerWrapper_.style['webkitBoxShadow'] =      this.markerWrapper_.style['MozBoxShadow'] =      this.getFlat() ? '' : this.getShadow();};RichMarker.prototype['flat_changed'] = RichMarker.prototype.flat_changed;/** * Sets the zIndex of the marker. * * @param {Number} index The index to set. */RichMarker.prototype.setZIndex = function(index) {  this.set('zIndex', index);};RichMarker.prototype['setZIndex'] = RichMarker.prototype.setZIndex;/** * Gets the zIndex of the marker. * * @return {Number} The zIndex of the marker. */RichMarker.prototype.getZIndex = function() {  return /** @type {Number} */ (this.get('zIndex'));};RichMarker.prototype['getZIndex'] = RichMarker.prototype.getZIndex;/** * zIndex changed event. */RichMarker.prototype.zIndex_changed = function() {  if (this.getZIndex() && this.ready_) {    this.markerWrapper_.style.zIndex = this.getZIndex();  }};RichMarker.prototype['zIndex_changed'] = RichMarker.prototype.zIndex_changed;/** * Whether the marker is draggable or not. * * @return {boolean} True if the marker is draggable. */RichMarker.prototype.getDraggable = function() {  return /** @type {boolean} */ (this.get('draggable'));};RichMarker.prototype['getDraggable'] = RichMarker.prototype.getDraggable;/** * Sets the marker to be draggable or not. * * @param {boolean} draggable If the marker is draggable or not. */RichMarker.prototype.setDraggable = function(draggable) {  this.set('draggable', !!draggable);};RichMarker.prototype['setDraggable'] = RichMarker.prototype.setDraggable;/** * Draggable property changed callback. */RichMarker.prototype.draggable_changed = function() {  if (this.ready_) {    if (this.getDraggable()) {      this.addDragging_(this.markerWrapper_);    } else {      this.removeDragListeners_();    }  }};RichMarker.prototype['draggable_changed'] =    RichMarker.prototype.draggable_changed;/** * Gets the postiton of the marker. * * @return {google.maps.LatLng} The position of the marker. */RichMarker.prototype.getPosition = function() {  return /** @type {google.maps.LatLng} */ (this.get('position'));};RichMarker.prototype['getPosition'] = RichMarker.prototype.getPosition;/** * Sets the position of the marker. * * @param {google.maps.LatLng} position The position to set. */RichMarker.prototype.setPosition = function(position) {  this.set('position', position);};RichMarker.prototype['setPosition'] = RichMarker.prototype.setPosition;/** * Position changed event. */RichMarker.prototype.position_changed = function() {  this.draw();};RichMarker.prototype['position_changed'] =    RichMarker.prototype.position_changed;/** * Gets the anchor. * * @return {google.maps.Size} The position of the anchor. */RichMarker.prototype.getAnchor = function() {  return /** @type {google.maps.Size} */ (this.get('anchor'));};RichMarker.prototype['getAnchor'] = RichMarker.prototype.getAnchor;/** * Sets the anchor. * * @param {RichMarkerPosition|google.maps.Size} anchor The anchor to set. */RichMarker.prototype.setAnchor = function(anchor) {  this.set('anchor', anchor);};RichMarker.prototype['setAnchor'] = RichMarker.prototype.setAnchor;/** * Anchor changed event. */RichMarker.prototype.anchor_changed = function() {  this.draw();};RichMarker.prototype['anchor_changed'] = RichMarker.prototype.anchor_changed;/** * Converts a HTML string to a document fragment. * * @param {string} htmlString The HTML string to convert. * @return {Node} A HTML document fragment. * @private */RichMarker.prototype.htmlToDocumentFragment_ = function(htmlString) {  var tempDiv = document.createElement('DIV');  tempDiv.innerHTML = htmlString;  if (tempDiv.childNodes.length == 1) {    return /** @type {!Node} */ (tempDiv.removeChild(tempDiv.firstChild));  } else {    var fragment = document.createDocumentFragment();    while (tempDiv.firstChild) {      fragment.appendChild(tempDiv.firstChild);    }    return fragment;  }};/** * Removes all children from the node. * * @param {Node} node The node to remove all children from. * @private */RichMarker.prototype.removeChildren_ = function(node) {  if (!node) {    return;  }  var child;  while (child = node.firstChild) {    node.removeChild(child);  }};/** * Sets the content of the marker. * * @param {string|Node} content The content to set. */RichMarker.prototype.setContent = function(content) {  this.set('content', content);};RichMarker.prototype['setContent'] = RichMarker.prototype.setContent;/** * Get the content of the marker. * * @return {string|Node} The marker content. */RichMarker.prototype.getContent = function() {  return /** @type {Node|string} */ (this.get('content'));};RichMarker.prototype['getContent'] = RichMarker.prototype.getContent;/** * Sets the marker content and adds loading events to images */RichMarker.prototype.content_changed = function() {  if (!this.markerContent_) {    // Marker content area doesnt exist.    return;  }  this.removeChildren_(this.markerContent_);  var content = this.getContent();  if (content) {    if (typeof content == 'string') {      content = content.replace(/^\s*([\S\s]*)\b\s*$/, '$1');      content = this.htmlToDocumentFragment_(content);    }    this.markerContent_.appendChild(content);    var that = this;    var images = this.markerContent_.getElementsByTagName('IMG');    for (var i = 0, image; image = images[i]; i++) {      // By default, a browser lets a image be dragged outside of the browser,      // so by calling preventDefault we stop this behaviour and allow the image      // to be dragged around the map and now out of the browser and onto the      // desktop.      google.maps.event.addDomListener(image, 'mousedown', function(e) {        if (that.getDraggable()) {          if (e.preventDefault) {            e.preventDefault();          }          e.returnValue = false;        }      });      // Because we don't know the size of an image till it loads, add a      // listener to the image load so the marker can resize and reposition      // itself to be the correct height.      google.maps.event.addDomListener(image, 'load', function() {        that.draw();      });    }    google.maps.event.trigger(this, 'domready');  }  if (this.ready_) {    this.draw();  }};RichMarker.prototype['content_changed'] = RichMarker.prototype.content_changed;/** * Sets the cursor. * * @param {string} whichCursor What cursor to show. * @private */RichMarker.prototype.setCursor_ = function(whichCursor) {  if (!this.ready_) {    return;  }  var cursor = '';  if (navigator.userAgent.indexOf('Gecko/') !== -1) {    // Moz has some nice cursors :)    if (whichCursor == 'dragging') {      cursor = '-moz-grabbing';    }    if (whichCursor == 'dragready') {      cursor = '-moz-grab';    }    if (whichCursor == 'draggable') {      cursor = 'pointer';    }  } else {    if (whichCursor == 'dragging' || whichCursor == 'dragready') {      cursor = 'move';    }    if (whichCursor == 'draggable') {      cursor = 'pointer';    }  }  if (this.markerWrapper_.style.cursor != cursor) {    this.markerWrapper_.style.cursor = cursor;  }};/** * Start dragging. * * @param {Event} e The event. */RichMarker.prototype.startDrag = function(e) {  if (!this.getDraggable()) {    return;  }  if (!this.dragging_) {    this.dragging_ = true;    var map = this.getMap();    this.mapDraggable_ = map.get('draggable');    map.set('draggable', false);    // Store the current mouse position    this.mouseX_ = e.clientX;    this.mouseY_ = e.clientY;    this.setCursor_('dragready');    // Stop the text from being selectable while being dragged    this.markerWrapper_.style['MozUserSelect'] = 'none';    this.markerWrapper_.style['KhtmlUserSelect'] = 'none';    this.markerWrapper_.style['WebkitUserSelect'] = 'none';    this.markerWrapper_['unselectable'] = 'on';    this.markerWrapper_['onselectstart'] = function() {      return false;    };    this.addDraggingListeners_();    google.maps.event.trigger(this, 'dragstart');  }};/** * Stop dragging. */RichMarker.prototype.stopDrag = function() {  if (!this.getDraggable()) {    return;  }  if (this.dragging_) {    this.dragging_ = false;    this.getMap().set('draggable', this.mapDraggable_);    this.mouseX_ = this.mouseY_ = this.mapDraggable_ = null;    // Allow the text to be selectable again    this.markerWrapper_.style['MozUserSelect'] = '';    this.markerWrapper_.style['KhtmlUserSelect'] = '';    this.markerWrapper_.style['WebkitUserSelect'] = '';    this.markerWrapper_['unselectable'] = 'off';    this.markerWrapper_['onselectstart'] = function() {};    this.removeDraggingListeners_();    this.setCursor_('draggable');    google.maps.event.trigger(this, 'dragend');    this.draw();  }};/** * Handles the drag event. * * @param {Event} e The event. */RichMarker.prototype.drag = function(e) {  if (!this.getDraggable() || !this.dragging_) {    // This object isn't draggable or we have stopped dragging    this.stopDrag();    return;  }  var dx = this.mouseX_ - e.clientX;  var dy = this.mouseY_ - e.clientY;  this.mouseX_ = e.clientX;  this.mouseY_ = e.clientY;  var left = parseInt(this.markerWrapper_.style['left'], 10) - dx;  var top = parseInt(this.markerWrapper_.style['top'], 10) - dy;  this.markerWrapper_.style['left'] = left + 'px';  this.markerWrapper_.style['top'] = top + 'px';  var offset = this.getOffset_();  // Set the position property and adjust for the anchor offset  var point = new google.maps.Point(left - offset.width, top - offset.height);  var projection = this.getProjection();  this.setPosition(projection.fromDivPixelToLatLng(point));  this.setCursor_('dragging');  google.maps.event.trigger(this, 'drag');};/** * Removes the drag listeners associated with the marker. * * @private */RichMarker.prototype.removeDragListeners_ = function() {  if (this.draggableListener_) {    google.maps.event.removeListener(this.draggableListener_);    delete this.draggableListener_;  }  this.setCursor_('');};/** * Add dragability events to the marker. * * @param {Node} node The node to apply dragging to. * @private */RichMarker.prototype.addDragging_ = function(node) {  if (!node) {    return;  }  var that = this;  this.draggableListener_ =    google.maps.event.addDomListener(node, 'mousedown', function(e) {      that.startDrag(e);    });  this.setCursor_('draggable');};/** * Add dragging listeners. * * @private */RichMarker.prototype.addDraggingListeners_ = function() {  var that = this;  if (this.markerWrapper_.setCapture) {    this.markerWrapper_.setCapture(true);    this.draggingListeners_ = [      google.maps.event.addDomListener(this.markerWrapper_, 'mousemove', function(e) {        that.drag(e);      }, true),      google.maps.event.addDomListener(this.markerWrapper_, 'mouseup', function() {        that.stopDrag();        that.markerWrapper_.releaseCapture();      }, true)    ];  } else {    this.draggingListeners_ = [      google.maps.event.addDomListener(window, 'mousemove', function(e) {        that.drag(e);      }, true),      google.maps.event.addDomListener(window, 'mouseup', function() {        that.stopDrag();      }, true)    ];  }};/** * Remove dragging listeners. * * @private */RichMarker.prototype.removeDraggingListeners_ = function() {  if (this.draggingListeners_) {    for (var i = 0, listener; listener = this.draggingListeners_[i]; i++) {      google.maps.event.removeListener(listener);    }    this.draggingListeners_.length = 0;  }};/** * Get the anchor offset. * * @return {google.maps.Size} The size offset. * @private */RichMarker.prototype.getOffset_ = function() {  var anchor = this.getAnchor();  if (typeof anchor == 'object') {    return /** @type {google.maps.Size} */ (anchor);  }  var offset = new google.maps.Size(0, 0);  if (!this.markerContent_) {    return offset;  }  var width = this.markerContent_.offsetWidth;  var height = this.markerContent_.offsetHeight;  switch (anchor) {   case RichMarkerPosition['TOP_LEFT']:     break;   case RichMarkerPosition['TOP']:     offset.width = -width / 2;     break;   case RichMarkerPosition['TOP_RIGHT']:     offset.width = -width;     break;   case RichMarkerPosition['LEFT']:     offset.height = -height / 2;     break;   case RichMarkerPosition['MIDDLE']:     offset.width = -width / 2;     offset.height = -height / 2;     break;   case RichMarkerPosition['RIGHT']:     offset.width = -width;     offset.height = -height / 2;     break;   case RichMarkerPosition['BOTTOM_LEFT']:     offset.height = -height;     break;   case RichMarkerPosition['BOTTOM']:     offset.width = -width / 2;     offset.height = -height;     break;   case RichMarkerPosition['BOTTOM_RIGHT']:     offset.width = -width;     offset.height = -height;     break;  }  return offset;};/** * Adding the marker to a map. * Implementing the interface. */RichMarker.prototype.onAdd = function() {  if (!this.markerWrapper_) {    this.markerWrapper_ = document.createElement('DIV');    this.markerWrapper_.style['position'] = 'absolute';  }  if (this.getZIndex()) {    this.markerWrapper_.style['zIndex'] = this.getZIndex();  }  this.markerWrapper_.style['display'] = this.getVisible() ? '' : 'none';  if (!this.markerContent_) {    this.markerContent_ = document.createElement('DIV');    this.markerWrapper_.appendChild(this.markerContent_);    var that = this;    google.maps.event.addDomListener(this.markerContent_, 'click', function(e) {      google.maps.event.trigger(that, 'click', e);    });    google.maps.event.addDomListener(this.markerContent_, 'mouseover', function(e) {      google.maps.event.trigger(that, 'mouseover', e);    });    google.maps.event.addDomListener(this.markerContent_, 'mouseout', function(e) {      google.maps.event.trigger(that, 'mouseout', e);    });  }  this.ready_ = true;  this.content_changed();  this.flat_changed();  this.draggable_changed();  var panes = this.getPanes();  if (panes) {    panes.overlayMouseTarget.appendChild(this.markerWrapper_);  }  google.maps.event.trigger(this, 'ready');};RichMarker.prototype['onAdd'] = RichMarker.prototype.onAdd;/** * Impelementing the interface. */RichMarker.prototype.draw = function() {  if (!this.ready_ || this.dragging_) {    return;  }  var projection = this.getProjection();  if (!projection) {    // The map projection is not ready yet so do nothing    return;  }  var latLng = /** @type {google.maps.LatLng} */ (this.get('position'));  var pos = projection.fromLatLngToDivPixel(latLng);  var offset = this.getOffset_();  // console.log(offset)  this.markerWrapper_.style['top'] = (pos.y + -42) + 'px';  this.markerWrapper_.style['left'] = (pos.x + -21) + 'px';  var height = this.markerContent_.offsetHeight;  var width = this.markerContent_.offsetWidth;  if (width != this.get('width')) {    this.set('width', width);  }  if (height != this.get('height')) {    this.set('height', height);  }};RichMarker.prototype['draw'] = RichMarker.prototype.draw;/** * Removing a marker from the map. * Implementing the interface. */RichMarker.prototype.onRemove = function() {  if (this.markerWrapper_ && this.markerWrapper_.parentNode) {    this.markerWrapper_.parentNode.removeChild(this.markerWrapper_);  }  this.removeDragListeners_();};RichMarker.prototype['onRemove'] = RichMarker.prototype.onRemove;/** * RichMarker Anchor positions * @enum {number} */var RichMarkerPosition = {  'TOP_LEFT': 1,  'TOP': 2,  'TOP_RIGHT': 3,  'LEFT': 4,  'MIDDLE': 5,  'RIGHT': 6,  'BOTTOM_LEFT': 7,  'BOTTOM': 8,  'BOTTOM_RIGHT': 9};window['RichMarkerPosition'] = RichMarkerPosition;
 |