JavaScript Canvas实现验证码


Posted in Javascript onAugust 02, 2020

在通常的登录界面我们都可以看到验证码,验证码的作用是检测是不是人在操作,防止机器等非人操作,防止数据库被轻而易举的攻破。

验证码一般用PHP和java等后端语言编写。

但是在前端,用canva或者SVG也可以绘制验证码。

绘制验证码不能是简单的随机字符串,而应该在绘制界面有一些干扰项:

如:干扰线段、干扰圆点、背景等等。

这里的这个demo的canvas验证码干扰项比较简单。

可以在图示中看到本例中的干扰项。

canvas验证码展示效果:

点击实现改变(重绘)验证码:

JavaScript Canvas实现验证码

在控制台运行函数输出返回值(验证码):

JavaScript Canvas实现验证码

源码 :

<!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>canvas验证码</title>
</head>

<body>
 <canvas width="200" height="60" id="check" style="border:1px solid #000;">您的浏览器不支持canvas标签!</canvas>
 <script>
 var ctx = document.getElementById("check").getContext("2d");
 var ctxW = document.getElementById("check").clientWidth;
 var ctxH = document.getElementById("check").clientHeight;

 /**
  * 产生一个随机数 可设置随机数区间
  * @param {[Number]} min [随机数区间下限]
  * @param {[Number]} max [随机数区间上限]
  * @return {[Number]} [返回一个在此区间的随机数]
  */
 function ranNum(min, max) {

  return Math.random() * (max - min) + min;

 }

 /**
  * 返回一个随机颜色 可设置颜色区间
  * @param {[Number]} min [颜色下限]
  * @param {[Number]} max [颜色上限]
  * @return {[String]} [随机颜色]
  */
 function ranColor(min, max) {

  var r = ranNum(min, max);

  var g = ranNum(min, max);

  var b = ranNum(min, max);

  // return "rgb(" + r + "," + g + "," + b + ")";
  return `rgb(${r},${g},${b})`;

 }

 /**
  * 随机字符串数组
  * @return {[Array]} [随机数组]
  */
 function ranStr() {

  var str = "QWERTYUIOPASDFGHJKLZXCVBNMqwertyuiopasdfghjklzxcvbnm0123456789";

  return str.split("").sort(function () {
  return Math.random() - 0.5
  });

 }

 /**
  * 绘制文本字符串
  * @param {[String]} canvasId [canvas的id]
  * @param {[Number]} canvasW [canvas的width]
  * @param {[Number]} canvasH [canvas的height]
  * @param {[Number]} num [绘制验证码的字数]
  * @param {[Number]} fsMin [字体大小下限]
  * @param {[Number]} fsMax [字体大小上限]
  * @param {[Number]} frMin [字体旋转偏移下限]
  * @param {[Number]} frMax [字体旋转偏移上限]
  * @param {[Number]} min [颜色下限]
  * @param {[Number]} max [颜色上限]
  * @return {[String]} [随机字符串]
  */
 function drawText(canvasId, canvasW, canvasH, num, fsMin, fsMax, frMin, frMax, min, max) {

  var str = "";

  for (var i = 0; i < num; i++) {

  var char = ranStr()[Math.floor(0, ranStr().length)];

  var fs = ranNum(fsMin, fsMax);

  canvasId.font = fs + "px Verdana";

  canvasId.fillStyle = ranColor(min, max);

  // 保存绘制的状态
  canvasId.save();

  // context.translate(x,y);
  // x 添加到水平坐标(x)上的值
  // y 添加到垂直坐标(y)上的值
  // 偏移
  canvasId.translate(canvasW / num * i + canvasW / 20, 0);

  // 变换角度
  canvasId.rotate(ranNum(frMin, frMax) * Math.PI / 180);

  // context.fillText(text,x,y,maxWidth);
  // text 规定在画布上输出的文本。
  // x 开始绘制文本的 x 坐标位置(相对于画布)。
  // y 开始绘制文本的 y 坐标位置(相对于画布)。
  // maxWidth 可选。允许的最大文本宽度,以像素计。
  canvasId.fillText(char, 0, (canvasH + fs) / 2.5, canvasW / num);

  // 返回之前保存过的路径状态和属性
  ctx.restore();

  str += char;

  }

  // console.log(str);
  return str;

 }

 /**
  * 绘制背景
  * @param {[String]} canvasId [canvas的id]
  * @param {[Number]} canvasW [canvas的width]
  * @param {[Number]} canvasH [canvas的height]
  * @param {[Number]} min [下限]
  * @param {[Number]} max [上限]
  */
 function drawBg(canvasId, canvasW, canvasH, min, max) {

  // 绘制canvas背景
  canvasId.fillStyle = ranColor(min, max);

  // 填充颜色
  canvasId.fillRect(0, 0, canvasW, canvasH);

 }

 /**
  * 绘制干扰 圆点
  * @param {[String]} canvasId [canvas的id]
  * @param {[Number]} canvasW [canvas的width]
  * @param {[Number]} canvasH [canvas的height]
  * @param {[Number]} num [绘制的数量]
  * @param {[Number]} r [圆点半径]
  * @param {[Number]} min [下限]
  * @param {[Number]} max [上线]
  */
 function drawCircle(canvasId, canvasW, canvasH, num, r, min, max) {

  for (var i = 0; i < num; i++) {

  // 开始绘制 (拿起笔)
  canvasId.beginPath();

  // context.arc(x,y,r,sAngle,eAngle,counterclockwise); (绘制)
  // x 圆的中心的 x 坐标。
  // y 圆的中心的 y 坐标。
  // r 圆的半径。
  // sAngle 起始角,以弧度计。(弧的圆形的三点钟位置是 0 度)。
  // eAngle 结束角,以弧度计。
  // counterclockwise 可选。规定应该逆时针还是顺时针绘图。False = 顺时针,true = 逆时针。
  canvasId.arc(ranNum(0, canvasW), ranNum(0, canvasH), r, 0, 2 * Math.PI);

  // 填充颜色
  canvasId.fillStyle = ranColor(min, max);

  // 填充
  canvasId.fill();

  // 闭合绘制 (放开笔)
  canvasId.closePath();

  }

 }

 /**
  * 绘制干扰 线段
  * @param {[String]} canvasId [canvas的id]
  * @param {[Number]} canvasW [canvas的width]
  * @param {[Number]} canvasH [canvas的height]
  * @param {[Number]} num [绘制的数量]
  * @param {[Number]} min [下限]
  * @param {[Number]} max [上线]
  */
 function drawLine(canvasId, canvasW, canvasH, num, min, max) {

  for (var i = 0; i < num; i++) {

  // 开始绘制 (拿起笔)
  canvasId.beginPath();

  // 绘制开始点
  canvasId.moveTo(ranNum(0, canvasW), ranNum(0, canvasH));

  // 绘制结束点
  canvasId.lineTo(ranNum(0, canvasW), ranNum(0, canvasH));

  canvasId.strokeStyle = ranColor(min, max);

  canvasId.stroke();

  canvasId.closePath();

  }

 }

 // 绘制验证码
 function drawCanvas() {

  // 清空canvas
  ctx.clearRect(0, 0, 200, 60);

  // 绘制背景
  drawBg(ctx, ctxW, ctxH, 200, 255);

  // 绘制干扰圆点
  drawCircle(ctx, ctxW, ctxH, 20, 5, 200, 255);

  // 绘制干扰线段
  drawLine(ctx, ctxW, ctxH, 20, 0, 255);

  // 绘制验证码
  var str = drawText(ctx, ctxW, ctxH, 4, 10, 50, -30, 30, 0, 100);

  return str;

 }

 drawCanvas();

 document.getElementById('check').onclick = drawCanvas;
 </script>
