/UseDjango-to-create-WebApplication

use the Django module to create a "Learning Log" web application

Primary LanguagePython

用 Django 创建“学习笔记”Web应用程序

编写一个名为“学习笔记”的Web应用程序,让用户能够记录感兴趣的主题,并在学习每个主题的过程中添加日志条目。“学习笔记”的主页对这个网站进行描述,并邀请用户注册或登录。用户登录后,可以创建新主题、添加新条目以及阅读既有的条目。

1. 使用Django

1.1 建立项目

1.1.1 创建项目

anaconda创建虚拟环境 useDjango

在useDjango中安装:

  • python --version = 3.7.2

  • Django --version = 2.2.0

在当前环境下执行如下命令:

django-admin startproject learing_log .

创建 learning_log 文件夹以及manage.py文件

  • manage.py —— 接受命令并将其交给Django的相关部分去运行
  • learning_log/settings.py —— 设置文件。指定Django如何与你的系统交互以及如何管理项目
  • learning_log/urls.py —— 告诉Django应该创建哪些网页来响应浏览器请求
  • learning_log/wsgi.py —— 帮助Django提供它创建的文件 ( wsgi —— web server gateway interface )

1.1.2 创建数据库

命令行中输入:

python manage.py migrate

新建一个数据库,让Django确保数据库与项目的当前状态匹配

1.1.3 查看项目

核实Django正确地创建了项目:

命令行输入:

python manage.py runserver

查看项目的状态

可以看到Django启动了一个名为 development server 的服务器,使能够查看系统中的项目,了解其工作情况。如果在浏览器中输入URL以请求页面,该Django服务器将进行响应:生成合适的页面,并将其发送给浏览器。

URL http://127.0.0.1:8000/ 表明项目将在计算机(即127.0.0.1)的端口8000上侦听当前系统发出的请求。

当要关闭这个服务器时,可切换到执行命令runserver 时所在的终端窗口,再退出。

1.2 创建应用程序

在保持runserver运行的前提下,打开新终端窗口,在manage.py所在目录下激活虚拟环境

命令行中输入:

python manage.py startapp learning_logs

让Django搭建创建应用程序所需的基础设施

此时在当前目录下创建learning_logs文件夹,该文件夹下包含:

  • learning_logs/admin.py
  • learning_logs/apps.py
  • learning_logs/models.py —— 定义要在应用程序中管理的数据
  • learning_logs/tests.py
  • learning_logs/views.py
  • learning_logs/migrations

1.2.1 定义模型

打开 learning_logs/models.py,初始文件如下:

from django.db import models

# Create your models here.

learning_logs/models.py导入models模块,并让我们创建模型,告诉Django如何处理应用程序中存储的数据。

建立用户将存储的主题模型如下:

from django.db import models

# Create your models here.
class Topic(models.Model):
    '''用户学习的主题'''
    text = models.CharField(max_length=200)
    date_added = models.DateTimeField(auto_now_add=True)

    def __str__(self):
        '''返回模型的字符串表示'''
        return self.text

在Topic类中添加两个属性:text(type:CharField) 和 date_added(type:DateTimeField),分别用以存储主题名和创建新主题时当前的日期和时间

Django通过调用方法__str__()来显示模型的简单表示

1.2.2 激活模型

让Django将前述应用程序包含到项目中激活模型后才能使用这些模型

在learning_log/settings.py文件的INSTALLED_APPS列表中添加个人创建的learning_logs应用程序,最终列表如下所示:

INSTALLED_APPS = [
    # my applications
    'learning_logs',

    # default applications
    'django.contrib.admin',
    'django.contrib.auth',
    'django.contrib.contenttypes',
    'django.contrib.sessions',
    'django.contrib.messages',
    'django.contrib.staticfiles',
]

注意:将自己创建的应用程序放在默认应用程序之前,覆盖默认应用程序的行为

接下来需让Django修改数据库,使其能够存储与模型Topic相关的信息。

在命令行中输入:

python manage.py makemigrations learning_logs

命令makemigrations让Django确定该如何修改数据库,使其能够存储与前面定义的新模型相关联的数据。输出表明Django创建了一个名为learning_logs/migrations/0001_initial.py的迁移文件,这个文件将在数据库中为模型Topic创建一个表。

下面应用这种迁移,让Django替我们修改数据库,在命令行中输入:

python manage.py migrate

综上,此后每当需要修改“学习笔记”管理的数据时,都采取如下三个步骤:

修改learning_logs/models.py,对learning_logs调用makemigrations,以及让Django迁移项目。

1.2.3 Django管理网站

Django提供的管理网站(admin site)让你能够轻松处理模型。管理网站只有网站管理员可以使用。1.2.3就建立管理网站,并通过它使用模型Topic来添加一些主题。

1.2.3.1 创建超级账户

在命令行中输入:

python manage.py createsuperuser

超级账户管理员创建如下:

Username: django_admin
Email address: 123456@gmail.com
Password: learning_log123
Password (again): learning_log123
1.2.3.2 向管理网站注册模型

打开 learning_logs/admin.py,初始文件如下:

from django.contrib import admin

# Register your models here.

为向管理网站注册Topic,在 learning_logs/admin.py 中首先导入要注册的模型。admin.site.register()让Django通过管理网站管理模型

from django.contrib import admin

# Register your models here.
from .models import Topic

admin.site.register(Topic)

现在,使用超级用户账户访问管理网站:访问 http://127.0.0.1:8000/admin/ ,登录超级账户。此时你能够添加和修改用户和用户组,还可管理与刚才定义的模型Topic相关的数据。

1.2.3.3 添加主题

向管理网站注册Topic 后,来添加第一个主题。为此,单击Topics进入主题页面。单击Add,出现用于添加新主题的表单。输入内容后Save保存。

1.2.4 定义模型Entry

用户需要在学习笔记中能够添加条目,定义多对一关系(一个特定主题有多个条目与之关联)的模型Entry,并放入learning_logs/models.py中。

class Entry(models.Model):
    '''学到某个主题的具体知识'''
    topic = models.ForeignKey(Topic, on_delete=models.CASCADE)
    text = models.TextField()
    date_added = models.DateTimeField(auto_now_add=True)

    class Meta:
        verbose_name_plural = "entries"

    def __str__(self):
        '''返回模型的字符串表示'''
        return f"{self.text[:50]}..."

像Topic 一样,Entry 也继承了Django基类Model。

第一个属性topic是个ForeignKey实例。foreign key指向数据库中的另一条记录,这里是将每个条目关联到特定主题。创建每个主题时,都分配了一个键(ID)。需要在两项数据之间建立联系时,Django使用与每项信息相关联的键。我们稍后将根据这些联系获取与特定主题相关联的所有条目。

实参on_delete=models.CASCADE让Django在删除主题的同时删除所有与之相关联的条目(级联删除cascading delete)。

属性text是TextField实例。

属性date_added让我们能够按创建顺序呈现条目,并在每个条目旁边放置时间戳。

在Entry类中嵌套了Meta类。Meta存储用于管理模型的额外信息。在这里,它让我们能够设置一个特殊属性,让Django在需要时使用Entries来表示多个条目。如果没有这个类,Django将使用Entrys来表示多个条目。

方法__str__()告诉Django,呈现条目时应显示哪些信息。

1.2.5 迁移模型Entry

在learning_logs/models.py中添加新模型后,如 1.2.2 中所说,需要再次迁移数据库。

命令行中输入:

python manage.py makemigrations learning_logs

再输入:

python manage.py migrate

完成数据库的迁移并查看输出

此时生成一个新的迁移文件learning_logs/migrations/0002_entry.py,它告诉Django如何修改数据库,使其能够存储与模型Entry相关的信息

1.2.6 向管理网站注册Entry

还需要注册模型Entry。为此learning_logs/admin.py文件中添加如下代码:

admin.site.register(Entry)

此时再次进入服务器 http://127.0.0.1:8000/admin/ ,将看到Learning_logs下列出了Entries。

单击Entries的Add链接,或者单击Entries再选择Add entry,将看到一个下拉列表,供你选择要为哪个主题创建条目,以及一个用于输入条目的文本框。

1.2.7 Django Shell

在活动状态的虚拟环境中执行时,命令python manage.py shell启动shell,就能够探索存储在项目数据库中的数据。

如下方代码导入了模块learning_logs.models中的模型Topic,再使用方法Topic.objects.all()获取模型Topic 的所有实例,这将返回一个称为查询集(queryset)的列表。此时可以像遍历列表一样遍历查询集。将返回的查询集存储在topics中,再打印每个主题的id属性和字符串表示。知道主题对象的ID后,就可使用方法Topic.objects.get()获取该对象并查看其属性。我们还可以查看与主题相关联的条目。前面给模型Entry定义了属性topic。这是一个ForeignKey,将条目与主题关联起来。利用这种关联,Django能够获取与特定主题相关联的所有条目。

# python manage.py shell

>>> from learning_logs.models import Topic
>>> Topic.objects.all()
<QuerySet [<Topic: Chess>, <Topic: Rock Climbing>]>
>>> topics = Topic.objects.all()
>>> for topic in topics:
...     print(topic.id, topic)
...
1 Chess
2 Rock Climbing
>>> t = Topic.objects.get(id=1)
>>> t.text
'Chess'
>>> t.date_added
datetime.datetime(2022, 6, 22, 11, 36, 53, 999755, tzinfo=<UTC>)
>>> t.entry_set.all()
<QuerySet [<Entry: The opening is the first part of the game, roughly...>, <Entry: In the opening phase of the game, it is important t...>]>
>>> 

要通过外键关系获取数据,可使用相关模型的小写名称、下划线和单词set。例如,假设有模型Pizza 和Topping,而Topping通过一个外键关联到Pizza。如果有一个名为my_pizza的Pizza对象,就可使用代码my_pizza.topping_set.all() 来获取这张比萨的所有配料。

