美文网首页
2018-08-04

2018-08-04

作者: Clysto | 来源:发表于2018-08-04 13:46 被阅读0次

移动端触摸事件

最近在写一个博客后台的单页面应用,在文章列表这一个页面每一条文章都会拥有三个操作,“预览”, “编辑”, “删除”,在PC端,他们看上去是这个样子的:

而在手机端展示的时候,就会显得很“挤”:

mobile.png

如果右侧的按钮改为左划呼出,那样的话,按钮区域就不会占用文字显示的空间,看上去也较为美观,先上成果图:

想要达到这样的效果,就要了解一下HTML DOM 事件对象中的touch事件。

我们先创建一个<div></div>,然后将event对象传进touchstart()这个函数,ontouchstart是一个DOM事件,当按下手指时,执行后面的函数:

<div ontouchstart="touchstart(event)" id="demo">
    touch me!
</div>

<script>
    function touchstart(e) {
        console.log(e.touches[0]);   
        //touches是当前位于屏幕上的所有手指的一个列表
    }
</script>

打印出来的结果为一个对象:

属性
clientX 141
clientY 60
force 1
identifier 0
pageX 141
pageY 60
radiusX 11.5
radiusY 11.5
rotationAngle: 0
screenX 548
screenY 198
target div#demo

其中一些属性的意义如下:

  • clientX:触摸目标在视口中的x坐标
  • clientY:触摸目标在视口中的y坐标
  • identifier:标识触摸的唯一ID
  • pageX:触摸目标在页面中的x坐标
  • pageY:触摸目标在页面中的y坐标
  • screenX:触摸目标在屏幕中的x坐标
  • screenY:触摸目标在屏幕中的y坐标
  • target:触摸的DOM节点目标

那么我们最需要的就是触摸目标在视口中的x坐标与y坐标,另外一些关于触摸的事件有:

  • 移动手指时,触发ontouchmove
  • 移走手指时,触发ontouchend

那么我们便可以监听这三个DOM事件,做一些有趣的操作~

做之前我们先加一点样式:

#demo {
    position: absolute;
    width: 256px;
    height: 256px;
    color: #7d8d9b;
    background-color: #eef3f8;
    border-radius: 3px;
    text-align: center;
    line-height: 256px;
}

监听三个事件,并且添加一条内联样式,便于我们在js中获取。

<div ontouchstart="touchstart(event)" 
     ontouchmove="touchmove(event)" 
     ontouchend="touchend(event)" 
     id="demo"
     style="top: 15px; left: 15px"> //内联样式
    touch me!
</div>

接着我们通过js来实现用手指拖动这个<div></div>的效果。在开始之前,我们需要定义一些变量记录手指按下瞬间,触摸点在视口的位置,以及拖动的过程中,触摸点在视口的位置,还要记录下拖动前元素的left以及top属性的值,以便在拖动元素时,计算出两个方向的偏移量。

var $demo = document.getElementById('demo');
var start_x,
    start_y,
    start_left,
    start_top,
    current_x,
    current_Y;

touchstart()函数中,只需记录下相应触摸点的信息:

function touchstart(e) {
    start_x = e.touches[0].clientX;
    start_y = e.touches[0].clientY;
    start_left = parseFloat($demo.style.left.replace('px', ''));
    start_top = parseFloat($demo.style.top.replace('px', ''));
}

在移动过程中我们获取移动过程中的触摸位置,并计算出偏移量,然后重新设置元素的left以及top属性的值,及触摸前的lefttop + 移动过程中的偏移量,也就是start_left/top + change_x/y

function touchmove(e) {
    current_x = e.touches[0].clientX;
    current_y = e.touches[0].clientY;
    var change_x = current_x - start_x;
    var change_y = current_y - start_y;
    $demo.style.left = start_left + change_x + 'px';
    $demo.style.top = start_top + change_y + 'px';
}

function touchend(e) {
    $demo.innerHTML = '别摸我!';
}

成果图:

简单的通过控制DOM元素的left以及top样式就可以实现这个有趣的效果!那么回到开头的命题,有了这些基础我们怎样完成左划呼出菜单的设想呢?上面这个demo已经实现了元素的拖动,我们只需要x方向的拖动,并且在手指离开的瞬间需要对元素移动距离进行判断,符合一定条件就可以呼出控制区域。首先先写好我们的html模板:

css样式:

