利用JS、CSS实现列表自动滑动滚动

零.业务需求

这几天在做大屏项目,对于大屏有很多信息需要实时滚动,废了点力气学的明明白白的,特来记录供大家学习。

0.1实现效果

一.逻辑分析

1.1滑动窗口和滚动条

当我们使用<table>或者<ul>标签时,我们可以制作一个滚动条,通过滚动条来实现我们想要查看的内容。

在滚动窗口内,“已经滚动过去”的区域我们称为:“scrollHeight”,“还未滚动到”的 区域我们称为:“scrollBottom”,“当前显示区域”称为:“scrollNow”。

我们再用一张图来描述一下,上面三个概念

可能有点丑...

为方便,上面这一整个区域(scrollTop、scrollNow、scrollBottom),我们统称为:“滑动窗口

1.2逻辑构思

如果我们有:“两份一模一样的数据顺序排列在一起形成一份数据”,再将这一份数据中的“元素”挨个展示在我们的“scrollNow”区域中,当这份数据的最后一个“元素”被展示时,我们重新让一整份数据从头开始,这样就在视觉上实现了滚动的效果

在这里,有一个关键性的问题:“两份数据的长度一定要大于显示区域的长度,否则将无法实现滑动

至于为什么,大家思考一下就行,当我们的数据滑动时,滑动过去的数据被“隐藏”了,但并不是消失了!

如果我们的“可显示区域”大于“数据”的长度,那么就会出现下面的情况:

当显示区域的长度足够容纳:“所有的数据”,此时还有“滑动”的必要吗???

我们“滑动”的原因是因为:“数据量太大,展示区域有限,故让它滑动方便查看

1.3伪代码实现

在js中,有几个属性:

scrollTop”用来“设置或者获取”元素的已滚动的上部不可见区域”的高度,这个概念就是我们上述讲的“scrollTop”(已经滚动过的区域)

scrollHeight”用来“返回”元素所有内容的“总高度”(包括折叠不见的),这个就是我们上述所说的(scrollTop+scrollNow+scrollBottom)

利用这几个属性,我们可以设定一个计时器用来计时,每次计时器加值时,scrollTop的值随之加值(向上滚动),从而实现了滚动

二.代码实现

2.1创建骨架

我们先创建一个div盒子,作为整个列表的骨架:

    <div id="review_box">
            <ul id="comment1">
                <li>第一条</li>
                <li>第二条</li>
                <li>第三条</li>
                <li>第四条</li>
                <li>第五条</li>
                <li>第六条</li>
            </ul>
            <ul id="comment2"></ul>
    </div>

其中“comment1<ul>原数据,“comment2<ul>原数据的副本,这两部分数据组合在一起形成我们的“一整个数据

<li>就是我们的“元素”,方便起见我这里内置数据,后面根据需要可以自己再写一个函数添加数据。

2.2创建样式

只有骨架是不行的,没有用到CSS美观是很丑的,如果不使用CSS,那么就是下图这个样子:

为此我们使用如下代码:

    <style>
        div {
            width: 100px;
            height: 100px;      /*列表的长度*/
            overflow: hidden;/* 必须 */
            margin: 50px auto;
            border: 1px solid red;
            text-align: center;
        }
        ul {
            margin: 0;
            padding: 0;         /*元素居中*/
            list-style: none;   /*去掉元素前面的标签*/
        }
        li{
            height: 20px;  /*元素高度,数据的高度等于元素个数乘元素高度*/
        }
    </style>

2.3JS代码

我们先创建一个“roll”滚动函数,用来实现控制“鼠标悬停在列表上时暂停滚动”、“鼠标离开列表时继续滚动

我们分别使用:“onmouseover”和“onmouseout两个事件来完成。

实现“向上滚动”和“向下滚动”在js中有些许不同,这里先以“向上滚动”为例

        function roll(t) {
            var ul1 = document.getElementById("comment1");
            var ul2 = document.getElementById("comment2");
            var ulbox = document.getElementById("review_box");
            ul2.innerHTML = ul1.innerHTML;
            ulbox.scrollTop = 0; // 开始无滚动时设为0
            var timer = setInterval(rollStart, t); // 设置定时器,参数t用在这为间隔时间(单位毫秒),参数t越小,滚动速度越快
            // 鼠标移入div时暂停滚动
            ulbox.onmouseover = function () {
                clearInterval(timer);
            }
            // 鼠标移出div后继续滚动
            ulbox.onmouseout = function () {
                timer = setInterval(rollStart, t);
            }
        }

