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)
上一篇 2022年4月9日 上午9:00
下一篇 2022年4月9日 上午9:00


相关推荐

  • Coze高阶工作流实战:变量+循环+批处理+聚合模块,手把手搭建文案创作、PPT 生成等 10+ 智能体

    Coze高阶工作流实战:变量+循环+批处理+聚合模块,手把手搭建文案创作、PPT 生成等 10+ 智能体

    2026年3月13日
    3
  • Windows下RStudio的下载与安装教程

    Windows下RStudio的下载与安装教程一、下载与安装R注意:R是RStudio的基础,必须先安装R,再安装RStudio。因为RStudio自身并不附带R程序。R的下载与安装可见博客:Windows下安装R二、下载RStudio安装包进入RStudio下载官网:添加链接描述点击“RStudioDesktopFree”下的“DOWNLOAD”开始下载对应自己的系统,选择合适的版本(我这里选择win10),等待安装包下载完成即可。三、安装RStudio双击运行下载好的安装包点击“下一步”选择好安装目录后点击“下

    2022年6月29日
    26
  • 【web前端技术】判断是否是手机端

    【web前端技术】判断是否是手机端通常在判断浏览终端类型时 会用后端来判断请求时从哪种设备发出的 根据请求中的 User Agent 来判断 但是在某些情况下 还是会用到前端检测终端类型 并进行操作 本文就是在解决这个问题 在查阅了大量网络资料后 我将这些资料进行了整理

    2026年3月18日
    2
  • 面试题:JVM,GC垃圾回收机制

    面试题:JVM,GC垃圾回收机制探索HotSpotJVM、RuntimeDataAreas与GC一、运行时数据区域二、哪些内存要回收?三、如何回收?垃圾收集算法、垃圾收集器、垃圾收集器参数与GC日志四、内存分配和回收策略对象优先在Eden上分配、大对象直接进入老年代、长期存活的对象将进入老年代

    2022年6月9日
    36
  • pycharm导入库变灰色_import python

    pycharm导入库变灰色_import pythonpycharm中import导入包呈现灰色问题之解决!问题描述:pycharm中单个py文件导入包时呈灰色,而别的文件却能正常显示,我按照CSDN博客上给的设置①右键点击项目,找下面的MarkDirectoryas选择SourceRoot”以及②点击File-InvalidteCaches/Restart…重启两种方法均不起作用,无法解决问题。我的解决方法:将鼠标移动到那行…

    2022年8月27日
    5
  • poj 2375

    poj 2375这道题是一道gu

    2022年6月13日
    22

发表回复

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

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