js+canvas实现滑动拼图验证码功能


Posted in Javascript onMarch 26, 2018

js+canvas实现滑动拼图验证码功能 

上图为网易云盾的滑动拼图验证码,其应该有一个专门的图片库,裁剪的位置是固定的。我的想法是,随机生成图片,随机生成位置,再用canvas裁剪出滑块和背景图。下面介绍具体步骤。

首先随便找一张图片渲染到canvas上,这里#canvas作为画布,#block作为裁剪出来的小滑块。

<canvas width="310" height="155" id="canvas"></canvas>
<canvas width="310" height="155" id="block"></canvas>
var canvas = document.getElementById('canvas')
 var block = document.getElementById('block')
 var canvas_ctx = canvas.getContext('2d')
 var block_ctx = block.getContext('2d')
 var img = document.createElement('img')
 img.onload = function() {
  canvas_ctx.drawImage(img, 0, 0, 310, 155)
  block_ctx.drawImage(img, 0, 0, 310, 155)
 };
 img.src = 'img.jpg'

js+canvas实现滑动拼图验证码功能 

下面考虑如何裁剪出拼图的形状,拼图形状比较复杂,首先我们画一个正方形,接着上边的代码写:

var x = 150, y = 40, w = 42, r = 10, PI = Math.PI 
 function draw(ctx) {
  ctx.beginPath()
  ctx.moveTo(x, y)
  ctx.lineTo(x + w, y)
  ctx.lineTo(x + w, y + w)
  ctx.lineTo(x, y + w)
  ctx.clip()
 }
 draw(canvas_ctx)
 draw(block_ctx)

x,y为正方形左上角的坐标,现在先随便写后边生成的时候用随机数,w为正方形的边长,r为后边画缺口的圆的半径。我们先把绘图过程用函数封装起来,方便后面同时操作背景和滑块。用clip()方法裁剪图片后生成一个正方形。

js+canvas实现滑动拼图验证码功能 

接下来画正方形上边和右边的圆形:

function draw(ctx) {
  ctx.beginPath()
  ctx.moveTo(x,y)
+  ctx.lineTo(x+w/2,y)
+  ctx.arc(x+w/2,y-r+2, r,0,2*PI) //
+  ctx.lineTo(x+w/2,y)
  ctx.lineTo(x+w,y)
+  ctx.lineTo(x+w,y+w/2)
+  ctx.arc(x+w+r-2,y+w/2,r,0,2*PI) //
+  ctx.lineTo(x+w,y+w/2)
  ctx.lineTo(x+w,y+w)
  ctx.lineTo(x,y+w)
  ctx.lineTo(x,y)
  ctx.clip()
 }

js+canvas实现滑动拼图验证码功能 

两处注释的位置将圆心往内偏移了2px,实现缺口的样式。然后是左边空心的部分,由于clip是裁剪路径内的部分,因此直接像上面画圆是不行的,我们开启一条新的路径,然后画圆将这个正方形“遮盖”出一个缺口,这里会用到 globalCompositeOperation 属性,'xor'顾名思义。代码接上边:

function draw(ctx) {
  ctx.beginPath()
  ...
  ctx.lineTo(x,y)
  ctx.clip()
+  ctx.beginPath()
+  ctx.arc(x,y+w/2, r,1.5*PI,0.5*PI) // 只需要画正方形内的半圆就行,方便背景图片的裁剪
+  ctx.globalCompositeOperation = "xor"
+  ctx.fill()
}

js+canvas实现滑动拼图验证码功能 

现在一个基本的拼图形状有了,我们调整#block的大小,并将裁剪出来的滑块放入#block中:

img.onload = function() {
  ctx.drawImage(img, 0, 0, 310, 155)
  block_ctx.drawImage(img, 0, 0, 310, 155)
+  var blockWidth = w + r * 2
+  var _y = y - r * 2 + 2 // 滑块实际的y坐标
+  var ImageData = block_ctx.getImageData(x, _y, blockWidth, blockWidth)
+  block.width = blockWidth
+  block_ctx.putImageData(ImageData, 0, _y)
 }

js+canvas实现滑动拼图验证码功能 

现在我们需要把左边画布展示原来的图片,并且抠掉中间滑块的部分,这里画路径的过程都是一样的,唯一不同只是clip()那里改成fill()即可实现效果,我们前面已经把画路径的过程封装成函数了,稍作改动即可:

- function draw(ctx) {
+ function draw(ctx, operation) {
  ...
- ctx.clip()
+ ctx.fillStyle = '#fff'
+ ctx[operation]()
  ...
}
+ draw(canvas_ctx, 'fill')
+ draw(block_ctx, 'clip')

js+canvas实现滑动拼图验证码功能 

接下来就是写样式了,略过:

js+canvas实现滑动拼图验证码功能 

然后就是写拖动事件,我们可以在鼠标按下时记录鼠标位置,然后在拖动时给滑块设置left值。最后在松开鼠标时,判断滑块此时的left值和最开始裁剪滑块时的x值,如果在一定范围内就算验证通过,否则验证失败。

最后再加上随机图片和随机剪切位置,基本就ok了。另外可以判断下鼠标移动时y轴的变化,以判断是否是“人”在操作,当然web安全这块神魔乱舞的,我就不班门弄斧了,只是做个简单判断就行。

由于没有给切片边缘加上边框或者阴影,导致某些图片的滑块可识别度不高,需要后边完善下(其实是我还没折腾出来- -),希望懂这块的大神帮我完善下//

js+canvas实现滑动拼图验证码功能 

后边代码比较杂乱,就不贴上来了, 查看完整代码点这里 演示地址 点这里

总结

以上所述是小编给大家介绍的js+canvas实现滑动拼图验证码功能,希望对大家有所帮助,如果大家有任何疑问请给我留言,小编会及时回复大家的。在此也非常感谢大家对三水点靠木网站的支持!

Javascript 相关文章推荐
Javascript写了一个清除“logo1_.exe”的杀毒工具(可扫描目录)
Feb 09 Javascript
javascript globalStorage类代码
Jun 04 Javascript
js DOM的学习笔记
Dec 22 Javascript
利用js实现遮罩以及弹出可移动登录窗口
Jul 08 Javascript
JQuery操作三大控件(下拉,单选,复选)的方法
Aug 06 Javascript
JS将表单导出成EXCEL的实例代码
Nov 11 Javascript
WordPress 单页面上一页下一页的实现方法【附代码】
Mar 10 Javascript
jQuery实现图片轮播效果代码(基于jquery.pack.js插件)
Jun 02 Javascript
JavaScript事件详细讲解
Jun 27 Javascript
jQuery 获取遍历获取table中每一个tr中的第一个td的方法
Oct 05 Javascript
Bootstrap 表单验证formValidation 实现表单动态验证功能
May 17 Javascript
js监听html页面的上下滚动事件方法
Sep 11 Javascript
JS从非数组对象转数组的方法小结
Mar 26 #Javascript
深入理解Node module模块
Mar 26 #Javascript
利用Console来Debug的10个高级技巧汇总
Mar 26 #Javascript
关于vuejs中v-if和v-show的区别及v-show不起作用问题
Mar 26 #Javascript
vue中使用iview自定义验证关键词输入框问题及解决方法
Mar 26 #Javascript
Vue中v-show添加表达式的问题(判断是否显示)
Mar 26 #Javascript
使用Vue构建可重用的分页组件
Mar 26 #Javascript
You might like
如何使用PHP中的字符串函数
2006/11/24 PHP
用PHP控制用户的浏览器--ob*函数的使用说明
2007/03/16 PHP
在PHP中检查PHP文件是否有语法错误的方法
2009/12/23 PHP
PHP文件操作实现代码分享
2011/09/01 PHP
关于PHP的curl开启问题探讨
2014/04/08 PHP
php面向对象中static静态属性和静态方法的调用
2015/02/08 PHP
PHP识别二维码的方法(php-zbarcode安装与使用)
2016/07/07 PHP
php使用curl详细解析及问题汇总
2016/08/11 PHP
PHP排序算法之简单选择排序(Simple Selection Sort)实例分析
2018/04/20 PHP
PHP树形结构tree类用法示例
2019/02/01 PHP
初探jquery——表单应用范例
2007/02/20 Javascript
JS图像无缝滚动脚本非常好用
2014/02/10 Javascript
Javascript中innerHTML用法实例分析
2015/01/12 Javascript
jQuery仿Flash上下翻动的中英文导航菜单实例
2015/03/10 Javascript
js实时获取并显示当前时间的方法
2015/07/31 Javascript
实例详解JavaScript获取链接参数的方法
2016/01/01 Javascript
javascript实现鼠标点击页面 移动DIV
2016/12/02 Javascript
浅析Jquery操作select
2016/12/13 Javascript
less简单入门(CSS 预处理语言)
2017/03/08 Javascript
vue 如何添加全局函数或全局变量以及单页面的title设置总结
2017/06/01 Javascript
jQuery实现frame之间互通的方法
2017/06/26 jQuery
详解webpack 多页面/入口支持&amp;公共组件单独打包
2017/06/29 Javascript
js实现右键弹出自定义菜单
2020/09/08 Javascript
[48:32]VGJ.T vs Fnatic 2018国际邀请赛小组赛BO2 第一场 8.16
2018/08/17 DOTA
安装ElasticSearch搜索工具并配置Python驱动的方法
2015/12/22 Python
对pandas进行数据预处理的实例讲解
2018/04/20 Python
python中如何使用分步式进程计算详解
2019/03/22 Python
Selenium获取登录Cookies并添加Cookies自动登录的方法
2020/12/04 Python
WEB控件及HTML服务端控件能否调用客户端方法?如果能,请解释如何调用?
2015/08/25 面试题
服装设计师职业生涯规划范文
2014/02/28 职场文书
对标管理实施方案
2014/03/12 职场文书
升学宴主持词
2014/04/02 职场文书
综治工作汇报材料
2014/10/27 职场文书
2014年派出所工作总结
2014/11/21 职场文书
博士给导师的自荐信
2015/03/06 职场文书
经理岗位职责范本
2015/04/15 职场文书