我们再创建一个“rollStart”函数,用来启动“自动滑动

        function rollStart() {
            // 上面声明的DOM对象为局部对象需要再次声明
            var ul1 = document.getElementById("comment1");
            var ul2 = document.getElementById("comment2");
            var ulbox = document.getElementById("review_box");
            // 正常滚动不断给scrollTop的值+1,当滚动高度大于列表内容高度时恢复为0
            if (ulbox.scrollTop >= ul1.scrollHeight) {
                ulbox.scrollTop = 0;
            } else {
                ulbox.scrollTop += 1;
            }
    }

最后,在代码的最前面加一句:

         window.onload = roll(20);

用来代码在页面加载完毕时自动启动,一切就完成啦!!

附:

如果想要列表向下滚动,我们只改动三处数值即可:

ulbox.scrollTop = ul1.scrollHeight; // 开始无滚动时设列表最长高度,用来表示到底了

 上面将初始“scrollTop”置为最大值,表示此时数据底部到底了,即无法继续向下滑动,只能向上滑动

// 正常滚动不断给scrollTop的值减一,当滚动高度小于等于0时,需要重新为最大高度
if (ulbox.scrollTop > 0) {
    ulbox.scrollTop = ul1.scrollHeight;
}
else {
    ulbox.scrollTop -= 1;
}

上面控制“scrollTop”的值每次减一,如果小于等于0了,说明此时数据顶部到底了,即无法继续向上滑动,只能向下滑动

值得注意:

若向下滑动,对应的数据也应是逆置的,否则将会出现部分数据无法正常滑动:

    <div id="review_box">
            <ul id="comment1">
                <li>第六条</li>
                <li>第五条</li>
                <li>第四条</li>
                <li>第三条</li>
                <li>第二条</li>
                <li>第一条</li>
            </ul>
            <ul id="comment2"></ul>
    </div>

 不过若此时(列表长度等于原数据长度[不是整个数据]的高度,那么数据可以不用逆置)

三.全部代码

下面代码,复制粘贴即可使用

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <meta name="viewport" content="width=device-width, initial-scale=1.0">
    <title>Document</title>
    <style>
        div {
            width: 100px;
            height: 100px;      /*列表的长度*/
            overflow: hidden;/* 必须 */
            margin: 50px auto;
            border: 1px solid red;
            text-align: center;
        }
        ul {
            margin: 0;
            padding: 0;         /*元素居中*/
            list-style: none;   /*去掉元素前面的标签*/
        }
        li{
            height: 30px;  /*元素高度,数据的高度等于元素个数乘元素高度*/
        }
    </style>
</head>
<body>
    <div id="review_box">
            <ul id="comment1">
                <li>第一条</li>
                <li>第二条</li>
                <li>第三条</li>
                <li>第四条</li>
                <li>第五条</li>
                <li>第六条</li>
            </ul>
            <ul id="comment2"></ul>
    </div>
    <script>
         window.onload = roll(20);
  
        function roll(t) {
            var ul1 = document.getElementById("comment1");
            var ul2 = document.getElementById("comment2");
            var ulbox = document.getElementById("review_box");
            ul2.innerHTML = ul1.innerHTML;
            ulbox.scrollTop = 0; // 开始无滚动时设为0
            var timer = setInterval(rollStart, t); // 设置定时器,参数t用在这为间隔时间(单位毫秒),参数t越小,滚动速度越快
            // 鼠标移入div时暂停滚动
            ulbox.onmouseover = function () {
                clearInterval(timer);
            }
            // 鼠标移出div后继续滚动
            ulbox.onmouseout = function () {
                timer = setInterval(rollStart, t);
            }
        }
 
        // 开始滚动函数
        function rollStart() {
            // 上面声明的DOM对象为局部对象需要再次声明
            var ul1 = document.getElementById("comment1");
            var ul2 = document.getElementById("comment2");
            var ulbox = document.getElementById("review_box");
            // 正常滚动不断给scrollTop的值+1,当滚动高度大于列表内容高度时恢复为0
            if (ulbox.scrollTop >= ul1.scrollHeight) {
                ulbox.scrollTop = 0;
            } else {
                ulbox.scrollTop += 1;
            }
    }
    </script>
