vue实现气泡运动撞击效果

本文实例为大家分享了vue实现气泡运动撞击效果的具体代码,供大家参考,具体内容如下

封装组件

<template>
  <ul id="main">
    <li v-for="(item, index) in circleData" :key="index" :class="{'active': item.is_latest_sign_user}">
      <div>
        <span>{{ item.nick_name }}</span>
        <span>签到</span>
      </div>
    </li>
  </ul>
</template>

<script>

export default {
  data() {
    return {
      circleData: [],
      circleDom: [],
      circleArr: [],
      //初始化运动的最大宽和高,初始定义0
      maxW: 0,
      maxH: 0,
      timer: null,
      timerArr: [],
      count: 0,
    };
  },
  mounted() {
    this.getLatest_sign_users('init')
    this.timer = setInterval((i) => {
      this.count++
      this.getLatest_sign_users()
    }, 5000)
  },
  methods: {
    getLatest_sign_users(type = '') {
      let data = []
      // nick_name: 用户名
      // is_latest_sign_user: 是否是新签到用户
      // gender: 0-女 1-男
      if (this.count === 0) {
        data = [
          {
            id: '1',
            nick_name: '萧一',
            is_latest_sign_user: true,
            gender: 0
          },
          {
            nick_name: '杨二',
            is_latest_sign_user: true,
            gender: 0
          },
          {
            nick_name: '张三',
            is_latest_sign_user: true,
            gender: 0
          }
        ]
      } else if (this.count === 1) {
        data = [
          {
            nick_name: '萧一',
            is_latest_sign_user: false,
            gender: 0
          },
          {
            nick_name: '杨二',
            is_latest_sign_user: false,
            gender: 0
          },
          {
            nick_name: '张三',
            is_latest_sign_user: false,
            gender: 0
          },
          {
            nick_name: '李四',
            is_latest_sign_user: true,
            gender: 1
          },
          {
            nick_name: '王五',
            is_latest_sign_user: true,
            gender: 0
          },
          {
            nick_name: '徐六',
            is_latest_sign_user: true,
            gender: 1
          },
          {
            nick_name: '刘七',
            is_latest_sign_user: true,
            gender: 1
          }
        ]
      } else if (this.count === 2) {
        data = [
          {
            nick_name: '萧一',
            is_latest_sign_user: false,
            gender: 0
          },
          {
            nick_name: '杨二',
            is_latest_sign_user: false,
            gender: 0
          },
          {
            nick_name: '张三',
            is_latest_sign_user: false,
            gender: 0
          },
          {
            nick_name: '李四',
            is_latest_sign_user: false,
            gender: 1
          },
          {
            nick_name: '王五',
            is_latest_sign_user: false,
            gender: 0
          },
          {
            nick_name: '徐六',
            is_latest_sign_user: false,
            gender: 1
          },
          {
            nick_name: '刘七',
            is_latest_sign_user: false,
            gender: 1
          },
          {
            nick_name: '何八',
            is_latest_sign_user: true,
            gender: 0
          },
          {
            nick_name: '柳九',
            is_latest_sign_user: true,
            gender: 0
          },
          {
            nick_name: '甄十',
            is_latest_sign_user: true,
            gender: 1
          },
          {
            nick_name: '十一',
            is_latest_sign_user: true,
            gender: 1
          },
          {
            nick_name: '十二',
            is_latest_sign_user: true,
            gender: 1
          }
        ]
      } else {
        data = [
          {
            nick_name: '萧一',
            is_latest_sign_user: false,
            gender: 0
          },
          {
            nick_name: '杨二',
            is_latest_sign_user: false,
            gender: 0
          },
          {
            nick_name: '张三',
            is_latest_sign_user: false,
            gender: 0
          },
          {
            nick_name: '李四',
            is_latest_sign_user: false,
            gender: 1
          },
          {
            nick_name: '王五',
            is_latest_sign_user: false,
            gender: 0
          },
          {
            nick_name: '徐六',
            is_latest_sign_user: false,
            gender: 1
          },
          {
            nick_name: '刘七',
            is_latest_sign_user: false,
            gender: 1
          },
          {
            nick_name: '何八',
            is_latest_sign_user: false,
            gender: 0
          },
          {
            nick_name: '柳九',
            is_latest_sign_user: false,
            gender: 0
          },
          {
            nick_name: '甄十',
            is_latest_sign_user: false,
            gender: 1
          },
          {
            nick_name: '十一',
            is_latest_sign_user: false,
            gender: 1
          },
          {
            nick_name: '十二',
            is_latest_sign_user: false,
            gender: 1
          }
        ]
      }
      this.circleData = [...data]
      if (type === 'init') {//初次加载时默认全是新增签到用户
        data.forEach(item => item.is_latest_sign_user = true)
      }
      this.$nextTick(() => {
        if (data.length) {
          this.initBubble()
        }
      })
    },
    initBubble() {
      let main = document.getElementById("main");
      let divDom = main.getElementsByClassName("active"); //获取新增加的dom
      if (!divDom.length) return;

      //清理每个球得定时器
      this.timerArr.forEach(item => {
        clearInterval(item)
      })
      this.timerArr = []

      //给新增加的dom设置宽高
      for (let i = 0; i < divDom.length; i++) {
        let colors = [
          "#EF250A",
          "#830AF6"
        ];
        divDom[i].style.boxShadow = "0 0 20px" + " " + colors[this.circleData[i].gender] + " " + "inset";
        // 10个以上尺寸变小
        divDom[i].style.width = "46px";
        divDom[i].style.height = "46px";
        divDom[i].style.fontSize = "12px";
        divDom[i].style.lineHeight = "16px";
        this.circleDom.push(divDom[i])
      }

      //根据浏览器窗口的大小自动调节小球的运动空间
      window.onresize = () => {
        this.maxW = main.clientWidth - divDom[0].clientWidth; //为了让小球不卡在浏览器边缘
        this.maxH = main.clientHeight - divDom[0].clientHeight; // 所以要减去自身的宽高
      };
      onresize();

      //数组对象的初始化
      for (let i = 0; i < this.circleDom.length; i++) {
        let obj = {};
        console.log(this.circleDom[i]);
        if (this.circleDom[i].getAttribute("class") === 'active') {
          obj.x = Math.floor(Math.random() * (this.maxW + 1)); //初始x坐标
          obj.y = Math.floor(Math.random() * (this.maxH + 1)); //初始y坐标
          obj.cx = obj.x + this.circleDom[0].offsetWidth / 2;//圆心x坐标
          obj.cy = obj.y + this.circleDom[0].offsetHeight / 2;//圆心y坐标
          obj.movex = Math.floor(Math.random() * 2); //x轴移动方向
          obj.movey = Math.floor(Math.random() * 2); //y轴移动方向
          obj.speed = 0.2; //随机速度
          obj.timer = null; //计时器
          obj.index = i; //索引值
          this.circleArr.push(obj)
          //小球位置初始化
          this.circleDom[i].style.left = obj.x + 'px';
          this.circleDom[i].style.top = obj.y + 'px';
        } else {
          //保留之前数据得位置信息,不刷新位置
          obj = this.circleArr[i]
        }
        this.move(obj);
      }
    },

    //移动函数
    move(balls) {
      //每个球单独有定时器
      balls.timer = setInterval(() => {
        if (balls.movex === 1) {
          //如果往右跑,则一直加速度,碰到边界,改为反方向运动
          balls.x += balls.speed;
          if (balls.x + balls.speed >= this.maxW) {
            //防止小球出界
            balls.x = this.maxW;
            balls.movex = 0; //小球运动方向发生改变
          }
        } else {
          balls.x -= balls.speed; // 1和0表示正反方向
          if (balls.x - balls.speed <= 0) {
            balls.x = 0;
            balls.movex = 1;
          }
        }
        if (balls.movey === 1) {
          balls.y += balls.speed;
          if (balls.y + balls.speed >= this.maxH) {
            balls.y = this.maxH;
            balls.movey = 0;
          }
        } else {
          balls.y -= balls.speed;
          if (balls.y - balls.speed <= 0) {
            balls.y = 0;
            balls.movey = 1;
          }
        }
        if (this.circleDom[balls.index]) {
          balls.cx = balls.x + this.circleDom[0].offsetWidth / 2;//小球圆心等于:运动中x的值加上自身的半径
          balls.cy = balls.y + this.circleDom[0].offsetHeight / 2;
          this.circleDom[balls.index].style.left = balls.x + "px"; //小球相对于屏幕的位置
          this.circleDom[balls.index].style.top = balls.y + "px";
          this.crash(balls.index); //每个小球进行碰撞检测
        }
      }, 25);
      this.timerArr.push(balls.timer)
    },
    //碰撞函数
    crash(a) {
      let container = [...this.circleArr]
      let ball1x = container[a].cx; //在数组中任意球的圆心坐标
      let ball1y = container[a].cy;//思路:先随便拿一个球,然后遍历所有球,拿这个球和所有球的圆心距离比较
      for (let i = 0; i < container.length; i++) {
        if (i !== a) { //判断取出来的球不是本身,才能和其他球进行距离判断
          let ball2x = container[i].cx; //将其他球的圆心坐标赋值给球2
          let ball2y = container[i].cy;
          //圆心距 求两个点之间的距离,开平方
          let distence = Math.sqrt((ball1x - ball2x) * (ball1x - ball2x) + (ball1y - ball2y) * (ball1y - ball2y));
          if (distence <= this.circleDom[0].offsetWidth) { //球心距离和求直径比较
            if (ball1x > ball2x) { //当前位于未知求的右方
              if (ball1y > ball2y) {//预设未知球撞当前球,然后当前球改变运动
                container[a].movex = 1; //1表示为正值,对应的右和下
                container[a].movey = 1;//0表示为负值,对应的左和上
              } else if (ball1y < ball2y) {
                container[a].movex = 1;
                container[a].movey = 0;
              } else {
                container[a].movex = 1;
              }
            } else if (ball1x < ball2x) {
              if (ball1y > ball2y) {
                container[a].movex = 0;
                container[a].movey = 0;
              } else if (ball1y < ball2y) {
                container[a].movex = 0;
                container[a].movey = 1;
              } else {
                container[a].movex = 0;
              }
            } else {
              if (ball1y > ball2y) {
                container[a].movey = 1;
              } else if (ball1y < ball2y) {
                container[a].movey = 0;
              }
            }
          }
        }
      }
    }
  },
  beforeDestroy() {
    //清理每个球得定时器
    this.timerArr.forEach(item => {
      clearInterval(item)
    })
    //清理签到数据
    clearInterval(this.timer)
  }
};
</script>
<style lang='less' scoped>
#main {
  position: relative;
  width: 100%;
  height: 100%;
  overflow: hidden;
  padding: 0;

  li {
    position: absolute;
    overflow: hidden;
    -moz-border-radius: 50%;
    -webkit-border-radius: 50%;
    border-radius: 50%;
    display: flex;
    align-items: center;
    justify-content: center;
    flex-wrap: wrap;

    &.active {
      animation: scaleBox 1s 1;
    }

    @keyframes scaleBox {
      0% {
        transform: scale(1);
      }
      50% {
        transform: scale(1.2);
      }
      100% {
        transform: scale(1);
      }
    }

    div {
      span {
        display: block;
        width: 100%;
        color: #fff;
        text-align: center;
      }
    }
  }
}
</style>

