1、变化的部分

2、上代码:

{# 引用模板 #}
{% extends 'base.html' %}
{% load staticfiles %}
{% load comment_tags %}
{% load likes_tags %}{% block header_extends %}<link rel="stylesheet" href="{% static 'blog/blog.css' %}"><link rel="stylesheet" href="{% static 'fontawesome-free-5.5.0-web/css/all.min.css' %}">{#  处理公式  #}<script src='https://cdnjs.cloudflare.com/ajax/libs/mathjax/2.7.5/MathJax.js?config=TeX-MML-AM_CHTML'async></script><script type="text/javascript" src="{% static "ckeditor/ckeditor-init.js" %}"></script><script type="text/javascript" src="{% static "ckeditor/ckeditor/ckeditor.js" %}"></script>{% endblock %}{# 标题 #}
{% block title %}{{ blog.title }}
{% endblock %}{# 内容#}
{% block content %}<div class="container"><div class="row"><div class="col-10 offset-1"><ul class="blog-info-description"><h3>{{ blog.title }}</h3><li>作者:{{ blog.author }}</li>{# 时间过滤器让时间按照自己需要的格式过滤 #}<li>发布日期:{{ blog.created_time|date:"Y-m-d H:i:s" }}</li><li>分类:<a href="{% url 'blogs_with_type' blog.blog_type.pk %}">{{ blog.blog_type }}</a></li><li>阅读({{ blog.get_read_num }})</li><li>评论({% get_comment_count blog %})</li></ul><div class="blog-content">{{ blog.content|safe }}</div><div class="like"onclick="likeChange(this,'{% get_content_type blog %}',{{ blog.pk }})"><i class="far fa-thumbs-up {% get_like_status blog %}"></i><span class="liked-num">{% get_like_count blog %}</span><span>喜欢</span></div><p>上一篇:{% if previous_blog %}<a href="{% url 'blog_detail' previous_blog.pk %}">{{ previous_blog.title }}</a>{% else %}<span>没有了</span>{% endif %}</p><p>下一篇:{% if next_blog %}<a href="{% url 'blog_detail' next_blog.pk %}">{{ next_blog.title }}</a>{% else %}<span>没有了</span>{% endif %}</p></div></div><div class="row"><div class="col-10 offset-1"><div class="comment-area"><h3 class="comment-area-title">提交评论</h3>{% if user.is_authenticated %}<form id="comment-form" action="{% url 'update_comment' %}" method="post"style="overflow: hidden">{% csrf_token %}<label for="form-control">{{ user.username }},欢迎评论~</label><div id="reply-content-container" style="display: none;"><p id="reply_title">回复:</p><div id="reply-content"></div></div>{% get_comment_form blog as comment_form %}{% for field in comment_form %}{{ field }}{% endfor %}<span id="comment-error" class="text-danger float-left"></span><input type="submit" value="评论" class="btn btn-primary float-right"></form>{% else %}您尚未登录,登录之后方可评论{# 提交登录的时候带上从哪里访问的路径 #}<a class="btn btn-primary" href="{% url 'login' %}?from={{ request.get_full_path }}">登录</a><span> or </span><a class="btn-danger btn" href="{% url 'register' %}?from={{ request.get_full_path }}">注册</a>{% endif %}</div><div class="-comment-area"><h3 class="comment-area-title">评论列表</h3><div id="comment-list">{% get_comment_list blog as comments %}{% for comment in comments %}<div id="root-{{ comment.pk }}" class="comment"><span>{{ comment.user.username }}</span><span>{{ comment.comment_time|date:"Y-m-d H:i:s" }}</span><div id="comment-{{ comment.pk }}">{{ comment.text|safe }}</div>{# 点赞 #}<div class="like"onclick="likeChange(this,'{% get_content_type comment %}',{{ comment.pk }})"><i class="far fa-thumbs-up {% get_like_status comment %}"></i><span class="liked-num">{% get_like_count comment %}</span></div><a href="javascript:reply({{ comment.pk }})">回复</a>{% for reply in comment.root_comment.all %}<div class="reply"><span>{{ reply.user.username }}</span><span>{{ reply.comment_time|date:"Y-m-d H:i:s" }}</span><span>回复:</span><span>{{ reply.reply_to.username }}</span><div id="comment-{{ reply.pk }}">{{ reply.text|safe }}</div>{# 点赞 #}<div class="like"onclick="likeChange(this,'{% get_content_type reply %}',{{ reply.pk }})"><i class="far fa-thumbs-up {% get_like_status reply %}"></i><span class="liked-num">{% get_like_count reply %}</span></div><a href="javascript:reply({{ reply.pk }})">回复</a></div>{% endfor %}</div>{% empty %}<span id="no-comment">暂无评论</span>{% endfor %}</div></div></div></div></div>
{% endblock %}{% block js %}<script>// 处理点赞function likeChange(obj, content_type, object_id) {let is_like = obj.getElementsByClassName('active').length === 0;$.ajax({url: "{% url 'like_change' %}",type: 'GET',data: {content_type: content_type,object_id: object_id,is_like: is_like,},cache: false,success: function (data) {console.log(data);if (data['status'] === 'SUCCESS') {// 更新点赞状态
                        let element = $(obj.getElementsByClassName('fa-thumbs-up'));if (is_like) {element.addClass('active');} else {element.removeClass('active');}// 更新点赞数量
                        let like_num = $(obj.getElementsByClassName('liked-num'));like_num.text(data['liked_num']);} else {if (data['code'] === 400) {$('#login_model').modal('show');} else {alert(data['msg']);}}},error: function (xhr) {console.log(xhr);}});}// 处理回复function reply(reply_comment_id) {$('#reply_comment_id').val(reply_comment_id);let html = $('#comment-' + reply_comment_id).html();$('#reply-content').html(html);$('#reply-content-container').show();  // 显示内容// 滚动富文本编辑器
            $('html').animate({scrollTop: $('#comment-form').offset().top - 60}, 300, function () {// 动画执行完毕后执行的方法// 让富文本编辑器获得焦点
                CKEDITOR.instances['id_text'].focus();});}function numFormat(num) {return ('00' + num).substr(-2);}function timeFormat(timestamp) {let datetime = new Date(timestamp * 1000);let year = datetime.getFullYear();let month = numFormat(datetime.getMonth() + 1);let day = numFormat(datetime.getDate());let hour = numFormat(datetime.getHours());let minute = numFormat(datetime.getMinutes());let second = numFormat(datetime.getSeconds());return `${year}-${month}-${day} ${hour}:${minute}:${second}`}// 提交评论
        $('#comment-form').submit(function () {// 获取错误框
            let comment_error = $('#comment-error');comment_error.text('');// 更新数据到textarea
            CKEDITOR.instances['id_text'].updateElement();let comment_text = CKEDITOR.instances['id_text'].document.getBody().getText().trim();// 判断是否为空if (!(CKEDITOR.instances['id_text'].document.getBody().find('img')['$'].length !== 0 || comment_text !== '')) {// 显示错误信息
                comment_error.text('评论内容不能为空');return false;}//异步提交
            $.ajax({url: "{% url 'update_comment' %}",type: 'POST',data: $(this).serialize(),// 序列化表单值
                cache: false, // 关闭缓存
                success: function (data) {let reply_comment = $('#reply_comment_id');if (data['status'] === 'SUCCESS') {console.log(data);//  插入数据//  es6写法
                        let like_html = `<div class="like"onclick="likeChange(this,'${data['content_type']}',${data["pk"]})"><i class="far fa-thumbs-up"></i><span class="liked-num">0</span></div>`;if (reply_comment.val() === '0') {// 插入评论
                            let comment_html = `<div id="root-${data["pk"]}" class="comment"><span>${data["username"]}</span><span>${timeFormat(data["comment_time"])}</span><div id="comment-${data["pk"]}">${data["text"]}</div>
                                ${like_html}<a href="javascript:reply(${data["pk"]})">回复</a></div>`;
$('#comment-list').prepend(comment_html);} else {// 插入回复
let reply_html = `<div class="reply"><span>${data["username"]}</span><span>${timeFormat(data["comment_time"])}</span><span>回复:</span><span>${data["reply_to"]}</span><div id="comment-${data["pk"]}">${data["text"]}</div>
                                        ${like_html}<a href="javascript:reply(${data["pk"]})">回复</a></div>`;
                            $('#root-' + data['root_pk']).append(reply_html);}//  清空编辑框的内容
                        CKEDITOR.instances['id_text'].setData('');$('#reply-content-container').hide(); // 回复完隐藏掉要回复的内容
                        reply_comment.val('0'); // 将回复标志重置0
                        $('#no-comment').remove(); // 如果有没回复标志,清除掉5
                        comment_error.text('评论成功');} else {// 显示错误信息
                        comment_error.text(data['message'])}},error: function (xhr) {console.log(xhr);}});return false;});</script><script>$(".nav-blog").addClass("active").siblings().removeClass("active");</script>
{% endblock %}

