UNPKG

leaflet

Version:

JavaScript library for mobile-friendly interactive maps

224 lines (183 loc) 6.88 kB
/* * @class Tooltip * @inherits DivOverlay * @aka L.Tooltip * Used to display small texts on top of map layers. * * @example * * ```js * marker.bindTooltip("my tooltip text").openTooltip(); * ``` * Note about tooltip offset. Leaflet takes two options in consideration * for computing tooltip offseting: * - the `offset` Tooltip option: it defaults to [0, 0], and it's specific to one tooltip. * Add a positive x offset to move the tooltip to the right, and a positive y offset to * move it to the bottom. Negatives will move to the left and top. * - the `tooltipAnchor` Icon option: this will only be considered for Marker. You * should adapt this value if you use a custom icon. */ // @namespace Tooltip L.Tooltip = L.DivOverlay.extend({ // @section // @aka Tooltip options options: { // @option pane: String = 'tooltipPane' // `Map pane` where the tooltip will be added. pane: 'tooltipPane', // @option offset: Point = Point(0, 0) // Optional offset of the tooltip position. offset: [0, 0], // @option direction: String = 'auto' // Direction where to open the tooltip. Possible values are: `right`, `left`, // `top`, `bottom`, `center`, `auto`. // `auto` will dynamicaly switch between `right` and `left` according to the tooltip // position on the map. direction: 'auto', // @option permanent: Boolean = false // Whether to open the tooltip permanently or only on mouseover. permanent: false, // @option sticky: Boolean = false // If true, the tooltip will follow the mouse instead of being fixed at the feature center. sticky: false, // @option interactive: Boolean = false // If true, the tooltip will listen to the feature events. interactive: false, // @option opacity: Number = 0.9 // Tooltip container opacity. opacity: 0.9 }, onAdd: function (map) { L.DivOverlay.prototype.onAdd.call(this, map); this.setOpacity(this.options.opacity); // @namespace Map // @section Tooltip events // @event tooltipopen: TooltipEvent // Fired when a tooltip is opened in the map. map.fire('tooltipopen', {tooltip: this}); if (this._source) { // @namespace Layer // @section Tooltip events // @event tooltipopen: TooltipEvent // Fired when a tooltip bound to this layer is opened. this._source.fire('tooltipopen', {tooltip: this}, true); } }, onRemove: function (map) { L.DivOverlay.prototype.onRemove.call(this, map); // @namespace Map // @section Tooltip events // @event tooltipclose: TooltipEvent // Fired when a tooltip in the map is closed. map.fire('tooltipclose', {tooltip: this}); if (this._source) { // @namespace Layer // @section Tooltip events // @event tooltipclose: TooltipEvent // Fired when a tooltip bound to this layer is closed. this._source.fire('tooltipclose', {tooltip: this}, true); } }, getEvents: function () { var events = L.DivOverlay.prototype.getEvents.call(this); if (L.Browser.touch && !this.options.permanent) { events.preclick = this._close; } return events; }, _close: function () { if (this._map) { this._map.closeTooltip(this); } }, _initLayout: function () { var prefix = 'leaflet-tooltip', className = prefix + ' ' + (this.options.className || '') + ' leaflet-zoom-' + (this._zoomAnimated ? 'animated' : 'hide'); this._contentNode = this._container = L.DomUtil.create('div', className); }, _updateLayout: function () {}, _adjustPan: function () {}, _setPosition: function (pos) { var map = this._map, container = this._container, centerPoint = map.latLngToContainerPoint(map.getCenter()), tooltipPoint = map.layerPointToContainerPoint(pos), direction = this.options.direction, tooltipWidth = container.offsetWidth, tooltipHeight = container.offsetHeight, offset = L.point(this.options.offset), anchor = this._getAnchor(); if (direction === 'top') { pos = pos.add(L.point(-tooltipWidth / 2 + offset.x, -tooltipHeight + offset.y + anchor.y)); } else if (direction === 'bottom') { pos = pos.subtract(L.point(tooltipWidth / 2 - offset.x, -offset.y)); } else if (direction === 'center') { pos = pos.subtract(L.point(tooltipWidth / 2 + offset.x, tooltipHeight / 2 - anchor.y + offset.y)); } else if (direction === 'right' || direction === 'auto' && tooltipPoint.x < centerPoint.x) { direction = 'right'; pos = pos.add([offset.x + anchor.x, anchor.y - tooltipHeight / 2 + offset.y]); } else { direction = 'left'; pos = pos.subtract(L.point(tooltipWidth + anchor.x - offset.x, tooltipHeight / 2 - anchor.y - offset.y)); } L.DomUtil.removeClass(container, 'leaflet-tooltip-right'); L.DomUtil.removeClass(container, 'leaflet-tooltip-left'); L.DomUtil.removeClass(container, 'leaflet-tooltip-top'); L.DomUtil.removeClass(container, 'leaflet-tooltip-bottom'); L.DomUtil.addClass(container, 'leaflet-tooltip-' + direction); L.DomUtil.setPosition(container, pos); }, _updatePosition: function () { var pos = this._map.latLngToLayerPoint(this._latlng); this._setPosition(pos); }, setOpacity: function (opacity) { this.options.opacity = opacity; if (this._container) { L.DomUtil.setOpacity(this._container, opacity); } }, _animateZoom: function (e) { var pos = this._map._latLngToNewLayerPoint(this._latlng, e.zoom, e.center); this._setPosition(pos); }, _getAnchor: function () { // Where should we anchor the tooltip on the source layer? return L.point(this._source && this._source._getTooltipAnchor && !this.options.sticky ? this._source._getTooltipAnchor() : [0, 0]); } }); // @namespace Tooltip // @factory L.tooltip(options?: Tooltip options, source?: Layer) // Instantiates a Tooltip object given an optional `options` object that describes its appearance and location and an optional `source` object that is used to tag the tooltip with a reference to the Layer to which it refers. L.tooltip = function (options, source) { return new L.Tooltip(options, source); }; // @namespace Map // @section Methods for Layers and Controls L.Map.include({ // @method openTooltip(tooltip: Tooltip): this // Opens the specified tooltip. // @alternative // @method openTooltip(content: String|HTMLElement, latlng: LatLng, options?: Tooltip options): this // Creates a tooltip with the specified content and options and open it. openTooltip: function (tooltip, latlng, options) { if (!(tooltip instanceof L.Tooltip)) { tooltip = new L.Tooltip(options).setContent(tooltip); } if (latlng) { tooltip.setLatLng(latlng); } if (this.hasLayer(tooltip)) { return this; } return this.addLayer(tooltip); }, // @method closeTooltip(tooltip?: Tooltip): this // Closes the tooltip given as parameter. closeTooltip: function (tooltip) { if (tooltip) { this.removeLayer(tooltip); } return this; } });