vue项目双滑块组件使用

 更新时间:2024年07月27日 09:45:03   作者:汉武大帝·  
这篇文章主要介绍了vue项目双滑块组件使用方式,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教

vue项目双滑块组件

先看效果图

说明

  • 两端值300,750 是可选范围的最大值;
  • 两个按钮上的值是已经选择的范围 ;

目前灰色背景条的宽度单位是px,

上面代码中,背景条宽度是写死的;

下面是优化后的代码

<template>
  <div
    :style="{
      display: 'flex',
      'justify-content': 'center',
      width: '100%',
    }"
  >
    <div
      class="slider-box"
      id="slider-box"
      :style="{
        width: '100%',
        background: outBg,
      }"
      @mousedown="handleClickZong"
    >
      <div
        :class="['slider-nei', isClick ? 'transtion' : '']"
        :style="{
          width: Math.abs(moveDistanceOne - moveDistanceTwo) + 20 + 'px',
          transform:
            moveDistanceTwo > moveDistanceOne
              ? `translateX(${moveDistanceOne}px)`
              : `translateX(${moveDistanceTwo}px)`,
          background: neiBg,
        }"
      ></div>
      <div
        :style="{
          transform: `translateX(${moveDistance}px)`,
        }"
        :class="{
          'point-box': true,
          zhua: isMove,
          transtion: isClick,
        }"
        v-if="!range"
        @mousedown.stop.prevent="handleMousedown"
      ></div>
      <div
        :style="{
          transform: `translateX(${moveDistanceOne}px)`,
          background: btnBg,position:'absolute','z-index': leftIndex
        }"
        :class="{
          'point-box': true,
          zhua: isMove,
          transtion: isClick,
        }"
        v-if="range"
        @mousedown.stop.prevent="handleMousedownOne"
        @touchstart="handleMousedownOne1"
        @mouseup="handleUp"
        @touchend="handleUp"
      >
        {{ Math.trunc(moveDistanceOne * rate) + min }}
      </div>
      <div
        :style="{
          transform: `translateX(${moveDistanceTwo}px)`,
          background: btnBg,position:'absolute','z-index': rightIndex
        }"
        :class="{
          'point-box': true,
          point: true,
          zhua: isMove,
          transtion: isClick,
        }"
        @mousedown.stop.prevent="handleMousedownTwo"
        @touchstart="handleMousedownTwo1"
        @mouseup="handleUp"
        @touchend="handleUp"
      >
        {{ Math.trunc(moveDistanceTwo * rate + min) }}
      </div>
    </div>
  </div>