blog下的blog_detail.html

from django.shortcuts import render, get_object_or_404
from django.core.paginator import Paginator
from django.conf import settings
from django.db.models import Count
from read_statistics.utils import read_statistics_once_read
from .models import Blog, BlogType# 分页部分公共代码
def blog_list_common_data(requests, blogs_all_list):paginator = Paginator(blogs_all_list, settings.EACH_PAGE_BLOGS_NUMBER)  # 第一个参数是全部内容,第二个是每页多少page_num = requests.GET.get('page', 1)  # 获取url的页面参数(get请求)page_of_blogs = paginator.get_page(page_num)  # 从分页器中获取指定页码的内容
current_page_num = page_of_blogs.number  # 获取当前页all_pages = paginator.num_pagesif all_pages < 5:page_range = list(range(max(current_page_num - 2, 1),min(all_pages + 1, current_page_num + 3)))  # 获取需要显示的页码 并且剔除不符合条件的页码else:if current_page_num <= 2:page_range = range(1, 5 + 1)elif current_page_num >= all_pages - 2:page_range = range(all_pages - 4, paginator.num_pages + 1)else:page_range = list(range(max(current_page_num - 2, 1),min(all_pages + 1, current_page_num + 3)))  # 获取需要显示的页码 并且剔除不符合条件的页码
blog_dates = Blog.objects.dates('created_time', 'month', order='DESC')blog_dates_dict = {}for blog_date in blog_dates:blog_count = Blog.objects.filter(created_time__year=blog_date.year, created_time__month=blog_date.month).count()blog_dates_dict = {blog_date: blog_count}return {'blogs': page_of_blogs.object_list,'page_of_blogs': page_of_blogs,'blog_types': BlogType.objects.annotate(blog_count=Count('blog')),  # 添加查询并添加字段'page_range': page_range,'blog_dates': blog_dates_dict}# 博客列表
def blog_list(requests):blogs_all_list = Blog.objects.all()  # 获取全部博客context = blog_list_common_data(requests, blogs_all_list)return render(requests, 'blog/blog_list.html', context)# 根据类型筛选
def blogs_with_type(requests, blog_type_pk):blog_type = get_object_or_404(BlogType, pk=blog_type_pk)blogs_all_list = Blog.objects.filter(blog_type=blog_type)  # 获取全部博客context = blog_list_common_data(requests, blogs_all_list)context['blog_type'] = blog_typereturn render(requests, 'blog/blog_with_type.html', context)# 根据日期筛选
def blogs_with_date(requests, year, month):blogs_all_list = Blog.objects.filter(created_time__year=year, created_time__month=month)  # 获取全部博客context = blog_list_common_data(requests, blogs_all_list)context['blogs_with_date'] = '{}年{}日'.format(year, month)return render(requests, 'blog/blog_with_date.html', context)# 博客详情
def blog_detail(requests, blog_pk):blog = get_object_or_404(Blog, pk=blog_pk)obj_key = read_statistics_once_read(requests, blog)context = {'blog': blog,'previous_blog': Blog.objects.filter(created_time__gt=blog.created_time).last(),'next_blog': Blog.objects.filter(created_time__lt=blog.created_time).first(),}response = render(requests, 'blog/blog_detail.html', context)response.set_cookie(obj_key, 'true')return response

blog下的views.py

"""
Django settings for myblog project.Generated by 'django-admin startproject' using Django 2.1.3.For more information on this file, see
https://docs.djangoproject.com/en/2.1/topics/settings/For the full list of settings and their values, see
https://docs.djangoproject.com/en/2.1/ref/settings/
"""import os# Build paths inside the project like this: os.path.join(BASE_DIR, ...)
BASE_DIR = os.path.dirname(os.path.dirname(os.path.abspath(__file__)))# Quick-start development settings - unsuitable for production
# See https://docs.djangoproject.com/en/2.1/howto/deployment/checklist/# SECURITY WARNING: keep the secret key used in production secret!
SECRET_KEY = 'ea+kzo_5k^6r7micfg@lar1(rfdc08@b4*+w5d11=0mp1p5ngr'# SECURITY WARNING: don't run with debug turned on in production!2.
DEBUG = TrueALLOWED_HOSTS = ['*']# Application definition

INSTALLED_APPS = ['django.contrib.admin','django.contrib.auth','django.contrib.contenttypes','django.contrib.sessions','django.contrib.messages','django.contrib.staticfiles','ckeditor','ckeditor_uploader','blog.apps.BlogConfig',  # 将自己创建的app添加到设置中'read_statistics.apps.ReadStatisticsConfig',  # 注册阅读统计app'comment.apps.CommentConfig',  # 注册评论'likes.apps.LikesConfig',  # 注册点赞'user.apps.UserConfig',  # 用户相关
]MIDDLEWARE = ['django.middleware.security.SecurityMiddleware','django.contrib.sessions.middleware.SessionMiddleware','django.middleware.common.CommonMiddleware','django.middleware.csrf.CsrfViewMiddleware','django.contrib.auth.middleware.AuthenticationMiddleware','django.contrib.messages.middleware.MessageMiddleware','django.middleware.clickjacking.XFrameOptionsMiddleware','blog.middleware.mymiddleware.My404',  # 添加自己的中间件
]ROOT_URLCONF = 'myblog.urls'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','user.context_processors.login_model_form',  # 自定义模板变量,直接用模板语言调用
            ],},},
]WSGI_APPLICATION = 'myblog.wsgi.application'# Database
# https://docs.djangoproject.com/en/2.1/ref/settings/#databases

