原生js实现轮播图及无缝滚动

news2025/1/10 14:06:12

我这里主要说轮播图和无缝滚动的实现思路,就采用最简单的轮播图了,当然实现的思路有很多种,我这也只是其中一种。

简单轮播图的大概结构是这样的,中间是图片,二边是箭头可以用来切换图片,下面的小圆点也可以用来切换图片。

1.简易的轮播图效果

先搭出html结构

 这里的左右箭头我采用的是svg图标

  <div class="container">
    <div class="carousel">
      <div class="item"><a href=""><img src="./3.jpg" alt=""></a></div>
      <div class="item"><a href=""><img src="./2.jpg" alt=""></a></div>
      <div class="item"><a href=""><img src="./1.jpg" alt=""></a></div>
    </div>
    <div class="left">
      <svg t="1693569521007" class="icon" viewBox="0 0 1024 1024" version="1.1" xmlns="http://www.w3.org/2000/svg"
        p-id="4000" width="20" height="20">
        <path
          d="M729.6 931.2l-416-425.6 416-416c9.6-9.6 9.6-25.6 0-35.2-9.6-9.6-25.6-9.6-35.2 0l-432 435.2c-9.6 9.6-9.6 25.6 0 35.2l432 441.6c9.6 9.6 25.6 9.6 35.2 0C739.2 956.8 739.2 940.8 729.6 931.2z"
          p-id="4001"></path>
      </svg>
    </div>
    <div class="right">
      <svg t="1693569535119" class="icon" viewBox="0 0 1024 1024" version="1.1" xmlns="http://www.w3.org/2000/svg"
        p-id="4245" width="16" height="16">
        <path
          d="M761.6 489.6l-432-435.2c-9.6-9.6-25.6-9.6-35.2 0-9.6 9.6-9.6 25.6 0 35.2l416 416-416 425.6c-9.6 9.6-9.6 25.6 0 35.2s25.6 9.6 35.2 0l432-441.6C771.2 515.2 771.2 499.2 761.6 489.6z"
          p-id="4246"></path>
      </svg>
    </div>
    <div class="indicator">
      <span class="active"></span>
      <span></span>
      <span></span>
    </div>
  </div>

然后是css样式

  * {
      margin: 0;
      padding: 0;
      box-sizing: border-box;
    }

    .container {
      width: 700px;
      height: 400px;
      margin: 10px auto;
      overflow: hidden;
      position: relative;
    }

    .container .carousel {
      width: 100%;
      height: 100%;
      display: flex;
    }

    .container .carousel .item img {
      width: 700px;
      height: 400px;
    }

    .container .indicator {
      height: 30px;
      position: absolute;
      bottom: 10px;
      left: 50%;
      transform: translateX(-50%);
    }

    .container .indicator span {
      border: 1px solid #fff;
      width: 20px;
      height: 20px;
      border-radius: 50%;
      display: inline-block;
    }

    .container .indicator span.active {
      background-color: pink;
    }

    .container .left {
      position: absolute;
      left: 10px;
      top: 50%;
    }

    .container .right {
      position: absolute;
      right: 10px;
      top: 50%;
    }

css的关键代码是overflow:hidden,我这里开启的flex弹性盒,使用它可以将多出来的图片进行隐藏,然后其中的一个圆圈元素加上了active

 下面是js代码

我们首先要获取到所有的dom元素

  var doms = {
    carousel: document.querySelector('.carousel'),
    indicator: document.querySelectorAll('.indicator span'),
    left: document.querySelector('.left'),
    right: document.querySelector('.right')
  }

最重要的就是轮播的函数

var curIndex = 0  //用于记录当前是第几个元素  
function moveTo(index) {
    //加上动画效果
    doms.carousel.style.transition = 'transform .5s'
    doms.carousel.style.transform = `translateX(-${index}00%)`
    //去除效果
    var active = document.querySelector('.indicator span.active')
    active.classList.remove('active')
    //选中当前效果
    doms.indicator[index].classList.add('active')
    curIndex = index
  }

我这里采用的是transform的translateX(-100%)来实现的轮播切换,也可以使用margin-left来控制都可以。

接下来可以给加上一个定时器来控制它进行自动轮播

 //添加图片自动轮播
  let timer = setInterval(() => {
    if (curIndex === doms.indicator.length - 1) {
      moveTo(0)
    } else (
      moveTo(curIndex + 1)
    )
  }, 2000)

