123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208 |
- /* AlloyFinger v0.1.3
- * By dntzhang
- * Github: https://github.com/AlloyTeam/AlloyFinger
- */
- ; (function () {
- function getLen(v) {
- return Math.sqrt(v.x * v.x + v.y * v.y);
- }
- function dot(v1, v2) {
- return v1.x * v2.x + v1.y * v2.y;
- }
- function getAngle(v1, v2) {
- var mr = getLen(v1) * getLen(v2);
- if (mr === 0) return 0;
- var r = dot(v1, v2) / mr;
- if (r > 1) r = 1;
- return Math.acos(r);
- }
- function cross(v1, v2) {
- return v1.x * v2.y - v2.x * v1.y;
- }
- function getRotateAngle(v1, v2) {
- var angle = getAngle(v1, v2);
- if (cross(v1, v2) > 0) {
- angle *= -1;
- }
- return angle * 180 / Math.PI;
- }
- var AlloyFinger = function (el, option) {
- this.element = typeof el == 'string' ? document.querySelector(el) : el;
- this.element.addEventListener("touchstart", this.start.bind(this), false);
- this.element.addEventListener("touchmove", this.move.bind(this), false);
- this.element.addEventListener("touchend", this.end.bind(this), false);
- this.element.addEventListener("touchcancel", this.cancel.bind(this), false);
- this.preV = { x: null, y: null };
- this.pinchStartLen = null;
- this.scale = 1;
- this.isDoubleTap = false;
- var noop = function () { };
- this.rotate = option.rotate || noop;
- this.touchStart = option.touchStart || noop;
- this.multipointStart = option.multipointStart || noop;
- this.multipointEnd = option.multipointEnd || noop;
- this.pinch = option.pinch || noop;
- this.swipe = option.swipe || noop;
- this.tap = option.tap || noop;
- this.doubleTap = option.doubleTap || noop;
- this.longTap = option.longTap || noop;
- this.singleTap = option.singleTap || noop;
- this.pressMove = option.pressMove || noop;
- this.touchMove = option.touchMove || noop;
- this.touchEnd = option.touchEnd || noop;
- this.touchCancel = option.touchCancel || noop;
- this.delta = null;
- this.last = null;
- this.now = null;
- this.tapTimeout = null;
- this.touchTimeout = null;
- this.longTapTimeout = null;
- this.swipeTimeout = null;
- this.x1 = this.x2 = this.y1 = this.y2 = null;
- this.preTapPosition = { x: null, y: null };
- };
- AlloyFinger.prototype = {
- start: function (evt) {
- if (!evt.touches) return;
- this.now = Date.now();
- this.x1 = evt.touches[0].pageX;
- this.y1 = evt.touches[0].pageY;
- this.delta = this.now - (this.last || this.now);
- this.touchStart(evt);
- if (this.preTapPosition.x !== null) {
- this.isDoubleTap = (this.delta > 0 && this.delta <= 250 && Math.abs(this.preTapPosition.x - this.x1) < 30 && Math.abs(this.preTapPosition.y - this.y1) < 30);
- }
- this.preTapPosition.x = this.x1;
- this.preTapPosition.y = this.y1;
- this.last = this.now;
- var preV = this.preV,
- len = evt.touches.length;
- if (len > 1) {
- this._cancelLongTap();
- var v = { x: evt.touches[1].pageX - this.x1, y: evt.touches[1].pageY - this.y1 };
- preV.x = v.x;
- preV.y = v.y;
- this.pinchStartLen = getLen(preV);
- this.multipointStart(evt);
- }
- this.longTapTimeout = setTimeout(function () {
- this.longTap(evt);
- }.bind(this), 750);
- },
- move: function (evt) {
- if (!evt.touches) return;
- var preV = this.preV,
- len = evt.touches.length,
- currentX = evt.touches[0].pageX,
- currentY = evt.touches[0].pageY;
- this.isDoubleTap = false;
- if (len > 1) {
- var v = { x: evt.touches[1].pageX - currentX, y: evt.touches[1].pageY - currentY };
- if (preV.x !== null) {
- if (this.pinchStartLen > 0) {
- evt.scale = getLen(v) / this.pinchStartLen;
- this.pinch(evt);
- }
- evt.angle = getRotateAngle(v, preV);
- this.rotate(evt);
- }
- preV.x = v.x;
- preV.y = v.y;
- } else {
- if (this.x2 !== null) {
- evt.deltaX = currentX - this.x2;
- evt.deltaY = currentY - this.y2;
- } else {
- evt.deltaX = 0;
- evt.deltaY = 0;
- }
- this.pressMove(evt);
- }
- this.touchMove(evt);
- this._cancelLongTap();
- this.x2 = currentX;
- this.y2 = currentY;
- if (evt.touches.length > 1) {
- this._cancelLongTap();
- evt.preventDefault();
- }
- },
- end: function (evt) {
- if (!evt.changedTouches) return;
- this._cancelLongTap();
- var self = this;
- if (evt.touches.length < 2) {
- this.multipointEnd(evt);
- }
- this.touchEnd(evt);
- //swipe
- if ((this.x2 && Math.abs(this.x1 - this.x2) > 30) ||
- (this.y2 && Math.abs(this.preV.y - this.y2) > 30)) {
- evt.direction = this._swipeDirection(this.x1, this.x2, this.y1, this.y2);
- this.swipeTimeout = setTimeout(function () {
- self.swipe(evt);
- }, 0)
- } else {
- this.tapTimeout = setTimeout(function () {
- self.tap(evt);
- // trigger double tap immediately
- if (self.isDoubleTap) {
- self.doubleTap(evt);
- clearTimeout(self.touchTimeout);
- self.isDoubleTap = false;
- } else {
- self.touchTimeout = setTimeout(function () {
- self.singleTap(evt);
- }, 250);
- }
- }, 0)
- }
- this.preV.x = 0;
- this.preV.y = 0;
- this.scale = 1;
- this.pinchStartLen = null;
- this.x1 = this.x2 = this.y1 = this.y2 = null;
- },
- cancel: function (evt) {
- clearTimeout(this.touchTimeout);
- clearTimeout(this.tapTimeout);
- clearTimeout(this.longTapTimeout);
- clearTimeout(this.swipeTimeout);
- this.touchCancel(evt);
- },
- _cancelLongTap: function () {
- clearTimeout(this.longTapTimeout);
- },
- _swipeDirection: function (x1, x2, y1, y2) {
- return Math.abs(x1 - x2) >= Math.abs(y1 - y2) ? (x1 - x2 > 0 ? 'Left' : 'Right') : (y1 - y2 > 0 ? 'Up' : 'Down')
- }
- };
- if (typeof module !== 'undefined' && typeof exports === 'object') {
- module.exports = AlloyFinger;
- } else {
- window.AlloyFinger = AlloyFinger;
- }
- })();
|