编写用户可请求的页面时,我们将使用这种语法。确认代码能获取所需的数据时,shell很有帮助。如果代码在shell中的行为符合预期,那么它们在项目文件中也能正确地工作。如果代码引发了错误或获取的数据不符合预期,那么在简单的shell环境中排除故障要比在生成页面的文件中排除故障容易得多。我们不会太多地使用shell,但应继续使用它来熟悉对存储在项目中的数据进行访问的Django语法。

注意:每次修改模型后,都需要重启shell,这样才能看到修改的效果。要退出shell会话,linux按Ctrl+D,Windows按Ctrl+Z,再按Enter。

1.3 创建学习笔记主页

使用Django创建页面的过程分三个阶段:定义URL,编写视图和编写模板。URL模式描述了URL是如何设计的,让Django知道如何将浏览器请求与网站URL匹配,以确定返回哪个页面。

每个URL都被映射到特定的视图——视图函数获取并处理页面所需的数据。视图函数通常使用模板来渲染页面,而模板定义页面的总体结构。

1.3.1 映射URL

当前基础URL http://127.0.0.1:8000/ 返回默认的Django网站,我们将其修改为映射到“学习笔记”的主页。

打开learning_log/urls.py,其代码如下:

"""learning_log URL Configuration

The `urlpatterns` list routes URLs to views. For more information please see:
    https://docs.djangoproject.com/en/2.2/topics/http/urls/
Examples:
Function views
    1. Add an import:  from my_app import views
    2. Add a URL to urlpatterns:  path('', views.home, name='home')
Class-based views
    1. Add an import:  from other_app.views import Home
    2. Add a URL to urlpatterns:  path('', Home.as_view(), name='home')
Including another URLconf
    1. Import the include() function: from django.urls import include, path
    2. Add a URL to urlpatterns:  path('blog/', include('blog.urls'))
"""
from django.contrib import admin
from django.urls import path

urlpatterns = [
    path('admin/', admin.site.urls),
]

导入了一个模块和一个函数,以便对管理网站的URL进行管理。

主体定义变量urlpatterns。在这个针对整个项目的learning_log/urls.py文件中,变量urlpatterns包含项目中应用程序的URL。

模块admin.site.urls定义了可在管理网站中请求的所有URL。

修改urlpatterns的值如下,使其包含learning_logs的URL

from django.contrib import admin
from django.urls import path, include

urlpatterns = [
    path('admin/', admin.site.urls),
    path('', include('learning_logs.urls')),
]

默认的urls.py包含在文件夹learning_log中,现在需要在文件夹learning_logs中再创建一个urls.py文件。再在learning_logs/urls.py中输入如下代码:

'''define URL patterns for learning_logs'''

from django.urls import path

from . import views


app_name = 'learning_logs'
urlpatterns = [
    # home page
    path('', views.index, name='index'),
]

导入函数path ,使用它将URL映射到视图。导入模块views(在learning_logs文件夹下)。

变量app_name让Django能够将learning_logs/urls.py文件同项目内其他应用程序中的同名文件区分开来。在这个模块中,变量urlpatterns是一个列表,包含可在应用程序learning_logs中请求的页面。实际的URL模式是对函数path()的调用,这个函数接受三个实参。第一个是字符串,帮助Django正确地路由(route)请求。收到请求的URL后,Django力图将请求路由给一个视图。为此,它搜索所有的URL模式,找到与当前请求匹配的那个。Django忽略项目的基础URL http://17.0.0.1:8000/ ,因此空字符串与基础URL匹配。其他URL都与这个模式不匹配。如果请求的URL与任何既有的URL模式都不匹配,Django将返回一个错误页面。path() 的第二个实参指定了要调用learning_logs/view.py中的哪个函数。请求的URL与前述正则表达式匹配时,Django将调用learning_logs/view.py中的函数index()。第三个实参将这个URL模式的名称指定为index,让我们能够在代码的其他地方引用它。每当需要提供到这个主页的链接时,都将使用这个名称,而不编写URL。

1.3.2 编写视图

视图函数接受请求中的信息,准备好生成页面所需的数据,再将这些数据发送给浏览器——这通常是使用定义页面外观的模板实现的。

learning_logs/views.py是执行命令python manage.py startapp时自动生成的,当前代码如下:

from django.shortcuts import render

# Create your views here.

当前,这个文件只导入了函数render() ,它根据视图提供的数据渲染响应。

我们在这个文件中添加为主页编写视图的代码:

from django.shortcuts import render

# Create your views here.
def index(request):
    '''main page of the learning log project'''
    return render(request, 'learning_logs/index.html')

URL请求与刚才定义的模式匹配时,Django将在文件learning_logs/views.py中查找函数index(),再将对象request传递给这个视图函数。这里不需要处理任何数据,因此这个函数只包含调用render()的代码。这里向函数render()提供了两个实参:对象request以及一个可用于创建页面的模板。

1.3.3 编写模板

1.3.2中最后已经提供了一个可用于创建页面的模板。现在来编写这个模板。

模板定义页面的外观,而每当页面被请求时,Django将填入相关的数据。模板让你 能够访问视图提供的任何数据。

在文件夹learning_logs中新建一个文件夹,并将其命名为templates。在文件夹templates中,再新建一个文件夹,并将其命名为learning_logs,则建立了Django能够明确解读的结构,即便项目很大、包含很多应用程序亦如此。

在最里面的文件夹learning_logs中,新建一个文件,并将其命名为index.html(这个文件的路径为learning_logs/templates/learning_logs/index.html),再在其中编写如下代码:

<p>Learning Log</p>

<p>Learning Log helps you keep track of your learning, for any topic you are learning about.</p>

标签

标识段落。标签

指出段落的开头位置,而标签

指出段落的结束位置。这里定义了两个段落:第一个充当标题,第二个阐述了用户可使用“学习笔记”来做什么。

现在,如果请求这个项目的基础URL http://127.0.0.1:8000/ ,将看到刚才创建的页面,而不是默认的Django页面。Django接受请求的URL,发现该URL与模式''匹配,因此调用函数views.index()。这将使用index.html包含的模板来渲染页面。

1.4 创建其他页面

将创建两个显示数据的页面,其中一个列出所有的主题,另一个显示特定主题的所有条目。对于每个页面,我们都将指定URL模式、编写一个视图函数并编写一个模板。但这样做之前,我们先创建一个父模板,项目中的其他模板都将继承它。

1.4.1 模板继承

创建网站时,一些通用元素几乎会在所有页面中出现。在这种情况下,可编写一个包含通用元素的父模板,并让每个页面都继承这个模板,而不必在每个页面中重复定义这些通用元素。

1.4.1.1 父模板

创建learning_logs\templates\learning_logs\base.html作为父模板,其他模板可继承它。其代码如下:

<p>
    <a href="{% url 'learning_logs:index' %}">Learning Log</a>
</p>

{% block content %}{% endblock content %}

第一部分创建一个包含项目名的段落,该段落也是到主页的链接。为创建链接,使用了一个模板标签 ,它是用花括号和百分号{% %}表示的。模板标签是一小段代码,生成要在页面中显示的信息。这里的模板标签生成一个URL,该URL与在learning_logs/urls.py中定义的名为index的URL模式匹配。在本例中,learning_logs是一个命名空间,而index是该命名空间中一个名称独特的URL模式。这个命名空间来自在文件learning_logs/urls.py中赋给app_name的值。

在简单的HTML页面中,链接是使用锚标签<a>定义的:

<a href="link_url">link text</a>

通过使用模板标签来生成URL,能很容易地确保链接是最新的:只需修改urls.py中的URL模式,Django就会在页面下次被请求时自动插入修改后的URL。在本项目中,每个页面都将继承base.html,因此从现在开始,每个页面都包含到主页的链接。

接着插入了一对块标签。这个块名为content,是一个占位符,其中包含的信息由子模板指定。子模板并非必须定义父模板中的每个块,因此在父模板中,可使用任意多个块来预留空间,而子模板可根据需要定义相应数量的块。

1.4.1.2 子模版

现在重写learning_logs\templates\learning_logs\index.html,使其继承learning_logs\templates\learning_logs\base.html。

修改代码如下:

{% extends "learning_logs/base.html" %}

{% block content %}
    <p>Learning Log helps you keep track of your learning, for any topic you are learning about.</p>
{% endblock content %}

子模板的第一行必须包含标签{% extends %},让Django知道它继承了哪个父模板。文件base.html位于learning_logs中,因此父模板路径中包含learning_logs。这行代码导入模板base.html的所有内容,让index.html能够指定要在content块预留的空间中添加的内容。

插入了一个名为content的{% block %}标签,以定义content块。不是从父模板继承的内容都包含在content块中,在这里是一个描述项目“学习笔记”的段落。

使用标签{% endblock content %}指出了内容定义的结束位置。在标签{% endblock %}中,并非必须指定块名,但如果模板包含多个块,指定块名有助于确定结束的是哪个块。

模板继承的优点:在子模板中,只需包含当前页面特有的内容。

1.4.2 显示所有主题的页面

显示用户创建的所有主题,是一个需要使用数据的页面。

1.4.2.1 URL模式

首先,定义显示所有主题的页面的URL。我们通常使用一个简单的URL片段来指出页面显示的信息,这里使用单词topics,因此URL http://localhost:8000/topics/ 将返回显示所有主题的页面。

在learning_logs/urls.py中修改代码如下:

'''define URL patterns for learning_logs'''

from django.urls import path

from . import views


app_name = 'learning_logs'
urlpatterns = [
    # home page
    path('', views.index, name='index'),

    # page that shows all topics
    path('topics/', views.topics, name='topics'),
]

这里在用于主页URL的字符串参数中添加了topics/。Django检查请求的URL时,这个模式与如下URL匹配:基础URL后面跟着topics。可在末尾包含斜杠,也可省略,但单词topics后面不能有任何东西,否则就与该模式不匹配。URL与该模式匹配的请求都将交给views.py中的函数topics()处理。

1.4.2.2 视图

函数topics()需要从数据库中获取数据并提交给模板,修改learning_logs/view.py如下:

from django.shortcuts import render

from .models import Topic

# Create your views here.
def index(request):
    '''show home page'''
    return render(request, 'learning_logs/index.html')