* {
    box-sizing: border-box;
}

#slide-wrap {
    border-top: 1px solid rgba(0, 0, 0, 0.07);
    border-bottom: 1px solid rgba(0, 0, 0, 0.07);
    overflow: hidden;
}

#slide-item {
    position: relative; /* 重要 */
    display: flex;
    width: 100%;
}

.info {
    flex: 0 0 100%;
    font-size: 30px;
    font-weight: 300;
    padding: 30px 15px;
}

.btns {
    display: flex;
    flex: 0 0 211px;
}

.separation {
    width: 1px;
    height: 100%;
    background-color: #e1e7ec;
}

.del {
    color: #ffe2e2;
    background-color: #e66565;
}

button {
    flex: 1;
    border: none;
    color: #636569;
    background-color: #eef3f8;
    outline: none;
}

html模板:

<div id="slide-wrap">
    <div id="slide-item">
        <div class="info">我是孤独的根号三</div>
        <div class="btns">
            <button>预览</button>
            <span class="separation"></span>
            <button>编辑</button>
            <button class="del">删除</button>
        </div>
    </div>
</div>

关于.info样式的设定,有必要补充一点说明,flex属性是flex-growflex-shrinkflex-basis 属性的简写属性,而我们设置的flex-shrink为0,也就是元素不会收缩他的宽度,会按照flex-basis中设置的宽度来显示,所以三个按钮就会被“顶”到右侧。

.info {
    flex: 0 0 100%; /* flex-grow|flex-shrink|flex-basis */
    font-size: 30px;
    font-weight: 300;
    padding: 30px 15px;
}

而外层元素#slide-wrapoverflow设为了hidden,所以按钮部分就会被修剪掉。

#slide-wrap {
    border-top: 1px solid rgba(0, 0, 0, 0.07);
    border-bottom: 1px solid rgba(0, 0, 0, 0.07);
    overflow: hidden;
}

如果我们将.info中的flex-shrink设置为1,也就是空间不足的时候会自动收缩,我们就能够看到三个按钮被显示出来了。

当然我们不需要这样的显示方式,如果我们要以滑动的方式显示按钮,就需要改变#slide-itemleft值,当我们将其设置为-211px时:

<div id="slide-item" style="left: -211px">
......
</div>

右侧按钮被显示了,那我们只需要让js代替我们完成这个过程!

监听触摸事件:

<div class="info"
     ontouchstart="touchstart(event)" 
     ontouchmove="touchmove(event)" 
     ontouchend="touchend(event)">我是孤独的根号三</div>

操作样式:

参照上一个demo的例子,将y方向移动的部分删去:

var $ele = document.getElementById('slide-item');
var start_x,
    start_left,
    current_x;

function touchstart(e) {
    start_x = e.touches[0].clientX;
    start_left = parseFloat($ele.style.left.replace('px', ''));
}

function touchmove(e) {
    current_x = e.touches[0].clientX;
    var change_x = current_x - start_x;
    $ele.style.left = start_left + change_x + 'px';
}

但是当手指离开时,元素并没有回到它该去的地方,所以在touchend()函数中我们要加以判断:

function touchend() {
    if (change_x < -70) {
        $ele.style.left = '-211px';
    } else {
        $ele.style.left = '0px';
    }
}

动画效果太僵硬了,添加一点过度,最终的js代码为:

var $ele = document.getElementById('slide-item');
var start_x,
    start_left,
    current_x,
    change_x;

function touchstart(e) {
    $ele.style.transition = 'left 0ms';
    start_x = e.touches[0].clientX;
    start_left = parseFloat($ele.style.left.replace('px', ''));
}

function touchmove(e) {
    current_x = e.touches[0].clientX;
    change_x = current_x - start_x;
    $ele.style.left = start_left + change_x + 'px';
}

function touchend(e) {
    $ele.style.transition = 'left 300ms';
    if (change_x < -70) {
        $ele.style.left = '-211px';
    } else {
        $ele.style.left = '0px';
    }
}

最终的效果为:

效果还不错哦(●'◡'●),当然还有许多细节可以优化,大家也可以给我指出欠妥的地方~


CodePen效果:在线demo(如果是PC端,在浏览器调试工具中模拟手机触屏才会有效果)

相关文章

网友评论

      本文标题:2018-08-04

      本文链接:https://www.haomeiwen.com/subject/zycmvftx.html