DATABASES = {# 'default': {#     'ENGINE': 'django.db.backends.sqlite3',#     'NAME': os.path.join(BASE_DIR, 'db.sqlite3'),# }'default': {'ENGINE': 'django.db.backends.mysql','NAME': 'myblogs',  # 要连接的数据库,连接前需要创建好'USER': 'root',  # 连接数据库的用户名'PASSWORD': 'felixwang',  # 连接数据库的密码'HOST': '127.0.0.1',  # 连接主机,默认本级'PORT': 3306  # 端口 默认3306
    }
}# Password validation
# https://docs.djangoproject.com/en/2.1/ref/settings/#auth-password-validators

AUTH_PASSWORD_VALIDATORS = [{'NAME': 'django.contrib.auth.password_validation.UserAttributeSimilarityValidator',},{'NAME': 'django.contrib.auth.password_validation.MinimumLengthValidator',},{'NAME': 'django.contrib.auth.password_validation.CommonPasswordValidator',},{'NAME': 'django.contrib.auth.password_validation.NumericPasswordValidator',},
]# Internationalization
# https://docs.djangoproject.com/en/2.1/topics/i18n/# LANGUAGE_CODE = 'en-us'
# 语言
LANGUAGE_CODE = 'zh-hans'# TIME_ZONE = 'UTC'
# 时区
TIME_ZONE = 'Asia/Shanghai'USE_I18N = TrueUSE_L10N = True# 不考虑时区
USE_TZ = False# Static files (CSS, JavaScript, Images)
# https://docs.djangoproject.com/en/2.1/howto/static-files/