def topics(request):
    '''show all topics'''
    topics = Topic.objects.order_by('date_added')
    context = {'topics': topics}
    return render(request, 'learning_logs/topics.html', context)

首先导入与所需数据相关联的模型。函数topics()包含一个形参:Django从服务器那里收到的request对象。

查询数据库——请求提供Topic对象,并根据属性date_added进行排序。返回的查询集被存储在topics中。

定义一个将发送给模板的上下文。上下文是一个字典,其中的键是将在模板中用来访问数据的名称,而值是要发送给模板的数据。这里只有一个键值对,包含一组将在页面中显示的主题。

创建使用数据的页面时,除了对象request和模板的路径外,还将变量context传递给render()

1.4.2.3 模板

显示所有主题的页面的模板接受字典context,以便使用topics()提供的数据。在learning_logs/templates/learning_logs中创建topics.html,在这个模板中显示主题。其代码如下:

{% extends "learning_logs/base.html" %}

{% block content %}

    <p>Topics</p>

    <ul>
        {% for topic in topics %}
            <li>{{ topic }}</li>
        {% empty %}
            <li>No topics have been added yet.</li>
        {% endfor %}
    </ul>

{% endblock content %}

首先使用标签{% extends %}来继承base.html,再开始定义content块。这个页面的主体是一个项目列表,其中列出了用户输入的主题。在标准HTML中,项目列表称为无序列表,用标签<ul></ul>表示。包含所有主题的项目列表始于<ul>

使用一个相当于for循环的模板标签,它遍历字典context中的列表topics。模板中使用的代码与Python代码存在一些重要差别:Python使用缩进来指出哪些代码行是for循环的组成部分;而在模板中,每个for循环都必须使用{% endfor %}标签来显式地指出其结束位置。因此在模板中,循环类似于下面这样:

{% for item in list %}
    do something with each item
{% endfor %}

在循环中,要将每个主题转换为项目列表中的一项。要在模板中打印变量,需要将变量名用双花括号括起。这些花括号不会出现在页面中,只是用于告诉Django我们使用了一个模板变量。因此每次循环时,代码{{ topic }}都被替换为topic的当前值。HTML标签<li></li>表示一个项目列表项。在标签对<ul></ul>内部,位于标签<li></li>之间的内容都是一个项目列表项。

使用模板标签{% empty %},它告诉Django在列表topics为空时该如何办。这里是打印一条消息,告诉用户还没有添加任何主题。最后两行分别结束for循环和项目列表。

下面需要修改父模板,使其包含到显示所有主题的页面的链接。修改learning_logs\templates\learning_logs\base.html代码如下:

<p>
    <a href="{% url 'learning_logs:index' %}">Learning Log</a> - 
    <a href="{% url 'learning_logs:topics' %}">Topics</a>
</p>

{% block content %}{% endblock content %}

在到主页的链接后面添加一个连字符,再添加一个到显示所有主题的页面的链接——使用的也是模板标签{% url %}。这行让Django生成一个链接,它与learning_logs/urls.py中名为topics的URL模式匹配。

现在如果刷新浏览器中的主页,将看到链接Topics。

1.4.3 显示特定主题的页面

接下来,需要创建一个专注于特定主题的页面,它显示该主题的名称以及所有条目。

此外,还将修改显示所有主题的页面,让每个项目列表项都变为到相应主题页面的链接。

1.4.3.1 URL模式

显示特定主题的页面的URL模式与前面的所有URL模式都稍有不同,因为它使用主题的id属性来指出请求的是哪个主题

在learning_logs/urls.py中添加与这个URL匹配的模式

urlpatterns = [
    # home page
    path('', views.index, name='index'),

    # page that shows all topics
    path('topics/', views.topics, name='topics'),

    # detail page for a single topic
    path('topics/<int:topic_id>/', views.topic, name='topic'),
]

这个URL模式中的字符串topics/<int:topic_id>/。这个字符串的第一部分让Django查找在基础URL后包含单词topics的URL,第二部分/<int:topic_id>/与包含在两个斜杠内的整数匹配,并将这个整数存储在一个名为topic_id的实参中。发现URL与这个模式匹配时,Django将调用视图函数topic(),并将存储在topic_id中的值作为实参传递给它。在这个函数中,将使用topic_id的值来获取相应的主题

1.4.3.2 视图

函数topic()需要从数据库中获取指定的主题以及与之相关联的所有条目,在learning_logs\views.py中定义函数topic():

def topic(request, topic_id):
    '''show a single topic and all its entries'''
    topic = Topic.objects.get(id=topic_id)
    entries = topic.entry_set.order_by('-date_added')
    context = {'topic': topic, 'entries': entries}
    return render(request, 'learning_logs/topic.html', context)

这是除request对象外还包含另一个形参的视图函数。这个函数接受表达式/<int:topic_id>/捕获的值,并将其存储到topic_id中。

使用get()来获取指定的主题,就像前面在Django shell中所做的那样。

获取与该主题相关联的条目,并根据date_added进行排序。date_added前面的减号指定按降序排列,即先显示最近的条目。

将主题和条目都存储在字典context中,再将这个字典发送给模板topic.html

注意:get和entry_set处的代码称为查询,因为它们向数据库查询了特定的信息。比起先编写视图和模板、再在浏览器中检查结果,在shell中执行代码可更快获得反馈。

1.4.3.3 模板

这个模板需要显示主题的名称和条目的内容。如果当前主题不包含任何条目,也需向用户指出:

创建learning_logs\templates\learning_logs\topic.html,写入特定主题的模板:

{% extends "learning_logs/base.html" %}

{% block content %}

    <p>Topic: {{ topic }}</p>

    <p>Entries:</p>

    <ul>
    {% for entry in entries %}
        <li>
            <p>{{ entry.date_added|date:'M d, Y H:i' }}</p>
            <p>{{ entry.text|linebreaks }}</p>
        </li>
    {% empty %}
        <li>There are no entries for this topic yet.</li>
    {% endfor %}
    </ul>

{% endblock content %}

首先继承base.html。

接下来,显示当前的主题,它存储在模板变量{{ topic }}中。变量topic包含在字典context中,所以可以使用。

定义一个显示每个条目的项目列表,并像前面显示所有主题一样遍历条目。每个项目列表项都将列出两项信息:条目的时间戳和完整的文本。

为列出时间戳,我们显示属性date_added的值。在Django模板中,竖线表示模板过滤器,即对模板变量的值进行修改的函数。过滤器date: 'M d,Y H:i'以类似于这样的格式显示时间戳:January 1, 2018 23:00

接下来的一行显示text 的完整值,而不仅仅是前50字符。过滤器linebreaks将包含换行符的长条目转换为浏览器能够理解的格式,以免显示为不间断的文本块。

使用模板标签{% empty %}打印一条消息,告诉用户当前主题还没有条目。

1.4.3.4 将显示所有主题的页面中的主题设置为链接

在浏览器中查看显示特定主题的页面前,需要修改模板learning_logs\templates\learning_logs\topics.html,让每个主题都链接到相应的页面,其代码修改如下:

{% extends "learning_logs/base.html" %}

{% block content %}

    <p>Topics</p>

    <ul>
        {% for topic in topics %}
            <li>
                <a href="{% url 'learning_logs:topic' topic.id %}">{{ topic }}</a>
            </li>
        {% empty %}
            <li>No topics have been added yet.</li>
        {% endfor %}
    </ul>

{% endblock content %}

使用模板标签url根据learning_logs中名为topic的URL模式生成了合适的链接。这个URL模式要求提供实参topic_id,因此在模板标签url中添加了属性'topic.id'。现在,主题列表中的每个主题都是链接了,链接到显示相应主题的页面,如 http://127.0.0.1:8000/topics/1/

如果现在刷新显示所有主题的页面,再单击其中的一个主题,就可看到相应的特定主题

注意:topics.html中添加的是属性topic.id而非实参topic_id。topic.id和topic_id之间存在细微而重要的差别。表达式topic.id检查主题并获取其ID值,而在代码中,变量topic_id是指向该ID的引用。

2. 用户账户的创建和设置

2.1 让用户输入数据

在建立用于创建用户账户的身份验证系统之前,先添加几个页面,让用户能够输入数据,包括添加新主题,添加新条目以及编辑既有条目。

用户不需要与管理网站交互,所以使用Django的表单创建工具来创建让用户能够输入数据的页面。

2.1.1 添加新主题

创建基于表单的页面的方法同样是:定义URL,编写视图函数以及编写模板。此外,与之前不同,需要导入包含表单的模块forms.py。

2.1.1.1 用于添加主题的表单

用户输入信息时,我们需要进行验证,确认提供的信息是正确的数据类型,而不是恶意的信息,如中断服务器的代码。然后,对这些有效信息进行处理,并将其保存到数据库的合适地方。这些工作很多都是由Django自动完成的。

在Diango中,创建表单的最简单方式是使用ModelForm,它根据已经定义好的模型中的信息自动创建表单。

在learning_logs文件夹中创建forms.py,编写第一个表单如下:

from django import forms

from .models import Topic


class TopicForm(forms.ModelForm):
    class Meta:
        model = Topic
        fields = ['text']
        labels = {'text': ''}

首先导入模块forms和模型Topic。

定义TopicForm类继承forms.ModelForm。最简单的ModelForm 版本只包含一个内嵌的Meta类,让Django根据哪个模型创建表单以及在表单中包含哪些字段。

根据模型Topic 创建表单,其中只包含字段text。

设置labels的第一个键值对让Django不要为字段text生成标签。

2.1.1.2 URL模式new_topic

在learning_logs/urls.py中创建页面new_topic的URL模式如下:

urlpatterns = [
    # -------- snip -------- #

    # page for adding a new topic
    path('new_topic/', views.new_topic, name='new_topic')
]

当用户要添加到新主题时,切换到 http://127.0.0.1:8000/new_topic/

这个URL模式将请求交给视图函数new_topic()。

2.1.1.3 视图函数new_topic()

函数new_topic() 需要处理两种情形。一是刚进入new_topic页面(在这种情况下应显示空表单);二是对提交的表单数据进行处理,并将用户重定向到页面topics。

