mirror of
https://github.com/lxsang/antd-web-apps
synced 2025-01-03 20:38:21 +01:00
328 lines
8.3 KiB
JavaScript
328 lines
8.3 KiB
JavaScript
/*
|
|
* Hamster.js v1.1.2
|
|
* (c) 2013 Monospaced http://monospaced.com
|
|
* License: MIT
|
|
*/
|
|
|
|
(function(window, document){
|
|
'use strict';
|
|
|
|
/**
|
|
* Hamster
|
|
* use this to create instances
|
|
* @returns {Hamster.Instance}
|
|
* @constructor
|
|
*/
|
|
var Hamster = function(element) {
|
|
return new Hamster.Instance(element);
|
|
};
|
|
|
|
// default event name
|
|
Hamster.SUPPORT = 'wheel';
|
|
|
|
// default DOM methods
|
|
Hamster.ADD_EVENT = 'addEventListener';
|
|
Hamster.REMOVE_EVENT = 'removeEventListener';
|
|
Hamster.PREFIX = '';
|
|
|
|
// until browser inconsistencies have been fixed...
|
|
Hamster.READY = false;
|
|
|
|
Hamster.Instance = function(element){
|
|
if (!Hamster.READY) {
|
|
// fix browser inconsistencies
|
|
Hamster.normalise.browser();
|
|
|
|
// Hamster is ready...!
|
|
Hamster.READY = true;
|
|
}
|
|
|
|
this.element = element;
|
|
|
|
// store attached event handlers
|
|
this.handlers = [];
|
|
|
|
// return instance
|
|
return this;
|
|
};
|
|
|
|
/**
|
|
* create new hamster instance
|
|
* all methods should return the instance itself, so it is chainable.
|
|
* @param {HTMLElement} element
|
|
* @returns {Hamster.Instance}
|
|
* @constructor
|
|
*/
|
|
Hamster.Instance.prototype = {
|
|
/**
|
|
* bind events to the instance
|
|
* @param {Function} handler
|
|
* @param {Boolean} useCapture
|
|
* @returns {Hamster.Instance}
|
|
*/
|
|
wheel: function onEvent(handler, useCapture){
|
|
Hamster.event.add(this, Hamster.SUPPORT, handler, useCapture);
|
|
|
|
// handle MozMousePixelScroll in older Firefox
|
|
if (Hamster.SUPPORT === 'DOMMouseScroll') {
|
|
Hamster.event.add(this, 'MozMousePixelScroll', handler, useCapture);
|
|
}
|
|
|
|
return this;
|
|
},
|
|
|
|
/**
|
|
* unbind events to the instance
|
|
* @param {Function} handler
|
|
* @param {Boolean} useCapture
|
|
* @returns {Hamster.Instance}
|
|
*/
|
|
unwheel: function offEvent(handler, useCapture){
|
|
// if no handler argument,
|
|
// unbind the last bound handler (if exists)
|
|
if (handler === undefined && (handler = this.handlers.slice(-1)[0])) {
|
|
handler = handler.original;
|
|
}
|
|
|
|
Hamster.event.remove(this, Hamster.SUPPORT, handler, useCapture);
|
|
|
|
// handle MozMousePixelScroll in older Firefox
|
|
if (Hamster.SUPPORT === 'DOMMouseScroll') {
|
|
Hamster.event.remove(this, 'MozMousePixelScroll', handler, useCapture);
|
|
}
|
|
|
|
return this;
|
|
}
|
|
};
|
|
|
|
Hamster.event = {
|
|
/**
|
|
* cross-browser 'addWheelListener'
|
|
* @param {Instance} hamster
|
|
* @param {String} eventName
|
|
* @param {Function} handler
|
|
* @param {Boolean} useCapture
|
|
*/
|
|
add: function add(hamster, eventName, handler, useCapture){
|
|
// store the original handler
|
|
var originalHandler = handler;
|
|
|
|
// redefine the handler
|
|
handler = function(originalEvent){
|
|
|
|
if (!originalEvent) {
|
|
originalEvent = window.event;
|
|
}
|
|
|
|
// create a normalised event object,
|
|
// and normalise "deltas" of the mouse wheel
|
|
var event = Hamster.normalise.event(originalEvent),
|
|
delta = Hamster.normalise.delta(originalEvent);
|
|
|
|
// fire the original handler with normalised arguments
|
|
return originalHandler(event, delta[0], delta[1], delta[2]);
|
|
|
|
};
|
|
|
|
// cross-browser addEventListener
|
|
hamster.element[Hamster.ADD_EVENT](Hamster.PREFIX + eventName, handler, useCapture || false);
|
|
|
|
// store original and normalised handlers on the instance
|
|
hamster.handlers.push({
|
|
original: originalHandler,
|
|
normalised: handler
|
|
});
|
|
},
|
|
|
|
/**
|
|
* removeWheelListener
|
|
* @param {Instance} hamster
|
|
* @param {String} eventName
|
|
* @param {Function} handler
|
|
* @param {Boolean} useCapture
|
|
*/
|
|
remove: function remove(hamster, eventName, handler, useCapture){
|
|
// find the normalised handler on the instance
|
|
var originalHandler = handler,
|
|
lookup = {},
|
|
handlers;
|
|
for (var i = 0, len = hamster.handlers.length; i < len; ++i) {
|
|
lookup[hamster.handlers[i].original] = hamster.handlers[i];
|
|
}
|
|
handlers = lookup[originalHandler];
|
|
handler = handlers.normalised;
|
|
|
|
// cross-browser removeEventListener
|
|
hamster.element[Hamster.REMOVE_EVENT](Hamster.PREFIX + eventName, handler, useCapture || false);
|
|
|
|
// remove original and normalised handlers from the instance
|
|
for (var h in hamster.handlers) {
|
|
if (hamster.handlers[h] == handlers) {
|
|
hamster.handlers.splice(h, 1);
|
|
break;
|
|
}
|
|
}
|
|
}
|
|
};
|
|
|
|
/**
|
|
* these hold the lowest deltas,
|
|
* used to normalise the delta values
|
|
* @type {Number}
|
|
*/
|
|
var lowestDelta,
|
|
lowestDeltaXY;
|
|
|
|
Hamster.normalise = {
|
|
/**
|
|
* fix browser inconsistencies
|
|
*/
|
|
browser: function normaliseBrowser(){
|
|
// detect deprecated wheel events
|
|
if (!('onwheel' in document || document.documentMode >= 9)) {
|
|
Hamster.SUPPORT = document.onmousewheel !== undefined ?
|
|
'mousewheel' : // webkit and IE < 9 support at least "mousewheel"
|
|
'DOMMouseScroll'; // assume remaining browsers are older Firefox
|
|
}
|
|
|
|
// detect deprecated event model
|
|
if (!window.addEventListener) {
|
|
// assume IE < 9
|
|
Hamster.ADD_EVENT = 'attachEvent';
|
|
Hamster.REMOVE_EVENT = 'detachEvent';
|
|
Hamster.PREFIX = 'on';
|
|
}
|
|
|
|
},
|
|
|
|
/**
|
|
* create a normalised event object
|
|
* @param {Function} originalEvent
|
|
* @returns {Object} event
|
|
*/
|
|
event: function normaliseEvent(originalEvent){
|
|
var event = {
|
|
// keep a reference to the original event object
|
|
originalEvent: originalEvent,
|
|
target: originalEvent.target || originalEvent.srcElement,
|
|
type: 'wheel',
|
|
deltaMode: originalEvent.type === 'MozMousePixelScroll' ? 0 : 1,
|
|
deltaX: 0,
|
|
deltaZ: 0,
|
|
preventDefault: function(){
|
|
if (originalEvent.preventDefault) {
|
|
originalEvent.preventDefault();
|
|
} else {
|
|
originalEvent.returnValue = false;
|
|
}
|
|
},
|
|
stopPropagation: function(){
|
|
if (originalEvent.stopPropagation) {
|
|
originalEvent.stopPropagation();
|
|
} else {
|
|
originalEvent.cancelBubble = false;
|
|
}
|
|
}
|
|
};
|
|
|
|
// calculate deltaY (and deltaX) according to the event
|
|
|
|
// 'mousewheel'
|
|
if (originalEvent.wheelDelta) {
|
|
event.deltaY = - 1/40 * originalEvent.wheelDelta;
|
|
}
|
|
// webkit
|
|
if (originalEvent.wheelDeltaX) {
|
|
event.deltaX = - 1/40 * originalEvent.wheelDeltaX;
|
|
}
|
|
|
|
// 'DomMouseScroll'
|
|
if (originalEvent.detail) {
|
|
event.deltaY = originalEvent.detail;
|
|
}
|
|
|
|
return event;
|
|
},
|
|
|
|
/**
|
|
* normalise 'deltas' of the mouse wheel
|
|
* @param {Function} originalEvent
|
|
* @returns {Array} deltas
|
|
*/
|
|
delta: function normaliseDelta(originalEvent){
|
|
var delta = 0,
|
|
deltaX = 0,
|
|
deltaY = 0,
|
|
absDelta = 0,
|
|
absDeltaXY = 0,
|
|
fn;
|
|
|
|
// normalise deltas according to the event
|
|
|
|
// 'wheel' event
|
|
if (originalEvent.deltaY) {
|
|
deltaY = originalEvent.deltaY * -1;
|
|
delta = deltaY;
|
|
}
|
|
if (originalEvent.deltaX) {
|
|
deltaX = originalEvent.deltaX;
|
|
delta = deltaX * -1;
|
|
}
|
|
|
|
// 'mousewheel' event
|
|
if (originalEvent.wheelDelta) {
|
|
delta = originalEvent.wheelDelta;
|
|
}
|
|
// webkit
|
|
if (originalEvent.wheelDeltaY) {
|
|
deltaY = originalEvent.wheelDeltaY;
|
|
}
|
|
if (originalEvent.wheelDeltaX) {
|
|
deltaX = originalEvent.wheelDeltaX * -1;
|
|
}
|
|
|
|
// 'DomMouseScroll' event
|
|
if (originalEvent.detail) {
|
|
delta = originalEvent.detail * -1;
|
|
}
|
|
|
|
// Don't return NaN
|
|
if (delta === 0) {
|
|
return [0, 0, 0];
|
|
}
|
|
|
|
// look for lowest delta to normalize the delta values
|
|
absDelta = Math.abs(delta);
|
|
if (!lowestDelta || absDelta < lowestDelta) {
|
|
lowestDelta = absDelta;
|
|
}
|
|
absDeltaXY = Math.max(Math.abs(deltaY), Math.abs(deltaX));
|
|
if (!lowestDeltaXY || absDeltaXY < lowestDeltaXY) {
|
|
lowestDeltaXY = absDeltaXY;
|
|
}
|
|
|
|
// convert deltas to whole numbers
|
|
fn = delta > 0 ? 'floor' : 'ceil';
|
|
delta = Math[fn](delta / lowestDelta);
|
|
deltaX = Math[fn](deltaX / lowestDeltaXY);
|
|
deltaY = Math[fn](deltaY / lowestDeltaXY);
|
|
|
|
return [delta, deltaX, deltaY];
|
|
}
|
|
};
|
|
|
|
if (typeof window.define === 'function' && window.define.amd) {
|
|
// AMD
|
|
window.define('hamster', [], function(){
|
|
return Hamster;
|
|
});
|
|
} else if (typeof exports === 'object') {
|
|
// CommonJS
|
|
module.exports = Hamster;
|
|
} else {
|
|
// Browser global
|
|
window.Hamster = Hamster;
|
|
}
|
|
|
|
})(window, window.document);
|