UNPKG

mapbox-gl

Version:
164 lines (130 loc) 5.31 kB
'use strict'; var DOM = require('../../util/dom'), browser = require('../../util/browser'), util = require('../../util/util'); module.exports = ScrollZoomHandler; var ua = typeof navigator !== 'undefined' ? navigator.userAgent.toLowerCase() : '', firefox = ua.indexOf('firefox') !== -1, safari = ua.indexOf('safari') !== -1 && ua.indexOf('chrom') === -1; /** * The `ScrollZoomHandler` allows a user to zoom the map by scrolling. * @class ScrollZoomHandler */ function ScrollZoomHandler(map) { this._map = map; this._el = map.getCanvasContainer(); util.bindHandlers(this); } ScrollZoomHandler.prototype = { /** * Enable the "scroll to zoom" interaction. * @example * map.scrollZoom.enable(); */ enable: function () { this.disable(); this._el.addEventListener('wheel', this._onWheel, false); this._el.addEventListener('mousewheel', this._onWheel, false); }, /** * Disable the "scroll to zoom" interaction. * @example * map.scrollZoom.disable(); */ disable: function () { this._el.removeEventListener('wheel', this._onWheel); this._el.removeEventListener('mousewheel', this._onWheel); }, _onWheel: function (e) { var value; if (e.type === 'wheel') { value = e.deltaY; // Firefox doubles the values on retina screens... if (firefox && e.deltaMode === window.WheelEvent.DOM_DELTA_PIXEL) value /= browser.devicePixelRatio; if (e.deltaMode === window.WheelEvent.DOM_DELTA_LINE) value *= 40; } else if (e.type === 'mousewheel') { value = -e.wheelDeltaY; if (safari) value = value / 3; } var now = browser.now(), timeDelta = now - (this._time || 0); this._pos = DOM.mousePos(this._el, e); this._time = now; if (value !== 0 && (value % 4.000244140625) === 0) { // This one is definitely a mouse wheel event. this._type = 'wheel'; // Normalize this value to match trackpad. value = Math.floor(value / 4); } else if (value !== 0 && Math.abs(value) < 4) { // This one is definitely a trackpad event because it is so small. this._type = 'trackpad'; } else if (timeDelta > 400) { // This is likely a new scroll action. this._type = null; this._lastValue = value; // Start a timeout in case this was a singular event, and dely it by up to 40ms. this._timeout = setTimeout(this._onTimeout, 40); } else if (!this._type) { // This is a repeating event, but we don't know the type of event just yet. // If the delta per time is small, we assume it's a fast trackpad; otherwise we switch into wheel mode. this._type = (Math.abs(timeDelta * value) < 200) ? 'trackpad' : 'wheel'; // Make sure our delayed event isn't fired again, because we accumulate // the previous event (which was less than 40ms ago) into this event. if (this._timeout) { clearTimeout(this._timeout); this._timeout = null; value += this._lastValue; } } // Slow down zoom if shift key is held for more precise zooming if (e.shiftKey && value) value = value / 4; // Only fire the callback if we actually know what type of scrolling device the user uses. if (this._type) this._zoom(-value, e); e.preventDefault(); }, _onTimeout: function () { this._type = 'wheel'; this._zoom(-this._lastValue); }, _zoom: function (delta, e) { if (delta === 0) return; var map = this._map; // Scale by sigmoid of scroll wheel delta. var scale = 2 / (1 + Math.exp(-Math.abs(delta / 100))); if (delta < 0 && scale !== 0) scale = 1 / scale; var fromScale = map.ease ? map.ease.to : map.transform.scale, targetZoom = map.transform.scaleZoom(fromScale * scale); map.zoomTo(targetZoom, { duration: 0, around: map.unproject(this._pos), delayEndEvents: 200 }, { originalEvent: e }); } }; /** * Zoom start event. This event is emitted just before the map begins a transition from one * zoom level to another, either as a result of user interaction or the use of methods such as `Map#jumpTo`. * * @event zoomstart * @memberof Map * @instance * @property {EventData} data Original event data, if fired interactively */ /** * Zoom event. This event is emitted repeatedly during animated transitions from one zoom level to * another, either as a result of user interaction or the use of methods such as `Map#jumpTo`. * * @event zoom * @memberof Map * @instance * @property {EventData} data Original event data, if fired interactively */ /** * Zoom end event. This event is emitted just after the map completes a transition from one * zoom level to another, either as a result of user interaction or the use of methods such as `Map#jumpTo`. * * @event zoomend * @memberof Map * @instance * @property {EventData} data Original event data, if fired interactively */