在learning_logs/views.py中导入函数redirect,用户提交主题后将使用这个函数重定向到页面topics。函数redirect将视图名作为参数,并将用户重定向到这个视图。还导入了刚创建的表单TopicForm。

learning_logs/views.py定义新函数new_topic()如下:

from .forms import TopicForm

def new_topic(request):
    '''add a new topic'''
    if request.method != 'POST':
        # No data submitted; create a blank form.
        form = TopicForm()
    else:
        # POST data submitted; process data.
        form = TopicForm(data=request.POST)
        if form.is_valid():
            form.save()
            return redirect('learning_logs:topics')

    # display a blank or invalid form
    context = {'form': form}
    return render(request, 'learning_logs/new_topic.html', context)

GET请求和POST请求:

创建Web应用程序时,将用到的两种主要请求类型是GET请求和POST请求。对于只是从服务器读取数据的页面,使用GET请求;在用户需要通过表单提交信息时,通常使用POST请求。处理所有表单时,都将指定使用POST方法。

函数new_topic()将请求对象作为参数。用户初次请求该页面时,其浏览器将发送GET请求;用户填写并提交表单时,其浏览器将发送POST请求。根据请求的类型,可确定用户请求的是空表单(GET请求)还是要求对填写好的表单进行处理(POST请求)。

new_topic()的第一个if测试确定请求方法是GET还是POST。如果请求方法不是POST,请求就可能是GET,因此需要返回一个空表单。(即便请求是其他类型的,返回空表单也不会有任何问题。)

下一行创建一个TopicForm实例,将其赋给变量form,再通过上下文字典将这个表单发送给模板。由于实例化TopicForm时没有指定任何实参,Django将创建一个空表单,供用户填写。

如果请求方法为POST,将执行else 代码块,对提交的表单数据进行处理。使用用户输入的数据(存储在request.POST中)创建一个TopicForm实例,这样对象form 将包含用户提交的信息。

要将提交的信息保存到数据库,必须先通过检查确定它们是有效的。方法is_valid() 核实用户填写了所有必不可少的字段(表单字段默认都是必不可少的),且输入的数据与要求的字段类型一致。这种自动验证避免了我们去做大量的工作。如果所有字段都有效,就可调用save(),将表单中的数据写入数据库。

保存数据后,就可离开这个页面了。为此,使用redirect()将用户的浏览器重定向到页面topics。在页面topics中,用户将在主题列表中看到他刚输入的主题。

在这个视图函数的末尾定义了变量context ,并使用稍后将创建的模板new_topic.html来渲染页面。这些代码不在if代码块内,因此无论是用户刚进入new_topic页面还是提交的表单数据无效,这些代码都将执行。用户提交的表单数据无效时,将显示一些默认的错误消息,帮助用户提供有效的数据。

2.1.1.4 模板new_topic

下面创建新模板learning_logs\templates\learning_logs\new_topic.html,用于显示刚创建的表单:

{% extends "learning_logs/base.html" %}

{% block content %}
    <p>Add a new topic:</p>

    <form action="{% url 'learning_logs:new_topic' %}" method="post">
        {% csrf_token %}
        {{ form.as_p }}
        <button name="submit">Add topic</button>
    </form>

{% endblock content %}

定义了一个HTML表单。实参action告诉服务器将提交的表单数据发送到哪里。这里将它发回给视图函数new_topic()。实参method让浏览器以POST请求的方式提交数据。

Django使用模板标签{% csrf_token %}来防止攻击者利用表单来获得对服务器未经授权的访问(这种攻击称为跨站请求伪造)。

只需创建模板变量{{form.as_p }},就可让Django自动创建显示表单所需的全部字段。修饰符as_p让Django以段落格式渲染所有表单元素,这是一种整洁地显示表单的简单方式。

Django不会为表单创建提交按钮,因此我们手动定义。

2.1.1.5 链接到页面new_topic

下面在页面topics中添加到页面new_topic的链接:

{% extends "learning_logs/base.html" %}

{% block content %}

    <p>Topics</p>

    <ul>
        {% for topic in topics %}
            <li>
                <a href="{% url 'learning_logs:topic' topic.id %}">{{ topic }}</a>
            </li>
        {% empty %}
            <li>No topics have been added yet.</li>
        {% endfor %}
    </ul>

    <a href="{% url 'learning_logs:new_topic' %}">Add a new topic</a>

{% endblock content %}

这个链接放在既有主题列表的后面。

2.1.2 添加新条目

通过在learning_logs/forms.py再添加一个类,并进行后续的定义URL、编写视图函数和编写模板,并链接到添加新条目的页面,使得用户账户可以添加新条目。

2.1.2.1 用于添加新条目的表单

需要创建一个与模型Entry相关联的表单,但这个表单的定制程度比TopicForm更高一些。

在learning_logs/forms.py再添加一个类EntryForm()

from .models import Entry

class EntryForm(forms.ModelForm):
    class Meta:
        model = Entry
        fields = ['text']
        labels = {'text': 'Entry:'}
        widgets = {'text': forms.Textarea(attrs={'cols': 80})}

新类EntryForm继承了forms.ModelForm,它包含的Meta类指出了表单基于的模型以及要在表单中包含哪些字段。这里给字段text指定了标签'Entry:'。

定义属性widgets 。小部件(widget)是一个HTML表单元素,如单行文本框、多行文本区域或下拉列表。通过设置属性widgets,可覆盖Django选择的默认小部件。通过让Django使用forms.Textarea,我们定制了字段'text'的输入小部件,将文本区域的宽度设置为80列,而不是默认的40列。这给用户提供了足够的空间来编写有意义的条目。

2.1.2.2 URL模式new_entry

在用于添加新条目的页面的URL模式中,需要包含实参topic_id,因为条目必须与特定的主题相关联。

在learning_logs/urls.py添加该URL模式如下:

urlpatterns = [
    # -------- snip -------- #

    # page for adding a new entry
    path('new_entry/<int:topic_id>/', views.new_entry, name='new_entry'),
]

这个URL模式与形如 http://127.0.0.1:8000/new_entry/id/ 的URL匹配,其中的id是一个与主题ID匹配的数。代码<int:topic_id>捕获一个数值,并将其赋给变量topic_id。请求的URL与这个模式匹配时,Django将请求和主题ID发送给函数new_entry()。

2.1.2.3 视图函数new_entry()

在learning_logs/views.py中定义并添加视图函数new_entry():

from .forms import EntryForm

def new_entry(request, topic_id):
    '''add a new entry for a particular topic'''
    topic = Topic.objects.get(id=topic_id)
    if request.method != 'POST':
        # No data submitted; create a blank form.
        form = EntryForm()
    else:
        # POST data submitted; process data.
        form = EntryForm(data=request.POST)
        if form.is_valid():
            new_entry = form.save(commit=False)
            new_entry.topic = topic
            new_entry.save()
            return redirect('learning_logs:topic', topic_id=topic_id)

    # display a blank or invalid form
    context = {'topic': topic, 'form': form}
    return render(request, 'learning_logs/new_entry.html', context)

导入刚刚创建的EntryForm类。

new_entry()的定义包含形参topic_id,用于存储从URL中获得的值。渲染页面和处理表单数据时,都需要知道针对的是哪个主题,因此使用topic_id来获得正确的主题。 用if request.method != 'POST'语句检查请求方法是POST还是GET。

如果是GET请求,就执行if代码块,创建一个空的EntryForm实例。如果请求方法为POST,就对数据进行处理:创建一个EntryForm实例,使用request对象中的POST数据来填充它。然后检查表单是否有效。如果有效,就设置条目对象的属性topic,再将条目对象保存到数据库。

调用save()时,传递实参commit=False,让Django创建一个新的条目对象,并将其赋给new_entry,但不保存到数据库中。将new_entry的属性topic设置为在这个函数开头从数据库中获取的主题,再调用save()且不指定任何实参。这将把条目保存到数据库,并将其与正确的主题相关联。

调用redirect(),它要求提供两个参数:要重定向到的视图和要给视图函数提供的参数。这里重定向到topic(),而这个视图函数需要参数topic_id。视图函数topic()渲染新增条目所属主题的页面,其中的条目列表包含新增的条目。

在视图函数new_entry()的末尾,我们创建了一个上下文字典,并使用模板new_entry.html渲染页面。这些代码将在用户刚进入页面或提交的表单数据无效时执行。

2.1.2.4 模板new_entry

在learning_logs\templates\learning_logs中创建new_entry.html,并在其中创建模板new_entry:

{% extends "learning_logs/base.html" %}

{% block content %}

    <p><a href="{% url 'learning_logs:topic' topic.id %}">{{ topic }}</a></p>

    <p>Add a new entry:</p>
    <form action="{% url 'learning_logs:new_entry' topic.id %}" method='post'>
        {% csrf_token %}
        {{ form.as_p }}
        <button name='submit'>Add entry</button>
    </form>
{% endblock content %}

在页面顶端显示主题,让用户知道自己是在哪个主题中添加条目。该主题名也是一个链接,可用于返回到该主题的主页面。

表单的实参action包含URL中的topic_id值,让视图函数能够将新条目关联到正确的主题。除此之外,这个模板与模板new_topic.html完全相同。

2.1.2.5 链接到页面new_entry

接下来,需要在显示特定主题的页面中添加到页面new_entry的链接。

我们将这个链接放在条目列表前面,因为在这种页面中,执行的最常见的操作是添加新条目。

修改learning_logs\templates\learning_logs\topic.html如下:

{% extends "learning_logs/base.html" %}

{% block content %}

    <p>Topic: {{ topic }}</p>

    <p>Entries:</p>
    <p>
        <a href="{% url 'learning_logs:new_entry' topic.id %}">Add new entry</a>
    </p>

    <ul>
    {% for entry in entries %}
        <li>
            <p>{{ entry.date_added|date:'M d, Y H:i' }}</p>
            <p>{{ entry.text|linebreaks }}</p>
        </li>
    {% empty %}
        <li>There are no entries for this topic yet.</li>
    {% endfor %}
    </ul>

{% endblock content %}

