使用 node-canvas 绘制验证码

高洛峰
풀어 주다: 2016-11-19 16:13:52
원래의
2277명이 탐색했습니다.

step 1 安装

在安装 node-canvas 之前,还需要安装一些依赖。不同的系统需要安装的不同,以 linux 和 mac 为例:

linux: sudo yum install cairo cairo-devel cairomm-devel libjpeg-turbo-devel pango pango-devel pangomm pangomm-devel giflib-devel

mac: brew install pkg-config cairo pango libpng jpeg giflib

其他参考 node-canvas#installation

安装完依赖后,执行 npm install canvas 即可。

step 2 画图

通过获取 canvas,可以得到 context 对象,然后就可以像在前端一样绘制图形了

const Canvas = require('canvas'); const canvas = new Canvas(100, 30), ctx = canvas.getContext('2d');
로그인 후 복사

实际上我用到的 api 和前端的 canvas 是一样的,绘制过程就不多解释,可以参考 canvas 的相关教程。

下面是绘制一个 a + b = ? 的验证码

ctx.font = '24px "Microsoft YaHei"'; // 绘制文本 let drawText = (text, x) => { ctx.save(); // 旋转角度 const angle = Math.random() / 10; // y 坐标 const y = 22; ctx.rotate(angle); ctx.fillText(text, x, y); ctx.restore(); } // 随机画线 let drawLine = () => { const num = Math.floor(Math.random() * 2 + 3); // 随机画几条彩色线条 for (let i = 0; i < num; i++) { const color = '#' + (Math.random() * 0xffffff << 0).toString(16); const y1 = Math.random() * height; const y2 = Math.random() * height; // 画线 ctx.strokeStyle = color; ctx.beginPath(); ctx.lineTo(0, y1); ctx.lineTo(width, y2); ctx.stroke(); } } // 数字的文本随机从小写汉字、大写汉字、数字里选择 const numArr = [ '〇一二三四五六七八九', '0123456789', '零壹贰叁肆伍陆柒捌玖' ]; // 第一个数字 const fir = Math.floor(Math.random() * 10); // 第二个数字 const sec = Math.floor(Math.random() * 10); // 随机选取运算 const operArr = ['加', '减', '乘']; const oper = Math.floor(Math.random() * operArr.length); drawLine(); drawText(numArr[Math.floor(Math.random() * numArr.length)][fir], 10); drawText(operArr[oper], 40); drawText(numArr[Math.floor(Math.random() * numArr.length)][sec], 70); drawText('=', 100); drawText('?', 130); // 验证码值的计算 let captcha; switch(oper) { case 0: captcha = fir + sec; break; case 1: captcha = fir - sec; break; case 2: captcha = fir * sec; break; } // 存入 session req.session.captcha = captcha;
로그인 후 복사

效果如下:

wKioL1guYhOjQXvkAAA8PZ7t7wQ318.png

step 3 返回图片

调用 canvas.toDataURL(),可以返回图片的 base64 格式数据。

res.send({ status: 200, data: canvas.toDataURL() })
로그인 후 복사

中文乱码

在将项目部署到 linux 后,发现输出显示的图片中的中文都变成了方框。

我参考了https://my.oschina.net/u/129529/blog/266843 这篇文章,但是没有全部运行,而是安装了

yum groupinstall "Chinese Support",yum groupinstall Fonts 这两个。

另外参考https://cnodejs.org/topic/53f98ad8bbdaa79d518c0836 问题里的 5 楼,使用了微软雅黑。

还有 issue#461,在字体两侧加上引号。

我按这三个做了,然后 重启项目 就好了~


관련 라벨:
원천:php.cn
본 웹사이트의 성명
본 글의 내용은 네티즌들의 자발적인 기여로 작성되었으며, 저작권은 원저작자에게 있습니다. 본 사이트는 이에 상응하는 법적 책임을 지지 않습니다. 표절이나 침해가 의심되는 콘텐츠를 발견한 경우 admin@php.cn으로 문의하세요.
최신 다운로드
더>
웹 효과
웹사이트 소스 코드
웹사이트 자료
프론트엔드 템플릿
회사 소개 부인 성명 Sitemap
PHP 중국어 웹사이트:공공복지 온라인 PHP 교육,PHP 학습자의 빠른 성장을 도와주세요!