Django自定義插件實(shí)現(xiàn)網(wǎng)站登錄驗(yàn)證碼功能
前言
網(wǎng)站登錄的時(shí)候我們常常會(huì)看到隨機(jī)的驗(yàn)證碼需要輸入后臺(tái)驗(yàn)證,如圖:

現(xiàn)在我們來實(shí)現(xiàn)在Django中通過自定制插件來實(shí)現(xiàn)隨機(jī)驗(yàn)證
check_code.py
基于PIL生成一個(gè)帶驗(yàn)證碼的圖片和驗(yàn)證碼,生成驗(yàn)證碼圖片需要Monaco.ttf字體(重要),可按自己要求更改check_code中的字體和字體文件位置
#!/usr/bin/env python
# -*- coding:utf-8 -*-
import random
from PIL import Image, ImageDraw, ImageFont, ImageFilter
_letter_cases = "abcdefghjkmnpqrstuvwxy" # 小寫字母,去除可能干擾的i,l,o,z
_upper_cases = _letter_cases.upper() # 大寫字母
_numbers = ''.join(map(str, range(3, 10))) # 數(shù)字
init_chars = ''.join((_letter_cases, _upper_cases, _numbers))
def create_validate_code(size=(120, 30),
chars=init_chars,
img_type="GIF",
mode="RGB",
bg_color=(255, 255, 255),
fg_color=(0, 0, 255),
font_size=18,
font_type="Monaco.ttf",
length=4,
draw_lines=True,
n_line=(1, 2),
draw_points=True,
point_chance=2):
"""
@todo: 生成驗(yàn)證碼圖片
@param size: 圖片的大小,格式(寬,高),默認(rèn)為(120, 30)
@param chars: 允許的字符集合,格式字符串
@param img_type: 圖片保存的格式,默認(rèn)為GIF,可選的為GIF,JPEG,TIFF,PNG
@param mode: 圖片模式,默認(rèn)為RGB
@param bg_color: 背景顏色,默認(rèn)為白色
@param fg_color: 前景色,驗(yàn)證碼字符顏色,默認(rèn)為藍(lán)色#0000FF
@param font_size: 驗(yàn)證碼字體大小
@param font_type: 驗(yàn)證碼字體,默認(rèn)為 ae_AlArabiya.ttf
@param length: 驗(yàn)證碼字符個(gè)數(shù)
@param draw_lines: 是否劃干擾線
@param n_lines: 干擾線的條數(shù)范圍,格式元組,默認(rèn)為(1, 2),只有draw_lines為True時(shí)有效
@param draw_points: 是否畫干擾點(diǎn)
@param point_chance: 干擾點(diǎn)出現(xiàn)的概率,大小范圍[0, 100]
@return: [0]: PIL Image實(shí)例
@return: [1]: 驗(yàn)證碼圖片中的字符串
"""
width, height = size # 寬高
# 創(chuàng)建圖形
img = Image.new(mode, size, bg_color)
draw = ImageDraw.Draw(img) # 創(chuàng)建畫筆
def get_chars():
"""生成給定長(zhǎng)度的字符串,返回列表格式"""
return random.sample(chars, length)
def create_lines():
"""繪制干擾線"""
line_num = random.randint(*n_line) # 干擾線條數(shù)
for i in range(line_num):
# 起始點(diǎn)
begin = (random.randint(0, size[0]), random.randint(0, size[1]))
# 結(jié)束點(diǎn)
end = (random.randint(0, size[0]), random.randint(0, size[1]))
draw.line([begin, end], fill=(0, 0, 0))
def create_points():
"""繪制干擾點(diǎn)"""
chance = min(100, max(0, int(point_chance))) # 大小限制在[0, 100]
for w in range(width):
for h in range(height):
tmp = random.randint(0, 100)
if tmp > 100 - chance:
draw.point((w, h), fill=(0, 0, 0))
def create_strs():
"""繪制驗(yàn)證碼字符"""
c_chars = get_chars()
strs = ' %s ' % ' '.join(c_chars) # 每個(gè)字符前后以空格隔開
font = ImageFont.truetype(font_type, font_size)
font_width, font_height = font.getsize(strs)
draw.text(((width - font_width) / 3, (height - font_height) / 3),
strs, font=font, fill=fg_color)
return ''.join(c_chars)
if draw_lines:
create_lines()
if draw_points:
create_points()
strs = create_strs()
# 圖形扭曲參數(shù)
params = [1 - float(random.randint(1, 2)) / 100,
0,
0,
0,
1 - float(random.randint(1, 10)) / 100,
float(random.randint(1, 2)) / 500,
0.001,
float(random.randint(1, 2)) / 500
]
img = img.transform(size, Image.PERSPECTIVE, params) # 創(chuàng)建扭曲
img = img.filter(ImageFilter.EDGE_ENHANCE_MORE) # 濾鏡,邊界加強(qiáng)(閾值更大)
return img, strs
創(chuàng)建urls和views
views.py:
# 將check_code包放在合適的位置,導(dǎo)入即可,我是放在utils下面 from utils import check_code def create_code_img(request): f = BytesIO() #直接在內(nèi)存開辟一點(diǎn)空間存放臨時(shí)生成的圖片 img, code = check_code.create_validate_code() #調(diào)用check_code生成照片和驗(yàn)證碼 request.session['check_code'] = code #將驗(yàn)證碼存在服務(wù)器的session中,用于校驗(yàn) img.save(f,'PNG') #生成的圖片放置于開辟的內(nèi)存中 return HttpResponse(f.getvalue()) #將內(nèi)存的數(shù)據(jù)讀取出來,并以HttpResponse返回
urls我的設(shè)置:url(r'^create_code_img/', views.create_code_img)
前端應(yīng)用驗(yàn)證碼和點(diǎn)擊自動(dòng)刷新
html:
<div class="row">
<div class="col-xs-7">
<input type="text" class="form-control" name="check_code" id="check_code" placeholder="請(qǐng)輸入驗(yàn)證碼">
</div>
<div class="col-xs-5">
<img id="check_code_img" src="/create_code_img/" onclick="refresh_check_code(this)">
{# src是url路徑,可得到驗(yàn)證碼圖片,點(diǎn)擊時(shí)調(diào)用refresh_check_code#}
</div><br></div>
javascript:
<script>
function refresh_check_code(ths) {
ths.src += '?';
{# src后面加問好會(huì)自動(dòng)刷新驗(yàn)證碼img的src#}
}
</script>
login的Views
login的Views進(jìn)行數(shù)據(jù)驗(yàn)證,然后做相應(yīng)的處理
post_check_code = request.POST.get('check_code')
session_check_code = request.session['check_code']
if post_check_code.lower() == session_check_code.lower() :
pass
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
使用Python通過win32 COM實(shí)現(xiàn)Word文檔的寫入與保存方法
今天小編就為大家分享一篇使用Python通過win32 COM實(shí)現(xiàn)Word文檔的寫入與保存方法,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過來看看吧2018-05-05
Python算法繪制特洛伊小行星群實(shí)現(xiàn)示例
這篇文章主要介紹了Python算法繪制特洛伊小行星群實(shí)現(xiàn)示例,這個(gè)小示例完成后非常的有意思也會(huì)使你在Python學(xué)習(xí)的道路上感到一絲絲小成就2021-10-10
python畫圖時(shí)linestyle,color和loc參數(shù)的設(shè)置方式
這篇文章主要介紹了python畫圖時(shí)linestyle,color和loc參數(shù)的設(shè)置方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2022-11-11
PyQt5實(shí)現(xiàn)tableWidget 居中顯示
這篇文章主要介紹了PyQt5實(shí)現(xiàn)tableWidget 居中顯示方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教2022-07-07
django商品分類及商品數(shù)據(jù)建模實(shí)例詳解
這篇文章主要介紹了django商品分類及商品數(shù)據(jù)建模實(shí)例代碼內(nèi)容,需要的朋友們學(xué)習(xí)參考下。2020-01-01