</template>
<script>
import { mapGetters } from "vuex";
export default {
  data() {
    return {
      range: true, //是否开启双滑块
      isClick: false, //是否直接点击
      isMove: false, //是否处于滑动状态
      flag: false, //
      moveDistance: 0,
      initX: 0, //记录鼠标点下时的坐标,用于移动中求差值
      moveStart: 0, //进度条的开始位置
      moveEnd: 0, //进度条的结束位置
      moveDistanceOne: 0, //滑块一的位置
      moveDistanceTwo: 50, //滑块二的位置
      rate: 1, // 比率
      outerWidth: 100,
      rate1: 1, // 比率
      outerWidth1: 100,
      valueRange: [],
      leftIndex: 0,
      rightIndex: 0.5,
    };
  },
  computed: {
    // 使用对象展开运算符将 getter 混入 computed 对象中
    ...mapGetters("common", [
      "userType"
    ]),
  },
  props: {
    // 可选择范围的较大值
    max: {
      type: Number,
      default: 750,
    },
    // 外层背景色
    outBg: {
      type: String,
      default: "#F0F0F0",
    },
    // 滑块背景色
    btnBg: {
      type: String,
      default: "#497cfe",
    },
    // 内层背景色
    neiBg: {
      type: String,
      default: "#E5EDFF",
    },
    // 可选择范围的较小值
    min: {
      type: Number,
      default: 150,
    },
    // 已经选择的范围
    valuesRange: {
      type: Array,
      default: [350, 550],
    },
  },
  watch: {
    valuesRange: {
      handler(val1) {
        this.valueRange = val1;
        this.init();
      },
      deep: true,
      immediate: true,
    },
  },
  mounted() {
    window.addEventListener("resize", this.init);
    window.addEventListener("mouseup", () => {
      if(this.flag) {
        this.handleUp2();
        this.flag = false;
      }
    });
    // this.init();
  },
  methods: {
    init() {
      this.$nextTick(() => {
        /* this.valueRange [1, 750] */
        if (this.valueRange && this.valueRange.length > 0) {
          let max_min = this.max - this.min;
          var slide_width = document.querySelector("#slider-box").clientWidth;
          var point_width = document.querySelector(".point").clientWidth;
          this.outerWidth = slide_width - point_width;
          this.outerWidth1 = slide_width - point_width -  point_width;
          this.rate = max_min / this.outerWidth;
          this.rate1 = max_min / this.outerWidth1;
          let vr = JSON.parse(JSON.stringify(this.valueRange));
          if(vr[0] > vr[1]) {
            this.valueRange = [vr[1],vr[0]];
          }
          this.moveDistanceOne = (this.valueRange[0] - this.min) / this.rate;
          this.moveDistanceTwo = (this.valueRange[1] - this.min) / this.rate;
          this.$forceUpdate();
        }
      });
    },
    //移动端 touchstart 点击监听 ontouchmove 移动监听 ontouchend 松开监听,e.changedTouches[0] 元素偏移的一些数据
    //PC端 mousedown 点击监听 onmousemove 移动监听 onmouseup 松开监听
    /**
     * 初始坐标就是鼠标点下时的clientX坐标,这样就可以做到将slider放在页面任何位置,拖动原点偏移的量也是正确的,
     * 因为原点移动距离是用鼠标移动的位置和鼠标点下的位置做差值计算,所以不用担心这里的clientX会因为slider放在
     * 别的地方而导致距离计算错误
     * @param e
     */
    //鼠标点击
    handleMousedown(e) {
      this.isMove = true;
      this.initX = e.clientX;
      const sliderDomWidth = this.outerWidth;
      // const sliderDomWidth = document.getElementsByClassName("slider-box")[0].clientWidth;
      document.onmousemove = (e) => {
        if (this.isMove) {
          this.moveDistance += e.clientX - this.initX;
          this.initX = e.clientX;
          if ((this.moveDistance / sliderDomWidth) * 100 > 100) {
            this.moveDistance = sliderDomWidth;
            return;
          }
          if ((this.moveDistance / sliderDomWidth) * 100 < 0) {
            this.moveDistance = 0;
            return;
          }
        }
      };
      document.onmouseup = (e) => {
        this.isMove = false;
        document.onmousemove = null;
      };
    },
    handleUp(){
      // console.log('this.valueRange', this.valueRange);
      // this.$emit("recommendRange", this.valueRange);
    },
    handleUp2(){
      this.$emit("recommendRange", this.valueRange);
    },
    handleUp3(){
      this.$emit("recommendRange", this.valueRange);
    },
    handleUp4(){
      this.$emit("recommendRange", this.valueRange);
    },
    /**
     * 当点击进度条某一个位置时,快捷设置进度条的位置,这里就需要用offsetX而不是clientX了。这里需要设置的便宜距离是鼠标距离目标元素的距离而不是鼠标在整个页面上的坐标
     * @param e
     */
    handleClickZong(e) {
      this.isClick = true;
      setTimeout(() => {
        this.isClick = false;
      }, 300);
      this.moveDistance = e.offsetX - 10;
    },
    //滑动滑块1
    handleMousedownOne1(e) {
      if (this.userType != 'VIP') {
        this.$store.commit('common/SET_SHOWEXCHANGEVIP', true);
        return;
      }
      this.flag = true;
      this.isMove = true;
      this.leftIndex = 0.5;
      this.rightIndex = 0;
      let currentDistance = e.changedTouches[0].clientX;
      document.ontouchmove = (e) => {
        let item = e.changedTouches[0];
        if (this.isMove) {
          const moveX = item.clientX - currentDistance;
          currentDistance = item.clientX;
          this.moveDistanceOne += moveX;
          if (this.moveDistanceOne < 0) {
            this.moveDistanceOne = 0;
            return;
          }
          if (this.moveDistanceOne > this.moveDistanceTwo) {
            this.moveDistanceOne = this.moveDistanceTwo;
            return;
          }
          this.valueRange = [this.moveDistanceOne, this.moveDistanceTwo];
        }
      };
      document.ontouchend = (e) => {
        this.isMove = false;
        // this.flag = false;
        document.onmousemove = null;
      };
    },
    handleMousedownOne(e) {
      console.log(e);
      if (this.userType != 'VIP') {
        this.$store.commit('common/SET_SHOWEXCHANGEVIP', true);
        return;
      }
      this.flag = true;
      this.isMove = true;
      this.leftIndex = 0.5;
      this.rightIndex = 0;
      let currentDistance = e.clientX;
      document.onmousemove = (e) => {
        if (this.isMove) {
          /* moveX 滑块横向移动的距离 */
          const moveX = e.clientX - currentDistance;
          currentDistance = e.clientX;
          this.moveDistanceOne += moveX;
          if (this.moveDistanceOne < 0) {
            this.moveDistanceOne = 0;
            return;
          }
          if (this.moveDistanceOne > this.moveDistanceTwo) {
            this.moveDistanceOne = this.moveDistanceTwo;
            return;
          }
          
          let arr_one = Math.trunc(this.moveDistanceOne * this.rate) + this.min;
          let arr_two = Math.trunc(this.moveDistanceTwo * this.rate) + this.min;
          this.valueRange = [arr_one, arr_two];
        }
      };
      document.onmouseup = (e) => {
        this.isMove = false;
        // this.flag = false;
        document.onmousemove = null;
      };
    },
    //滑动滑块2
    handleMousedownTwo1(e) {
      if (this.userType != 'VIP') {
        this.$store.commit('common/SET_SHOWEXCHANGEVIP', true);
        return;
      }
      this.isMove = true;
      this.flag = true;
      this.leftIndex = 0;
      this.rightIndex = 0.5;
      let currentDistance = e.changedTouches[0].clientX;
      document.ontouchmove = (e) => {
        if (this.isMove) {
          let item = e.changedTouches[0];
          const moveX = item.clientX - currentDistance;
          currentDistance = item.clientX;
          this.moveDistanceTwo += moveX;
          if (this.moveDistanceTwo > this.outerWidth) {
            this.moveDistanceTwo = this.outerWidth;
            return;
          }
          if (this.moveDistanceTwo < this.moveDistanceOne) {
            this.moveDistanceTwo = this.moveDistanceOne;
            return;
          }
          if (this.moveDistanceTwo < 0) {
            this.moveDistanceTwo = 0;
            return;
          }
          this.valueRange = [this.moveDistanceOne, this.moveDistanceTwo];
        }
      };
      document.ontouchend = (e) => {
        this.isMove = false;
        // this.flag = false;
        document.onmousemove = null;
      };
    },
    //滑动滑块2
    handleMousedownTwo(e) {
      if (this.userType != 'VIP') {
        this.$store.commit('common/SET_SHOWEXCHANGEVIP', true);
        return;
      }
      this.flag = true;
      this.isMove = true;
      this.leftIndex = 0;
      this.rightIndex = 0.5;
      let currentDistance = e.clientX;
      document.onmousemove = (e) => {
        if (this.isMove) {
          const moveX = e.clientX - currentDistance;
          currentDistance = e.clientX;
          this.moveDistanceTwo += moveX;
          if (this.moveDistanceTwo > this.outerWidth) {
            this.moveDistanceTwo = this.outerWidth;
            return;
          }
          if (this.moveDistanceTwo < this.moveDistanceOne) {
            this.moveDistanceTwo = this.moveDistanceOne;
            return;
          }
          if (this.moveDistanceTwo < 0) {
            this.moveDistanceTwo = 0;
            return;
          }
          let arr_one = Math.trunc(this.moveDistanceOne * this.rate) + this.min;
          let arr_two = Math.trunc(this.moveDistanceTwo * this.rate) + this.min;
          this.valueRange = [arr_one, arr_two];
        }
      };
      document.onmouseup = (e) => {
        this.isMove = false;
        // this.flag = false;
        document.onmousemove = null;
      };
    },
  },
};
</script>
<style lang="scss" scoped>
@function autoPx($num) {
  @return (($num / 1200) * 24rem);
}
.slider-box {
  height: autoPx(10);
  border-radius: autoPx(20);
  position: relative;
}
 
