Canvas实现红包雨

需求

  • 实现红包雨效果
  • 红包有不同的倾斜角度
  • 红包下落的速度不同
  • 红包横向位置随机

思路

  • 使用canvas + requestAnimationFrame实现
  • 红包使用图片。防止出现图片读取失败,跨域等问题。这里将图片转成base64再使用
  • 位置、倾斜度、速度等随机,可以使用Math.random()生成随机数实现
  • 红包的旋转角度通过旋转画布context.rotate()实现,绘制完后再旋转回来

倾斜后的lefttop值计算如下图

image.png

实现

  • src/main.js文件中对requestAnimationFrame做兼容处理
// requestAnimationFrame兼容处理 if (!window.requestAnimationFrame) { window.requestAnimationFrame = ( window.webkitRequestAnimationFrame || window.mozRequestAnimationFrame || window.msRequestAnimationFrame|| window.oRequestAnimationFrame || function (callback) { return setTimeout(callback, Math.floor(1000 / 60)) } ) }
  • src/utils/index.js文件中定义几个常用的方法
// 加载图片 export function loadImage (url) { return new Promise((resolve, reject) => { let img = new Image() img.src = url img.onload = () => { resolve(img) } img.onerror = () => { reject(new Error('load image error!')) } }) } // 获取一定范围内的随机数,保留2位小数 export function getRandom (min, max) { if (min > max) { [min, max] = [max, min] } const random = Math.random() * (max - min) + min return Number(random.toFixed(2)) }
  • html部分
<div class="canvas canvas-gift"> <canvas ref="canvasGift"></canvas> </div>
  • javascript部分
import { loadImage, getRandom } from 'utils/index' export default { data () { return { canvas: null, ctx: null, sum: 15, images: [], img: { // 红包图片的地址,宽度和高度 url: '', width: 35, height: 43 } } }, mounted() { this.initCanvas() this.initImages() this.executeFrame() }, methods: { initCanvas() { this.canvas = this.$refs.canvasGift this.ctx = this.canvas.getContext('2d') // 这里获取 canvas 父标签的宽度和高度,赋值给 canvas const parent = this.canvas.parentElement this.canvas.width = parent.offsetWidth this.canvas.height = parent.offsetHeight } // 将访入 this.images 里面的对象初始化单独拎出来作为一个方法 initImageItem (image) { return { image, tilt: getRandom(3, 6), // 速度加成。视情况而定。越小速度越慢 scale: getRandom(-45, 45), // 倾斜度,最多向左或向右倾斜45° left: getRandom(0, window.innerWidth - this.img.width), top: getRandom(10 - this.img.height, -this.img.height) // } }, async initImages () { const image = await loadImage(this.img.url) // 每隔150ms 加入一个image 直到 this.images 数组项的数量达到 this.sum const timer = setInterval(() => { if (this.images.length < this.sum && !this.end) { this.images.push(this.initImageItem(image)) } else { clearInterval(timer) } }, 150) }, executeFrame() { // 页面大小变化话,重新计算 canvas 的长度和宽度 if (this.canvas.width !== window.innerWidth || this.canvas.height !== window.innerHeight) { this.initCanvas() } // 清空所有 this.ctx.clearRect(0, 0, this.canvas.width, this.canvas.height) // 将 this.images 里面的所有项绘制到画布中 this.images.forEach((item, index) => { // 角度转成弧度 const r = item.scale * Math.PI / 180 // 计算倾斜后的坐标值 const temp = item.top - item.left * Math.tan(r) const top = temp * Math.cos(r) const left = (item.left / Math.cos(r)) + temp * Math.sin(r) // 保存状态 this.ctx.save() // 设置画布上的(0,0)位置,也就是旋转的中心点 // this.ctx.translate(0, 0) this.ctx.rotate(r) // 绘制图片 this.ctx.drawImage(item.image, left, top, this.img.width, this.img.height) // 恢复状态 this.ctx.restore() if (item.top + item.tilt <= window.innerHeight) { // 还没有到达底部,继续向下 item.top = item.top + item.tilt } else { // 到达底部后,重新赋值。重新随机从上面出现 this.images[index] = this.initImageItem(item.image) } }) // 递归渲染 window.requestAnimationFrame(this.executeFrame) } } }

创作不易,若本文对你有帮助,欢迎打赏支持作者!

 分享给好友: