canvas绘制刮刮卡,超过一定面积显示全图

说明:栗子转自简书,在他的基础上添加了重置和超过一定面积显示全图,仅做学习使用。(推荐7.2的代码)

1.前端时间做的一个项目需要支持多终端,网页版需要使用html5中canvas画布对象对一组数据进行渲染还原,但是在实际实现过程中遇到了一个问题,canvas中没有mask(遮罩)层的概念,所以一些效果实现不了,最后翻看文档的时候发现可以通过Context对象的globalCompositeOperation属性或者Context的clip()裁剪路径方法实现遮罩的效果。

2.globalCompositeOperation属性介绍

接下来先详细了解下Context的globalCompositeOperation的各种值描述,由于项目不便演示最后我们通过它来实现一个刮刮卡的效果来学习使用它。

属性值描述显示效果
source-over (default)新图形会覆盖在原有内容之上

destination-over会在原有内容之下绘制新图形

source-in新图形会仅仅出现与原有内容重叠的部分。其它区域都变成透明的

destination-in原有内容中与新图形重叠的部分会被保留,其它区域都变成透明的

source-out结果是只有新图形中与原有内容不重叠的部分会被绘制出来

destination-out原有内容中与新图形不重叠的部分会被保留

source-atop新图形中与原有内容重叠的部分会被绘制,并覆盖于原有内容之上

destination-atop原有内容中与新内容重叠的部分会被保留,并会在原有内容之下绘制新图形

lighter两图形中重叠部分作加色处理

darker两图形中重叠的部分作减色处理

xor重叠的部分会变成透明

copy只有新图形会被保留,其它都被清除掉


蓝色 表示先绘制的图形、红色 表示后绘制的图形

3.浏览器支持: Internet Explorer 9FirefoxOperaChromeSafari 支持globalCompositeOperation 属性

通过Context的globalCompositeOperation我们可以灵活的掌握绘制图形之间层叠显示关系,做出很多漂亮的显示效果。接下来我们就使用globalCompositeOperation=destination-out来实现一个刮刮卡的效果。

4.globalCompositeOperation属性应用

刮刮卡实现效果

5.实现原理

  1. 在页面上放一个div容器,设置这个div的宽高、把机器猫的图片设为背景,
  2. 在div中放一个canvas标签,设置canvas的宽高和父容器div的一样。
  3. 获取canvas的context对象绘制一个以灰色为背景宽高和canvas宽高相同的矩形,这样机器猫背景图就被遮住了,只能看见一个灰色的背景。
  4. canvas绑定鼠标mousedown,mousemove和mouseup事件(移动端绑定事件分别是:touchstart,touchmove,touchend),设置鼠标按下标志,鼠标按下或者鼠标按下并且移动时记录鼠标坐标值。
  5. 鼠标点击或者按住鼠标移动的时候开始绘图,绘图的时候设置context.globalCompositeOperation='destination-out'根据上面属性的解释,原有图形(灰色矩形)与新图形(画的线条)不重叠的部分会被保留,所以画过线条的部分不会被保留就可以看见下面机器猫图片背景了。
  6. 鼠标抬起设置鼠标按下标志为false,清空坐标数组。

6.代码实现

6.1这里的方式是通过点击的次数来计算,是否全部清除的

<!DOCTYPE html>
<html lang="en">

<head>
    <meta charset="UTF-8">
    <meta name="viewport" content="width=device-width, initial-scale=1.0">
    <meta http-equiv="X-UA-Compatible" content="ie=edge">
    <title>Document</title>
    <style>
        * { margin: 0; padding: 0; }  .box { width: 100%; min-height: 360px; background-image: url("http://demo.sc.chinaz.com/Files/DownLoad/webjs1/201801/jiaoben5647/img/1.jpg"); background-repeat: no-repeat; background-size: 30% 30%; background-position: center; }
    </style>
</head>

