实现验证码方法之canvas (源码分享)
时间:2018-08-20
本文章向大家介绍实现验证码方法之canvas (源码分享),需要的朋友可以参考一下。
在通常的登录界面我们都可以看到验证码,验证码的作用是检测是不是人在操作,防止机器等非人操作,防止数据库被轻而易举的攻破。
验证码一般用PHP和java等后端语言编写。
但是在前端,用canva或者SVG也可以绘制验证码。
绘制验证码不能是简单的随机字符串,而应该在绘制界面有一些干扰项:
如:干扰线段、干扰圆点、背景等等。
这里的这个demo的canvas验证码干扰项比较简单。
可以在图示中看到本例中的干扰项。
canvas验证码展示效果:
点击实现改变(重绘)验证码:
在控制台运行函数输出返回值(验证码):
源码 :
1 <!DOCTYPE html> 2 <html lang="en"> 3 4 <head> 5 <meta charset="UTF-8"> 6 <meta name="viewport" content="width=device-width, initial-scale=1.0"> 7 <meta http-equiv="X-UA-Compatible" content="ie=edge"> 8 <title>canvas验证码</title> 9 </head> 10 11 <body> 12 <canvas width="200" height="60" id="check" style="border:1px solid #000;">您的浏览器不支持canvas标签!</canvas> 13 <script> 14 var ctx = document.getElementById("check").getContext("2d"); 15 var ctxW = document.getElementById("check").clientWidth; 16 var ctxH = document.getElementById("check").clientHeight; 17 18 /** 19 * 产生一个随机数 可设置随机数区间 20 * @param {[Number]} min [随机数区间下限] 21 * @param {[Number]} max [随机数区间上限] 22 * @return {[Number]} [返回一个在此区间的随机数] 23 */ 24 function ranNum(min, max) { 25 26 return Math.random() * (max - min) + min; 27 28 } 29 30 /** 31 * 返回一个随机颜色 可设置颜色区间 32 * @param {[Number]} min [颜色下限] 33 * @param {[Number]} max [颜色上限] 34 * @return {[String]} [随机颜色] 35 */ 36 function ranColor(min, max) { 37 38 var r = ranNum(min, max); 39 40 var g = ranNum(min, max); 41 42 var b = ranNum(min, max); 43 44 // return "rgb(" + r + "," + g + "," + b + ")"; 45 return `rgb(${r},${g},${b})`; 46 47 } 48 49 /** 50 * 随机字符串数组 51 * @return {[Array]} [随机数组] 52 */ 53 function ranStr() { 54 55 var str = "QWERTYUIOPASDFGHJKLZXCVBNMqwertyuiopasdfghjklzxcvbnm0123456789"; 56 57 return str.split("").sort(function () { 58 return Math.random() - 0.5 59 }); 60 61 } 62 63 /** 64 * 绘制文本字符串 65 * @param {[String]} canvasId [canvas的id] 66 * @param {[Number]} canvasW [canvas的width] 67 * @param {[Number]} canvasH [canvas的height] 68 * @param {[Number]} num [绘制验证码的字数] 69 * @param {[Number]} fsMin [字体大小下限] 70 * @param {[Number]} fsMax [字体大小上限] 71 * @param {[Number]} frMin [字体旋转偏移下限] 72 * @param {[Number]} frMax [字体旋转偏移上限] 73 * @param {[Number]} min [颜色下限] 74 * @param {[Number]} max [颜色上限] 75 * @return {[String]} [随机字符串] 76 */ 77 function drawText(canvasId, canvasW, canvasH, num, fsMin, fsMax, frMin, frMax, min, max) { 78 79 var str = ""; 80 81 for (var i = 0; i < num; i++) { 82 83 var char = ranStr()[Math.floor(0, ranStr().length)]; 84 85 var fs = ranNum(fsMin, fsMax); 86 87 canvasId.font = fs + "px Verdana"; 88 89 canvasId.fillStyle = ranColor(min, max); 90 91 // 保存绘制的状态 92 canvasId.save(); 93 94 // context.translate(x,y); 95 // x 添加到水平坐标(x)上的值 96 // y 添加到垂直坐标(y)上的值 97 // 偏移 98 canvasId.translate(canvasW / num * i + canvasW / 20, 0); 99 100 // 变换角度 101 canvasId.rotate(ranNum(frMin, frMax) * Math.PI / 180); 102 103 // context.fillText(text,x,y,maxWidth); 104 // text 规定在画布上输出的文本。 105 // x 开始绘制文本的 x 坐标位置(相对于画布)。 106 // y 开始绘制文本的 y 坐标位置(相对于画布)。 107 // maxWidth 可选。允许的最大文本宽度,以像素计。 108 canvasId.fillText(char, 0, (canvasH + fs) / 2.5, canvasW / num); 109 110 // 返回之前保存过的路径状态和属性 111 ctx.restore(); 112 113 str += char; 114 115 } 116 117 // console.log(str); 118 return str; 119 120 } 121 122 /** 123 * 绘制背景 124 * @param {[String]} canvasId [canvas的id] 125 * @param {[Number]} canvasW [canvas的width] 126 * @param {[Number]} canvasH [canvas的height] 127 * @param {[Number]} min [下限] 128 * @param {[Number]} max [上限] 129 */ 130 function drawBg(canvasId, canvasW, canvasH, min, max) { 131 132 // 绘制canvas背景 133 canvasId.fillStyle = ranColor(min, max); 134 135 // 填充颜色 136 canvasId.fillRect(0, 0, canvasW, canvasH); 137 138 } 139 140 /** 141 * 绘制干扰 圆点 142 * @param {[String]} canvasId [canvas的id] 143 * @param {[Number]} canvasW [canvas的width] 144 * @param {[Number]} canvasH [canvas的height] 145 * @param {[Number]} num [绘制的数量] 146 * @param {[Number]} r [圆点半径] 147 * @param {[Number]} min [下限] 148 * @param {[Number]} max [上线] 149 */ 150 function drawCircle(canvasId, canvasW, canvasH, num, r, min, max) { 151 152 for (var i = 0; i < num; i++) { 153 154 // 开始绘制 (拿起笔) 155 canvasId.beginPath(); 156 157 // context.arc(x,y,r,sAngle,eAngle,counterclockwise); (绘制) 158 // x 圆的中心的 x 坐标。 159 // y 圆的中心的 y 坐标。 160 // r 圆的半径。 161 // sAngle 起始角,以弧度计。(弧的圆形的三点钟位置是 0 度)。 162 // eAngle 结束角,以弧度计。 163 // counterclockwise 可选。规定应该逆时针还是顺时针绘图。False = 顺时针,true = 逆时针。 164 canvasId.arc(ranNum(0, canvasW), ranNum(0, canvasH), r, 0, 2 * Math.PI); 165 166 // 填充颜色 167 canvasId.fillStyle = ranColor(min, max); 168 169 // 填充 170 canvasId.fill(); 171 172 // 闭合绘制 (放开笔) 173 canvasId.closePath(); 174 175 } 176 177 } 178 179 /** 180 * 绘制干扰 线段 181 * @param {[String]} canvasId [canvas的id] 182 * @param {[Number]} canvasW [canvas的width] 183 * @param {[Number]} canvasH [canvas的height] 184 * @param {[Number]} num [绘制的数量] 185 * @param {[Number]} min [下限] 186 * @param {[Number]} max [上线] 187 */ 188 function drawLine(canvasId, canvasW, canvasH, num, min, max) { 189 190 for (var i = 0; i < num; i++) { 191 192 // 开始绘制 (拿起笔) 193 canvasId.beginPath(); 194 195 // 绘制开始点 196 canvasId.moveTo(ranNum(0, canvasW), ranNum(0, canvasH)); 197 198 // 绘制结束点 199 canvasId.lineTo(ranNum(0, canvasW), ranNum(0, canvasH)); 200 201 canvasId.strokeStyle = ranColor(min, max); 202 203 canvasId.stroke(); 204 205 canvasId.closePath(); 206 207 } 208 209 } 210 211 // 绘制验证码 212 function drawCanvas() { 213 214 // 清空canvas 215 ctx.clearRect(0, 0, 200, 60); 216 217 // 绘制背景 218 drawBg(ctx, ctxW, ctxH, 200, 255); 219 220 // 绘制干扰圆点 221 drawCircle(ctx, ctxW, ctxH, 20, 5, 200, 255); 222 223 // 绘制干扰线段 224 drawLine(ctx, ctxW, ctxH, 20, 0, 255); 225 226 // 绘制验证码 227 var str = drawText(ctx, ctxW, ctxH, 4, 10, 50, -30, 30, 0, 100); 228 229 return str; 230 231 } 232 233 drawCanvas(); 234 235 document.getElementById('check').onclick = drawCanvas; 236 </script> 237 </body> 238 239 </html>
- 输入一个数字,然后计算出从1到输入数字的和,要求,如果输入的数字小于1,则重新输入,直到输入正确的数字为止
- Linux基础(day76)
- zabbix设置QQ邮箱告警
- 关于JSON CSRF的一些思考
- linux学习第七十篇:expect脚本同步文件,expect脚本指定host和要同步的文件,构建文件分发系统,批量远程执行命令
- linux学习第六十九篇:分发系统介绍,expect脚本远程登录,expect脚本远程执行命令,expect脚本传递参数
- linux学习第六十八篇:告警系统邮件引擎,运行告警系统
- linux学习第六十七篇:告警系统主脚本,告警系统配置文件,告警系统监控项目
- linux学习第六十六篇:shell中的函数,shell中的数组,告警系统需求分析
- linux学习第六十五篇:for循环,while循环, break跳出循环,continue结束本次循环
- linux学习第六十四篇:Shell脚本中的逻辑判断,文件目录属性判断, if特殊用法,case判断
- linux学习第六十三篇:Shell脚本介绍,Shell脚本结构和执行,date命令用法,Shell脚本中的变量
- 熔断Hystrix使用尝鲜
- 报警系统QuickAlarm之默认报警规则扩展
- HTML 教程
- HTML 简介
- html div 标签介绍
- html span 标签介绍
- html a 超链接标签
- HTML Br换行标签介绍
- HTML P段落标签介绍
- HTML br与p标签区别
- Html H 标题标签
- html px em pt长度单位
- HTML form 标签
- HTML radio 单选框
- HTML B 加粗标签
- HTML strong加粗粗体标签
- HTML em 强调标签
- HTML i 斜体标签
- HTML u下划线标签
- HTML s 删除线标签
- Html img 图片标签
- Html上标注sup与下标注sub标签
- HTML nobr 禁止换行标签
- HTML hr 水平线标签
- HTML label 标签
- HTML input 标签
- HTML textarea 标签
- HTML select下拉列表标签
- HTML checkbox 多选框
- HTML font color 标签
- HTML iframe 框架标签
- HTML Table 表格
- HTML dl dt dd 标签
- HTML ol li有序列表标签
- HTML ul li 无序列表标签
- HTML 注释
- CSS 教程
- CSS 简介
- CSS 语法
- CSS Id 和 Class选择器
- CSS 样式的创建
- CSS background 背景介绍
- CSS 文本样式
- CSS font 字体
- CSS A 链接
- CSS ul ol列表样式
- CSS TABLE 样式
- CSS 框模型
- CSS border 边框
- CSS Outlines 轮廓
- CSS 外边距 Margin
- CSS Padding 内边距
- CSS 分组和嵌套选择器
- CSS 尺寸 (Dimension)
- CSS Display 属性
- CSS Position 定位
- CSS Float 浮动
- CSS 水平对齐(Horizontal Align)
- CSS 组合选择符
- CSS 伪类
- CSS 伪元素
- CSS 导航栏
- CSS 下拉菜单
- CSS 图片廊
- CSS 图像透明/不透明
- CSS sprite 图像拼合技术
- CSS 媒体类型
- CSS 属性选择器
- CSS 实例
- 一天一大 leet(正则表达式匹配)难度:困难 DAY-20
- mongoDB (三) mongoDB分片集群
- mongoDB (四) mongoDB认证
- 一天一大 leet(二叉树中的最大路径和)难度:困难 DAY-21
- 一天一大 leet(把数字翻译成字符串)难度:中等 DAY-9
- 一天一大 leet(模式匹配)难度:中等 DAY-22
- 一天一大 leet(二进制求和)难度:简单 DAY-23
- xmake v2.3.7 发布, 新增 tinyc 和 emscripten 工具链支持
- 一天一大 leet(最接近的三数之和)难度:中等 DAY-24
- Django渲染模板
- 一天一大 leet(移除重复节点)难度:简单 DAY-26
- Spring Boot 项目瘦身指南,瘦到不可思议!
- BigData--Hadoop数据压缩
- 面试官问我什么是JMM
- django+nginx+gunicorn部署配置