cavas的驗證碼效果demo

最近做了cavas的驗證碼效果,如下圖:
在這裏插入圖片描述
圈的形狀和顏色隨機,可設計需要選中文字的數量。核心代碼如下:

//畫空心圓
			class setAnimationArc{
				constructor() {}
				
				#step=1;
				#startAngle=0;
				#endAngle=0;
				#add = Math.PI * 2 / 100;
				#counterClockwise = false;
				
				drawShadow(){
					this.ctx.shadowOffsetX = 0; // 設置水平位移
					this.ctx.shadowOffsetY = 0; // 設置垂直位移
					this.ctx.shadowBlur = 10; // 設置模糊度
				}
				
				clearShadow(){
					this.ctx.shadowOffsetX = 0; // 設置水平位移
					this.ctx.shadowOffsetY = 0; // 設置垂直位移
					this.ctx.shadowBlur = 0; // 設置模糊度
				}
				
				#x=0;
				#y=0;
				#n=100;
				#varName=null;
				
				actiondo(fn) {
					this.#step = 1;
					this.#startAngle = 0;
					this.ctx.strokeStyle = '#' + ('00000' + (Math.random() * 0x1000000 << 0).toString(16)).slice(-6); //圓圈顏色                
					this.ctx.shadowColor = "#ff8b7c"||'#' + ('00000' + (Math.random() * 0x1000000 << 0).toString(16)).slice(-6); // 設置陰影顏色
					this.ctx.fillStyle = '#' + ('00000' + (Math.random() * 0x1000000 << 0).toString(16)).slice(-6);
					//圓心位置
					this.#x = this.mouseX ;
					this.#y = this.mouseY;
					clearInterval(this.#varName);
					this.#varName = setInterval(()=>{
						this.animation(fn);
					}, this.animation_interval);
				}
				
				animation (fn) {
					if (this.#step <= this.#n) {
						this.#endAngle = this.#startAngle + this.#add;
						this.drawArc(this.#startAngle, this.#endAngle);
						this.#startAngle = this.#endAngle;
						this.#step++;
					} else {
						fn&&fn();
						clearInterval(this.#varName);
					}
				}
				
				drawArc(s, e) {
					this.ctx.beginPath();
					this.ctx.arc(this.#x, this.#y, this.radius, s, e, this.#counterClockwise);
					this.ctx.stroke();
					this.ctx.closePath();
					this.ctx.fillStyle="gray";
					this.ctx.fill();  
				}
				
				toDrawArc(){
					return new Promise((r,c)=>{
						this.ctx.lineWidth = 5.0;
						this.drawShadow();
						this.actiondo(()=>{
							r();
						});
					// this.clearShadow();
					});
				}
			}
			
			//按照圖片畫圓
			class setAnimationArcImage{
				constructor() {
					this.getImages().then(()=>{
						this.#isLoad=true;
					});
				}
				
				#img1=""
				#img2=""
				#img3=""
				
				circleImg(ctx, img, x, y, r) {
					x=x-r;
					y=y-r;
				  ctx.save();
				  var d =2 * r;
				  var cx = x + r;
				  var cy = y + r;
				  this.ctx.arc(cx, cy, r, 0, 2 * Math.PI);
				  this.ctx.clip();
				  this.ctx.drawImage(img, x, y, d, d);
				  this.ctx.restore();
				}
				
				sector (x, y, radius, sDeg, eDeg) {
				    // 初始保存
				    this.ctx.save();
				    // 位移到目標點
				    this.ctx.translate(x, y);
				    this.ctx.beginPath();
				    // 畫出圓弧
				    this.ctx.arc(0,0,radius,sDeg, eDeg);
				    // 再次保存以備旋轉
				    this.ctx.save();
				    // 旋轉至起始角度
				    this.ctx.rotate(eDeg);
				    // 移動到終點,準備連接終點與圓心
				    this.ctx.moveTo(radius,0);
				    // 連接到圓心
				    this.ctx.lineTo(0,0);
				    // 還原
				    this.ctx.restore();
				    // 旋轉至起點角度
				    this.ctx.rotate(sDeg);
				    // 從圓心連接到起點
				    this.ctx.lineTo(radius,0);
				    this.ctx.closePath();
				    // 還原到最初保存的狀態
				    this.ctx.restore();
				}
				#degrees=-50;
				startDraw(fn){
					this.#degrees+=8;
					this.sector(this.mouseX,this.mouseY,this.radius,-0.5*Math.PI,Math.PI*(this.#degrees/100))
					this.ctx.fill();
					if(this.#degrees<=150){
					    setTimeout(()=>{
							this.startDraw(fn);
						},this.animation_interval);
					}else{
						fn&&fn()
					}
				}
				
				#img1El = new Image();
				#img2El = new Image();
				#img3El = new Image();
				getImages(){
				let $this=this;
					return new Promise(async(r,c)=>{
						try{
						await this.loadImg(this.#img1El,this.#img1);
						await this.loadImg(this.#img2El,this.#img2);
						await this.loadImg(this.#img3El,this.#img3);
						r()
						}
						catch(e){
							this.getImages();
							c(e);
						}
					})
					
				}
				
				loadImg(imgs,src){
					imgs.src=src;
					return new Promise((r,c)=>{
						imgs.onload=function(){
							r(this);
						}
						imgs.onerror=function(){
							c(this)
						}
					});
				}
				
				getColorImg(img,width,height,color) {
					let canvas=this.toSetImg(img,width,height);
					let ctx=canvas.getContext('2d');
					let imageD = ctx.getImageData(0, 0, canvas.width, canvas.height);
					var pdata = imageD.data;
					var colorArr = color;
					for (var j = 0; j < pdata.length; j += 4) {
						pdata[j] = colorArr[0];
						pdata[j + 1] = colorArr[1];
						pdata[j + 2] = colorArr[2];
					}
					ctx.putImageData(imageD, 0, 0);
					return canvas
				}
				
				toSetImg(img,width,height){
					   var canvas=document.createElement("canvas");
					   var ctx=canvas.getContext('2d');
					   canvas.width=this.canvas.width;
					   canvas.height=this.canvas.height;
					   ctx.drawImage(img,this.mouseX-this.radius,this.mouseY-this.radius,width,height);
					   return canvas
				}
				
				toDraw(fn){
					let imgMap=[this.#img1El,this.#img2El,this.#img3El];
					console.log(this.themeColor);
					var pattern = this.ctx.createPattern(this.getColorImg(imgMap[Math.floor(Math.random() * imgMap.length + 1)-1],this.radius*2,this.radius*2,this.getColor(this.themeColor)), "no-repeat");
					this.ctx.fillStyle=pattern;
					this.#degrees=-50;
					this.startDraw(fn)
				}
				
				#isLoad=false;
				toDrawArc(){
					this.ctx.save();
					return new Promise((r,v)=>{
						if(this.#isLoad){
							this.toDraw(()=>{
								r()
								this.ctx.restore();
							});
						}else{
							this.getImages().then(()=>{
								this.#isLoad=true;
								this.toDraw(()=>{
									r()
									this.ctx.restore();
								});
							})
						}
					});
				}
			}
			
			
			class textValiCode extends setAnimationArcImage{
				constructor(obj = {}) {
					super()
					if (obj) {
						let {
							href,
							elem,
							step
						} = obj;
						if (href) this.getImg(href);
						if (elem) this.getElem(elem);
						if (step) this.setStep(step);
					}
					
					this.init();
				}

				canvas = null;
				ctx = null;

				copy({form,to}){
					//form [x,y,w,h]
					let imgData=this.ctx.getImageData.apply(null,form);
					//to [x,y]
					ctx.putImageData.apply(null,[imgData,...to]);
				}
	
				setCanvas() {
					let canvas = this.canvas = document.createElement("canvas");
					canvas.className="textValiCodeCanvas";
					this.ctx = canvas.getContext("2d");
					Array.from(this.elem.querySelectorAll('.textValiCodeCanvas')).forEach(v=>{
						v.remove();
					});
					this.elem.appendChild(canvas)
				}

				setImg() {
					let image = new Image();
					image.src = this.href;
					let that = this;
					image.onload = function() {
						let width = that.canvas.width = this.width;
						let hight = that.canvas.height = this.height;
						that.ctx.rect(0, 0, that.canvas.width, that.canvas.height);
						that.ctx.fillStyle = "rgba(0,0,0,0)";
						that.ctx.fill();
						that.ctx.drawImage(image, 0, 0, this.width, this.height);
					}
				}

				href = "";
				elem = ""

				getImg(href) {
					this.href = href;
					return this;
				}
				getElem(elem) {
					this.elem = (typeof elem == "string" ? document.querySelector(elem) : elem);
					return this;
				}

				mouseX = 0;
				mouseY = 0;

				getPont = e => {
					let box = this.canvas.getBoundingClientRect()
					this.mouseX = (e.clientX - box.left) * this.canvas.width / box.width
					this.mouseY = (e.clientY - box.top) * this.canvas.height / box.height
					this.setPont(e, [this.mouseX, this.mouseY]);
				}

				bind() {
					this.canvas.addEventListener('click', this.getPont, false);
					this.canvas.addEventListener('click', this.getPont, false)
				}

				timer = null;

				callBack = () => {}

				call(callBack) {
					this.callBack = callBack;
					return this;
				}

				clickEven = () => {}
				onClick(clickEven) {
					this.clickEven = clickEven;
					return this;
				}
				
				#retMaps=[];
				
				reset(href){
					this.#retMaps=[];
					if(href)this.href=href;
					this.stepIndex = 0;
					return this.init();
				}
				
				config(opt,isReset){
					if(typeof opt == "object"){
						for ( let n in opt){
							if(n in this){
								if(typeof n!=="function"){
									this[n]=opt[n];
								}
							}
						}
					}else if(typeof opt =="function"){
						opt&&opt.call(this);
					}
					if(isReset!==false)return this.reset();
					return this;
				}
				
				fontColor="";
				fontSize="30px";
				themeColor="#000"
				
				setTheme(color="",op){
					if(color.match(/(rgb)|(#)/)){
						this.themeColor=this.getColor(color,op)
					}else{
						console.log(this.randomColor());
						this.themeColor=this.randomColor();
					}
				}
				
				randomColor(){
					return '#' + ('00000' + (Math.random() * 0x1000000 << 0).toString(16)).slice(-6);
				}
				
				// 將rgb顏色轉成hex
				colorRGB2Hex(color) {
					var rgb = color.split(',');
					var r = parseInt(rgb[0].split('(')[1]);
					var g = parseInt(rgb[1]);
					var b = parseInt(rgb[2].split(')')[0]);
				
					var hex = "#" + ((1 << 24) + (r << 16) + (g << 8) + b).toString(16).slice(1);
					return hex;
				}
				// 將hex顏色轉成rgb
				hexToRgba(hex, opacity) {
					const extendHex = shortHex =>
						'#' +
						shortHex
						.slice(shortHex.startsWith('#') ? 1 : 0)
						.split('')
						.map(x => x + x)
						.join('');
					if (hex.length <= 4) hex = extendHex(hex)
					var RGBA = "rgba(" + parseInt("0x" + hex.slice(1, 3)) + "," + parseInt("0x" + hex.slice(3, 5)) + "," + parseInt(
						"0x" + hex.slice(5, 7)) + "," + opacity + ")";
					return {
						red: parseInt("0x" + hex.slice(1, 3)),
						green: parseInt("0x" + hex.slice(3, 5)),
						blue: parseInt("0x" + hex.slice(5, 7)),
						rgba: RGBA
					}
				}
				
				getColor(str, opacity) {
					if (str.match(/\#/g)) {
						let color = this.hexToRgba(str, opacity);
						return [color.red, color.green, color.blue, opacity || 1];
					} else {
						let colorStr = str.match(/(\(.*\))/g);
						if (colorStr) {
							return JSON.parse(colorStr[0].replace(/\(/g, "[").replace(/\)/g, "]"));
						}
					}
				}
				
				async setPont(e, pont) {
					if (this.stepIndex < this.step) {
						await this.toDrawArc();
						this.ctx.save();
						this.ctx.font = `bold ${this.fontSize} Microsoft YaHei`;
						let wi = this.ctx.measureText(this.stepIndex+1).width / 2;
						this.ctx.fillStyle=this.fontColor||this.themeColor;
						this.ctx.fillText(this.stepIndex+1, this.mouseX - wi, this.mouseY + wi);
						this.ctx.restore();
						this.stepIndex++;
						this.clickEven(e, pont,this.stepIndex);
						this.#retMaps.push(pont)
						if(this.stepIndex==this.step){
							this.callBack(e,this.#retMaps,this.stepIndex);
						}
					}
				}

				step = 3;
				stepIndex = 0;
				radius=30;
				animation_interval=10;
				setStep(step) {
					this.step = step;
					return this;
				}

				init() {
					this.setCanvas();
					this.setImg();
					this.bind();
					return this;
				}
			}
			
export default textValiCode
	<div class="textValiCode" ref="con"></div>
	//調用
	<script>
		function randomFrom(lowerValue,upperValue)
		{
		    return Math.floor(Math.random() * (upperValue - lowerValue + 1) + lowerValue);
		}
		var obj = new textValiCode({
			href:"./test.jpg",//背景底圖
			elem: ".textValiCode"//容器
		}).call(function(e,map) {
			setTimeout(()=>{
				if(confirm(`座標依次爲${JSON.stringify(map)},確定要重置嘛?`)){
					this.reset();
				}
			},100)
		}).onClick(function(e, pont,stepIndex) {
			this.config({radius:randomFrom(30,100)},false);
			this.setTheme();
		}).config({step:4});
	</script>
發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章