<body>
    <!-- 背景图 -->
    <div id='box' class="box"></div>

    <!-- 重置按钮 -->
    <button id="btn">重置</button>

    <script src="https://cdn.bootcss.com/jquery/3.4.0/jquery.min.js"></script>
    <script>
        // 判断鼠标或手者指点击的次数,从而清除画布
        var totalNum = 0;

        // 画布信息
        var myCanvasObject;
        var ctx;

        // 画布的宽高,根据背景图来获取
        var width = document.getElementById("box").offsetWidth - 2;
        var height = document.getElementById("box").offsetHeight;

        // 鼠标或者手指事件的相关参数
        var isDown = false; //鼠标是否按下标志
        var pointerArr = []; //鼠标移动坐标数组
        var xPointer = 0;//鼠标当前x坐标
        var yPointer = 0;//鼠标当前y坐标

        // 进入页面先初始化刮刮卡
        init();

        // 清空画布
        $("#btn").on("click", function () {
            if (myCanvasObject) {
                //绘制黑色矩形 ,将蒙层充满 
                initBlackRect();
            } else {
                init();
            }
        })

        // 绘制画布的黑色矩形
        function initBlackRect() {
            ctx.beginPath();
            ctx.fillStyle = "#939393";
            ctx.rect(0, 0, width, height);
            ctx.closePath();
            ctx.fill();
        }

        // 清空画布,重置点击次数
        function clearRect() {
            ctx.clearRect(0, 0, width, height);
            totalNum = 0;
        }

        // 初始化画布的方法
        function init() {

            if (!document.getElementById("myCanvas")) {
                var canvas = document.createElement("canvas");
                canvas.setAttribute("width", width + "px");
                canvas.setAttribute("height", height + "px");
                canvas.setAttribute("style", "border:1px solid green");
                canvas.id = "myCanvas";
                document.getElementById("box").appendChild(canvas);
            }

            myCanvasObject = document.getElementById("myCanvas");
            ctx = myCanvasObject.getContext("2d");

            //绘制黑色矩形    
            initBlackRect();

            //pc,移动事件兼容写法
            var hastouch = "ontouchstart" in window ? true : false,
                tapstart = hastouch ? "touchstart" : "mousedown",
                tapmove = hastouch ? "touchmove" : "mousemove",
                tapend = hastouch ? "touchend" : "mouseup";

            //鼠标按下
            myCanvasObject.addEventListener(tapstart, function (event) {
                var e = window.event || event;//2017-12-06
                this.style.cursor = "move";
                isDown = true;
                xPointer = hastouch ? e.targetTouches[0].pageX : e.clientX - this.offsetLeft;
                yPointer = hastouch ? e.targetTouches[0].pageY : e.clientY - this.offsetTop;;
                pointerArr.push([xPointer, yPointer]);
                circleReset(ctx);
            });

            //鼠标按下后拖动
            myCanvasObject.addEventListener(tapmove, function (event) {
                var e = window.event || event;//2017-12-06
                if (isDown) {
                    xPointer = hastouch ? e.targetTouches[0].pageX : e.clientX - this.offsetLeft;;
                    yPointer = hastouch ? e.targetTouches[0].pageY : e.clientY - this.offsetTop;;
                    pointerArr.push([xPointer, yPointer]);
                    circleReset(ctx);
                }
            });

            //鼠标抬起取消事件
            myCanvasObject.addEventListener(tapend, function (event) {
                totalNum += 1;
                isDown = false;
                pointerArr = [];

                // 这里可以设置一个点击的最大次数
                // 当到达最大点击次数时,清除画布
                // 画布清空,只需要重置一下宽高即可
                // 这里是通过手指点击的次数来计算的,但你知道手指点击时绘制的面积,然后就可计算点击几次能达到你所需的面积大小
                // 虽然这种统计的面积方法不太准确,但在接受范围内
                if (totalNum == 2) {
                    clearRect()
                }
            });

            //圆形橡皮檫
            function circleReset(ctx) {
                ctx.save();
                ctx.beginPath();
                ctx.moveTo(pointerArr[0][0], pointerArr[0][1]);
                ctx.lineCap = "round";   //设置线条两端为圆弧
                ctx.lineJoin = "round";   //设置线条转折为圆弧
                ctx.lineWidth = 60;
                ctx.globalCompositeOperation = "destination-out";
                if (pointerArr.length == 1) {
                    ctx.lineTo(pointerArr[0][0] + 1, pointerArr[0][1] + 1);
                } else {
                    for (var i = 1; i < pointerArr.length; i++) {
                        ctx.lineTo(pointerArr[i][0], pointerArr[i][1]);
                        ctx.moveTo(pointerArr[i][0], pointerArr[i][1]);
                    }
                }
                ctx.closePath();
                ctx.stroke();
                ctx.restore();
            }
        }
    </script>
</body>
</html>

6.2.转自地址:https://www.jianshu.com/p/3fef68afd139

7.另一种实现方式  (强烈推荐)

