zoukankan      html  css  js  c++  java
  • Vue 实现进度环

    <template>
      <div class="percentloop">
        <div class="circle-left">
          <div ref="leftcontent"></div>
        </div>
        <div class="circle-right">
          <div ref="rightcontent"></div>
        </div>
        <div class="number">
          {{ percent }} %
        </div>
      </div>
    </template>
    
    <script>
    export default {
      props: {
        percentNum: {
          type: [String, Number],
          default: 0
        },
        speed: { // 建议取值为0-3
          type: [String, Number],
          default: 1
        }
      },
      data () {
        return {
          percent: 0,
          initDeg: 0,
          timeId: null,
          animationing: false
        }
      },
      methods: {
        transformToDeg (percent) {
          let deg = 0
          if (percent >= 100) {
            deg = 360
          } else {
            deg = parseInt(360 * percent / 100)
          }
          return deg
        },
        transformToPercent (deg) {
          let percent = 0
          if (deg >= 360) {
            percent = 100
          } else {
            percent = parseInt(100 * deg / 360)
          }
          return percent
        },
        rotateLeft (deg) { // 大于180时,执行的动画
          this.$refs.leftcontent.style.transform = 'rotate(' + (deg - 180) + 'deg)'
        },
        rotateRight (deg) { // 小于180时,执行的动画
          this.$refs.rightcontent.style.transform = 'rotate(' + deg + 'deg)'
        },
        goRotate (deg) {
          this.animationing = true
          this.timeId = setInterval(() => {
            if (deg > this.initDeg) { // 递增动画
              this.initDeg += Number(this.speed)
              if (this.initDeg >= 180) {
                this.rotateLeft(this.initDeg)
                this.rotateRight(180) // 为避免前后两次传入的百分比转换为度数后的值不为步距的整数,可能出现的左右转动不到位的情况。
              } else {
                this.rotateRight(this.initDeg)
              }
            } else { // 递减动画
              this.initDeg -= Number(this.speed)
              if (this.initDeg >= 180) {
                this.rotateLeft(this.initDeg)
              } else {
                this.rotateLeft(180) // 为避免前后两次传入的百分比转换为度数后的值不为步距的整数,可能出现的左右转动不到位的情况。
                this.rotateRight(this.initDeg)
              }
            }
            this.percent = this.transformToPercent(this.initDeg) // 百分比数据滚动动画
            const remainer = Number(deg) - this.initDeg
            if (Math.abs(remainer) < this.speed) {
              this.initDeg += remainer
              if (this.initDeg > 180) {
                this.rotateLeft(deg)
              } else {
                this.rotateRight(deg)
              }
              this.animationFinished()
            }
          }, 10)
        },
        animationFinished () {
          this.percent = this.percentNum // 百分比数据滚动动画
          this.animationing = false
          clearInterval(this.timeId)
          this.$emit('animationFinished') // 动画完成的回调
        }
      },
      created () {
        this.goRotate(this.transformToDeg(this.percentNum))
      },
      watch: {
        'percentNum': function (val) {
          if (this.animationing) return
          this.goRotate(this.transformToDeg(val))
        }
      }
    }
    </script>
    
    <style scoped lang="scss">
    .percentloop {
      position: relative;
      width: 100%;
      height: 100%;
      border-radius: 50%;
      overflow: hidden;
      .circle-left, .circle-right {
        position: absolute;
        top: 0;
        left: 0;
        width: 50%;
        height: 100%;
        background-color: red;
        overflow: hidden;
        &>div {
          width: 100%;
          height: 100%;
          background-color: #8a8a8a;
          transform-origin: right center;
          /*transition: all .5s linear;*/
        }
      }
      .circle-right {
        left: 50%;
        &>div {
          transform-origin: left center;
        }
      }
      .number {
        position: absolute;
        top: 9%;
        bottom: 9%;
        left: 9%;
        right: 9%;
        background-color: #fff;
        border-radius: 50%;
        overflow: hidden;
        display: flex;
        align-items: center;
        justify-content: center;
        color: #000;
      }
    }
    </style>
    View Code
  • 相关阅读:
    vue语法
    第3章 语言基础(上)
    第2章 HTML中的JavaScript
    第1章 什么是JavaScript
    附录 A ES6附加特性
    第14章 跨浏览器开发技巧
    第13章 历久弥新的事件
    第12章 DOM操作
    第11章 代码模块化
    第10章 正则表达式
  • 原文地址:https://www.cnblogs.com/jervy/p/14259620.html
Copyright © 2011-2022 走看看