思路
- 使用
canvas
实现
- 通过样式控制
canvas
标签在内容上方
- 没有刮奖前,使用纯色和文字填充。填充完毕后再显示下面的内容
- 刮奖过程中使用一个刮奖效果的图片覆盖。防止出现图片读取失败,跨域等问题。这里将图片转成
base64
再使用
- 绑定
touchstart
、touchmove
、touchend
事件处理刮奖动作
PC
端绑定mousedown
、mousemove
、mouseup
事件处理刮奖动作
- 计算刮奖区域大小,超过总面积的一定比例,就全部打开
实现
<template>
<div class="canvas-container">
<canvas
v-show="canvasShow"
ref="canvas"
class="canvas"
@mousedown.stop="handleMouseDown"
@touchstart.stop="handleMouseDown"
@mousemove.stop="handleMouseMove"
@touchmove.stop="handleMouseMove"
@mouseup.stop="handleMouseUp"
@touchend.stop="handleMouseUp"
></canvas>
<div v-if="dataShow" class="canvas-m">
<slot></slot>
</div>
</div>
</template>
export default {
data() {
return {
isDrawing: false,
lastPoint: null,
canvas: null,
ctx: null,
brush: null,
canvasShow: true,
dataShow: false
}
},
mounted() {
this.initCanvas()
},
methods: {
initCanvas() {
const w = document.documentElement.clientWidth || document.body.clientWidth
this.canvasShow = true
this.canvas = this.$refs.canvas
this.ctx = this.canvas.getContext('2d')
const parent = this.canvas.parentElement
this.canvas.width = parent.offsetWidth
this.canvas.height = parent.offsetHeight
this.ctx.fillStyle = '#eed6c1'
this.ctx.fillRect(0, 0, this.canvas.width + 1, this.canvas.height + 1)
this.ctx.font = 'normal 18px Arial'
this.ctx.fillStyle = '#fff'
this.ctx.textAlign = 'center'
this.ctx.textBaseline = 'middle'
this.ctx.fillText('刮刮查看详情', this.canvas.width / 2, this.canvas.height / 2)
this.dataShow = true
this.brush = new Image()
this.brush.src = ``
},
getMouse (e, canvas) {
let offsetX = 0
let offsetY = 0
if (canvas.offsetParent !== undefined) {
do {
offsetX += canvas.offsetLeft
offsetY += canvas.offsetTop
} while ((canvas = canvas.offsetParent))
}
const mx = (e.pageX || e.touches[0].clientX) - offsetX
const my = (e.pageY || e.touches[0].clientY) - offsetY
return { x: mx, y: my }
},
distanceBetween (point1, point2) {
return Math.sqrt(Math.pow(point2.x - point1.x, 2) + Math.pow(point2.y - point1.y, 2))
},
angleBetween (point1, point2) {
return Math.atan2(point2.x - point1.x, point2.y - point1.y)
},
getFilledInPixels (stride) {
if (!stride || stride < 1) {
stride = 1
}
const pixels = this.ctx.getImageData(0, 0, this.canvas.width, this.canvas.height)
const pdata = pixels.data
const l = pdata.length
const total = (l / stride)
let count = 0
for (let i = count = 0; i < l; i += stride) {
if (parseInt(pdata[i]) === 0) {
count++
}
}
return Math.round((count / total) * 100)
},
handlePercentage (filledInPixels) {
filledInPixels = filledInPixels || 0
if (filledInPixels > 50) {
this.canvasShow = false
}
},
handleMouseDown (e) {
this.isDrawing = true
this.lastPoint = this.getMouse(e, this.canvas)
},
handleMouseMove (e) {
if (!this.isDrawing) {
return
}
const currentPoint = this.getMouse(e, this.canvas)
const dist = this.distanceBetween(this.lastPoint, currentPoint)
const angle = this.angleBetween(this.lastPoint, currentPoint)
let x
let y
for (let i = 0; i < dist; i++) {
x = this.lastPoint.x + (Math.sin(angle) * i) - 25
y = this.lastPoint.y + (Math.cos(angle) * i) - 25
this.ctx.globalCompositeOperation = 'destination-out'
this.ctx.drawImage(this.brush, x, y)
}
this.lastPoint = currentPoint
this.handlePercentage(this.getFilledInPixels(32))
},
handleMouseUp (e) {
this.isDrawing = false
}
}
}
发表评论