123 lines
		
	
	
		
			3.2 KiB
		
	
	
	
		
			Vue
		
	
	
		
		
			
		
	
	
			123 lines
		
	
	
		
			3.2 KiB
		
	
	
	
		
			Vue
		
	
	
|  | <template> | ||
|  |   <div class="s-canvas"> | ||
|  |     <canvas id="s-canvas" :width="contentWidth" :height="contentHeight" /> | ||
|  |   </div> | ||
|  | </template> | ||
|  | 
 | ||
|  | <script setup> | ||
|  | import { onMounted, watch } from "vue"; | ||
|  | 
 | ||
|  | defineOptions({ | ||
|  |   name: "AppVerificationCode", | ||
|  | }); | ||
|  | const props = defineProps({ | ||
|  |   identifyCode: { type: String, default: "1234" }, | ||
|  |   fontSizeMin: { type: Number, default: 25 }, | ||
|  |   fontSizeMax: { type: Number, default: 30 }, | ||
|  |   backgroundColorMin: { type: Number, default: 255 }, | ||
|  |   backgroundColorMax: { type: Number, default: 255 }, | ||
|  |   colorMin: { type: Number, default: 0 }, | ||
|  |   colorMax: { type: Number, default: 160 }, | ||
|  |   lineColorMin: { type: Number, default: 100 }, | ||
|  |   lineColorMax: { type: Number, default: 255 }, | ||
|  |   dotColorMin: { type: Number, default: 0 }, | ||
|  |   dotColorMax: { type: Number, default: 255 }, | ||
|  |   contentWidth: { type: Number, default: 112 }, | ||
|  |   contentHeight: { type: Number, default: 31 }, | ||
|  | }); | ||
|  | // 生成一个随机数
 | ||
|  | const randomNum = (min, max) => { | ||
|  |   return Math.floor(Math.random() * (max - min) + min); | ||
|  | }; | ||
|  | // 生成一个随机的颜色
 | ||
|  | const randomColor = (min, max) => { | ||
|  |   const r = randomNum(min, max); | ||
|  |   const g = randomNum(min, max); | ||
|  |   const b = randomNum(min, max); | ||
|  |   return "rgb(" + r + "," + g + "," + b + ")"; | ||
|  | }; | ||
|  | const drawPic = () => { | ||
|  |   const canvas = document.getElementById("s-canvas"); | ||
|  |   const ctx = canvas.getContext("2d"); | ||
|  |   ctx.textBaseline = "bottom"; | ||
|  |   // 绘制背景
 | ||
|  |   ctx.fillStyle = randomColor( | ||
|  |     props.backgroundColorMin, | ||
|  |     props.backgroundColorMax | ||
|  |   ); | ||
|  |   ctx.fillRect(0, 0, props.contentWidth, props.contentHeight); | ||
|  |   // 绘制文字
 | ||
|  |   for (let i = 0; i < props.identifyCode.length; i++) { | ||
|  |     drawText(ctx, props.identifyCode[i], i); | ||
|  |   } | ||
|  |   drawLine(ctx); | ||
|  |   drawDot(ctx); | ||
|  | }; | ||
|  | const drawText = (ctx, txt, i) => { | ||
|  |   ctx.fillStyle = randomColor(props.colorMin, props.colorMax); | ||
|  |   ctx.font = randomNum(props.fontSizeMin, props.fontSizeMax) + "px SimHei"; | ||
|  |   const x = (i + 1) * (props.contentWidth / (props.identifyCode.length + 1)); | ||
|  |   const y = randomNum(props.fontSizeMax, props.contentHeight - 5); | ||
|  |   const deg = randomNum(-45, 45); | ||
|  |   // 修改坐标原点和旋转角度
 | ||
|  |   ctx.translate(x, y); | ||
|  |   ctx.rotate((deg * Math.PI) / 180); | ||
|  |   ctx.fillText(txt, 0, 0); | ||
|  |   // 恢复坐标原点和旋转角度
 | ||
|  |   ctx.rotate((-deg * Math.PI) / 180); | ||
|  |   ctx.translate(-x, -y); | ||
|  | }; | ||
|  | const drawLine = (ctx) => { | ||
|  |   // 绘制干扰线
 | ||
|  |   for (let i = 0; i < 5; i++) { | ||
|  |     ctx.strokeStyle = randomColor(props.lineColorMin, props.lineColorMax); | ||
|  |     ctx.beginPath(); | ||
|  |     ctx.moveTo( | ||
|  |       randomNum(0, props.contentWidth), | ||
|  |       randomNum(0, props.contentHeight) | ||
|  |     ); | ||
|  |     ctx.lineTo( | ||
|  |       randomNum(0, props.contentWidth), | ||
|  |       randomNum(0, props.contentHeight) | ||
|  |     ); | ||
|  |     ctx.stroke(); | ||
|  |   } | ||
|  | }; | ||
|  | const drawDot = (ctx) => { | ||
|  |   // 绘制干扰点
 | ||
|  |   for (let i = 0; i < 80; i++) { | ||
|  |     ctx.fillStyle = randomColor(0, 255); | ||
|  |     ctx.beginPath(); | ||
|  |     ctx.arc( | ||
|  |       randomNum(0, props.contentWidth), | ||
|  |       randomNum(0, props.contentHeight), | ||
|  |       1, | ||
|  |       0, | ||
|  |       2 * Math.PI | ||
|  |     ); | ||
|  |     ctx.fill(); | ||
|  |   } | ||
|  | }; | ||
|  | watch( | ||
|  |   () => props.identifyCode, | ||
|  |   () => { | ||
|  |     drawPic(); | ||
|  |   } | ||
|  | ); | ||
|  | onMounted(() => { | ||
|  |   drawPic(); | ||
|  | }); | ||
|  | </script> | ||
|  | 
 | ||
|  | <style scoped lang="scss"> | ||
|  | .s-canvas { | ||
|  |   height: 38px; | ||
|  |   padding-top: 3px; | ||
|  | 
 | ||
|  |   canvas { | ||
|  |     margin-top: 1px; | ||
|  |     margin-right: 10px; | ||
|  |   } | ||
|  | } | ||
|  | </style> |