STATIC_URL = '/static/'
STATICFILES_DIRS = [os.path.join(BASE_DIR, "static")
]# media
MEDIA_URL = '/media/'
MEDIA_ROOT = os.path.join(BASE_DIR, 'media')# 配置ckeditor
CKEDITOR_UPLOAD_PATH = 'upload/'# 自定义参数
EACH_PAGE_BLOGS_NUMBER = 7# 设置数据库缓存
CACHES = {'default': {'BACKEND': 'django.core.cache.backends.db.DatabaseCache','LOCATION': 'my_read_num_cache_table',}
}# ckeditor 代码高亮,以及公式
CKEDITOR_CONFIGS = {'default': {'skin': 'moono','tabSpaces': 4,'mathJaxLib': 'https://cdnjs.cloudflare.com/ajax/libs/mathjax/2.7.5/MathJax.js?config=TeX-MML-AM_CHTML','toolbar': (['div', 'Source', '-', 'Save', 'NewPage', 'Preview', '-', 'Templates'],['Cut', 'Copy', 'Paste', 'PasteText', 'PasteFromWord', '-', 'Print','SpellChecker', 'Scayt'],['Undo', 'Redo', '-', 'Find', 'Replace', '-', 'SelectAll', 'RemoveFormat','-', 'Maximize', 'ShowBlocks', '-', "CodeSnippet", 'Mathjax', 'Subscript','Superscript'],['Form', 'Checkbox', 'Radio', 'TextField', 'Textarea', 'Select', 'Button','ImageButton', 'HiddenField'],['Bold', 'Italic', 'Underline', 'Strike', '-'],['NumberedList', 'BulletedList', '-', 'Outdent', 'Indent', 'Blockquote'],['JustifyLeft', 'JustifyCenter', 'JustifyRight', 'JustifyBlock'],['Link', 'Unlink', 'Anchor'],['Image', 'Flash', 'Table', 'HorizontalRule', 'Smiley', 'SpecialChar','PageBreak'], ['Styles', 'Format', 'Font', 'FontSize'],['TextColor', 'BGColor'],),'extraPlugins': ','.join(['codesnippet','mathjax','dialog','dialogui','lineutils',]),},'comment_ckeditor': {'toolbar': 'custom','toolbar_custom': [['Bold', 'Italic', 'Underline', 'Strike', 'Subscript', 'Superscript'],['TextColor', 'BGColor', 'RemoveFormat'],['NumberedList', 'BulletedList'],['Link', 'Unlink'],['Smiley', 'SpecialChar', 'Blockquote'],],'width': 'auto','height': '180','tabSpace': 4,'removePlugins': 'elementspath','resize_enabled': False,}
}