实现效果

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持我们。

(0)

相关推荐

  • vue+elementui 对话框取消 表单验证重置示例

    最近在写增删改查,在新增的时候要弹出对话框填写form表单信息,发现对话框右上角的小X和右下角的取消不是一个事件,我想在点击它们两个的时候都可以重置表单,最终解决,如下. vue: <el-dialog :title="titleName[dialogStatus]" :visible.sync="dialogFormVisible" @close="closeDialog" :close-on-click-modal="fals

  • vue如何实现关闭对话框后刷新列表

    目录 关闭对话框后刷新列表 父窗口代码 子窗口代码 关闭打开的窗口后刷新父页面 解决办法 关闭对话框后刷新列表 有些场景需要实现用户弹窗确定后自动刷新列表,父窗口绑定fevent即可 父窗口代码 <template>      <div>  <el-button @click="$refs.editform.dialogFormVisible = true" >编辑用户</el-button>  <editform  @fevent

  • Vue3编写气泡对话框组件

    Vue3气泡对话框组件,使用 TypeScript枚举限定类型,样式用到了 TailwindCSS 组件代码 <template>   <div class="mt-5 mb-5 p-2 bg-white border-solid border-gray-300 border-l border-t border-r border-b border-light-blue-500 rounded-md relative">     <div :class=&q

  • vue3动态加载对话框的方法实例

    目录 简介 常规方式使用对话框 异步动态加载 使用方式 TestModal.vue 使用结果 动态操作对话框的实现 DzModalService.ts main.ts 总结 简介 介绍使用vue3的异步组件动态管理对话框组件,简化对话框组件使用方式.本文使用的是vue3.typescript.element_plus完成的示例. 常规方式使用对话框 一般情况下,使用对话框组件,会使用v-model进行双向绑定,通过visible变量控制对话框的显示和关闭.常规方式有一个弊端,自定义组件中使用<e

  • vue+ElementUI 关闭对话框清空验证,清除form表单的操作

    前面跟大家提到过 elementUI验证的问题,那么今天就来看看 点击对话框和关闭按钮 怎么清空验证,清空form表单,避免二次点击还会有 验证错误的提示 1.首先在你的对话框 取消按钮 加一个click事件,例如:(ps::callOf里面的addGroupData和ref一 一对应起来) <div slot="footer" class="dialog-footer"> <el-button @click="callOf('addGr

  • vue中的模态对话框组件实现过程

    写在前面 对话框是很常用的组件 , 在很多地方都会用到,一般我们可以使用自带的alert来弹出对话框,但是假如是设计出的图该怎么办呢 ,所以我们需要自己写一个对话框,并且如果有很多地方都用到,那我们很有必要写成一个通用的组件形式,在需要的地方之间引用. 现在我们来动手实现一个对话框组件 ,按照之前的习惯,我们先看下实现的效果图 1.首先,通过template定义一个组件 <template id="dialog"> <div class="dialog&qu

  • vue实现element-ui对话框可拖拽功能

    element-ui对话框可拖拽及边界处理 应业务需求,需要实现对话框可拖拽问题,应element-ui没有提供官方支持,于是便参考大神的文章,得出了适合业务需要的解决方案.很多大神给出的代码是没有解决边界问题的,但是不解决边界问题存在一个bug,拖到不可视区域后边再也拖不回来了,不信你们可以试试. 在实现的功能的情况下,封装成了js文件,然后再main.js中引入后可全局使用. 还是上代码吧 功能实现代码directives.js代码如下: import Vue from 'vue'; //

  • vue+elementui(对话框中form表单的reset问题)

    目录 对话框中form表单的reset问题 解决原理 解决办法 element UI form表单重置无效 实例化是 说下解决 对话框中form表单的reset问题 一般在新增和编辑的时候用的都是同一个对话框和form表单,而在先点击编辑的时候form表单的resetfileds函数就会失效 解决原理 实际上结构是(通过vue类比) data里面有一个form表单的初始值, methods里面定义了一个resetfileds的函数 resetfileds函数的作用:记录在mounted生命周期执

  • Vue对话框组件使用方法详解

    本文实例为大家分享了Vue对话框组件的使用,供大家参考,具体内容如下 效果如下图所示:(整体样式模仿ant-design-vue Modal样式,同时阴影覆盖浏览器窗口) ①创建组件Dialog.vue: <template>   <div class="m-dialog-mask">     <div class="m-modal">       <div class="m-modal-content"

  • element-ui和vue表单(对话框)验证提示语(残留)清除操作

    问题是这样的: 如果一个页面有多个按钮打开相同的element-ui的Dialog对话框,那么如果第一个点击"顺序"按钮出现红色验证提示语后, 再点击"取消",或者点击页面空白处此对话框消失,而后再点击同一个"顺序"按钮亦或别的"顺序"按钮, 那么此红色提示依然存在,显然是无法忍受的,那么怎么去除呢, 这里也用到一个vue的一个语法watch,从字面意思上理解就是起到监控的作用,监控这个diaLog对话框的打开和关闭. 在 &

  • 解决Vue开发中对话框被遮罩层挡住的问题

    在Vue的开发中,一旦我们用到对话框,经常出现的问题是对话框被遮罩层挡住,无论是Element-UI dialog还是bootstrap的Modal,如下图所示: 造成这个问题的原因是对话框组件的父元素的position有fixed或者relative值,比较简单易行的办法如下:对于bootstrap Modal需要添加css语句 .modal-backdrop {  z-index: -1;} 而对于Element UI该组件已经在属性层面提供了解决办法,只要添加:modal-append-t

  • vue 指令之气泡提示效果的实现代码

    菜鸟学习之路 //L6zt github 自己 在造组件轮子,也就是瞎搞. 自己写了个slider组件,想加个气泡提示.为了复用和省事特此写了个指令来解决. 预览地址 项目地址github 我叫给它胡博 效果图片 我对指令的理解: 前不久看过 一部分vnode实现源码,奈何资质有限...看不懂. vnode的生命周期-----> 生成前.生成后.生成真正dom.更新 vnode.更新dom . 销毁. 而Vue的指令则是依赖于vnode 的生命周期, 无非也是有以上类似的钩子. 代码效果 指令挂

随机推荐