通过html5 touch事件封装手势识别组件
html5移动端新增了touchstart,touchmove,touchend事件,利用这3个事件,判断手指的点击和划动轨迹,我们可以封装各种手势的识别功能,
这3个事件和pc端的mousedown,mousemove,mouveup非常类似,不同的是touch事件可以有多个点击的点,而鼠标每次只有一个点,我们即然是做组件封装,就要考虑在pc上调试的情况,否则用手机调试非常不方便,通过对mouse事件的处理,可以一套代码同时兼容pc端和移动端。
下面来逐步封装一个滑动手势(swipe)的组件
1.判断是否触摸屏
我们使用能力检测,检测是否支持touchstart事件,就可以知道是否是触摸屏,因为触摸事件可以通过document.ontouchstart=function(){} 这样的方式定义,用in操作符判断即可,对于win8,触屏能力会在navigator对象中生成一个msPointerEnabled属性。
if ('ontouchstart' in window || 'ontouchstart' in document) { //iOS & android supportsTouch = true; } else if(window.navigator.msPointerEnabled) { //Win8 supportsTouch = true; }
2.同时兼容鼠标和触摸屏的事件绑定
我们根据上一步的判断,如果支持toucestart就绑定对应的touchstart,touchmove,touchend事件,如果不支持,则绑定对应的3个鼠标事件
if(isSupportTouch()){ el.addEventListener('touchstart',touchStart); el.addEventListener('touchend',touchEnd); el.addEventListener('touchmove',touchMove); }else{ el.addEventListener('mousedown',touchStart); el.addEventListener('mouseup',touchEnd); el.addEventListener('mousemove',touchMove); }
3.获取点击的点位置信息(兼容鼠标和触摸屏)
从事件参数中可以得到位置信息,如果是鼠标,则通过e.pageX,e.pageY获取点击位置相对于页面根节点的坐标,如果是触摸屏,则e.touches对象是一个点击点位置的数组,包含多个手指的点击位置,我们暂时只处理一只手指的情况,所以取e.touches[0].pageX,e.touches[0].pageY.
function touchStart(e){ var t=e.touches?e.touches[0]:e; startPoint={x:t.pageX,y:t.pageY}; }
4.判断手指滑动方向
在toucemove事件中判断手指划动,toucemove事件会连续触发,为了过滤掉划动距离太短的无效滑动,我们可以判断pageX和pageY和上一次位置的偏移量超过两个像素才认为是有效事件,然后再判断滑动方向,当前点击位置的(x,y)坐标,减去上一个位置的(x,y)坐标,如果x轴的差值大,就认为是左右滑,如果是y轴的差值大就认为是上下滑,再进一步判断差值 为正数则是左或上,差值为负数则为右或下。代码如下:
function getSwipeDirection(x1, x2, y1, y2) { return Math.abs(x1 - x2) >= Math.abs(y1 - y2) ? (x1 - x2 > 0 ? 'left' : 'right') : (y1 - y2 >0 ? 'up' : 'down') }
5.jquery插件封装
为了更方便使用,可以封装成jquery插件,我们常说的jquery对象其实是指继随自jquery原型的对象,jquery的原型是指$.fn,只要扩展$.fn即可,
如$.fn.methodName=function(){//code}
或用$.fn.extend({
methodName:funciton(){//code}
})
完整代码如下:
function TouchEvent(){ var self=this,element=$(this); var el=element[0],isTouching,isSwipe,startTime,startPoint,currentPoint; if(arguments.length>1){ var eventType=arguments[0]; } var callback=arguments[arguments.length-1]; function doAction(type,args){ args.type=type; if(eventType){ if(eventType==type){ callback.call(self,args); } }else{ callback.call(self,args); } } function getSwipeDirection(x1, x2, y1, y2) { return Math.abs(x1 - x2) >= Math.abs(y1 - y2) ? (x1 - x2 > 0 ? 'left' : 'right') : (y1 - y2 >0 ? 'up' : 'down') } function isSupportTouch(){ var supportsTouch = false; if ('ontouchstart' in window || 'ontouchstart' in document) { //iOS & android supportsTouch = true; } else if(window.navigator.msPointerEnabled) { //Win8 supportsTouch = true; } return supportsTouch; } function touchStart(e){ isTouching=true; startTime=new Date(); var t=e.touches?e.touches[0]:e; startPoint={x:t.pageX,y:t.pageY}; } function touchMove(e){ if(isTouching){ var t=e.touches?e.touches[0]:e; var p={x:t.pageX,y:t.pageY}; currentPoint=p; var x1=startPoint.x,x2=currentPoint.x,y1=startPoint.y,y2=currentPoint.y; if(Math.abs(x1-x2)>2 || Math.abs(y1-y2)>2){ isSwipe=true; var direction=getSwipeDirection(x1,x2,y1,y2); //console.log(direction); e.direction=direction; doAction("swipe",e); } } } function touchEnd(e){ isTouching=false; if(!isSwipe){ e["long"]=new Date()-startTime>1000; doAction("tap",e); //console.log("tap"); }else{ var x1=startPoint.x,x2=currentPoint.x,y1=startPoint.y,y2=currentPoint.y; var direction=getSwipeDirection(x1,x2,y1,y2); console.log(direction) doAction("swipeEnd",{direction:direction}); } isSwipe=false; } if(isSupportTouch()){ el.addEventListener('touchstart',touchStart); el.addEventListener('touchend',touchEnd); el.addEventListener('touchmove',touchMove); //el.addEventListener('touchcancel',actionFinsh); }else{ el.addEventListener('mousedown',touchStart); el.addEventListener('mouseup',touchEnd); el.addEventListener('mousemove',touchMove); } } $.fn.touchEvent = TouchEvent;