settings.py

"""myblog URL ConfigurationThe `urlpatterns` list routes URLs to views. For more information please see:https://docs.djangoproject.com/en/2.1/topics/http/urls/
Examples:
Function views1. Add an import:  from my_app import views2. Add a URL to urlpatterns:  path('', views.home, name='home')
Class-based views1. Add an import:  from other_app.views import Home2. Add a URL to urlpatterns:  path('', Home.as_view(), name='home')
Including another URLconf1. Import the include() function: from django.urls import include, path2. Add a URL to urlpatterns:  path('blog/', include('blog.urls'))
"""
from django.contrib import admin
from django.urls import path, include
from django.conf import settings
from django.conf.urls.static import static
from . import viewsurlpatterns = [path('', views.home, name='home'),  # 主页路径path('admin/', admin.site.urls),path('ckeditor', include('ckeditor_uploader.urls')),  # 配置上传urlpath('blog/', include('blog.urls')),  # 博客app路径path('comment/', include('comment.urls')),  # 博客app路径path('likes/', include('likes.urls')),  # 博客app路径path('user/', include('user.urls')),  # 博客app路径
]# 设置ckeditor的上传
urlpatterns += static(settings.MEDIA_URL, document_root=settings.MEDIA_ROOT)

myblog下的urls.py

# -*- coding: utf-8 -*-
# @Time    : 18-11-7 下午4:12
# @Author  : Felix Wangfrom django.shortcuts import render
from django.contrib.contenttypes.models import ContentType
from read_statistics.utils import get_seven_days_read_data, get_x_days_hot_data
from blog.models import Blogdef home(requests):blog_content_type = ContentType.objects.get_for_model(Blog)dates, read_nums = get_seven_days_read_data(blog_content_type)context = {'read_nums': read_nums,'dates': dates,'today_hot_data': get_x_days_hot_data(0),  # 获取今日热门'yesterday_hot_data': get_x_days_hot_data(1),  # 获取昨日热门'seven_days_hot_data': get_x_days_hot_data(7),  # 获取周热门'one_month_hot_data': get_x_days_hot_data(30),  # 获取月热门
    }return render(requests, 'home.html', context)

myblog下的views.py