7.1通过 cavans的getImageData 以像素点的个数来计算,挂过的面积来衡量 

7.2代码部分

<!DOCTYPE html>
<html lang="en">

	<head>
		<meta charset="UTF-8">
		<meta name="viewport" content="width=device-width, initial-scale=1.0">
		<meta http-equiv="X-UA-Compatible" content="ie=edge">
		<title>Document</title>
		<style>
			* {
				margin: 0;
				padding: 0;
			}
			
			.box {
				width: 100%;
				min-height: 360px;
				background-image: url("http://demo.sc.chinaz.com/Files/DownLoad/webjs1/201801/jiaoben5647/img/1.jpg");
				background-repeat: no-repeat;
				background-size: 30% 30%;
				background-position: center;
			}
		</style>
	</head>

	<body>
		<!-- 背景图 -->
		<div id='box' class="box"></div>

		<!-- 重置按钮 -->
		<button id="btn">重置</button>

		<script src="https://cdn.bootcss.com/jquery/3.4.0/jquery.min.js"></script>
		<script>
			// 画布信息
			var myCanvasObject;
			var ctx;

			// 画布的宽高,根据背景图来获取
			var width = document.getElementById("box").offsetWidth - 2;
			var height = document.getElementById("box").offsetHeight;

			// 鼠标或者手指事件的相关参数
			var isDown = false; //鼠标是否按下标志
			var pointerArr = []; //鼠标移动坐标数组
			var xPointer = 0; //鼠标当前x坐标
			var yPointer = 0; //鼠标当前y坐标

			// 进入页面先初始化刮刮卡
			init();

			// 清空画布
			$("#btn").on("click", function() {
				if(myCanvasObject) {
					//绘制黑色矩形 ,将蒙层充满 
					initBlackRect();
				} else {
					init();
				}
			})

			// 绘制画布的黑色矩形
			function initBlackRect() {
				ctx.beginPath();
				ctx.fillStyle = "#939393";
				ctx.rect(0, 0, width, height);
				ctx.closePath();
				ctx.fill();
			}

			// 清空画布,重置点击次数
			function clearRect() {
				ctx.clearRect(0, 0, width, height);
			}

			// 初始化画布的方法
			function init() {

				if(!document.getElementById("myCanvas")) {
					var canvas = document.createElement("canvas");
					canvas.setAttribute("width", width + "px");
					canvas.setAttribute("height", height + "px");
					canvas.setAttribute("style", "border:1px solid green");
					canvas.id = "myCanvas";
					document.getElementById("box").appendChild(canvas);
				}

				myCanvasObject = document.getElementById("myCanvas");
				ctx = myCanvasObject.getContext("2d");

				//绘制黑色矩形    
				initBlackRect();

				//pc,移动事件兼容写法
				var hastouch = "ontouchstart" in window ? true : false,
					tapstart = hastouch ? "touchstart" : "mousedown",
					tapmove = hastouch ? "touchmove" : "mousemove",
					tapend = hastouch ? "touchend" : "mouseup";

				//鼠标按下
				myCanvasObject.addEventListener(tapstart, function(event) {
					var e = window.event || event; //2017-12-06
					this.style.cursor = "move";
					isDown = true;
					xPointer = hastouch ? e.targetTouches[0].pageX : e.clientX - this.offsetLeft;
					yPointer = hastouch ? e.targetTouches[0].pageY : e.clientY - this.offsetTop;;
					pointerArr.push([xPointer, yPointer]);
					circleReset(ctx);
				});

				//鼠标按下后拖动
				myCanvasObject.addEventListener(tapmove, function(event) {
					var e = window.event || event; //2017-12-06
					if(isDown) {
						xPointer = hastouch ? e.targetTouches[0].pageX : e.clientX - this.offsetLeft;;
						yPointer = hastouch ? e.targetTouches[0].pageY : e.clientY - this.offsetTop;;
						pointerArr.push([xPointer, yPointer]);
						circleReset(ctx);
						
						// 在这里是通过像素点来计算,挂过的面积方式来的,比较好
						handleFilledPercentage(getFilledPercentage());
					}
				});

				//鼠标抬起取消事件
				myCanvasObject.addEventListener(tapend, function(event) {
					isDown = false;
					pointerArr = [];
				});

				//圆形橡皮檫
				function circleReset(ctx) {
					ctx.save();
					ctx.beginPath();
					ctx.moveTo(pointerArr[0][0], pointerArr[0][1]);
					ctx.lineCap = "round";   //设置线条两端为圆弧
					ctx.lineJoin = "round";   //设置线条转折为圆弧
					ctx.lineWidth = 60;
					ctx.globalCompositeOperation = "destination-out";
					if(pointerArr.length == 1) {
						ctx.lineTo(pointerArr[0][0] + 1, pointerArr[0][1] + 1);
					} else {
						for(var i = 1; i < pointerArr.length; i++) {
							ctx.lineTo(pointerArr[i][0], pointerArr[i][1]);
							ctx.moveTo(pointerArr[i][0], pointerArr[i][1]);
						}
					}
					ctx.closePath();
					ctx.stroke();
					ctx.restore();
				}

				// 计算已经刮过的区域占整个区域的百分比
				function getFilledPercentage() {
					let imgData = ctx.getImageData(0, 0, width, height);
					// imgData.data是个数组,存储着指定区域每个像素点的信息,数组中4个元素表示一个像素点的rgba值
					let pixels = imgData.data;
					let transPixels = [];
					for(let i = 0; i < pixels.length; i += 4) {
						// 严格上来说,判断像素点是否透明需要判断该像素点的a值是否等于0,
						// 为了提高计算效率,这儿设置当a值小于128,也就是半透明状态时就可以了
						if(pixels[i + 3] < 128) {
							transPixels.push(pixels[i + 3]);
						}
					}
					return(transPixels.length / (pixels.length / 4) * 100).toFixed(2) + '%'
				}
				// 设置阈值,去除灰色涂层
				function handleFilledPercentage(percentage) {
					percentage = percentage || 0;
					if(parseInt(percentage) > 50) {
						// 当像素点的个数超过  50% 时,清空画布,显示底图
						ctx.clearRect(0, 0, width, height);
					}
				}
			}
		</script>
	</body>