</body>
</html>

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:/a/518689.html

如若内容造成侵权/违法违规/事实不符,请联系我们进行投诉反馈qq邮箱809451989@qq.com,一经查实,立即删除!

相关文章

蓝桥杯第十四届C++A组(未完)

【规律题】平方差 题目描述 给定 L, R&#xff0c;问 L ≤ x ≤ R 中有多少个数 x 满足存在整数 y,z 使得 。 输入格式 输入一行包含两个整数 L, R&#xff0c;用一个空格分隔。 输出格式 输出一行包含一个整数满足题目给定条件的 x 的数量。 样例输入 1 5 样例输出 …

Redis中的Sentinel(二)

Sentinel 初始化Sentinel状态。 在应用了Sentinel的专用代码之后&#xff0c;接下来&#xff0c;服务器会初始化一个sentinel.c/sentinelState结构(简称Sentinel状态),这个结构 保存了服务器中所有和Sentinel功能有关的状态(服务器的一般状态仍然由redis.h/redisServer保存);…

【java数据结构-二叉树(上)】

java数据结构-二叉树&#xff08;上&#xff09; 二叉树的概念二叉树的节点介绍 二叉树构造如何使用兄弟表示法构造二叉树两种特别的二叉树二叉树的基本性质&#xff1a; 二叉树的存储二叉树的遍历&#xff1a;前序遍历&#xff1a;中序遍历&#xff1a;后序遍历&#xff1a;层…

最新ChatGPT4.0工具使用教程:GPTs,Midjourney绘画,AI换脸,GPT语音对话,文档分析一站式系统

一、前言 ChatGPT3.5、GPT4.0、相信对大家应该不感到陌生吧&#xff1f;简单来说&#xff0c;GPT-4技术比之前的GPT-3.5相对来说更加智能&#xff0c;会根据用户的要求生成多种内容甚至也可以和用户进行创作交流。 然而&#xff0c;GPT-4对普通用户来说都是需要额外付费才可以…

基于RDMA的云服务能力实践与探索

01 背景 随着基于大数据大模型构建的数据系统越来越有商业价值&#xff0c;机器学习的玩家也越来越多&#xff0c;数据量越来越大。为解决海量数据在服务器之间的同步效率问题&#xff0c;RDMA(Remote Direct Memory Access) 技术逐渐走进了网络技术人员的视野。RDMA为什么…

yolov8多分支任务头训练

目前已知的yolov8可以针对多个任务进行单独训练,但是暂时还没有开放针对多个任务头同时进行训练的教程,本文章针对yolov8的多任务训练进行详细介绍。 先放上效果图: 三个任务,分别是目标检测、可行驶区域、车道线,具体步骤请往下看。 一、环境配置 从如下github下载代码…

Flutter Don‘t use ‘BuildContext‘s across async gaps.

Flutter提示Don‘t use ‘BuildContext‘s across async gaps.的解决办法—flutter里state的mounted属性

10-热点文章-定时计算

xxl-Job分布式任务调度 1 今日内容 1.1 需求分析 目前实现的思路&#xff1a;从数据库直接按照发布时间倒序查询 问题1&#xff1a; 如何访问量较大&#xff0c;直接查询数据库&#xff0c;压力较大 问题2&#xff1a; 新发布的文章会展示在前面&#xff0c;并不是热点文章 …

交叉验证(Cross-Validation)

交叉验证的基本概念 交叉验证通常用于评估机器学习模型在未知数据上的性能。它将数据集分成k个不同的子集&#xff0c;然后进行k次训练和验证。在每次迭代中&#xff0c;选择一个子集作为测试集&#xff0c;其余的子集作为训练集。这样&#xff0c;每个子集都用作过测试集&…

Debian安装1panel管理面板教程-最新

1Panel 是一个现代化、开源的 Linux 服务器运维管理面板。 1Panel面板是一个强大的服务器管理工具&#xff0c;它通过提供一站式管理、易于使用的界面、高度的可定制性、安全可靠的性能、强大的扩展性以及活跃的社区支持&#xff0c;为用户提供了一个高效、便捷的管理解决方案…