现在用户可添加新主题,还可在每个主题中添加任意数量的条目。

2.1.3 编辑条目

下面创建让用户账户能够编辑既有条目的页面。

2.1.3.1 URL模式edit_entry

该页面的URL传递要编辑的条目的ID,所以对learning_logs/urls.py进行修改如下:

urlpatterns = [
    # -------- snip -------- #

    # page for editing an entry
    path('edit_entry/<int:entry_id>/', views.edit_entry, name='edit_entry')
]

在URL(如 http://127.0.0.1:8000/edit_entry/1/ )中传递的ID存储在形参entry_id中。

这个URL模式将与其匹配的请求发送给视图函数edit_entry()。

2.1.3.2 视图函数edit_entry()

页面edit_entry收到GET请求时,edit_entry()将返回一个表单,让用户能够对条目进行编辑;收到POST请求(条目文本经过修订)时,则将修改后的文本保存到数据库。

为实现如下功能,在learning_logs/views.py中定义函数edit_entry()如下:

from .models import Entry

def edit_entry(request, entry_id):
    '''edit an existing entry'''
    entry = Entry.objects.get(id=entry_id)
    topic = entry.topic

    if request.method != 'POST':
        # Initial request; pre-fill form with the current entry.
        form = EntryForm(instance=entry)
    else:
        # POST data submitted; process data.
        form = EntryForm(instance=entry, data=request.POST)
        if form.is_valid():
            form.save()
            return redirect('learning_logs:topic', topic_id=topic.id)

    context = {'entry': entry, 'topic': topic, 'form': form}
    return render(request, 'learning_logs/edit_entry.html', context)

首先导入模型Entry。

entry获取用户要修改的条目对象以及与其相关联的主题。

在请求方法为GET时将执行的if代码块中,使用实参instance=entry创建一个EntryForm实例。这个实参让Django创建一个表单,并使用既有条目对象中的信息填充它。用户将看到既有的数据,并且能够编辑。

处理POST请求时,传递实参instance=entry和data=request.POST,让Django根据既有条目对象创建一个表单实例,并根据request.POST中的相关数据对其进行修改。然后,检查表单是否有效。如果有效,就调用save()且不指定任何实参,因为条目已关联到特定的主题。然后,重定向到显示条目所属主题的页面,用户将在其中看到其编辑的条目的新版本。

如果要显示表单让用户编辑条目或者用户提交的表单无效,就创建上下文字典并使用模板edit_entry.html渲染页面。

2.1.3.3 创建模板edit_entry

在learning_logs\templates\learning_logs创建edit_entry.html文件并在其中创建edit_entry模板。

{% extends "learning_logs/base.html" %}

{% block content %}

    <p><a href="{% url 'learning_logs:topic' topic.id %}">{{ topic }}</a></p>

    <p>Edit entry:</p>

    <form action="{% url 'learning_logs:edit_entry' entry.id %}" method='post'>
        {% csrf_token %}
        {{ form.as_p }}
        <button name="submit">Save changes</button>
    </form>

{% endblock content %}

实参action将表单发送给函数edit_entry()处理。在标签{% url %}中,将条目ID作为一个实参,让视图函数edit_entry()能够修改正确的条目对象。

将提交按钮的标签设置成Save changes,旨在提醒用户:单击该按钮将保存所做的编辑,而不是创建一个新条目。

2.1.3.4 链接到页面edit_entry

在显示特定主题的页面中给每个条目添加到页面edit_entry的链接。

为此,修改learning_logs\templates\learning_logs\topic.html如下:

{% extends "learning_logs/base.html" %}

{% block content %}

    <p>Topic: {{ topic }}</p>

    <p>Entries:</p>
    <p>
        <a href="{% url 'learning_logs:new_entry' topic.id %}">Add new entry</a>
    </p>

    <ul>
    {% for entry in entries %}
        <li>
            <p>{{ entry.date_added|date:'M d, Y H:i' }}</p>
            <p>{{ entry.text|linebreaks }}</p>
            <p>
                <a href="{% url 'learning_logs:edit_entry' entry.id %}">Edit entry</a>
            </p>
        </li>
    {% empty %}
        <li>There are no entries for this topic yet.</li>
    {% endfor %}
    </ul>

{% endblock content %}

将编辑链接放在了每个条目的日期和文本后面。在循环中,使用模板标签{% url %}根据URL模式edit_entry和当前条目的ID属性(entry.id)来确定URL。链接文本为Edit entry,它出现在页面中每个条目的后面。

至此,“学习笔记”已具备了需要的大部分功能。用户可添加主题和条目,还可根据需要查看任何条目。

2.2 创建用户账户

建立用户注册和身份验证系统,让用户能够注册账户,进而登录和注销。

为此,新建一个应用程序,其中包含与处理用户账户相关的所有功能。这个应用程序将尽可能使用Django自带的用户身份验证系统来完成工作。

还将对模型Topic稍做修改,让每个主题都归属于特定用户。

2.2.1 应用程序users

命令行中使用startapp命令创建名为users的应用程序:

python manage.py startapp users

接着需要在learning_log/settings.py中,将这个新的应用程序添加到INSTALLED_APPS中,使得Django将应用程序users包含到项目中:

INSTALLED_APPS = [
    # my applications
    'learning_logs',
    'users',

    # default django applications
    'django.contrib.admin',
    'django.contrib.auth',
    'django.contrib.contenttypes',
    'django.contrib.sessions',
    'django.contrib.messages',
    'django.contrib.staticfiles',
]

接下来,修改learning_log/urls.py,使其包含将为应用程序定义的URL,即与任何users打头的URL都匹配:

urlpatterns = [
    path('admin/', admin.site.urls),
    path('users/', include('users.urls')),
    path('', include('learning_logs.urls')),
]

2.2.2 实现登录页面

使用Django提供的默认视图login ,因此这个应用程序的URL模式稍有不同。

在learning_log/users文件夹下,新建urls.py:

'''define URL patterns for users'''

from django.urls import path, include


app_name = 'users'
urlpatterns = [
    # include default auth urls
    path('', include('django.contrib.auth.urls')),
]

导入函数path和include,以便包含Django定义的一些默认的身份验证URL。这些默认的URL包含具名的URL模式,'login' 和'logout'。

我们将变量app_name设置成'users',让Django能够将这些URL与其他应用程序的URL区分开来。即便是Django提供的默认URL,将其包含在应用程序users的文件中后,也可通过命名空间users进行访问。

登录页面的URL模式与URL http://127.0.0.1:8000/users/login/匹配 。这个URL中的单词users让Django在users/urls.py中查找,而单词login让它将请求发送给Django的默认视图login。

2.2.2.1 模板login.html

用户请求登录页面时,Django将使用一个默认的视图函数,但我们依然需要为这个页面提供模板。默认的身份验证视图在文件夹registration中查找模板,因此我们需要创建这个文件夹。为此,在目录users中新建一个名templates的目录,再在这个目录中新建一个名为registration的目录。下面是模板login.html,应将其存储到目录users\templates\registration中:

{% extends "learning_logs/base.html" %}

{% block content %}

    {% if form.errors %}
        <p>Your username and password didn't match. Please try again.</p>
    {% endif %}
    
    <form method="post" action="{% url 'users:login' %}">
        {% csrf_token %}
        {{ form.as_p }}
    
        <button name="submit">Log in</button>
        <input type="hidden" name="next" value="{% url 'learning_logs:index' %}" />
    </form>
    
{% endblock content %}

这个模板继承了base.html,旨在确保登录页面的外观与网站的其他页面相同。

注意:一个应用程序中的模板可继承另一个应用程序中的模板。

如果设置表单的errors属性,就显示一条错误消息,指出输入的用户名密码对与数据库中存储的任何用户名密码对都不匹配。

我们要让登录视图对表单进行处理,因此将实参action设置为登录页面的URL。登录视图将一个表单发送给模板。在模板中,我们显示这个表单并添加一个提交按钮。

包含了一个隐藏的表单元素'next',其中的实参value告诉Django在用户成功登录后将其重定向到什么地方。在本项目中,用户将返回主页。

2.2.2.2 链接到登录界面

下面在learning_logs\templates\learning_logs\base.html中添加到登录页面的链接,让所有页面都包含它。

用户已登录时,我们不想显示这个链接,因此将它嵌套在一个{% if %}标签中:

<p>
    <a href="{% url 'learning_logs:index' %}">Learning Log</a> - 
    <a href="{% url 'learning_logs:topics' %}">Topics</a> - 
    {% if user.is_authenticated %}
        Hello, {{ user.username }}.
    {% else %}
        <a href="{% url 'users:login' %}">Log in</a>
    {% end if %}
</p>

{% block content %}{% endblock content %}

在Django身份验证系统中,每个模板都可使用变量user。这个变量有一个is_authenticated属性:如果用户已登录,该属性将为True,否则为False。这让你能够向已通过身份验证的用户显示一条消息,而向未通过身份验证的用户显示另一条消息。

这里向已登录的用户显示问候语。对于已通过身份验证的用户,还设置了属性username。这里使用该属性来个性化问候语,让用户知道自己已登录。

对于尚未通过身份验证的用户,显示到登录页面的链接。

2.2.2.3 使用登录页面

前面建立了一个用户账户,下面来登录一下,看看登录页面是否管用。

访问 http://127.0.0.1:8000/admin ,登出管理员超级账户

登出后,访问 http://127.0.0.1:8000/users/login ,即可看到登录页面

2.2.3 登出

现在需要提供一个让用户登出的途径。为此,我们将在learning_logs\templates\learning_logs\base.html中添加一个登出链接。用户单击这个链接时,将进入一个确认其已登出的页面。

2.2.3.1 在base.html中添加登出链接

下面在learning_logs\templates\learning_logs\base.html中添加注销链接,让每个页面都包含它。将注销链接放在{% if user.is_authenticated %}部分中,这样只有已登录的用户才能看到它,默认的具名登出URL模式为'logout':

    {% if user.is_authenticated %}
        Hello, {{ user.username }}.
        <a href="{% url 'users:logout' %}">Log out</a>
    {% else %}
2.2.3.2 登出确认页面

成功登出后,用户希望获悉这一点。因此默认的注销视图使用模板logged_out.html渲染注销确认页面。

在users\templates\registration目录中创建logged_out.html:

{% extends "learning_logs/base.html" %}

{% block content %}
    <p>You have been logged out. Thank you for visiting!</p>
{% endblock content %}

2.2.4 注册页面

下面创建一个页面供新用户注册。我们将使用Django提供的表单UserCreationForm,但编写自己的视图函数和模板。

2.2.4.1 注册页面的URL模式

在users\urls.py中添加注册页面的URL模式:

from . import views

app_name = 'users'
urlpatterns = [
    # include default auth urls
    path('', include('django.contrib.auth.urls')),
    # registration page
    path('register/', views.register, name='register'),
]

注册页面的URL模式与URL http://127.0.0.1:8000/users/register/ 匹配,并将请求发送给视图函数register() 。

2.2.4.2 视图函数register()

在注册页面首次被请求时,视图函数register()需要显示一个空的注册表单,并在用户提交填写好的注册表单时对其进行处理。如果注册成功,这个函数还需让用户自动登录。

在users/views.py中输入代码如下:

from django.shortcuts import render, redirect
from django.contrib.auth import login
from django.contrib.auth.forms import UserCreationForm


def register(request):
    """register a new user"""
    if request.method != 'POST':
        # display blank registration form
        form = UserCreationForm()
    else:
        # process completed form
        form = UserCreationForm(data=request.POST)
        
        if form.is_valid():
            new_user = form.save()
            # log the user in and then redirect to home page
            login(request, new_user)
            return redirect('learning_logs:index')

    # display a blank or invalid form
    context = {'form': form}
    return render(request, 'registration/register.html', context)

首先导入函数render()和redirect(),然后导入函数login(),以便在用户正确填写了注册信息时让其自动登录。

还导入默认表单UserCreationForm。在函数register()中,检查要响应的是否是POST请求。如果不是,就创建一个UserCreationForm 实例,且不给它提供任何初始数据。

如果响应的是POST请求,就根据提交的数据创建一个UserCreationForm实例,并检查这些数据是否有效。就本例而言,有效是指用户名未包含非法字符,输入的两个密码相同,以及用户没有试图做恶意的事情。

如果提交的数据有效,就调用表单的方法save(),将用户名和密码的散列值保存到数据库中。方法save()返回新创建的用户对象,我们将它赋给了new_user。保存用户的信息后,调用函数login()并传入对象request和new_user,为用户创建有效的会话,从而让其自动登录。最后,将用户重定向到主页,而主页的页眉中显示了一条个性化的问候语,让用户知道注册成 功了。

在这个函数的末尾,我们渲染了注册页面:它要么显示一个空表单,要么显示提交的无效表单。

2.2.4.3 注册模板

下面创建注册页面的模板users\templates\registration\register.html:

{% extends "learning_logs/base.html" %}

{% block content %}

    <form method="post" action="{% url 'users:register' %}">
        {% csrf_token %}
        {{ form.as_p }}

        <button name="submit">Register</button>
        <input type="hidden" name="next" value="{% url 'learning_logs:index' %}" />
    </form>

{% endblock content %}

方法as_p让Django在表单中正确地显示所有的字段,包括错误消息——如果用户没有正确地填写表单。

2.2.4.4 链接到注册页面

在用户没有登录时,需要显示到注册页面的链接。

在learning_logs\templates\learning_logs\base.html中修改代码如下:

<p>
    <a href="{% url 'learning_logs:index' %}">Learning Log</a> -
    <a href="{% url 'learning_logs:topics' %}">Topics</a> - 
    {% if user.is_authenticated %}
        Hello, {{ user.username }}.
        <a href="{% url 'users:logout' %}">Log out</a>
    {% else %}
        <a href="{% url 'users:register' %}">Register</a> -
        <a href="{% url 'users:login' %}">Log in</a>
    {% endif %}
</p>
  
{% block content %}{% endblock content %}

现在,已登录的用户看到的是个性化的问候语和注销链接,而未登录的用户看到的是注册链接和登录链接。

2.3 让用户拥有自己的数据

用户应该能够输入其专有的数据,因此我们将创建一个系统,确定各项数据所属的用户,再限制对页面的访问,让用户只能使用自己的数据。

将修改模型Topic,让每个主题都归属于特定用户。这也将影响条目,因为每个条目都属于特定的主题。我们先来限制对一些页面的访问。

2.3.1 使用login_required限制访问

Django提供了装饰器@login_required,让你能够轻松地只允许已登录用户访问某些页面。装饰器decorator是放在函数定义前面的指令,Python在函数运行前根据它来修改函数代码的行为。

2.3.1.1 限制访问显示所有主题的页面

每个主题都归特定用户所有,因此应只允许已登录的用户请求显示所有主题的页面。

在learning_logs/views.py中添加如下代码:

from django.shortcuts import render, redirect
from django.contrib.auth.decorators import login_required

# -------- snip -------- #

@login_required
def topics(request):
# -------- snip -------- #

首先导入函数login_required()。将login_required()作为装饰器应用于视图函数topics()——在它前面加上符号@和login_required,让Python在运行topics()的代码之前运行login_required()的代码。

login_required()的代码检查用户是否已登录,仅当用户已登录时,Django才运行topics()的代码。如果用户未登录,就重定向到登录页面。

为实现这种重定向,需要修改settings.py,让Django知道到哪里去查找登录页面。为此在learning_log/settings.py最末尾添加如下代码:

# my settings
LOGIN_URL = 'users:login'

现在,如果未登录的用户请求装饰器@login_required保护的页面,Django将重定向到settings.py中的LOGIN_URL指定的URL。

要测试这个设置,可注销并进入主页,再单击链接Topics,这将重定向到登录页面。然后,使用你的账户登录,并再次单击主页中的Topics链接,你将看到显示所有主题的页面。

2.3.1.2 全面限制对“学习笔记”项目的访问

在项目“学习笔记”中,将不限制对主页和注册页面的访问,并限制对其他所有页面的访问。

在learning_logs/views.py中,对除index()外的每个视图都应用装饰器@login_required(即在每个定义函数前都加上:

@login_required

此时,如果在未登录的情况下尝试访问这些页面,将被重定向到登录页面。

另外,也不能单击到new_topic等页面的链接。如果你输入URL http://127.0.0.1:8000/new_topic/ ,将被重定向到登录页面。

2.3.2 将数据关联到用户

现在,需要将数据关联到提交它们的用户。只需将最高层的数据关联到用户,更低层的数据就会自动关联到用户。

下面修改模型Topic,在其中添加一个关联到用户的外键。这样做之后,必须对数据库进行迁移。最后,必须修改某些视图,使其只显示与当前登录的用户相关联的数据。

2.3.2.1 修改模型Topic

对于learning_logs/models.py中的Topic模型进行修改。

首先导入django.contrib.auth中的模型User,然后在Topic中添加字段owner,它建立到模型User的外键关系。用户被删除时,所有与之相关联的主题也会被删除。

其代码修改如下:

from django.contrib.auth.models import User


class Topic(models.Model):
    '''a topic the user is learning about'''
    text = models.CharField(max_length=200)
    date_added = models.DateTimeField(auto_now_add=True)
    owner = models.ForeignKey(User, on_delete=models.CASCADE)

    def __str__(self):
        '''return a string representation of the model'''
        return self.text
2.3.2.2 确定当前有多少用户

迁移数据库时,Django将对数据库进行修改,使其能够存储主题和用户之间的关联。为执行迁移,Django需要知道该将各个既有主题关联到哪个用户。最简单的办法是,将既有主题都关联到同一个用户,如超级用户。为此,需要知道该用户的ID。

下面来查看已创建的所有用户的ID。

启动一个Django shell会话:

python manage.py shell

在命令行中输入:

>>> from django.contrib.auth.models import User
>>> User.objects.all()
<QuerySet [<User: django_admin>, ...(other not-superaccount users) ]>
>>> for user in User.objects.all():
...     print(user.username, user.id)
... 
django_admin 1
(other not-superaccount user) 2
>>>

通过遍历列表并打印每位用户的用户名和ID。Django询问要将既有主题关联到哪个用户时,我们将指定其中一个ID值。

2.3.2.3 迁移数据库

知道用户ID后,就可迁移数据库了。本项目中我们暂时将模型Topic关联到特定账户,而非在文件learning_logs/models.py中指定默认账户。

首先在命令行中输入:

python manage.py makemigrations learning_logs

当python询问如何迁移数据库时,选择第一个选项。

You are trying to add a non-nullable field 'owner' to topic without a default; we can't do that (the database needs something to populate existing rows).
Please select a fix:
 1) Provide a one-off default now (will be set on all existing rows with a null value for this column)
 2) Quit, and let me add a default in models.py