也可以给下面的小圆圈和左右箭头加上对应的点击事件

  //添加点击事件
  doms.indicator.forEach((item, index) => {
    item.addEventListener('click', () => {
      moveTo(index)
    })
  })
  //添加点击事件
  doms.left.addEventListener('click', () => {
    moveTo(curIndex-1)
  })
  doms.right.addEventListener('click', () => {
     moveTo(curIndex+1)
  })

到这里其实已经实现的简易轮播图的基本功能,但是还并不完美,会存在防抖以及无法无缝滚动的效果

2.无缝滚动及防抖

无缝滚动的实现思路就是采用克隆的功能及改变动画效果来实现的

就像这样,将最后一张复制出来放到最前面,但是展示的还是1.jpg,第一张复制放到最后面

 在切换到最后一张或者第一张时,首先将过度动画关掉切换,然后迅速开启过度动画轮播下一张,这样眼睛是无法察觉出来的,因为其速度很快。

 首先是克隆

  //克隆图片,实现无缝滚动
  function clone() {
    var first = doms.carousel.firstElementChild.cloneNode(true);
    //复制最后一张
    var last = doms.carousel.lastElementChild.cloneNode(true);
    //插入到最后
    doms.carousel.appendChild(first);
    //插入到最前
    doms.carousel.insertBefore(last, doms.carousel.firstElementChild)
    //将复制的第一张的位置调整
    last.style.position = 'absolute';
    last.style.transform = `translateX(-100%)`
  }
  clone()

克隆的关键在于要将克隆的第一张图片改变一下位置

然后实现右边箭头的无缝滚动

  //实现右边的无缝滚动
  var count = doms.indicator.length
  function rightMove() {
    //首先去除动画效果
    if (curIndex === count - 1) {
      doms.carousel.style.transform = `translateX(100%)`;
      doms.carousel.style.transition = 'none';
      //强制渲染,否则可能不会执行
      doms.carousel.clientHeight;
      moveTo(0)
    } else {
      moveTo(curIndex + 1)
    }
  }

右边实现后左边就很简单了

  //实现左边的无缝滚动
  function leftMove() {

    if (curIndex === 0) {
      doms.carousel.style.transform = `translateX(-${count}00%)`;
      doms.carousel.style.transition = 'none';
      //强制渲染
      doms.carousel.clientHeight;
      moveTo(count - 1)
    } else {
      moveTo(curIndex - 1)
    }
  }

然后我们的定时器就需要进行一下改变,让他往右边轮播‘

  //添加图片自动轮播
  let timer = setInterval(() => {
    rightMove()
  }, 2000)

然后在需要防抖的地方进行一下防抖,其实我这种很简单的防抖就是将定时器关闭掉,在点击任务完成后再开启定时器。

 //添加点击事件
  doms.indicator.forEach((item, index) => {
    item.addEventListener('click', () => {
      //关闭定时器
      clearInterval(timer)
      moveTo(index)
      //执行完后开启定时器
      timer = setInterval(() => {
        rightMove()
      }, 2000)
    })
  })

左右箭头的点击事件


  //添加点击事件
  doms.left.addEventListener('click', () => {
    //关闭定时器
    clearInterval(timer)
    leftMove()
    //开启定时器
    timer = setInterval(() => {
      rightMove()
    }, 2000)
  })
  doms.right.addEventListener('click', () => {
    //关闭定时器
    clearInterval(timer)
    rightMove()
    //开启定时器
    timer = setInterval(() => {
      rightMove()
    }, 2000)
  })

到这里就实现了简单的轮播图效果

整体代码:

<!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>
    * {
      margin: 0;
      padding: 0;
      box-sizing: border-box;
    }

    .container {
      width: 700px;
      height: 400px;
      margin: 10px auto;
      overflow: hidden;
      position: relative;
    }

    .container .carousel {
      width: 100%;
      height: 100%;
      display: flex;

      /* transition: .5s; */
    }

    .container .carousel .item img {
      width: 700px;
      height: 400px;
    }

    .container .indicator {
      height: 30px;
      position: absolute;
      bottom: 10px;
      left: 50%;
      transform: translateX(-50%);
    }

    .container .indicator span {
      /* background-color: #fff; */
      border: 1px solid #fff;
      width: 20px;
      height: 20px;
      border-radius: 50%;
      display: inline-block;
    }

    .container .indicator span.active {
      background-color: pink;
    }

    .container .left {
      position: absolute;
      left: 10px;
      top: 50%;
    }

    .container .right {
      position: absolute;
      right: 10px;
      top: 50%;
    }
  </style>
