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 相关文章推荐
Javascript 定时器调用传递参数的方法
Nov 12 Javascript
JS模拟面向对象全解(二、类型与赋值)
Jul 13 Javascript
全面兼容的javascript时间格式化函数(比较实用)
May 14 Javascript
jquery 中ajax执行的优先级
Jun 22 Javascript
jquery图片滚动放大代码分享(1)
Aug 25 Javascript
jQuery简单动画变换效果实例分析
Jul 04 Javascript
Bootstrap源码解读下拉菜单(4)
Dec 23 Javascript
js 博客内容进度插件详解
Feb 19 Javascript
Vue中Quill富文本编辑器的使用教程
Sep 21 Javascript
JavaScript链式调用原理与实现方法详解
May 16 Javascript
AutoJs实现刷宝短视频的思路详解
May 22 Javascript
react中useState使用:如何实现在当前表格直接更改数据
Aug 05 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
C#使用PHP服务端的Web Service通信实例
2014/04/08 PHP
ThinkPHP路由机制简介
2016/03/23 PHP
javascript数组的扩展实现代码集合
2008/06/01 Javascript
JS array 数组详解
2009/03/22 Javascript
JavaScript 类的定义和引用 JavaScript高级培训 自定义对象
2010/04/27 Javascript
js中查找最近的共有祖先元素的实现代码
2010/12/30 Javascript
自己动手制作jquery插件之自动添加删除行的实现
2011/10/13 Javascript
jQuery使用数组编写图片无缝向左滚动
2012/12/11 Javascript
Json对象与Json字符串互转(4种转换方式)
2013/03/27 Javascript
子窗体与父窗体传值示例js代码
2013/08/01 Javascript
基于bootstrap插件实现autocomplete自动完成表单
2016/05/07 Javascript
解决nodejs中使用http请求返回值为html时乱码的问题
2017/02/18 NodeJs
微信小程序中的swiper组件详解
2017/04/14 Javascript
js模仿微信朋友圈计算时间显示几天/几小时/几分钟/几秒之前
2017/04/27 Javascript
Vue文件配置全局变量的实例
2018/09/06 Javascript
利用d3.js实现蜂巢图表带动画效果
2019/09/03 Javascript
js实现QQ邮箱邮件拖拽删除功能
2020/08/27 Javascript
Python中使用 Selenium 实现网页截图实例
2014/07/18 Python
跟老齐学Python之从格式化表达式到方法
2014/09/28 Python
Python遍历指定文件及文件夹的方法
2015/05/09 Python
深入浅出学习python装饰器
2017/09/29 Python
python3 http提交json参数并获取返回值的方法
2018/12/19 Python
Python获取航线信息并且制作成图的讲解
2019/01/03 Python
Python合并同一个文件夹下所有PDF文件的方法
2019/03/11 Python
TensorFlow通过文件名/文件夹名获取标签,并加入队列的实现
2020/02/17 Python
Python基于百度AI实现OCR文字识别
2020/04/02 Python
浅谈python opencv对图像颜色通道进行加减操作溢出
2020/06/03 Python
python 获取剪切板内容的两种方法
2020/11/28 Python
CSS 3.0文字悬停跳动特效代码
2020/10/26 HTML / CSS
HTML5实现QQ聊天气泡效果
2017/06/26 HTML / CSS
HTML5 移动页面自适应手机屏幕四类方法总结
2017/08/17 HTML / CSS
汉森批发:Hansen Wholesale
2018/05/24 全球购物
三十年同学聚会感言
2015/07/30 职场文书
Ajax常用封装库——Axios的使用
2021/05/08 Javascript
Pytorch中的学习率衰减及其用法详解
2021/06/05 Python
新手入门Jvm-- JVM对象创建与内存分配机制
2021/06/18 Java/Android