</html>

8.也可通过这个

8.1 TumoH.min.js

var TumoH = function(e) { function t(e) { for(var t = e.offsetLeft, o = e.offsetTop; e = e.offsetParent;) t += e.offsetLeft, o += e.offsetTop; return { oLeft: t, oTop: o } }  function o() { var e = "ontouchstart" in window ? !0 : !1, t = e ? "touchstart" : "mousedown", o = e ? "touchmove" : "mousemove", f = e ? "touchend" : "mouseup"; d.lineCap = "round", d.lineJoin = "round", d.lineWidth = 2 * u, d.globalCompositeOperation = "destination-out", r.addEventListener(t, function(t) { function u(t) { clearTimeout(i), t.preventDefault(), x2 = e ? t.targetTouches[0].pageX - m : t.clientX - r.offsetLeft, y2 = e ? t.targetTouches[0].pageY - p : t.clientY - r.offsetTop, d.save(), d.moveTo(n, a), d.lineTo(x2, y2), d.stroke(), d.restore(), n = x2, a = y2 } clearTimeout(i), t.preventDefault(), n = e ? t.targetTouches[0].pageX - m : t.clientX - r.offsetLeft, a = e ? t.targetTouches[0].pageY - p : t.clientY - r.offsetTop, d.save(), d.beginPath(), d.arc(n, a, 1, 0, 2 * Math.PI), d.fill(), d.restore(), r.addEventListener(o, u), r.addEventListener(f, function() { r.removeEventListener(o, u), i = setTimeout(function() { for(var e = d.getImageData(0, 0, r.width, r.height), t = 0, o = 0; o < e.width; o += l) for(var n = 0; n < e.height; n += l) { var a = 4 * (n * e.width + o); e.data[a + 3] > 0 && t++ } t / (e.width * e.height / (l * l)) < c && T && (s(), T = !1) }, g) }) }) } var n, a, i, r = document.getElementById(e.cavId), f = e.ImgSRC, s = e.callBack, e = e || {}, u = e.oR || 20, c = 1 - e.maxPro || .8, h = r.parentNode, d = r.getContext("2d"), g = 100, l = 30; r.width = h.clientWidth, r.height = h.clientHeight; var v = new Image; v.src = f, v.onload = function() { d.drawImage(v, 0, 0, r.width, r.height), o() }; var m = t(r).oLeft, p = t(r).oTop, T = !0 };

8.2 html 部分

