Как включить правый щелчок (и средний щелчок) мыши в Hammer.js
Я только начинаю использовать Hammer.js для приложения, которое может использоваться на различных устройствах, и я хотел сделать его максимально функциональным для сенсорного экрана и мыши. Немного погуглив о том, как заставить Hammer распознавать средние и правые щелчки мыши и не находя ответа, я написал свое собственное решение, которое расширяет hammer.js и добавляет свойство "button" к событию, инициируемому мышью. Обратите внимание, что вам придется отключить контекстное меню отдельно, если вы хотите использовать правые клики.
Если у кого-то есть лучшее решение или он видит какие-либо ошибки, пожалуйста, отправьте их...
2 ответа
//extend to allow right click
//input mouse map is not a public property of Hammer, so copy it here
var MOUSE_INPUT_MAP = {
mousedown: Hammer.INPUT_START,
mousemove: Hammer.INPUT_MOVE,
mouseup: Hammer.INPUT_END
};
//override
Hammer.inherit(Hammer.MouseInput, Hammer.Input, {
/**
* handle mouse events
* @param {Object} ev
*/
handler: function MEhandler(ev) {
var eventType = MOUSE_INPUT_MAP[ev.type];
//modified to handle all buttons
//left=0, middle=1, right=2
if (eventType & Hammer.INPUT_START) {
//firefox sends button 0 for mousemove, so store it here
if(this.pressed === false) this.button = ev.button;
this.pressed = true;
}
if (eventType & Hammer.INPUT_MOVE && ev.which === 0) {
eventType = Hammer.INPUT_END;
}
// mouse must be down, and mouse events are allowed (see the TouchMouse input)
if (!this.pressed || !this.allow) {
return;
}
if (eventType & Hammer.INPUT_END) {
this.pressed = false;
this.button = false;
}
this.callback(this.manager, eventType, {
button: this.button,
pointers: [ev],
changedPointers: [ev],
pointerType: 'mouse',
srcEvent: ev
});
}
});
Чтобы IE11 и IE10 работали одинаково, мне пришлось сделать следующее.
Hammer.inherit(Hammer.PointerEventInput, Hammer.Input, {
/**
* handle mouse events
* @param {Object} ev
*/
handler: function PEhandler(ev) {
var store = this.store;
var removePointer = false;
var eventTypeNormalized = ev.type.toLowerCase().replace('ms', '');
var eventType = POINTER_INPUT_MAP[eventTypeNormalized];
var pointerType = IE10_POINTER_TYPE_ENUM[ev.pointerType] || ev.pointerType;
//modified to handle all buttons
//left=0, middle=1, right=2
if (eventType & Hammer.INPUT_START) {
//firefox sends button 0 for mousemove, so store it here
this.button = ev.button;
}
var isTouch = (pointerType == Hammer.INPUT_TYPE_TOUCH);
// get index of the event in the store
var storeIndex = inArray(store, ev.pointerId, 'pointerId');
// start and mouse must be down
if (eventType & Hammer.INPUT_START && (ev.button === 0 || ev.button === 1 || ev.button === 2 || isTouch)) {
if (storeIndex < 0) {
store.push(ev);
storeIndex = store.length - 1;
}
} else if (eventType & (Hammer.INPUT_END | Hammer.INPUT_CANCEL)) {
removePointer = true;
}
// it not found, so the pointer hasn't been down (so it's probably a hover)
if (storeIndex < 0) {
return;
}
// update the event in the store
store[storeIndex] = ev;
this.callback(this.manager, eventType, {
button: this.button +1,
pointers: store,
changedPointers: [ev],
pointerType: pointerType,
srcEvent: ev
});
if (removePointer) {
// remove from the store
store.splice(storeIndex, 1);
}
}
});
Надеюсь, это поможет кому-то.
Добавьте к этому следующий патч для обработки PointerEvents (также копируя необходимые значения / функции, где Hammer их не раскрывает):
var POINTER_INPUT_MAP = {
pointerdown: Hammer.INPUT_START,
pointermove: Hammer.INPUT_MOVE,
pointerup: Hammer.INPUT_END,
pointercancel: Hammer.INPUT_CANCEL,
pointerout: Hammer.INPUT_CANCEL
};
var INPUT_TYPE_TOUCH = 'touch';
var INPUT_TYPE_PEN = 'pen';
var INPUT_TYPE_MOUSE = 'mouse';
var INPUT_TYPE_KINECT = 'kinect';
// in IE10 the pointer types is defined as an enum
var IE10_POINTER_TYPE_ENUM = {
2: INPUT_TYPE_TOUCH,
3: INPUT_TYPE_PEN,
4: INPUT_TYPE_MOUSE,
5: INPUT_TYPE_KINECT // see https://twitter.com/jacobrossi/status/480596438489890816
};
/**
* find if a array contains the object using indexOf or a simple polyFill
* @param {Array} src
* @param {String} find
* @param {String} [findByKey]
* @return {Boolean|Number} false when not found, or the index
*/
function inArray(src, find, findByKey) {
if (src.indexOf && !findByKey) {
return src.indexOf(find);
} else {
var i = 0;
while (i < src.length) {
if ((findByKey && src[i][findByKey] == find) || (!findByKey && src[i] === find)) {
return i;
}
i++;
}
return -1;
}
}
Hammer.inherit(Hammer.PointerEventInput, Hammer.Input, {
/**
* handle mouse events
* @param {Object} ev
*/
handler: function PEhandler(ev) {
var store = this.store;
var removePointer = false;
var eventTypeNormalized = ev.type.toLowerCase().replace('ms', '');
var eventType = POINTER_INPUT_MAP[eventTypeNormalized];
var pointerType = IE10_POINTER_TYPE_ENUM[ev.pointerType] || ev.pointerType;
//var isTouch = (pointerType == Hammer.INPUT_TYPE_TOUCH);
// get index of the event in the store
var storeIndex = inArray(store, ev.pointerId, 'pointerId');
// start and mouse must be down
if (eventType & Hammer.INPUT_START)
{
this.button = ev.button;
if (storeIndex < 0) {
store.push(ev);
storeIndex = store.length - 1;
}
} else if (eventType & (Hammer.INPUT_END | Hammer.INPUT_CANCEL)) {
this.button = false;
removePointer = true;
}
// it not found, so the pointer hasn't been down (so it's probably a hover)
if (storeIndex < 0) {
return;
}
// update the event in the store
store[storeIndex] = ev;
this.callback(this.manager, eventType, {
button: this.button,
pointers: store,
changedPointers: [ev],
pointerType: pointerType,
srcEvent: ev
});
if (removePointer) {
// remove from the store
store.splice(storeIndex, 1);
}
}
});