前言
上文已经实现了登录页面的基本逻辑,现在我们来实现图片验证码
图片验证码
网上有太多教程,就直接放图片验证码的实现函数了。 我们要操作的是创建utils文件夹,里面创建code.py文件,放check_code函数,check_code的font_file需要传入一个字符文件ttf,网上随便下载一个导入即可,其他字段默认值根据自己需要改变。
import random
from PIL import Image, ImageDraw, ImageFont, ImageFilter
def check_code(font_file, width=120, height=30, char_length=4, font_size=28):
code = []
img = Image.new(mode='RGB', size=(width, height), color=(255, 255, 255))
draw = ImageDraw.Draw(img, mode='RGB')
def rndChar():
"""
生成随机字母
:return:
"""
return chr(random.randint(65, 90))
def rndColor():
"""
生成随机颜色
:return:
"""
return (random.randint(0, 255), random.randint(10, 255), random.randint(64, 255))
font = ImageFont.truetype(font_file, font_size)
for i in range(char_length):
char = rndChar()
code.append(char)
h = random.randint(0, 4)
draw.text([i * width / char_length, h], char, font=font, fill=rndColor())
for i in range(40):
draw.point([random.randint(0, width), random.randint(0, height)], fill=rndColor())
for i in range(40):
draw.point([random.randint(0, width), random.randint(0, height)], fill=rndColor())
x = random.randint(0, width)
y = random.randint(0, height)
draw.arc((x, y, x + 4, y + 4), 0, 90, fill=rndColor())
for i in range(5):
x1 = random.randint(0, width)
y1 = random.randint(0, height)
x2 = random.randint(0, width)
y2 = random.randint(0, height)
draw.line((x1, y1, x2, y2), fill=rndColor())
img = img.filter(ImageFilter.EDGE_ENHANCE_MORE)
return img, ''.join(code)
然后在urls.py中添加路径
from django.urls import path
from web import views
urlpatterns = [
path('login/', views.login),
path('image/code/', views.img_code),
]
在这里插入代码片
在views.py中定义函数
from django.shortcuts import render, redirect, HttpResponse
from io import BytesIO
from web import models
def img_code(request):
img, code_string = check_code(font_file='web/static/font/AaManYuShouXieTi.ttf')
stream = BytesIO()
img.save(stream, 'png')
return HttpResponse(stream.getvalue())
在login.html文件中添加验证码区域
<!DOCTYPE html>
<html>
<head>
<title>Japan_is_shit</title>
</head>
<body>
<div>
<h2>用户登陆</h2>
<form method="post">
{% csrf_token %}
<div>
<label>用户名</label>
<div>
<input type="" name="user">
</div>
</div>
<div>
<label>密码</label>
<div>
<input type="" name="pwd">
</div>
</div>
{#验证码#}
<div>
<label>验证码</label>
<img src="/image/code/">
<div>
<input type="" name="code">
</div>
</div>
<div>
<div>
<span>{{error_msg}}</span>
<button>登录</button>
</div>
</div>
</form>
</div>
</body>
</html>
实现效果如下图所示 那现在的任务就是写登录逻辑,让我们去到views.py 中写验证图片验证码的登录逻辑:验证码应该是最先被验证的,验证码不通过,直接提示验证码不通过,验证码通过后才验证用户名和密码。
对此做出的改变是在img_code函数中把code_string定义为全局变量,并在login函数中调用code_string与获取的code字段进行比较(大小写一致),不同就重新返回带提示的login.html页面,相同就验证用户和密码。
from django.shortcuts import render, redirect, HttpResponse
from io import BytesIO
from web import models
from web.utils.code import check_code
code_string = ''
def img_code(request):
global code_string
img, code_string = check_code(font_file='web/static/font/AaManYuShouXieTi.ttf')
stream = BytesIO()
img.save(stream, 'png')
return HttpResponse(stream.getvalue())
def login(request):
if request.method == "GET":
return render(request, 'login.html')
name = request.POST.get('user')
pwd = request.POST.get('pwd')
code = request.POST.get('code')
if code.upper() != code_string.upper():
return render(request, 'login.html', {"error_msg": "验证码错误"})
data_list = models.Admin.objects.all()
for data in data_list:
if data.username == name and data.password == pwd:
return redirect('https://blog.csdn.net/qq_43504837')
return render(request, 'login.html', {"error_msg": "用户名或错误"})
总结
给页面加图片验证码的过程: 1.确保有一个生成图片验证码的函数check_code 2.在urls.py中添加路径image/code/ ,并映射函数img_code 3.在views.py 中添加img_code函数 4.在 img_code 函数中利用check_code生成图片和字符 5.图片写入返回前端,字符则用来与前端传过来的验证码比较 6.验证码比较通过,才比较用户和密码
|