vue花瓣飘落/彩带撒落特效

需求:点击按钮弹出对话框,实现全屏飘落花瓣/彩带

参考文档:用Vue实现网页——花瓣飘落背景_哔哩哔哩_bilibili

方法:一、定义花瓣飘落js

let stop, staticx // eslint-disable-line no-unused-vars
const img = new Image()
img.src = ''
function Sakura (x, y, s, r, fn) {
  this.x = x
  this.y = y
  this.s = s
  this.r = r
  this.fn = fn
}

Sakura.prototype.draw = function (cxt) {
  cxt.save()
  // 这个数值是花瓣大小,电脑端网页 40 效果最好
  const xc = 25 * this.s
  cxt.translate(this.x, this.y)
  cxt.rotate(this.r)
  cxt.drawImage(img, 0, 0, xc, xc)
  cxt.restore()

}

Sakura.prototype.update = function () {
  this.x = this.fn.x(this.x, this.y)
  this.y = this.fn.y(this.y, this.y)
  this.r = this.fn.r(this.r)
  if (this.x > window.innerWidth ||
    this.x < 0 ||
    this.y > window.innerHeight ||
    this.y < 0
  ) {
    this.r = getRandom('fnr')
    if (Math.random() > 0.4) {
      this.x = getRandom('x')
      this.y = 0
      this.s = getRandom('s')
      this.r = getRandom('r')
    } else {
      this.x = window.innerWidth
      this.y = getRandom('y')
      this.s = getRandom('s')
      this.r = getRandom('r')
    }
  }
}

// let SakuraList
const SakuraList = function () {
  this.list = []
}
SakuraList.prototype.push = function (sakura) {
  this.list.push(sakura)
}
SakuraList.prototype.update = function () {
  let i = 0; const len = this.list.length
  for (; i < len; i++) {
    this.list[i].update()
  }
}
SakuraList.prototype.draw = function (cxt) {
  let i = 0; const len = this.list.length
  for (; i < len; i++) {
    this.list[i].draw(cxt)
  }
}
SakuraList.prototype.get = function (i) {
  return this.list[i]
}
SakuraList.prototype.size = function () {
  return this.list.length
}

function getRandom (option) {
  let ret, random
  switch (option) {
    case 'x':
      ret = Math.random() * window.innerWidth
      break
    case 'y':
      ret = Math.random() * window.innerHeight
      break
    case 's':
      ret = Math.random()
      break
    case 'r':
      ret = Math.random() * 6
      break
    case 'fnx':
      random = -0.5 + Math.random()
      ret = function (x) {
        return x + 0.5 * random - 1.7
      }
      break
    case 'fny':
      random = 1.5 + Math.random() * 0.7
      ret = function (x, y) {
        return y + random
      }
      break
    case 'fnr':
      random = Math.random() * 0.03
      ret = function (r) {
        return r + random
      }
      break
  }
  return ret
}

export function flowerDropPink () {
  const requestAnimationFrame = window.requestAnimationFrame ||
    window.mozRequestAnimationFrame ||
    window.webkitRequestAnimationFrame ||
    window.msRequestAnimationFrame ||
    window.oRequestAnimationFrame
  const canvas = document.createElement('canvas')
  // const cxt
  staticx = true
  canvas.height = window.innerHeight
  canvas.width = window.innerWidth
  canvas.setAttribute('style', 'position: fixed;left: 0;top: 0;pointer-events: none;z-index: 9999')
  canvas.setAttribute('id', 'canvas_sakura')
  document.getElementsByTagName('body')[0].appendChild(canvas)
  const cxt = canvas.getContext('2d')
  const sakuraList = new SakuraList()
  for (let i = 0; i < 50; i++) {
    // let sakura, randomX, randomY, randomS, randomR, randomFnx, randomFny, randomFnR
    const randomX = getRandom('x')
    const randomY = getRandom('y')
    const randomR = getRandom('r')
    const randomS = getRandom('s')
    const randomFnx = getRandom('fnx')
    const randomFny = getRandom('fny')
    const randomFnR = getRandom('fnr')
    const sakura = new Sakura(randomX, randomY, randomS, randomR, {
      x: randomFnx,
      y: randomFny,
      r: randomFnR
    })
    sakura.draw(cxt)
    sakuraList.push(sakura)
  }
  stop = requestAnimationFrame(function reverse () {
    cxt.clearRect(0, 0, canvas.width, canvas.height)
    sakuraList.update()
    sakuraList.draw(cxt)
    stop = requestAnimationFrame(reverse)
  })
}

window.onresize = function () {
  const canvas = document.createElement('canvas')
  canvas.width = window.innerWidth
  canvas.height = window.innerHeight
}


export function stopPink () {
  if (staticx) {
    const child = document.getElementById('canvas_sakura')
    child.parentNode.removeChild(child)
    window.cancelAnimationFrame(stop)
    staticx = false
  } else {
    flowerDropPink()
  }
}

        注:1、图片src可自定义;

                2、通过document.getElementsByTagName('body')[0].appendChild(canvas)指定在哪个div块内部添加该特效;

                3、设置canvas的z-index为最大,以确保特效在最上层显示。

二、vue中引入js并在打开dialog时执行飘落方法

<template>
  <div class="app-container pull-auto opp">
    <basic-container>
      <el-button type="primary" @click="test">测试</el-button>
      <el-dialog
        @close="closeDrop"
        :visible.sync="dialogVisible">
        <div style="text-align: center;margin-top: 200px;font-size: 30px;font-weight: 600;margin-bottom: 100px;display: flex;justify-content: center;align-items: center;">内容
        </div>
      </el-dialog>
    </basic-container>
  </div>
</template>

<script>
import {flowerDropBlue, stopBlue} from '@/const/flowDropBlue'
import {flowerDropGreen, stopGreen} from '@/const/flowDropGreen'
import {flowerDropPurple, stopPurple} from '@/const/flowDropPurple'
import {flowerDropRed, stopRed} from '@/const/flowDropRed'
import {flowerDropYellow, stopYellow} from '@/const/flowDropYellow'
import {flowerDropPink, stopPink} from '@/const/flowDropPink'

export default {
  name: 'test',
  data() {
    return {
      dialogVisible: false
    }
  },
  methods: {
    closeDrop() {
      stopBlue()
      stopGreen()
      stopPurple()
      stopRed()
      stopYellow()
      stopPink()
    },
    test() {
      this.dialogVisible = true
      this.$nextTick(() => {
        flowerDropBlue()
        flowerDropGreen()
        flowerDropPurple()
        flowerDropRed()
        flowerDropYellow()
        flowerDropPink()
      })
    }
  }
}
</script>

<style lang="scss">
</style>

        注:这里我尝试一个canvas绘制多个图片,发现显示有点奇怪,所以引入了多个js,绘制不同的花瓣/彩带实现飘落,并在dialog关闭时停止飘落。

最终效果(是动态的):

  • 8
    点赞
  • 13
    收藏
    觉得还不错? 一键收藏
  • 2
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 2
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值