</head>

<body>
  <div class="container">
    <div class="carousel">
      <div class="item"><a href=""><img src="./3.jpg" alt=""></a></div>
      <div class="item"><a href=""><img src="./2.jpg" alt=""></a></div>
      <div class="item"><a href=""><img src="./1.jpg" alt=""></a></div>
    </div>
    <div class="left">
      <svg t="1693569521007" class="icon" viewBox="0 0 1024 1024" version="1.1" xmlns="http://www.w3.org/2000/svg"
        p-id="4000" width="20" height="20">
        <path
          d="M729.6 931.2l-416-425.6 416-416c9.6-9.6 9.6-25.6 0-35.2-9.6-9.6-25.6-9.6-35.2 0l-432 435.2c-9.6 9.6-9.6 25.6 0 35.2l432 441.6c9.6 9.6 25.6 9.6 35.2 0C739.2 956.8 739.2 940.8 729.6 931.2z"
          p-id="4001"></path>
      </svg>
    </div>
    <div class="right">
      <svg t="1693569535119" class="icon" viewBox="0 0 1024 1024" version="1.1" xmlns="http://www.w3.org/2000/svg"
        p-id="4245" width="16" height="16">
        <path
          d="M761.6 489.6l-432-435.2c-9.6-9.6-25.6-9.6-35.2 0-9.6 9.6-9.6 25.6 0 35.2l416 416-416 425.6c-9.6 9.6-9.6 25.6 0 35.2s25.6 9.6 35.2 0l432-441.6C771.2 515.2 771.2 499.2 761.6 489.6z"
          p-id="4246"></path>
      </svg>
    </div>
    <div class="indicator">
      <span class="active"></span>
      <span></span>
      <span></span>
    </div>
  </div>
</body>
<script>
  var doms = {
    carousel: document.querySelector('.carousel'),
    indicator: document.querySelectorAll('.indicator span'),
    left: document.querySelector('.left'),
    right: document.querySelector('.right')
  }
  var curIndex = 0
  function moveTo(index) {
    //加上动画效果
    doms.carousel.style.transition = 'transform .5s'
    doms.carousel.style.transform = `translateX(-${index}00%)`
    //去除效果
    var active = document.querySelector('.indicator span.active')
    active.classList.remove('active')
    //选中当前效果
    doms.indicator[index].classList.add('active')
    curIndex = index
  }
  //添加点击事件
  doms.indicator.forEach((item, index) => {
    item.addEventListener('click', () => {
      //关闭定时器
      clearInterval(timer)
      moveTo(index)
      //执行完后开启定时器
      timer = setInterval(() => {
        rightMove()
      }, 2000)
    })
  })
  //添加图片自动轮播
  let timer = setInterval(() => {
    rightMove()
  }, 2000)

  //克隆图片,实现无缝滚动
  function clone() {
    var first = doms.carousel.firstElementChild.cloneNode(true);
    //复制最后一张
    var last = doms.carousel.lastElementChild.cloneNode(true);
    //插入到最后
    doms.carousel.appendChild(first);
    //插入到最前
    doms.carousel.insertBefore(last, doms.carousel.firstElementChild)
    //将复制的第一张的位置调整
    last.style.position = 'absolute';
    last.style.transform = `translateX(-100%)`
  }
  clone()

  //实现右边的无缝滚动
  var count = doms.indicator.length
  function rightMove() {
    //首先去除动画效果
    if (curIndex === count - 1) {
      doms.carousel.style.transform = `translateX(100%)`;
      doms.carousel.style.transition = 'none';
      //强制渲染
      doms.carousel.clientHeight;
      moveTo(0)
    } else {
      moveTo(curIndex + 1)
    }
  }
  //实现左边的无缝滚动
  function leftMove() {

    if (curIndex === 0) {
      doms.carousel.style.transform = `translateX(-${count}00%)`;
      doms.carousel.style.transition = 'none';
      //强制渲染
      doms.carousel.clientHeight;
      moveTo(count - 1)
    } else {
      moveTo(curIndex - 1)
    }
  }

  //添加点击事件
  doms.left.addEventListener('click', () => {
    //关闭定时器
    clearInterval(timer)
    leftMove()
    //开启定时器
    timer = setInterval(() => {
      rightMove()
    }, 2000)
  })
  doms.right.addEventListener('click', () => {
    //关闭定时器
    clearInterval(timer)
    rightMove()
    //开启定时器
    timer = setInterval(() => {
      rightMove()
    }, 2000)
  })
