web移动端:touchmove实现局部滚动

总结一下最近项目用到的一个功能点,具体要求如下:body中会呈现一个可滚动的长页面,在点击某个按钮的时候,会出现一个弹出框,由于弹出框的内容较长,会出现滚动条,但是要保证位于弹框下部的body在弹框滚动的时候不触发滚动事件实现效果图如下:一.css部分html,body{margin:0;padding:0;}.btn{position:fixed;

大家好,又见面了,我是你们的朋友全栈君。

总结一下最近项目用到的一个功能点,具体要求如下:
body中会呈现一个可滚动的长页面,在点击某个按钮的时候,会出现一个弹出框,由于弹出框的内容较长,会出现滚动条,但是要保证位于弹框下部的body在弹框滚动的时候不触发滚动事件

实现效果图如下:
这里写图片描述这里写图片描述

一.css部分

html,body{ margin:0; padding: 0; }
.btn{ position: fixed; bottom: 0; width: 100%; padding: .5rem 0; background-color: pink; text-align: center; color: #fff; font-weight: 700; }
.replyCeng{ display: none; position: fixed; top:0; width:100%; height: 100vh; background-color: rgba(0,0,0,0.6); z-index: 100; }
.replyContainer{ background-color: #fff; position: absolute; bottom: 0; width: 100%; height: 24rem; text-align: center; overflow: hidden; }
.replyList{ position: absolute; top:3.2rem; width: 100%; }
.replyTitle{ position: absolute; top: 0; width:100%; padding:1.0625rem 0; border-bottom:1px solid #e0e0e2; z-index: 202; color: #575b61; font-size: .9375rem; font-weight: 700; background-color: #fff; }
.closeReply{ position: absolute; width: 1rem; height: 1rem; top: .25rem; right: .25rem; z-index: 202; margin: 1rem; }
.reply-item{ padding:.9375rem 1.625rem; font-size: .9375rem; line-height: 1.5; color: #666666; border-bottom:1px solid #e0e0e2; }
.replyBtn{ /* padding: 1.0625rem 1.625rem;*/ font-size: .9375rem; color: #1daffa; font-weight: 700; width:100%; position: absolute; bottom: 0; background-color: #fff; z-index: 202; border-top: 1px solid #e0e0e2; }
.replyBtn div{ margin: 1.0625rem 1.625rem; padding: .6rem; /* width: 100%;*/ border: 1px solid #1daffa; border-radius: .3rem; }

二.html部分

<div>
    <img src="image/langImg.jpg" style="width:100%;height:auto;"/>
</div>
<div class="btn" onclick="replyCengShow()">button</div>
<div class="replyCeng">
    <div class="replyContainer">
        <div class="replyTitle">睡前小贴士</div>
        <img class="closeReply" src="image/close.png" />
        <div class="replyList" id="wrapper">
            <div class="reply-item" onclick="selectReply()">要注意哦,睡觉前不要吃水果,可以适当喝牛奶</div>
            <div class="reply-item">要注意哦,睡觉前不要吃水果,可以适当喝牛奶</div>
            <div class="reply-item">要注意哦,睡觉前不要吃水果,可以适当喝牛奶</div>
            <div class="reply-item">要注意哦,睡觉前不要吃水果,可以适当喝牛奶</div>
            <div class="reply-item">把黑暗的光线试想成大自然的安眠药,从而提示你的身体去大量地制造褪黑激素,这是一种能帮助人平静下来的激素。来自旧金山的专门从事焦虑和失眠的心理学家、心理学博士Steve Orma表示,在入睡前将灯关掉能提高褪黑激素的产量,这样人就会感到倦意。支持这一说法的研究有:2011年发表在《临床内分泌与代谢》期刊上的一篇研究报告表明,在黄昏时刻到睡觉之前暴露在明亮的灯光下会极大地压缩褪黑激素的产量,使人变得极其兴奋。</div>
            <div class="reply-item">把黑暗的光线试想成大自然的安眠药,从而提示你的身体去大量地制造褪黑激素,这是一种能帮助人平静下来的激素。来自旧金山的专门从事焦虑和失眠的心理学家、心理学博士Steve Orma表示,在入睡前将灯关掉能提高褪黑激素的产量,这样人就会感到倦意。支持这一说法的研究有:2011年发表在《临床内分泌与代谢》期刊上的一篇研究报告表明,在黄昏时刻到睡觉之前暴露在明亮的灯光下会极大地压缩褪黑激素的产量,使人变得极其兴奋。</div>
        </div>
        <div class="replyBtn">
            <div>管理</div>
        </div>
    </div>
</div>

三.js部分

1.获得触屏的初始坐标

$(".replyList").on("touchstart",function(e){
    startX = e.originalEvent.changedTouches[0].pageX,
    startY = e.originalEvent.changedTouches[0].pageY;
});

2.移动屏幕添加相应事件
touchmove事件的要点主要是移动方向的判断和滑动边界的判断
<1>移动方向的判断

var replyListTop=parseInt($(".replyList").css("top"));
$(".replyList").bind("touchmove",function(e){ 
     
    moveEndX = e.originalEvent.changedTouches[0].pageX;
    moveEndY = e.originalEvent.changedTouches[0].pageY;
    var moveX;
    var moveY;
    moveX=moveEndX-startX;
    moveY=moveEndY-startY;
    //向上滑动
    if(moveY < 0){  
    }
    //向下滑动
    else if(moveY > 0){
    }
});

<2>移动边界的判断
A.向上滑动到上边界
注:replyScrollTop为文中replyListTop
注:replyScrollTop为文中replyListTop

此时replyListTop为负数

canSeeHeight == replyHeight + replyListTop //(此时正好处于边界状态)

如果再往上滑动,我们将会看到越来越少的内容,也就是:

canSeeHeight > replyHeight + replyListTop

于是得到上边界值:

hasTop= canSeeHeight <= $(this).height()+replyListTop ? true:false

B.向下滑动到下边界
这里写图片描述
由于滚动区域是包裹在replyContainer里面的,所以滚动区域的初始top值为标题区域的高度,只要标题区域的高度 == 滚动区域的top值 ,就代表向下滑动到下边界

hasBottom = 3.2*fontHeight >= replyListTop ? true:false ;//title区域用rem布局,fontHeight为html字体大小
//向上滑动
if(moveY < 0 && hasTop){
    $(".replyList").css("top",replyListTop+"px");
    replyListTop= replyListTop+moveY;
}
//向下滑动
else if(moveY > 0 && hasBottom){
    $(".replyList").css("top",replyListTop);
    replyListTop= replyListTop+moveY;
}

<3>限制滑动过大越界

//向上滑动
if(moveY < 0 && hasTop){
    $(".replyList").css("top",replyListTop+"px");
    replyListTop= replyListTop+moveY;
    //判断加上移动后的位置是否越界,如果越界,以边界值为top值
    replyListTop= canSeeHeight <= ($(this).height()+replyListTop) ? replyListTop : (-Math.abs($(this).height()-canSeeHeight));
}
//向下滑动
else if(moveY > 0 && hasBottom){
    $(".replyList").css("top",replyListTop);
    replyListTop= replyListTop+moveY;
    //判断加上移动后的位置是否越界,如果越界,以边界值为top值
    replyListTop= 3.2*fontHeight >= replyListTop ? replyListTop : 3.2*fontHeight;
}

<4>js完整代码

function replyCengShow(){ 
   
    var startX;
    var startY;
    var moveEndX;
    var moveEndY;
    var testLeft;
    var fontHeight=parseInt($("html").css("fontSize"));
    $(".replyCeng").css("display","block");
    $('body').bind("touchmove",function(e){ 
     
            e.preventDefault();  
        });
    $(".replyList").on("touchstart",function(e){ 
   
        startX = e.originalEvent.changedTouches[0].pageX,
        startY = e.originalEvent.changedTouches[0].pageY;
    });

    var replyListTop=parseInt($(".replyList").css("top"));
    $(".replyList").bind("touchmove",function(e){ 
     
        moveEndX = e.originalEvent.changedTouches[0].pageX;
        moveEndY = e.originalEvent.changedTouches[0].pageY;
        var moveX;
        var moveY;
        moveX=moveEndX-startX;
        moveY=moveEndY-startY;

        canSeeHeight=$(".replyContainer").height()-($(".replyBtn").height()+2.125*fontHeight);
        hasTop= canSeeHeight <= $(this).height()+replyListTop ? true:false
        hasBottom = 3.2*fontHeight >= replyListTop ? true:false ;
        //向上滑动
        if(moveY < 0 && hasTop){
            $(".replyList").css("top",replyListTop+"px");
            replyListTop= replyListTop+moveY;
            replyListTop= canSeeHeight <= ($(this).height()+replyListTop) ? replyListTop : (-Math.abs($(this).height()-canSeeHeight));
        }
        //向下滑动
        else if(moveY > 0 && hasBottom){
            $(".replyList").css("top",replyListTop);
            replyListTop= replyListTop+moveY;
            replyListTop= 3.2*fontHeight >= replyListTop ? replyListTop : 3.2*fontHeight;
        }
    });

    $(".closeReply").unbind("click").bind("click",function(){ 
   
        $(".replyCeng").css("display","none");
        $('body').unbind("touchmove");
    });
}

四.不足

在滚动的过程中能够感到明显的卡顿

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请联系我们举报,一经查实,本站将立刻删除。

发布者:全栈程序员-站长,转载请注明出处:https://javaforall.net/127092.html原文链接:https://javaforall.net

(0)
全栈程序员-站长的头像全栈程序员-站长


相关推荐

  • god is a girl 是什么意思_god is a girl 歌词中文

    god is a girl 是什么意思_god is a girl 歌词中文godisagirl题意:解码。将题目中的样例做差输出,打表可得到112581321所以是按斐波那契额值解码的,一开始直接算的菲波那切数没有取模,这样的话90左右就会超longlong所以数组开了100,交了一发runtime,所以得把数组开大,就对斐波那契数取个模就好了#include&lt;bits/stdc++.h&gt;usingnamespacestd;in…

    2022年10月8日
    0
  • Java 基础高频面试题(2021年最新版)

    最新Java基础高频面试题

    2022年4月6日
    34
  • 易中天 品三国

    易中天 品三国(一)大江东去观众朋友们大家好,从今天开始我们讲三国。说起三国,我们就会想起著名的赤壁之战,而说起赤壁之战,我们就会想起苏东坡那首著名的《赤壁怀古》——  “大江东去,浪淘尽,千古风流人物。故垒西边,人道是,三国周郎赤壁。乱石崩云,惊涛裂岸,卷起千堆雪。江山如画,一时多少豪杰。  遥想公瑾当年,小乔初嫁了,雄姿英发。羽扇纶巾,谈笑间,墙橹灰飞烟灭。故国神游,多情应笑我,早生华发。

    2022年6月9日
    44
  • javaweb-爬虫-2-63

    javaweb-爬虫-2-63

    2021年5月18日
    137
  • Python海龟画图集合

    Python海龟画图集合Python海龟画图集合1.小猪佩奇2.彩色螺旋线3.太极4.美国队长盾牌1.小猪佩奇#coding:utf-8importturtleastt.screensize(400,300)t.pensize(4)#设置画笔的大小t.colormode(255)#设置GBK颜色范围为0-255t.color((255,155,192),"pink…

    2022年6月28日
    28
  • Android开发环境配置(以windows为例)

    Android开发环境配置(以windows为例)Android开发环境配置工具   如果你准备从事Android开发,那么无论选择在eclipse下开发,还是选择在AndroidStudio下开发,都可以参照以下步骤进行Android开发环境的配置。Android开发环境配置过程1.准备笔记本或台式机  使用笔记本还是台式机,视个人需求而定,但我要强调的是在配置上不要手软,要舍得下手。一台流畅的电脑,会让

    2022年7月23日
    8

发表回复

您的邮箱地址不会被公开。 必填项已用 * 标注

关注全栈程序员社区公众号