华为云1核2G免费使用一年

个人用户专享云服务器、云数据库产品每天上午9:30开抢&#xff0c;其他产品每天0点开放领取&#xff0c;企业用户所有产品每天0点开放领取&#xff1b; 云产品体验名额有限&#xff0c;领完即止。详情&#xff1a;https://www.vpspick.com/vps/591.html 通用入门型 T6 云服务…

实验06_IPv6实验

目录 实验拓扑 实验需求 实验配置及其现象验证&#xff1a; (1)R1---R4的IPv6的地址配置&#xff0c;根据设备编码配置IPv6 (2)R6 通过无状态自动获取 IPv6 地址 (3)OSPFv3 配置&#xff0c;通过 OSPFv3 实现全网通&#xff08;区域划分根据拓扑图&#xff09; 实验拓扑 …

Linux存储的基本管理

实验环境&#xff1a; 系统里添加两块硬盘 ##1.设备识别## 设备接入系统后都是以文件的形式存在 设备文件名称&#xff1a; SATA/SAS/USB /dev/sda,/dev/sdb ##s SATA, dDISK a第几块 IDE /dev/hd0,/dev/hd1 ##h hard VIRTIO-BLOCK /de…

蓝桥杯(5):python动态规划DF[2:背包问题]

1 0-1背包介绍【每件物品只能拿1件或者不拿】 1.1 简介 贪心是不可以的&#xff01;&#xff01;&#xff01; 1.2 状态 及状态转移 转移解释&#xff1a;要么不选 则上一个直接转移过来【dp[i-1][j]】&#xff0c;要么是选这个之后体积为j 则上一个对应的就是【dp[i-1][j-wi]…

Vue 样式技巧总结与整理[中级局]

SFC&#xff08;单文件组件&#xff09;由 3 个不同的实体组成&#xff1a;模板、脚本和样式。三者都很重要&#xff0c;但后者往往被忽视&#xff0c;即使它可能变得复杂&#xff0c;且经常导致挫折和 bug。 更好的理解可以改善代码审查并减少调试时间。 这里有 7 个奇技淫巧…

CSS设置网页颜色

目录 前言&#xff1a; 1.颜色名字&#xff1a; 2.十六进制码&#xff1a; 3.RGB&#xff1a; 4.RGBA&#xff1a; 5.HSL&#xff1a; 1.hue&#xff1a; 2.saturation&#xff1a; 3.lightness&#xff1a; 6.HSLA&#xff1a; 前言&#xff1a; 我们在电脑显示器&…

高血压的常见症状,你是否了解并警惕?

高血压是一种慢性心血管疾病&#xff0c;它的形成和发展与很多种因素密切相关。导致高血压的重要因素之一就是不良的生活饮食习惯。 高血压严重危害着人的心脏、肾脏、大脑等这些人体重要的器官&#xff0c;会容易带来心脏病、肾功能衰竭等一系列疾病。今天我在来为大家讲解一…

CorelDRAW2024序列号破解版下载以及CorelDRAW新功能介绍

CorelDRAW Graphics Suite 2024破解版介绍 CorelDRAW Graphics Suite 2024 破解版是领先的专业图形设计软件套件&#xff0c;旨在提供多个精心设计的工具和丰富的功能&#xff0c;轻松提高您的技能睡哦平&#xff0c;您可以更好的完成矢量插图、完美的布局、高级的照片编辑和高…

Jeecg commonController 任意文件上传漏洞复现

0x01 产品简介 Jeecg(J2EE Code Generation)是一款基于代码生成器的低代码开发平台,使用JEECG可以简单快速地开发出企业级的Web应用系统。 0x02 漏洞概述 JEECG(J2EE Code Generation) 是开源的代码生成平台,目前官方已停止维护。由于 /api 接口鉴权时未过滤路径遍历,攻…

中高级前端? 这些一元运算符,你真的搞清楚了吗

前言 一元运算符&#xff0c;不太起眼&#xff0c;作用很大&#xff0c;请别忽视她&#xff01; 走近她&#xff0c;爱上她&#xff01; 定义 只需要一个操作数的运算符称为一元运算符。 还是代码容易懂&#xff1a; 1 // 一个操作数1 2 // 两个操作数一元运算符清单 运…