</body>

</html>

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持三水点靠木。

Javascript 相关文章推荐
jQuery性能优化28条建议你值得借鉴
Feb 16 Javascript
根据身份证号自动输出相关信息(籍贯,出身日期,性别)
Nov 15 Javascript
jquery实现手机发送验证码的倒计时代码
Feb 12 Javascript
jQuery手机拨号界面特效代码分享
Aug 27 Javascript
深入理解JavaScript函数参数(推荐)
Jul 26 Javascript
D3.js实现直方图的方法详解
Sep 25 Javascript
JS瀑布流实现方法实例分析
Dec 19 Javascript
node.js发送邮件email的方法详解
Jan 06 Javascript
js实现增加数字显示的环形进度条效果
Feb 05 Javascript
JavaScript设计模式之模板方法模式原理与用法示例
Aug 07 Javascript
浅谈目前可以使用ES10的5个新特性
Jun 25 Javascript
国内常用的js类库大全(CDN公共库)
Jun 24 Javascript
JS实现方形抽奖效果
Aug 27 #Javascript
对vuejs的v-for遍历、v-bind动态改变值、v-if进行判断的实例讲解
Aug 27 #Javascript
Vuex 在Vue 组件中获得Vuex 状态state的方法
Aug 27 #Javascript
JS实现匀速与减速缓慢运动的动画效果封装示例
Aug 27 #Javascript
vue展示dicom文件医疗系统的实现代码
Aug 27 #Javascript
vue中子组件的methods中获取到props中的值方法
Aug 27 #Javascript
最适应的vue.js的form提交涉及多种插件【推荐】
Aug 27 #Javascript
You might like
PHP 魔术函数使用说明
2010/05/14 PHP
ThinkPHP中实例Model方法的区别说明
2010/08/21 PHP
php正则匹配文章中的远程图片地址并下载图片至本地
2015/09/29 PHP
Zend Framework动作助手Redirector用法实例详解
2016/03/05 PHP
Laravel 5.4重新登录实现跳转到登录前页面的原理和方法
2017/07/13 PHP
PHP+redis实现的悲观锁机制示例
2018/06/12 PHP
jquery中dom操作和事件的实例学习 下拉框应用
2011/12/01 Javascript
使用JavaScript动态设置样式实现代码(2)
2013/01/25 Javascript
Jquery实现图片放大镜效果的思路及代码(自写)
2013/10/18 Javascript
实现checkbox全选、反选、取消JavaScript小脚本异常
2014/04/10 Javascript
js创建一个input数组并绑定click事件的方法
2014/06/12 Javascript
浅谈javascript构造函数与实例化对象
2015/06/22 Javascript
js绘制圆形和矩形的方法
2015/08/05 Javascript
JavaScript使用DeviceOne开发实战(四)仿优酷视频应用
2015/12/02 Javascript
JavaScript判断是否是微信浏览器
2016/06/13 Javascript
ES6概念 Symbol.keyFor()方法
2016/12/25 Javascript
JavaScript输入框字数实时统计更新
2017/06/17 Javascript
浅谈vue.js中v-for循环渲染
2017/07/26 Javascript
浅谈vue的几种绑定变量的值 防止其改变的方法
2018/03/01 Javascript
基于Vue自定义指令实现按钮级权限控制思路详解
2018/05/23 Javascript
解决angularjs中同步执行http请求的方法
2018/08/13 Javascript
深入浅析Vue 中 ref 的使用
2019/04/29 Javascript
Vue的transition-group与Virtual Dom Diff算法的使用
2019/12/09 Javascript
vue-cli设置css不生效的解决方法
2020/02/07 Javascript
[40:10]2015国际邀请赛全明星表演赛
2015/08/07 DOTA
[09:31]2016国际邀请赛中国区预选赛Yao赛后采访 答题送礼
2016/06/27 DOTA
使用Python脚本将绝对url替换为相对url的教程
2015/04/24 Python
Python单元测试简单示例
2018/07/03 Python
Python对象中__del__方法起作用的条件详解
2018/11/01 Python
Pytorch 实现sobel算子的卷积操作详解
2020/01/10 Python
Python代码中如何读取键盘录入的值
2020/05/27 Python
使用Python操作MySQL的小技巧
2020/09/10 Python
以工厂直接定价的传奇性能:Ben Hogan Golf
2019/01/04 全球购物
《找不到快乐的波斯猫》教学反思
2014/02/24 职场文书
群教个人对照检查材料
2014/08/20 职场文书
解决Golang中goroutine执行速度的问题
2021/05/02 Golang