{% load staticfiles %}<!DOCTYPE html>
<html lang="en">
<head><meta charset="UTF-8"><!-- 根据屏幕自动响应布局 --><meta name="viewport" content="width=device-width, initial-scale=1, shrink-to-fit=no"><title>{#  用来放标题  #}{% block title %}{% endblock %}</title>{# 加载css代码 #}<link rel="stylesheet" href="{% static 'bootstrap4.1/bootstrap.min.css' %}"><link rel="stylesheet" href="{% static 'css/base.css' %}">{% block header_extends %}{#    用来做头部扩展,如加载某些静态文件     #}{% endblock %}
</head>
<body>{# 导航栏 #}
<nav class="navbar navbar-expand-lg navbar-light bg-light sticky-top"><a class="navbar-brand" href="{% url 'home' %}">Felix Blog</a><button class="navbar-toggler" type="button" data-toggle="collapse" data-target="#navbarSupportedContent"aria-controls="navbarSupportedContent" aria-expanded="false" aria-label="Toggle navigation"><span class="navbar-toggler-icon"></span></button><div class="collapse navbar-collapse" id="navbarSupportedContent"><ul class="navbar-nav mr-auto"><li class="nav-item nav-home"><a href="{% url 'home' %}" class="nav-link">首页</a></li><li class="nav-item nav-blog"><a href="{% url 'blog_list' %}" class="nav-link">博客</a></li></ul><ul class="navbar-nav">{% if not user.is_authenticated %}<li class="nav-item"><a class="nav-link" href="{% url 'login' %}?from={{ request.get_full_path }}">登录</a></li><li class="nav-item"><a class="nav-link" href="{% url 'register' %}?from={{ request.get_full_path }}">注册</a></li>{% else %}<li class="nav-item dropdown"><a class="nav-link dropdown-toggle" href="#" id="navbarDropdown" role="button"data-toggle="dropdown"aria-haspopup="true" aria-expanded="false">{{ user.username }}</a><div class="dropdown-menu" aria-labelledby="navbarDropdown"><a class="dropdown-item" href="{% url 'user_info' %}">个人资料</a><div class="dropdown-divider"></div><a class="dropdown-item" href="{% url 'logout' %}?from={{ request.get_full_path }}">登出</a></div></li>{% endif %}</ul></div>
</nav>{# 用来放内容 #}
{% block content %}{% endblock %}<!-- Modal 登录模态框 -->
<div class="modal fade" id="login_model" tabindex="-1" role="dialog"aria-labelledby="exampleModalCenterTitle" aria-hidden="true"><div class="modal-dialog modal-dialog-centered" role="document"><div class="modal-content"><form id="login_model_form" action="" method="post"><div class="modal-header"><h5 class="modal-title">登录</h5><button type="button" class="close" data-dismiss="modal" aria-label="Close"><span aria-hidden="true">&times;</span></button></div><div class="modal-body">{% csrf_token %}{% for field in login_model_form %}<label for="{{ field.id_for_label }}">{{ field.label }}</label>{{ field }}{% endfor %}<span id="login_model_tip" class="text-danger"></span></div><div class="modal-footer"><button type="submit" class="btn btn-primary">登录</button><button type="button" class="btn btn-secondary" data-dismiss="modal">关闭</button></div></form></div></div>
</div>
<script>$('#login_model_form').submit(function (event) {$('#login_model_tip').text('');event.preventDefault(); // 阻止原事件提交
        $.ajax({url: '{% url 'login_for_model' %}',type: 'POST',data: $(this).serialize(),cache: false,success: function (data) {if (data['status'] === 'SUCCESS') {window.location.reload();} else {$('#login_model_tip').text('用户名或密码不正确')}}});})
</script>{# js代码放在后面可以增加性能 #}
<!-- jQuery first, then Popper.js, then Bootstrap JS -->
<script src="{% static 'js/jquery-3.3.1.min.js' %}"></script>
<script src="{% static 'bootstrap4.1/popper.min.js' %}"></script>
<script src="{% static 'bootstrap4.1/bootstrap.min.js' %}"></script>{# 导入资源建议放在js代码前 #}
{# 用来放js代码 #}
{% block js %}{% endblock %}</body>
</html>

base.html

# -*- coding: utf-8 -*-
# @Time    : 18-11-25 下午2:55
# @Author  : Felix Wangfrom .forms import LoginFormdef login_model_form(requests):return {'login_model_form': LoginForm()}

context_processors.py

# -*- coding: utf-8 -*-
# @Time    : 18-11-20 下午8:10
# @Author  : Felix Wangfrom django import forms
from django.contrib import auth
from django.contrib.auth.models import Userclass LoginForm(forms.Form):username = forms.CharField(label='用户名', required=True,widget=forms.TextInput(attrs={'class': 'form-control', 'placeholder': '请输入用户名'}))# widget指定input标签类型password = forms.CharField(label='密码',widget=forms.PasswordInput(attrs={'class': 'form-control', 'placeholder': '请输入密码'}))def clean(self):  # 验证数据username = self.cleaned_data['username']password = self.cleaned_data['password']user = auth.authenticate(username=username, password=password)if user is None:raise forms.ValidationError('用户名或密码错误')self.cleaned_data['user'] = user  # 将验证过的user放入clean_datareturn self.cleaned_dataclass RegisterForm(forms.Form):# 用户名字段username = forms.CharField(label='用户名',max_length=30,min_length=3,required=True,widget=forms.TextInput(attrs={'class': 'form-control', 'placeholder': '请输入用户名'}))# 邮箱字段email = forms.EmailField(label='邮箱',min_length=3,required=True,widget=forms.EmailInput(attrs={'class': 'form-control', 'placeholder': '请输入邮箱'}))# 密码字段password = forms.CharField(label='密码',min_length=6,required=True,widget=forms.PasswordInput(attrs={'class': 'form-control', 'placeholder': '请输入密码'}))# 再次输入密码password_again = forms.CharField(label='确认密码',min_length=6,required=True,widget=forms.PasswordInput(attrs={'class': 'form-control', 'placeholder': '请再输入一次密码'}))def clean_username(self):username = self.cleaned_data['username']if User.objects.filter(username=username).exists():raise forms.ValidationError('用户名已存在')return usernamedef clean_email(self):email = self.cleaned_data['email']if User.objects.filter(email=email).exists():raise forms.ValidationError('邮箱已存在')return emaildef clean_password_again(self):password = self.cleaned_data['password']password_again = self.cleaned_data['password_again']if password != password_again:raise forms.ValidationError('两次输入的密码不一致')return password_again

user下的forms.py

# -*- coding: utf-8 -*-
# @Time    : 18-11-4 下午5:22
# @Author  : Felix Wangfrom django.urls import path
from . import viewsurlpatterns = [path('login/', views.login, name='login'),  # 登录path('logout/', views.logout, name='logout'),  # 登录path('login_for_model/', views.login_for_model, name='login_for_model'),  # 登录path('register/', views.register, name='register'),  # 登录path('user_info/', views.user_info, name='user_info'),  # 登录
]

user下的urls.py

# -*- coding: utf-8 -*-
# @Time    : 18-11-7 下午4:12
# @Author  : Felix Wangfrom django.shortcuts import render, redirect
from django.http import JsonResponse
from django.contrib import auth
from django.contrib.auth.models import User
from django.urls import reverse
from .forms import LoginForm, RegisterFormdef login(requests):# 如果是form表单提交验证登录if requests.method == 'POST':login_form = LoginForm(requests.POST)if login_form.is_valid():  # 验证是否通过# 因为在form表单验证过了,所以不用自己再验证user = login_form.cleaned_data.get('user')auth.login(requests, user)return redirect(requests.GET.get('from', reverse('home')))else:login_form.add_error(None, '用户名或密码不正确')else:login_form = LoginForm()context = {'login_form': login_form,}return render(requests, 'user/login.html', context)def login_for_model(requests):login_form = LoginForm(requests.POST)# 如果是form表单提交验证登录if login_form.is_valid():  # 验证是否通过# 因为在form表单验证过了,所以不用自己再验证user = login_form.cleaned_data.get('user')auth.login(requests, user)data = {'status': 'SUCCESS',}else:data = {'status': 'ERROR',}return JsonResponse(data)def register(requests):if requests.method == 'POST':reg_form = RegisterForm(requests.POST)if reg_form.is_valid():username = reg_form.cleaned_data['username']email = reg_form.cleaned_data['email']password = reg_form.cleaned_data['password']# 创建用户user = User.objects.create_user(username=username, email=email, password=password)user.save()# 登录用户user = auth.authenticate(username=username, password=password)auth.login(requests, user)# 登录之后跳转return redirect(requests.GET.get('from', reverse('home')))else:reg_form = RegisterForm()context = {'reg_form': reg_form,}return render(requests, 'user/register.html', context)def logout(requests):auth.logout(requests)return redirect(requests.GET.get('from', reverse('home')))def user_info(requests):context = {}return render(requests, 'user/user_info.html', context)

views.py

3、解释

使用到了模板变量

如果需要在模板里直接使用模板变量,在app中新建context_processors.py,然后定义一个方法,返回一个字典。

然后在settings中配置。

实例:

# -*- coding: utf-8 -*-
# @Time    : 18-11-25 下午2:55
# @Author  : Felix Wangfrom .forms import LoginFormdef login_model_form(requests):return {'login_model_form': LoginForm()}

context_processors.py

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','user.context_processors.login_model_form',  # 自定义模板变量,直接用模板语言调用
            ],},},
]

settings中添加设置

{% login_model_form %}

模板中使用

转载于:https://www.cnblogs.com/felixwang2/p/10015978.html

搭建自己的博客(二十七):增加登录注册以及个人资料按钮相关推荐

  1. 如何改typecho主题头像_零成本搭建hexo个人博客(二)--修改主题

    hexo默认主题landscape 估计很多小伙伴在搭建部署hexo完成后,会看到上面的页面后,然后就感觉,啊?就这?这个页面也太难看了.有没有什么方法可以改一下这个页面的UI啥的,方法是有的: 第一 ...

  2. 从零开始免费搭建自己的博客(一)——本地搭建hexo框架

    ​ 本文是博客搭建系列文章第一篇,其他文章链接: 从零开始免费搭建自己的博客(一)--本地搭建 Hexo 框架 从零开始免费搭建自己的博客(二)--基于 GitHub pages 建站 从零开始免费搭 ...

  3. 从零开始免费搭建自己的博客(五)——Typora + PicGo + GitHub/Gitee图床

    ​ 本文是博客搭建系列文章第五篇,其他文章链接: 从零开始免费搭建自己的博客(一)--本地搭建 Hexo 框架 从零开始免费搭建自己的博客(二)--基于 GitHub pages 建站 从零开始免费搭 ...

  4. 5分钟 0元搭建个人独立博客网站(二)

    文/北妈 阅读本文需要 5.1分钟 一 接着第一篇 5分钟建立独立网站系列,戳这里:<5分钟 0元搭建个人独立博客网站(一)> 这个Hexo.GitPages系列,其实网络很多教程,但都不 ...

  5. 如何搭建一个独立博客——简明Github Pages与Hexo教程

    (?)[-] http://cnfeat.com/2014/05/10/2014-05-11-how-to-build-a-blog/ 摘要:这是一篇很详尽的独立博客搭建教程,里面介绍了域名注册.DN ...

  6. 【Microsoft Azure 的1024种玩法】六十.通过Azure Virtual Machines快速搭建个人Ghost博客系统

    [简介] Ghost 是一套基于Node.js 语言开发构建的开源博客系统,它的整体架构为前端管理系统基于Ember.js, 后端的模板引擎采用的handlebars, 数据库是基于MySQL的,本篇 ...

  7. github+hexo搭建自己的博客网站(六)进阶配置(搜索引擎收录,优化你的url,添加RSS)...

    详细的可以查看hexo博客的演示:https://saucxs.github.io/ 绑定了域名: http://www.chengxinsong.cn hexo+github博客网站源码(可以clo ...

  8. 阿里云ECS服务器搭建wordpress个人博客网站【详细图文教程】

    阿里云ECS服务器搭建wordpress个人博客网站[小白专用的图文教程] 在阿里云上搭建使用个人博客主要分为以下几个步骤: 1.购买阿里云ECS主机 2.购买域名 3.申请备案 4.环境配置 5.安 ...

  9. 个人博客代码_Solo小众开源博客系统:手把手教你搭建自己的博客系统

    博客是很多程序员朋友的最爱,通过博客可以梳理自己的技术体系,将自己日常开发过程中的技术心得或者对于一个技术问题的解决思路记录下来,对于以后碰到此类问题有一个参考的作用.同时,博客系统的内容是对外的,我 ...

最新文章

  1. 【系列索引】结合项目实例 回顾传统设计模式 打造属于自己的模式类系列
  2. 1行代码消除PyTorch的CUDA内存溢出报错,这个GitHub项目刚发布就揽星600+
  3. 游戏设计艺术(第3版) The Art of Game Design
  4. 如何从失焦的图像中恢复景深并将图像变清晰?
  5. msdn 上的关于delegate和event的一个例子,理解后加点评论
  6. 如何在两个目录中删除其中一个目录中同名文件
  7. suse 新增用户oracle,在SUSE Linux中让其他用户能运行Oracle命令
  8. 日常生活 -- 嵌入式再学习前言
  9. MySQL之深入解析一条SQL的执行流程
  10. php is_dir 判断是否存在这目录
  11. mysql支持的时区列表_mysql按天分组支持时区
  12. vue.js框架搭建
  13. linux RPM包安装、更新、删除等操作命令简明总结, 如何查看yum安装的软件路径 ?...
  14. ndk-build生成.so
  15. 《金字塔原理》学习笔记 | 第4篇—演示的逻辑
  16. C++ 冰河木马的实现
  17. R语言逻辑回归、ROC曲线和十折交叉验证
  18. excel复制公式递增_快速向下复制Excel公式
  19. 【微服务】微服务熔断器--Hystrix
  20. 手摸手教学之:梳理数据指标体系

热门文章

  1. Team Foundation Server 2010 安装、部署与配置(一):安装计划 .
  2. HttpModule的认识
  3. 带绿色箭头指示的滑动门DIV效果
  4. 通过模板页master page和主题theme来实现网站的风格切换
  5. Leetcode --9
  6. 计算机网络——OSI参考模型和TCP/IP协议
  7. 自然语言处理 —— 2.2 使用词嵌入
  8. java通过匹配合并数据(数据预处理)
  9. pytorch自动求梯度—详解
  10. The Use Case Definition in UML