Как включить правый щелчок (и средний щелчок) мыши в 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);
        }
    }
});
Другие вопросы по тегам