侧边栏已经正确地显示了最新文章列表、归档、分类等信息。现在来完善归档和分类功能,当用户点击归档下的某个日期或者分类下的某个分类时,跳转到文章列表页面,显示该日期或者分类下的全部文章。
要显示某个归档日期下的文章列表,思路和显示主页文章列表是一样的,回顾一下主页视图的代码:
主页视图函数中我们通过 <code>post.objects.all()</code> 获取全部文章,而在我们的归档和分类视图中,我们不再使用 <code>all</code> 方法获取全部文章,而是使用 <code>filter</code> 来根据条件过滤。先来看归档视图:
写好视图函数后就是配置好 url:
这个归档视图对应的 url 的正则表达式和 <code>detail</code> 视图函数对应的 url 是类似的,这在之前我们讲过。两个括号括起来的地方是两个命名组参数,django 会从用户访问的 url 中自动提取这两个参数的值,然后传递给其对应的视图函数。例如如果用户想查看 2017 年 3 月下的全部文章,他访问 /archives/2017/3/,那么 <code>archives</code> 视图函数的实际调用为:<code>archives(request, year=2017, month=3)</code>。
在模板找到归档列表部分的代码,修改超链接的 <code>href</code> 属性,让用户点击超链接后跳转到文章归档页面:
这里 {% url %} 这个模板标签的作用是解析视图函数 <code>blog:archives</code> 对应的 url 模式,并把 url 模式中的年和月替换成 <code>date.year</code>,<code>date.month</code> 的值。例如 blog:archives 表示 blog 应用下的 archives 函数,这个函数对应的 url 模式为 <code>^archives/(?p<year>[0-9]{4})/(?p<month>[0-9]{1,2})/$</code>,假设 <code>date.year=2017</code>,<code>date.month=5</code>,那么 {% url 'blog:archives' date.year date.month %} 模板标签返回的值为/archives/2017/5/。
为什么要使用 {% url %} 模板标签呢?事实上,我们把超链接的 href 属性设置为 <code>/archives/{{ date.year }}/{{ date.month }}/</code> 同样可以达到目的,但是这种写法是硬编码的。虽然现在 blog:archives 视图函数对应的 url 模式是这种形式,但是如果哪天这个模式改变了呢?如果使用了硬编码的写法,那你需要把每一处 <code>/archives/{{ date.year }}/{{ date.month }}/</code> 修改为新的模式。但如果使用了 {% url %} 模板标签,则不用做任何修改。
测试一下,点击侧边栏归档的日期,跳转到归档页面,发现报了个错误,提示没有安装 pytz。激活虚拟环境,使用 <code>pip install pytz</code> 安装即可。
重启一下开发服务器,再次测试,发现可以显示归档下的文章列表了。
同样的写好分类页面的视图函数:
这里我们首先根据传入的 pk 值(也就是被访问的分类的 id 值)从数据库中获取到这个分类。<code>get_object_or_404</code> 函数和 detail 视图中一样,其作用是如果用户访问的分类不存在,则返回一个 404 错误页面以提示用户访问的资源不存在。然后我们通过 <code>filter</code> 函数过滤出了该分类下的全部文章。同样也和首页视图中一样对返回的文章列表进行了排序。
url 配置如下:
这个分类页面对应的 url 模式和文章详情页面对应的 url 模式十分类似,你可以自己分析分析它是如何工作的,在此就不赘述了。
修改相应模板:
同样,{% url %} 模板标签的用法和写归档页面时的用法是一样的。现在尝试点击相应的链接,就可以跳转到归档或者分类页面了。
如果遇到问题,请通过下面的方式寻求帮助。
将问题的详细描述通过邮件发送到 [email protected],一般会在 24 小时内回复。