基于vue实现滑块动画效果

主要实现:通过鼠标移移动、触摸元素、鼠标释放、离开元素事件来进行触发

创建了一个滑动盒子,其中包含一个滑块图片。通过鼠标按下或触摸开始事件,开始跟踪滑块的位置和鼠标/触摸位置之间的偏移量。然后,通过计算偏移量和起始时的位移值,设置滑动盒子的 transform 属性来实现滑动效果。使用 transition 属性,还可以为滑块添加滑动动画。
效果:

运行效果图

html 复制代码
<template>
  <div id="app">
    <div class="slider-container img">
      <div class="slider-track">
        <img class="slider-handle" src="./assets/dot.svg" alt="滑块">
      </div>
    </div>
  </div>
</template>

<script>

export default {
  name: 'App',
  components: {
  },
  data(){
    return {
      sliderContainer:'',
      sliderTrack:'',
      sliderHandle:'',
      isDragging:false,
      startPosY:0,
      startTranslateY:0,
      minTranslateY:0,
      maxTranslateY:0,
    }
  },
  created() {

  },
  mounted() {
    this.sliderContainer = document.querySelector('.slider-container');
    this.sliderTrack = document.querySelector('.slider-track');
    this.sliderHandle = document.querySelector('.slider-handle');

    this.isDragging = false;
    this.startPosY = 0;
    this.startTranslateY = 0;
    this.minTranslateY = 0; // 顶部限制为30px,css样式修改后默认为0
    this.maxTranslateY = this.sliderContainer.clientHeight - this.sliderHandle.offsetHeight - 20; // 底部限制为20px

    this.sliderHandle.addEventListener('mousedown', this.startDrag);
    this.sliderHandle.addEventListener('touchstart', this.startDrag);
  },
  methods: {
    startDrag(event) {
      event.preventDefault();

      if (event.type === 'mousedown') {
        this.startPosY = event.clientY;
      } else if (event.type === 'touchstart') {
        this.startPosY = event.touches[0].clientY;
      }

      this.startTranslateY = this.getTranslateY();
      this.isDragging = true;

      document.addEventListener('mousemove', this.drag);
      document.addEventListener('touchmove', this.drag);
      document.addEventListener('mouseup', this.stopDrag);
      document.addEventListener('touchend', this.stopDrag);
    },
    drag(event) {
      if (!this.isDragging) return;

      var currentPosY = 0;

      if (event.type === 'mousemove') {
        currentPosY = event.clientY;
      } else if (event.type === 'touchmove') {
        currentPosY = event.touches[0].clientY;
      }

      var offsetY = currentPosY - this.startPosY;
      var translateY = this.startTranslateY + offsetY;

      // 边缘限制
      if (translateY < this.minTranslateY) {
        translateY = this.minTranslateY;
      } else if (translateY > this.maxTranslateY) {
        translateY = this.maxTranslateY;
      }

      this.setTranslateY(translateY);
    },
    stopDrag() {
      this.isDragging = false;

      document.removeEventListener('mousemove', this.drag);
      document.removeEventListener('touchmove', this.drag);
      document.removeEventListener('mouseup', this.stopDrag);
      document.removeEventListener('touchend', this.stopDrag);
    },

    getTranslateY() {
      var style = window.getComputedStyle(this.sliderTrack);
      var transform = style.transform || style.webkitTransform;
      var matrix = transform.match(/^matrix\((.+)\)$/);

      if (matrix) {
        return parseFloat(matrix[1].split(', ')[5]); // 获取Y轴位移值
      } else {
        return 0;
      }
    },
    setTranslateY(translateY) {
      let y = translateY
      if (y > 30 && y < 60) {
        console.log("午间");
        y = 60
      } else if (y > 60 && y < 90 ) {
        console.log("午间");
        y = 60
      } else if (y > 90 && y <= 130) {
        console.log("晚间");
        y = 120
      } else if (y < 30) {
        console.log("早间");
        y = 0
      }
      this.sliderTrack.style.transform = 'translate3d(0, ' + y + 'px, 0)'; // 设置Y轴位移值
    }

  }
}
</script>

<style>
#app {
  font-family: Avenir, Helvetica, Arial, sans-serif;
  -webkit-font-smoothing: antialiased;
  -moz-osx-font-smoothing: grayscale;
  color: #2c3e50;
}

.img {
  width: 100px;
  height: 200px;
  background-color: antiquewhite;
  background: url("./assets/TimeOfDay.svg") no-repeat;
  background-size: contain;
  position: relative;
  transition: all 0.5s;
}

.hk_img {
  position: absolute;
  top: 30px;
}

.slider-container {
  width: 100px;
  height: 200px;
  overflow: hidden;
  position: relative;
  background-color: #f0f0f0;
}

.slider-track {
  width: 100%;
  height: 100%;
  position: absolute;
  top: 0;
  left: 0;
  transition: transform 0.3s ease-out;
  touch-action: none;
  /* 禁用默认的滑动行为 */
}

.slider-handle {
  width: 50px;
  height: 50px;
  position: absolute;
  top: 15px;
  /* 距离顶部30px */
  left: 50%;
  transform: translateX(-50%);
}

.slider-handle:hover {
  cursor: pointer;
}</style>
相关推荐
潜意识起点17 分钟前
精通 CSS 阴影效果:从基础到高级应用
前端·css
奋斗吧程序媛22 分钟前
删除VSCode上 origin/分支名,但GitLab上实际上不存在的分支
前端·vscode
IT女孩儿31 分钟前
JavaScript--WebAPI查缺补漏(二)
开发语言·前端·javascript·html·ecmascript
m0_748256563 小时前
如何解决前端发送数据到后端为空的问题
前端
请叫我飞哥@3 小时前
HTML5适配手机
前端·html·html5
@解忧杂货铺5 小时前
前端vue如何实现数字框中通过鼠标滚轮上下滚动增减数字
前端·javascript·vue.js
F-2H6 小时前
C语言:指针4(常量指针和指针常量及动态内存分配)
java·linux·c语言·开发语言·前端·c++
苹果酱05676 小时前
「Mysql优化大师一」mysql服务性能剖析工具
java·vue.js·spring boot·mysql·课程设计
gqkmiss7 小时前
Chrome 浏览器插件获取网页 iframe 中的 window 对象
前端·chrome·iframe·postmessage·chrome 插件
m0_748247559 小时前
Web 应用项目开发全流程解析与实战经验分享
开发语言·前端·php