</script>

</html>

 

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

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

【广州华锐互动】VR全景工厂虚拟导览,虚拟现实技术提升企业数字化信息管理水平

随着工业4.0的到来&#xff0c;VR工厂全景制作成为了越来越多工业企业的选择。传统的工厂管理方式往往存在诸多问题&#xff0c;如信息不对称、安全隐患等。为了解决这些问题&#xff0c;VR工厂全景制作应运而生&#xff0c;它通过结合虚拟现实现实技术和数据采集技术&#xff…

PostGIS空间数据中基础常用函数介绍

目录 前言 一、基础数据 1、数据结构准备 2、基础数据构造 二、常用空间函数 1、st_srid 获取空间对象SRID 2、st_asgeojson geojson转换 3、st_aswkt wkt支持 4、st_area 面积计算 5、ST_Buffer 缓冲区 6、其它函数 总结 前言 近些年&#xff0c;面向GIS的应用如雨后…

画图工具draw.io UML图 使用

点击下载 git 下载地址&#xff1a;https://github.com/jgraph/drawio-desktop/releases 在线版本&#xff08;推荐&#xff0c;可以储存&#xff0c;可以共享&#xff09;&#xff1a;https://app.diagrams.net/ 流程图相关

【校招VIP】产品面试之职业规划

考点介绍&#xff1a; 对于刚入行的产品同学&#xff0c;由于行业知识和经验不足&#xff0c;只能执行上层的产品策略&#xff0c;但是与团队的沟通是非常重要的&#xff0c;产品经理就是沙丁鱼中的鲶鱼&#xff0c;必须要能够把控整个团队的开发节奏&#xff0c;知道如何最大化…

12 权重衰退

过拟合的应对方法——weight_decay 权重衰退是最广泛使用的正则化方法之一。 模型容量受参数个数和参数范围影响&#xff0c;通过L2正则项限制w的取值范围&#xff0c;权重w每次更新乘以小于1的数&#xff0c;w的数值范围不会太大&#xff0c;从而降低模型复杂度&#xff0c;…

读研Zotero 插件安装

Zotero 插件下载&#xff1a;https://zoteroplugins.netlify.app/#/ https://www.zotero.org/download/ Zotero 全部插件列表&#xff1a;https://www.zotero.org/support/plugins 安装教程&#xff1a;https://b23.tv/U9Nj2NE

将 Python 与 RStudio IDE 配合使用(R与Python系列第一篇)

目录 前言&#xff1a; 1-安装reticulate包 2-安装Python 3-选择Python的默认版本&#xff08;配置Python环境&#xff09; 4-使用Python 4.1 运行一个简单的Python脚本 4.2 在RStudio上安装Python模块 4.3 在 R 中调用 Python 模块 4.4 在RStudio上调用Python脚本写的…

1. 安装Zookeeper

​ 1.下载 点击下载Zookeeper 单机版安装 安装Zookeeper前需要先安装jdk上传安装包rz解压安装包:tar -zxvf apache-zookeeper-3.6.0-bin.tar.gz -C /opt/app/zookeeper zookeeper目录结构:a. bin: 放置运行脚本和工具脚本b. conf: zookeeper 默认读取配置的目录,里面会有…

截图工具 snipaste

一款十分好用小巧的截图工具 snipaste 一般F1、F3 就能满足大多数使用清情形

【ArcGIS Pro二次开发】(65):进出平衡SHP转TXT、TXT转SHP

最近一个小伙伴提了这么一个需求&#xff0c;需要把TXT和SHP进行互转。 这种TXT文件其实遇到了好几个版本&#xff0c;都有一点小差异。之前已经做过一个TXT转SHP的工具&#xff0c;但好像不适用。于是针对这个版本&#xff0c;做了互转的2个工具。 【SHP转TXT】 一、要实现的…

Nginx从入门到精通之Nginx应用场景和配置参数最佳实践