Select an option: 1
Please enter the default value now, as valid Python
The datetime and django.utils.timezone modules are available, so you can do e.g. timezone.now
Type 'exit' to exit this prompt
>>> 1
Migrations for 'learning_logs':
  learning_logs\migrations\0003_topic_owner.py
    - Add field owner to topic

在输出中,Django指出你试图给既有模型Topic添加一个不可为空的字段,而该字段没有默认值。Django提供了两种选择:要么现在提供默认值,要么退出并在models.py中添加默认值。我们选择了第一个选项,因此Django让我们输入默认值。

为将所有既有主题都关联到管理用户django_admin,我们输入用户ID值1。可以使用已创建的任何用户的ID,而非必须是超级用户。

接下来,Django使用这个值来迁移数据库,并生成了迁移文件0003_topic_owner.py,它在模型Topic中添加字段owner。

接下来执行迁移。在命令行中输入:

python manage.py migrate

为验证迁移符合预期,可开启django shell会话,并输入:

>>> from learning_logs.models import Topic
>>> for topic in Topic.objects.all():
...    print(topic, topic.owner)
... 
Chess django_admin
Rock Climbing django_admin
>>>

遍历所有的既有主题,并打印每个主题及其所属的用户。如你所见,现在每个主题都属于超级管理员用户django_admin。

