需求
- 实现红包雨效果
- 红包有不同的倾斜角度
- 红包下落的速度不同
- 红包横向位置随机
思路
- 使用
canvas
+requestAnimationFrame
实现 - 红包使用图片。防止出现图片读取失败,跨域等问题。这里将图片转成
base64
再使用 - 位置、倾斜度、速度等随机,可以使用
Math.random()
生成随机数实现 - 红包的旋转角度通过旋转画布
context.rotate()
实现,绘制完后再旋转回来
倾斜后的left
和top
值计算如下图
实现
- 在
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)
}
}
}
发表评论