高级Nginx应用场景和配置技巧 文章目录 高级Nginx应用场景和配置技巧1. 配置参数最佳实践2. Nginx配置示例3. 常见场景1. 静态文件服务2. 动态内容代理3. SSL加密4. URL重写5. 访问控制6. 请求限制7. 日志记录8. 压缩9. 定向与错误页面10. 跨域配置11. 长连接处理12. 代理WebSo…

【文心一言大模型插件制作初体验】制作面试错题本大模型插件

文心一言插件开发初体验 效果图 注意&#xff1a;目前插件仅支持在本地运行&#xff0c;虽然只能自用&#xff0c;但仍然是一个不错的选择。&#xff08;什么&#xff1f;你说没有用&#xff1f;这不可能&#xff01;文心一言app可以支持语音&#xff0c;网页端结合手机端就可…

【算法与数据结构】654、LeetCode最大二叉树

文章目录 一、题目二、解法三、完整代码 所有的LeetCode题解索引&#xff0c;可以看这篇文章——【算法和数据结构】LeetCode题解。 一、题目 二、解法 思路分析&#xff1a;【算法与数据结构】106、LeetCode从中序与后序遍历序列构造二叉树这两道题有些类似&#xff0c;相关代…

Anaconda虚拟环境下导入opencv

文章目录 解决方法测试 解决方法 1、根据自己虚拟环境对于的python版本与电脑对应的位长选择具体的版本&#xff0c;例如python3.9选择cp39&#xff0c;64位电脑选择64 下载地址&#xff1a;资源地址 若是不确定自己虚拟环境对应的python版本&#xff0c;可以输入下列命令&…

【启扬方案】启扬多尺寸安卓屏一体机,助力仓储物料管理系统智能化管理

随着企业供应链管理的不断发展&#xff0c;对仓储物料管理的要求日益提高。企业需要实时追踪和管理物料的流动&#xff0c;提高物流效率、降低库存成本和减少库存的风险。因此&#xff0c;仓储物料管理系统的实现成为必要的手段。 仓储物料管理系统一体机作为一种新型的物料管理…

Java中网络的基本介绍。网络通信,网络,ip地址,域名,端口,网络通信协议,TCP/IP传输过程,网络通信协议模型,TCP协议,UDP协议

- 网络通信 概念&#xff1a;网络通信是指通过计算机网络进行信息传输的过程&#xff0c;包括数据传输、语音通话、视频会议等。在网络通信中&#xff0c;数据被分成一系列的数据包&#xff0c;并通过网络传输到目的地。在数据传输过程中&#xff0c;需要确保数据的完整性、准…

【STM32】SPI初步使用 读写FLASH W25Q64

硬件连接 (1) SS( Slave Select)&#xff1a;从设备选择信号线&#xff0c;常称为片选信号线&#xff0c;每个从设备都有独立的这一条 NSS 信号线&#xff0c;当主机要选择从设备时&#xff0c;把该从设备的 NSS 信号线设置为低电平&#xff0c;该从设备即被选中&#xff0c;即…

0.96寸IIC-OLED屏幕

文章目录 一、硬件介绍1.1 0.96寸IIC-OLED屏幕1.2 主控1.3 取模工具 二、软件程序2.1 oled.c2.2 oled.h2.3 font.c2.4 font.h 一、硬件介绍 1.1 0.96寸IIC-OLED屏幕 1.2 主控 使用stm32f103c8t6单片机进行控制, IIC驱动使用软件模拟的方式。 1.3 取模工具 文字取模工具&…

算法通关村第10关【青铜】| 快速排序各种写法

思路&#xff1a; 指定一个数字&#xff0c;将数组比他小的放到左边&#xff0c;比他大的放到右边&#xff0c;实现归位 然后再指定一个数字递归&#xff0c;一直遍历完数组 最好的情况每次指定的都是中间位置的数字&#xff0c;划分完后两边长度相等&#xff0c;2T(n/2) O…

解决crosstalk的方法及原理分析

我正在「拾陆楼」和朋友们讨论有趣的话题,你⼀起来吧? 拾陆楼知识星球入口 crosstalk是干扰线与受绕线之间由于信号跳变产生的耦合电容引起的。 解决crosstalk的方法从两方面入手,一方面降低耦合电容,一方面降低timing window的overlap。 静态时序分析: 串扰延迟分析 以…