<!DOCTYPE html>
<html>

	<head>
		<meta charset="UTF-8">
		<title></title>
		<style type="text/css">
			*{ margin: 0; padding: 0; }  .rowCenter { display: flex; display: -webkit-flex; display: -moz-flex; -webkit-box-pack: center; justify-content: center; -webkit-justify-content: center; -moz-justify-content: center; -ms-justify-content: center; -o-justify-content: center; }  .rowAndColCenter { display: flex; display: -webkit-flex; display: -moz-flex; flex-direction: column; -webkit-flex-direction: column; -moz-flex-direction: column; justify-content: center; -webkit-justify-content: center; -moz-justify-content: center; align-items: center; -webkit-align-items: center; -moz-align-items: center; }  .indexB{ background-color: rgba(0,0,0,0.5); padding-bottom: 20px; }  .indexB-guabgOut{ width: 574px; height: 252px; margin-top: 100px; position: relative; background-color: #fff; }  .indexB-guabgStart { width: 537px; height: 220px; position: absolute; left: 18px; top: 16px; }  .indexB-guabgEnd { width: 537px; height: 220px; line-height: 220px; text-align: center; color: #ff722c; font-weight: bold; font-size: 50px; box-shadow: 0px 0px 14px 3px rgba(0,0,0,0.1); -webkit-box-shadow: 0px 0px 14px 3px rgba(0,0,0,0.1); -moz-box-shadow: 0px 0px 14px 3px rgba(0,0,0,0.1); background-repeat: no-repeat; background-position: center; background-size: 124px 124px; background-image: url(''); } .chongzhi{ width: 200px; height: 80px; line-height: 80px; text-align: center; background-color: orange; color: #fff; font-size: 20px; }
		</style>
		<script src="https://cdn.bootcss.com/jquery/3.4.1/jquery.min.js" type="text/javascript" charset="utf-8"></script>
	</head>

	<body>
		<div class="indexB rowCenter" style="">				
			<!-- 刮刮卡背景图 -->
			<div class="indexB-guabgOut rowAndColCenter" style="">				
				<!-- 未刮奖 -->
				<div class="indexB-guabgStart bgimgeAbsCenter100">
					<canvas class="canvas" id="myCanvas">sorry, 浏览器不支持canvas</canvas>
				</div>
				<!-- 刮奖结果 -->
				<div class="indexB-guabgEnd line1" style=""></div>
			</div>
		</div>
		<div class="chongzhi">重置</div>
		<script src="TumoH.min.js" type="text/javascript" charset="utf-8"></script>
		<script type="text/javascript">
			// 刮刮卡,未刮状态的图片
			var ImgSRC = '';
			
			// loading图片
			var loadingbg = '';
			
			// 白底
			var guabgEndbg_ff = '';
			
			canvasStart();
			function canvasStart(){
				TumoH({
			  		cavId:'myCanvas',		//canvas的id(string)
			 		oR:40,					//涂抹半径大小(number)
			 		ImgSRC:ImgSRC,			//涂抹的图片地址src(string)
			 		callBack:function(){	//涂抹达到比例后执行的函数(function)
			 			finish_tumo();
			 		},				
			 		maxPro:0.1				//涂抹比例(number)
				})
			}
			
			//涂抹
			function finish_tumo(){
	 			setTimeout(function(){
	 				$("#myCanvas").hide();
 					setTimeout(function(){
 						$(".indexB-guabgEnd").css({'background-image':'url('+guabgEndbg_ff+')','background-size': '100% 100%'});
 						$(".indexB-guabgEnd").text("恭喜获得大奖");
 					},100)
	 			},100);			
			}
			
			$(".chongzhi").on("click",function(){
				$("#myCanvas").show();
				canvasStart();
				setTimeout(function(){
					setTimeout(function(){
	 					$(".indexB-guabgEnd").css({'background-image':'url('+loadingbg+')','background-size': '124px 124px'});
	 					$(".indexB-guabgEnd").text("");
	   				},100);
				},100);
			});
		</script>
	</body>

</html>

8.3 效果

  

8.4 说明

在H5活动中尽量使用flex少使用 position: absolute 来;

9.也可以

9.1html

(1)Lottery.js

function Lottery(id, cover, coverType, width, height, drawPercentCallback) {
    this.conId = id;
    this.conNode = document.getElementById(this.conId);
    this.cover = cover;
    this.coverType = coverType;
    this.background = null;
    this.backCtx = null;
    this.mask = null;
    this.maskCtx = null;
    this.lottery = null;
    this.lotteryType = 'image';
    this.width = width || 300;
    this.height = height || 100;
    this.clientRect = null;
    this.drawPercentCallback = drawPercentCallback;
}

Lottery.prototype = {
    createElement: function (tagName, attributes) {
        var ele = document.createElement(tagName);
        for (var key in attributes) {
            ele.setAttribute(key, attributes[key]);
        }
        return ele;
    },
    getTransparentPercent: function(ctx, width, height) {
        var imgData = ctx.getImageData(0, 0, width, height),
            pixles = imgData.data,
            transPixs = [];
        for (var i = 0, j = pixles.length; i < j; i += 4) {
            var a = pixles[i + 3];
            if (a < 128) {
                transPixs.push(i);
            }
        }
        var percent = (transPixs.length / (pixles.length / 4) * 100).toFixed(2);
        
        if(parseInt(percent)>50){
        	ctx.clearRect(0, 0, width, height);
        }
        
        return percent;
    },
    resizeCanvas: function (canvas, width, height) {
        canvas.width = width;
        canvas.height = height;
        canvas.getContext('2d').clearRect(0, 0, width, height);
    },
    drawPoint: function (x, y) {
        this.maskCtx.beginPath();
        var radgrad = this.maskCtx.createRadialGradient(x, y, 0, x, y, 30);
        radgrad.addColorStop(0, 'rgba(0,0,0,0.6)');
        radgrad.addColorStop(1, 'rgba(255, 255, 255, 0)');
        this.maskCtx.fillStyle = radgrad;
        this.maskCtx.arc(x, y, 30, 0, Math.PI * 2, true);
        this.maskCtx.fill();
        if (this.drawPercentCallback) {
            this.drawPercentCallback.call(null, this.getTransparentPercent(this.maskCtx, this.width, this.height));
        }
    },
    bindEvent: function () {
        var _this = this;
        var device = (/android|webos|iphone|ipad|ipod|blackberry|iemobile|opera mini/i.test(navigator.userAgent.toLowerCase()));
        var clickEvtName = device ? 'touchstart' : 'mousedown';
        var moveEvtName = device? 'touchmove': 'mousemove';
        if (!device) {
            var isMouseDown = false;
            document.addEventListener('mouseup', function(e) {
                isMouseDown = false;
            },  { passive: false });
        } else {
            document.addEventListener("touchmove", function(e) {
                if (isMouseDown) {
                    e.preventDefault();
                }
            },  { passive: false });
            document.addEventListener('touchend', function(e) {
                isMouseDown = false;
            },  { passive: false });
        }
        this.mask.addEventListener(clickEvtName, function (e) {
            isMouseDown = true;
            var docEle = document.documentElement;
            if (!_this.clientRect) {
                _this.clientRect = {
                    left: 0,
                    top:0
                };
            }
            var x = (device ? e.touches[0].clientX : e.clientX) - _this.clientRect.left + docEle.scrollLeft - docEle.clientLeft;
            var y = (device ? e.touches[0].clientY : e.clientY) - _this.clientRect.top + docEle.scrollTop - docEle.clientTop;
            _this.drawPoint(x, y);
        }, false);

        this.mask.addEventListener(moveEvtName, function (e) {
            if (!device && !isMouseDown) {
                return false;
            }
            var docEle = document.documentElement;
            if (!_this.clientRect) {
                _this.clientRect = {
                    left: 0,
                    top:0
                };
            }
            var x = (device ? e.touches[0].clientX : e.clientX) - _this.clientRect.left + docEle.scrollLeft - docEle.clientLeft;
            var y = (device ? e.touches[0].clientY : e.clientY) - _this.clientRect.top + docEle.scrollTop - docEle.clientTop;
            _this.drawPoint(x, y);
        }, false);
    },
    drawLottery: function () {
        this.background = this.background || this.createElement('canvas', {
            style: 'position:absolute;left:0;top:0;'
        });
        this.mask = this.mask || this.createElement('canvas', {
            style: 'position:absolute;left:0;top:0;'
        });

        if (!this.conNode.innerHTML.replace(/[\w\W]| /g, '')) {
            this.conNode.appendChild(this.background);
            this.conNode.appendChild(this.mask);
            this.clientRect = this.conNode ? this.conNode.getBoundingClientRect() : null;
            this.bindEvent();
        }

        this.backCtx = this.backCtx || this.background.getContext('2d');
        this.maskCtx = this.maskCtx || this.mask.getContext('2d');

        if (this.lotteryType == 'image') {
            var image = new Image(),
                _this = this;
            image.onload = function () {
                _this.resizeCanvas(_this.background, _this.width, _this.height);
                _this.backCtx.drawImage(this, 0, 0,_this.width, _this.height);
                _this.drawMask();
            }
            image.src = this.lottery;
        } else if (this.lotteryType == 'text') {
            this.width = this.width;
            this.height = this.height;
            this.resizeCanvas(this.background, this.width, this.height);
            this.backCtx.save();
            this.backCtx.fillStyle = '#FFF';
            this.backCtx.fillRect(0, 0, this.width, this.height);
            this.backCtx.restore();
            this.backCtx.save();
            var fontSize = 30;
            this.backCtx.font = 'Bold ' + fontSize + 'px Arial';
            this.backCtx.textAlign = 'center';
            this.backCtx.fillStyle = '#F60';
            this.backCtx.fillText(this.lottery, this.width / 2, this.height / 2 + fontSize / 2);
            this.backCtx.restore();
            this.drawMask();
        }
    },
    drawMask: function() {
        this.resizeCanvas(this.mask, this.width, this.height);
        if (this.coverType == 'color') {
            this.maskCtx.fillStyle = this.cover;
            this.maskCtx.fillRect(0, 0, this.width, this.height);
            this.maskCtx.globalCompositeOperation = 'destination-out';
        } else if (this.coverType == 'image'){
            var image = new Image(),
                _this = this;
            image.onload = function () {
                _this.maskCtx.drawImage(this, 0, 0);
                _this.maskCtx.globalCompositeOperation = 'destination-out';
            }
            image.src = this.cover;
        }
    },
    init: function (lottery, lotteryType) {
        this.lottery = lottery;
        this.lotteryType = lotteryType || 'image';
        this.drawLottery();
    }
}

(2)html

<!DOCTYPE html>
<html>

	<head>
		<meta charset="utf-8" />
		<meta name="viewport" content="width=device-width, initial-scale=1.0, user-scalable=no" />
		<title>Lottery Demo</title>
		<style type="text/css"> *{ margin: 0; padding: 0; } body { height: 1000px; }  #lotteryContainer { position: relative; width: 300px; height: 100px; position: absolute; left: 50%; margin-left: -150px; top: 200px; }  #drawPercent { color: #F60; } </style>
	</head>

	<body>
		<button id="freshBtn">刷新彩票</button><label>已刮开 <span id="drawPercent">0%</span> 区域。</label>
		<div id="lotteryContainer"></div>
		<script src="Lottery.js"></script>
		<script>
			window.onload = function() {
				var lottery = new Lottery('lotteryContainer', '#CCC', 'color', 200, 200, drawPercent);
				lottery.init('http://www.baidu.com/img/bdlogo.gif', 'image');

				document.getElementById('freshBtn').onclick = function() {
					drawPercentNode.innerHTML = '0%';
					lottery.init(getRandomStr(10), 'text');
				}

				var drawPercentNode = document.getElementById('drawPercent');

				function drawPercent(percent) {
					drawPercentNode.innerHTML = percent + '%';
				}
			}

			function getRandomStr(len) {
				var text = "";
				var possible = "ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789";
				for(var i = 0; i < len; i++)
					text += possible.charAt(Math.floor(Math.random() * possible.length));
				return text;
			}
		</script>
	</body>
</html>

9.2效果

10.说明:

(1)画布清空:https://blog.csdn.net/inuyasha1121/article/details/53925538

(2)面积的处理问题

    ①(参考问题回答区域):https://bbs.csdn.net/topics/390998715

    ②也可这样处理:https://ask.csdn.net/questions/352384

(3)推荐:

    ①这个也比较好:https://blog.csdn.net/XuM222222/article/details/82353729

    ②推荐地址:https://github.com/xm2by/fragment/blob/master/canvas%E5%AE%9E%E8%B7%B5/%E5%88%AE%E5%88%AE%E5%8D%A1.html

    ③ 推荐:

https://blog.csdn.net/zhouziyu2011/article/details/67640749

(4)转载地址

①  https://github.com/artwl/Lottery

②  https://www.cnblogs.com/jscode/p/3580878.html

 

已标记关键词 清除标记
相关推荐
©️2020 CSDN 皮肤主题: 技术黑板 设计师:CSDN官方博客 返回首页