.slider-nei {
  position: absolute;
  left: 0;
  top: 0;
  width: 0;
  height: autoPx(10);
  border-radius: autoPx(20);
}
 
.point-box {
  width: autoPx(43);
  height: autoPx(20);
  border-radius: autoPx(20);
  position: absolute;
  top: -(autoPx(5));
  color: #fff;
  font-size: autoPx(12);
  line-height: autoPx(20);
  text-align: center;
  user-select: none;
  cursor: pointer;
}
 
.point-box:hover {
  cursor: grab;
}
 
.zhua:active {
  cursor: grabbing;
}
 
.transtion {
  transition: all 0.3s;
}
</style>

总结

以上为个人经验,希望能给大家一个参考,也希望大家多多支持脚本之家。

相关文章

  • vue通过 API 监听数组的变化

    vue通过 API 监听数组的变化

    这篇文章主要介绍了vue通过 API 监听数组的变化,在 Vue 中,你可以通过监听数组的变化来更新界面,Vue 提供了一些特殊的语法以及 API 来实现对数组的监听,本文通过实例代码介绍的非常详细,需要的朋友可以参考下
    2023-05-05
  • vue3+vite项目中按需引入vant报错:Failed to resolve import的解决方案

    vue3+vite项目中按需引入vant报错:Failed to resolve import的解决方案

    最近在vue项目中引入vant的时候发现报错了,经过尝试发现了问题,现将完整引入流程提供给大家参考,下面这篇文章主要给大家介绍了关于vue3+vite项目中按需引入vant报错:Failed to resolve import的解决方案,需要的朋友可以参考下
    2022-12-12
  • vuex实现简易计数器

    vuex实现简易计数器

    这篇文章主要为大家详细介绍了vuex实现一个简易计数器,文中示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2016-10-10
  • vue中的自定义属性并获得属性的值方式

    vue中的自定义属性并获得属性的值方式

    这篇文章主要介绍了vue中的自定义属性并获得属性的值方式,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2022-06-06
  • Vue2项目配置@指向src路径方式

    Vue2项目配置@指向src路径方式

    这篇文章主要介绍了Vue2项目配置@指向src路径方式,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2022-12-12
  • 如何在Vue项目中使用vuex

    如何在Vue项目中使用vuex

    这篇文章主要介绍了如何在Vue项目中使用vuex问题,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2023-01-01
  • vue如何实现observer和watcher源码解析

    vue如何实现observer和watcher源码解析

    这篇文章主要为大家详细介绍了vue如何实现observer和watcher源码的相关资料,分析vue的observe实现源码,聊聊如何一步一步实现$watch,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2017-03-03
  • vue2中的el-select组件数据太多,如何实现滚动加载数据效果

    vue2中的el-select组件数据太多,如何实现滚动加载数据效果

    这篇文章主要介绍了vue2中的el-select组件数据太多,如何实现滚动加载数据效果,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教
    2024-04-04
  • uni-app项目中引入Vant UI组件库完美避坑指南(纯净版)

    uni-app项目中引入Vant UI组件库完美避坑指南(纯净版)

    网上百度uniapp使用vant时,很多答案都是在根路径下创建文件夹,而且都是基于小程序环境的,其实uniapp可以直接使用的,这篇文章主要给大家介绍了关于uni-app项目中引入Vant UI组件库完美避坑指南的相关资料,需要的朋友可以参考下
    2024-02-02
  • vue文件使用iconfont解析

    vue文件使用iconfont解析

    这篇文章主要介绍了vue文件使用iconfont解析,具有很好的参考价值,希望对大家有所帮助。如有错误或未考虑完全的地方,望不吝赐教
    2022-04-04

最新评论