注意:你可以重置数据库而不是迁移它,但如果这样做,既有的数据都将丢失。一种不错的做法是,学习如何在迁移数据库的同时确保用户数据的完整性。如果你确实想要一个全新的数据库,可执行命令python manage.py flush,这将重建数据库的结构。如果这样做,就必须重新创建超级用户,且原来的所有数据都将丢失。

2.3.3 只允许用户访问自己的主题

当前,不管以哪个用户的身份登录,都能够看到所有的主题。下面改变这一点,只向用户显示属于其自己的主题。

在learning_logs/views.py中,修改函数topics()如下:

@login_required
def topics(request):
    '''show all topics'''
    topics = Topic.objects.filter(owner=request.user).order_by('date_added')
    context = {'topics': topics}
    return render(request, 'learning_logs/topics.html', context)

用户登录后,request对象将有一个user属性,其中存储了有关该用户的信息。查询Topic.objects.filter(owner=request.user)让Django只从数据库中获取owner属性为当前用户的Topic对象。由于没有修改主题的显示方式,无须对显示所有主题的页面的模板做任何修改。

要查看结果,以所有既有主题关联到的用户的身份登录,并访问显示所有主题的页面,你将看到所有的主题。然后,登出并以另一个用户的身份登录,该页面将不列出任何主题。

2.3.4 保护用户的主题

我们还没有限制对显示单个主题的页面的访问,因此任何已登录的用户都可输入类似于 http://localhost:8000/topics/1/ 的URL,来访问显示相应主题的页面。

为修复这种问题,我们在learning_logs/views.py中的视图函数topic()获取请求的条目前执行检查。

from django.http import Http404


def topic(request, topic_id):
    '''show a single topic and all its entries'''
    topic = Topic.objects.get(id=topic_id)
    # make sure the topic belongs to the current user
    if topic.owner != request.user:
        raise Http404

    entries = topic.entry_set.order_by('-date_added')
    context = {'topic': topic, 'entries': entries}
    return render(request, 'learning_logs/topic.html', context)

服务器上没有请求的资源时,标准的做法是返回404响应。这里导入了异常Http404,并在用户请求其不应查看的主题时引发这个异常。收到主题请求后,在渲染页面前检查该主题是否属于当前登录的用户。如果请求的主题不归当前用户所有,就引发Http404异常,让Django返回一个404错误页面。

现在,如果你试图查看其他用户的主题条目,将看到Django发送的消息Page Not Found。

2.3.5 保护页面edit_entry

页面edit_entry的URL形式为 http://127.0.0.1:8000/edit_entry/entry_id/ ,其中entry_id 是一个数。下面来保护这种页面,禁止用户通过输入类似于前面的URL来访问其他用户的条目。

修改learning_logs/views.py中的函数edit_entry()如下:

def edit_entry(request, entry_id):
    '''edit an existing entry'''
    entry = Entry.objects.get(id=entry_id)
    topic = entry.topic

    if topic.owner != request.user:
        raise Http404

    if request.method != 'POST':
        # Initial request; pre-fill form with the current entry.
        form = EntryForm(instance=entry)
    else:
        # POST data submitted; process data.
        form = EntryForm(instance=entry, data=request.POST)
        if form.is_valid():
            form.save()
            return redirect('learning_logs:topic', topic_id=topic.id)

    context = {'entry': entry, 'topic': topic, 'form': form}
    return render(request, 'learning_logs/edit_entry.html', context)

首先获取指定的条目以及与之相关联的主题,再检查主题的所有者是否是当前登录的用户。如果不是,就引发Http404异常。

2.3.6 将新主题关联到当前账户

当前,用于添加新主题的页面存在问题——没有将新主题关联到特定用户。如果你尝试添加新主题,将看到错误消息IntegrityError,指出learning_logs_topic.user_id不能为NULL(NOT NULL constraint failed: learning_logs_topic.owner_id)。

Django的意思是说,创建新主题时,必须给owner字段指定值。我们可通过request对象获悉当前用户,因此有一个修复该问题的简单方案。

通过修改learning_logs/views.py中的函数new_topic()来将新主题关联到当前账户:

def new_topic(request):
    '''add a new topic'''
    if request.method != 'POST':
        # No data submitted; create a blank form.
        form = TopicForm()
    else:
        # POST data submitted; process data.
        form = TopicForm(data=request.POST)
        if form.is_valid():
            new_topic = form.save(commit=False)
            new_topic.owner = request.user
            new_topic.save()
            return redirect('learning_logs:topics')

    # display a blank or invalid form
    context = {'form': form}
    return render(request, 'learning_logs/new_topic.html', context)

首先调用form.save()并传递实参commit=False,因为要先修改新主题,再将其保存到数据库。接下来,将新主题的owner属性设置为当前用户。最后,对刚定义的主题实例调用save()。现在,主题包含所有必不可少的数据,将被成功保存。

这个项目现在允许任何用户注册,而每个用户想添加多少新主题都可以。每个用户都只能访问自己的数据,无论是查看数据、输入新数据还是修改旧数据时都如此。

3. 设置应用程序的样式并部署

当前的“学习笔记”并未设置样式,故只能在本地运行。

所以当前需要用Bootstrap库来设置样式,并且将“学习笔记”项目部署到Heroku,将其推送到服务器,使其可以联网使用。

3.1 设置项目样式

在应用程序能够正确运行的基础上,下面来设置外观。

3.1.1 应用程序django-bootstrap4

将使用django-bootstrap4将Bootstrap集成到项目中。这个应用程序下载必要的Bootstrap文件,将其放到项目的合适位置,让你能够在项目的模板中使用样式设置指令。

首先在虚拟环境中安装django-bootstrap4,这里我们安装0.0.7版本:

pip install django-bootstrap4==0.0.7

然后,为在项目中包含应用程序django-bootstrap4,在learning_log/settings.py中的INSTALL_APPS中新建一个名为“第三方应用程序”的片段,用于指定其他开发人员开发的应用程序,并添加'bootstrap4'。务必将这个片段放在“我的应用程序”和“Django默认添加的应用程序”之间。代码如下:

INSTALLED_APPS = [
    # my applications
    'learning_logs',
    'users',

    # third party apps
    'bootstrap4',

    # default django applications
    'django.contrib.admin',
    'django.contrib.auth',
    'django.contrib.contenttypes',
    'django.contrib.sessions',
    'django.contrib.messages',
    'django.contrib.staticfiles',
]

3.1.2 使用Bootstrap设置“学习笔记”的样式

Bootstrap是一个大型样式设置工具集,还提供了大量模板,可应用于项目以创建独特的总体风格。对Bootstrap初学者来说,这些模板比样式设置工具用起来容易得多。要查看Bootstrap提供的模板,可访问其官方网站,单击Examples并找到Navbars。我们将使用模板Navbars static,它提供了简单的顶部导航栏以及用于放置页面内容的容器。

3.1.3 修改base.html

需要修改learning_logs\templates\learning_logs\base.html,来使用Bootstrap的模板。

3.1.3.1 定义HTML头部

通过定义HTML头部,使得显示“学习笔记”的每个页面时,浏览器标题栏都显示该网站名。

此外,还要添加一些在模板中使用Bootstrap所需的信息。

将learning_logs\templates\learning_logs\base.html重写如下:

{% load bootstrap4 %}

<!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>Learning Log</title>

    {% bootstrap_css %}
    {% bootstrap_javascript jquery='full' %}

</head>

首先加载django-bootstrap4中的模板标签集。

接下来,将这个文件声明为使用英语编写的HTML文档。

HTML文件分为两个主要部分:头部(head)和主体(body)。HTML文件的头部不包含任何内容,只是向浏览器提供正确显示页面所需的信息。

title元素使得在浏览器中打开网站“学习笔记”的页面时,浏览器的标题栏将显示该元素的内容。

使用django-bootstrap4的一个自定义模板标签,让Django包含所有的Bootstrap样式文件。接下来的标签启用你可能在页面中使用的所有交互式行为,如可折叠的导航栏。最后为结束标签</head>

3.1.3.2 定义导航栏

需要同时支持较窄的手机屏幕和较宽的台式计算机显示器。

将分为三个部分定义导航栏。

第一部分:

<body>

    <nav class="navbar navbar-expand-md navbar-light bg-light mb-4 border">
  
        <a class="navbar-brand" href="{% url 'learning_logs:index'%}">
                Learning Log</a>
  
        <button class="navbar-toggler" type="button" data-toggle="collapse"
                data-target="#navbarCollapse" aria-controls="navbarCollapse"
                aria-expanded="false" aria-label="Toggle navigation">
            <span class="navbar-toggler-icon"></span></button>

第一个元素为起始标签<body>。HTML文件的主体包含用户将在页面上看到的内容。

<nav>元素,表示页面的导航链接部分。对于这个元素内的所有内容,都将根据此处的navbar和navbar-expand-md等选择器定义的Bootstrap样式规则来设置样式。选择器(selector)决定了样式规则将应用于页面上的哪些元素。选择navbar-light和bg-light使用一种浅色主题来设置导航栏的颜色。mb-4中的mb表示下边距(margin-bottom),这个选择器确保导航栏和页面其他部分之间有一些空白区域。选择器border在浅色背景周围添加很细的边框,将导航栏与页面其他部分分开。

指定在导航栏最左端显示项目名,并将其设置为到主页的链接,因为它将出现在这个项目的每个页面中。选择器navbar-brand设置这个链接的样式,使其比其他链接更显眼,这是一种网站推广方式。

定义一个按钮,它将在浏览器窗口太窄、无法水平显示整个导航栏时显示出来。如果用户单击这个按钮,将出现一个下拉列表,其中包含所有的导航元素。在用户缩小浏览器窗口或在屏幕较小的移动设备上显示网站时,collapse会导致导航栏折叠起来。

第二部分:

        <div class="collapse navbar-collapse" id="navbarCollapse">
            <ul class="navbar-nav mr-auto">
                <li class="nav-item">
                    <a class="nav-link" href="{% url 'learning_logs:topics'%}">
                        Topics</a></li>
            </ul>

