Django_day05

Django

登录注册功能

配置settings.py
在最后加上

# 配置没有登录则跳转到登录
LOGIN_URL = '/users/login/'

配置templates

TEMPLATES = [
    {
        'BACKEND': 'django.template.backends.django.DjangoTemplates',
        'DIRS': [os.path.join(BASE_DIR, 'templates')],
        'APP_DIRS': True,
        'OPTIONS': {
            'context_processors': [
                'django.template.context_processors.debug',
                'django.template.context_processors.request',
                'django.contrib.auth.context_processors.auth',
                'django.contrib.messages.context_processors.messages',
            ],
        },
    },
]

M(models):
models.py

V(views):
urls.py
views.py
forms.py

T(templates):
base.html
index.html
login.html
register.html

关联关系:url——>浏览器——>views.py&urls.py——>templates&html——>forms.py&views.py&urls.py——>templates&html

urls.py

from django.conf.urls import url
from django.contrib.auth.decorators import login_required

from users import views

urlpatterns = [
    # 注册
    url(r'^register/', views.register, name='register'),
    # 登录
    url(r'^login/', views.login, name='login'),
    # 首页
    url(r'^index/', login_required(views.index), name='index'),
    # 注销
    url(r'^logout', login_required(views.logout), name='logout'),
]

首先导入相应的库
login_required() 顾名思义,需要登录后才能访问

forms.py

from django import forms
from django.contrib.auth.models import User


class UserForm(forms.Form):
    """
    校验注册信息
    """
    # username = forms.CharField(required=True)
    # password = forms.CharField(required=True)
    # password2 = forms.CharField(required=True)

    username = forms.CharField(required=True,
                               max_length=5,
                               min_length=2,
                               error_messages={
                                   'required': '用户必填',
                                   'max_length': '长度不能超过5位',
                                   'min_length': '长度不能少于2位'
                               })
    password = forms.CharField(required=True,
                               min_length=6,
                               error_messages={
                                   'required': '密码必填',
                                   'min_length': '长度不能少于6位'
                               })
    password2 = forms.CharField(required=True,
                                min_length=6,
                                error_messages={
                                    'required': '密码必填',
                                    'min_length': '长度不能少于6位'
                                })

    def clean(self):
        # 校验用户名是否已经注册过
        user = User.objects.filter(username=self.cleaned_data.get('username'))
        if user:
            # 已经被注册
            raise forms.ValidationError({'username': '用户名已经存在,请直接登录'})
            pass
        else:
            # 没有被注册
            pass
        # 校验密码和确认密码是否相同
        if self.cleaned_data.get('password') != self.cleaned_data.get('password2'):
            raise forms.ValidationError({'password2': '两次密码不一致'})
        return self.cleaned_data


class CheckUserForm(forms.Form):
    username = forms.CharField(required=True,
                               max_length=5,
                               min_length=2,
                               error_messages={
                                   'required': '用户必填',
                                   'max_length': '长度不能超过5位',
                                   'min_length': '长度不能少于2位'
                               })
    password = forms.CharField(required=True,
                               min_length=6,
                               error_messages={
                                   'required': '密码必填',
                                   'min_length': '长度不能少于6位'
                               })

    def clean(self):
        # 校验用户名是否已经注册过
        user = User.objects.filter(username=self.cleaned_data.get('username')).first()
        if user:
            # 已经被注册
            pass
        else:
            # 没有被注册
            raise forms.ValidationError({'username': '用户名不存在!'})

forms.py的作用是检测和过滤从html页面传过来的值,并且返回对应的错误提示。得到期望的数据。

views.py

from django.contrib import auth
from django.contrib.auth.models import User
from django.http import HttpResponseRedirect
from django.shortcuts import render
from django.urls import reverse

from users.forms import UserForm, CheckUserForm


def register(request):
    if request.method == 'GET':
        return render(request, 'register.html')

    if request.method == 'POST':
        # 校验页面中传递的参数是否填写完整
        # username = request.POST.get('username')
        form = UserForm(request.POST)
        # is_valid():判断表单是否验证通过
        if form.is_valid():
            username = form.cleaned_data.get('username')
            password = form.cleaned_data.get('password')
            User.objects.create_user(username=username, password=password)
            # 实现跳转
            return HttpResponseRedirect(reverse('users:login'))
        else:
            return render(request, 'register.html', {'form': form})


def login(request):
    if request.method == 'GET':
        return render(request, 'login.html')
    if request.method == 'POST':
        # 表单验证,用户名和密码是否填写,校验用户名是否注册
        form = CheckUserForm(request.POST)
        if form.is_valid():
            # 校验用户名和密码,判断返回的对象是否为空,如果不为空,则为user对象
            user = auth.authenticate(
                username=form.cleaned_data['username'],
                password=form.cleaned_data['password']
            )
            if user:
                # 校验成功 用户名和密码是正确的 加密方法是Django自带的,所以用Django自带的方法
                auth.login(request, user)
                return HttpResponseRedirect(reverse('users:index'))
            else:
                # 密码错误
                return render(request, 'login.html', {'error': '密码错误'})
        else:
            return render(request, 'login.html', {'form': form})


def index(request):
    if request.method == 'GET':
        return render(request, 'index.html')


def logout(request):
    if request.method == 'GET':
        # 注销
        auth.logout(request)
        return HttpResponseRedirect(reverse('users:login'))

views.py负责响应请求,根据request中的不同请求完成不同的操作。

知识小结:
forms.py:

  • 1.新建UserForm类和CheckUserForm类,并且都继承forms.Form。
  • 2.使用forms中的CharField方法对页面中传来的数据进行过滤,并返回错误提示。
  • 3.clean方法,对用户信息进行查重。

views.py:

  • 1.实例化UserForm和CheckUserForm,调用is_valid()方法判断表单是否验证通过,通过为True,不通过为False且返回错误报告(return render(request, 'register.html', {'form': form}))
  • 2.auth.authenticate()校验用户名和密码,判断返回的对象是否为空,如果不为空,则为user对象
user = auth.authenticate(
                username=form.cleaned_data['username'],
                password=form.cleaned_data['password']
            )
  • 3.auth.login(request, user) 用户登入
    校验成功,用户名和密码是正确的则允许登入。request中user有匿名用户(AnonymousUser)变为指定用户。 加密方法是Django自带的,所以用Django自带的方法
  • 4.auth.logout(request) 注销
    将request中的user用户变为匿名用户(AnonymousUser),实现登出。
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,362评论 5 477
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,330评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,247评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,560评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,580评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,569评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,929评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,587评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,840评论 1 297
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,596评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,678评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,366评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,945评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,929评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,165评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 43,271评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,403评论 2 342

推荐阅读更多精彩内容