首先开启了导航栏的一个新区域。div是division的缩写。我们创建页面时,将其分隔成多个区域,并指定要应用于各个区域的样式和行为规则。在<div>起始标签中定义的样式和行为规则将影响下一个结束标签</div>之前的所有元素。这里指定了屏幕或窗口太窄时将折叠起来的导航栏部分的起始位置。

定义了一组链接。Bootstrap通过<ul><li></li></ul>将导航元素定义为无序列表项,但使用的样式规则让它们一点也不像列表。导航栏中的每个链接或元素都能以列表项的方式定义。这里只有一个列表项——到显示所有主题的页面的链接。

第三部分:

            <ul class="navbar-nav ml-auto">
                {% if user.is_authenticated %}
                    <li class="nav-item">
                        <span class="navbar-text">Hello, {{ user.username }}.</span>
                    </li>
                    <li class="nav-item">
                        <a class="nav-link" href="{% url 'users:logout' %}">Log out</a>
                    </li>
                {% else %}
                    <li class="nav-item">
                        <a class="nav-link" href="{% url 'users:register' %}">Register</a>
                    </li>
                    <li class="nav-item">
                        <a class="nav-link" href="{% url 'users:login' %}">Log in</a></li>
                {% endif %}
            </ul>
        </div>
        
    </nav>

使用起始标签<ul>定义了另一组链接(你可根据需要在页面中包含任意数量的链接编组),这组链接与登录和注册相关,出现在导航栏最右端。选择器ml-auto表示自动左边距(margin-leftautomatic),它根据导航栏包含的其他元素设置左边距,确保这组链接位于屏幕右边。

if代码块与以前使用的条件代码块相同,它根据用户是否已登录显示相应的消息。这个代码块比以前长一些,因为它现在包含一些样式规则。

<span>元素,用于设置区域内一系列文本或元素的样式。因为<div>元素创建区域,而<span>元素不会。这里只是要设置导航栏中信息性文本(如已登录用户的名称)的样式,旨在让其外观与链接不同,以免用户忍不住去单击,因此使用了<span>

要在导航栏中添加其他链接,可在既有的<ul>元素中添加<li>元素,并使用这里演示的样式设置指令。

3.1.3.3 定义页面的主要部分

在base.html中,还需添加一些代码:定义两个块,供各个页面放置其特有的内容。

添加如下部分:

    <main role="main" class="container">
        <div class="pb-2 mb-2 border-bottom">
            {% block page_header %}{% endblock page_header %}
        </div>
        <div>
            {% block content %}{% endblock content %}
        </div>
    </main>
        
</body>
        
</html>

<main>元素用于定义页面主体的最重要部分。此处指定了Bootstrap选择器container,这是一种对页面元素进行编组的简单方式。我们将在这个容器中放置两个<div>元素。

第一个<div>元素包含一个page_header块,我们会在大多数页面中使用它来指定标题。为突出标题,设置内边距。内边距(padding)指的是元素内容和边框之间的距离。选择器pb-2是一个Bootstrap指令,将元素的下内边距设置为适度的值。外边距(margin)指的是元素的边框与其他元素之间的距离。我们只想在标题下面添加边框,因此使用选择器border-bottom,它在page_header块的下面添加较细的边框。

另一个<div>元素包含content 块。我们没有对这个块指定样式,因此在具体的页面中,可根据需要设置内容的样式。

如果现在在浏览器中加载“学习笔记”的主页,你将看到一个专业级导航栏。请尝试将窗口调整得非常窄,此时导航栏将变成一个按钮。如果你单击这个按钮,将打开一个下拉列表,其中包含所有的导航链接。

3.1.4 使用jumbotron设置主页的样式

下面使用Bootstrap元素jumbotron来修改主页。jumbotron元素是一个大框,在页面中显得鹤立鸡群。它可以包含任何东西,通常用于在主页中呈现简要的项目描述和让用户行动起来的元素。

修改learning_logs\templates\learning_logs\index.html如下:

{% extends "learning_logs/base.html" %}

{% block page_header %}
    <div class="jumbotron">
        <h1 class="display-3">Track your learning.</h1>

        <p class="lead">Make your own Learning Log, and keep a list of the
            topics you're learning about. Whenever you learn something new
            about a topic, make an entry summarizing what you've learned.</p>

        <a class="btn btn-lg btn-primary" href="{% url 'users:register' %}"
            role="button">Register &raquo;</a>
    </div>
{% endblock page_header %}

告诉Django接下来要定义page_header块包含的内容。

jumbotron就是应用了一系列样式设置指令的<div>元素。这里使用选择器jumbotron应用这组来自Bootstrap库的样式设置指令。

这个jumbotron包含三个元素。第一个是一条简短的消息——Track your learning,让首次访问者大致知道“学习笔记”是做什么用的。h1类表示一级标题,而选择器display-3让这个标题显得更窄更高。

添加一条更长的消息,让用户更详细地知道使用学习笔记可以做什么。

通过创建一个按钮(而不是文本链接)邀请用户注册账户。它与导航栏中的链接Register一样链接到的注册页面,但是按钮更显眼,并且让用户知道要使用这个项目首先需要如何做。这里的选择器让这个按钮很大,召唤用户赶快行动起来。代码&raquo;是一个HTML实体,表示两个右尖括号>>

结束page_header块。我们不想在这个页面中添加其他内容,因此不需要定义content块。

3.1.5 设置登录页面的样式

下面修改users\templates\registration\login.html来改进登录表单,使得表单与页面的其他部分一致:

{% extends "learning_logs/base.html" %}

{% load bootstrap4 %}

{% block page_header %}
    <h2>Log in to your account.</h2>
{% endblock page_header %}

{% block content %}

    <form method="post" action="{% url 'users:login' %}" class="form">
        {% csrf_token %}
        {% bootstrap_form form %}
        {% buttons %}
            <button name="submit" class="btn btn-primary">Log in</button>
        {% endbuttons %}
    
        <input type="hidden" name="next" 
            value="{% url 'learning_logs:index' %}" />
    </form>
    
{% endblock content %}

在这个模板中加载bootstrap4模板标签。

定义page_header块,指出这个页面是做什么用的。注意,我们从这个模板中删除了代码块{% if form.errors %},因为django-bootstrap4会自动管理表单错误。

添加属性class="form",再使用模板标签{% bootstrap_form %}来显示表单,它替换了标签{{ form.as_p }}。模板标签{% bootstrap_form %}将Bootstrap样式规则应用于各个表单元素。

bootstrap4起始模板标签{% buttons %}将Bootstrap样式应用于按钮。

3.1.6 设置显示所有主题的页面的样式

下面来确保用于查看信息的页面也有合适的样式

首先来设置显示所有主题的页面,修改learning_logs\templates\learning_logs\topics.html如下:

{% extends "learning_logs/base.html" %}

{% block page_header %}
    <h1>Topics</h1>
{% endblock page_header %}

{% block content %}
    <ul>
        {% for topic in topics %}
            <li><h3>
                <a href="{% url 'learning_logs:topic' topic.id %}">{{ topic }}</a>
            </h3></li>
        {% empty %}
            <li><h3>No topics have been added yet.</h3></li>
        {% endfor %}
    </ul>

    <h3><a href="{% url 'learning_logs:new_topic' %}">Add a new topic</a></h3>

{% endblock content %}

不需要标签{% load bootstrap4 %},因为这个文件中没有使用任何bootstrap4自定义标签。我们将标题Topics移到page_header块中,并给它指定标题样式,而没有使用简单的段落标签。将每个主题都设置为<h3>元素,使其在页面上显得大一些。对于添加新主题的链接,也做同样的处理。

3.1.7 设置显示单个主题的页面中的条目样式

比起大部分页面,显示单个主题的页面包含更多内容,因此需要做样式设置工作要更多一些。

我们将使用Bootstrap的卡片(card)组件来突出每个条目。

卡片是带灵活的预定义样式的<div>,非常适合用于显示主题的条目。

修改learning_logs\templates\learning_logs\topic.html如下:

{% extends 'learning_logs/base.html' %}

{% block page_header %}
    <h3>{{ topic }}</h3>
{% endblock page_header %}

{% block content %}
    <p>
        <a href="{% url 'learning_logs:new_entry' topic.id %}">Add new entry</a>
    </p>

    <ul>
    {% for entry in entries %}
        <div class="card mb-3">
            <h4 class="card-header">
                {{ entry.date_added|date:'M d, Y H:i' }}
                <small><a href="{% url 'learning_logs:edit_entry' entry.id %}">
                    edit entry</a></small>
            </h4>
            <div class="card-body">
                {{ entry.text|linebreaks }}
            </div>
        </div>
    {% empty %}
        <p>There are no entries for this topic yet.</p>
    {% endfor %}

{% endblock content %}

首先将主题放在page_header块中,并删除该模板中以前使用的无序列表结构。

创建一个带选择器card的<div>元素(而不是将每个条目作为一个列表项),其中包含两个嵌套的元素:一个包含条目的创建日期以及用于编辑条目的链接,另一个包含条目的内容。

嵌套的第一个元素是个标题。它是带选择器card-header的<h4>元素,包含条目的创建日期以及用于编辑条目的链接。用于编辑条目的链接放在标签<small>内,这让它看起来比时间戳小一些。

第二个嵌套的元素是一个带选择器card-body的<div>元素,将条目的内容放在一个简单的框内。注意我们只修改了影响页面外观的元素,对在页面中包含信息的Django代码未做任何修改。

注意:要使用其他Bootstrap模板,可采用类似流程:将要使用的模板复制到base.html中并修改包含实际内容的元素,以使用该模板来显示项目的信息,然后使用Bootstrap的样式设置工具来设置各个页面中内容的样式。

3.2 部署“学习笔记”

鉴于特殊网络原因,以及相比于国内云服务器,heroku都算不上一个好的选择,故本项目到这里就结束了。至此,本项目除了只能在本地计算机中运行使用外,与最终的服务器版本并无差别。

如果想尝试使用heroku服务器部署项目到全互联网,请参考heroku-deploy分支的内容。