django-taggit模块是一个可重用的应用程序,它主要提供一个标签模型和一个管理器,可以轻松地向任意模型添加标签。
https://github.com/alex/django-taggit
目录
添加taggit到setting.py中的INSTALLED_APPS
安装django-taggit
python -m pip install django-taggit
添加taggit到setting.py中的INSTALLED_APPS
INSTALL_APPS = [
#...
'taggit',
]
编辑models.py,将标签应用到post模型
from taggit.managers import TaggableManager
class Post(models.Mocel):
#...
tags = TaggableManager()
标记管理器将允许您从Post对象中添加、检索和删除标记。
更新数据库
python manage.py makemigrations blog
python manage.py migrate
重构blog应用程序的views.py文件
导入标签模型表单django-taggit,并将post_list视图更改为根据标签筛选帖子。视图最终是这样的。
def post_list(request,tag_slug=None):
object_list = Post.published.all()
tag = None
if tag_slug:
tag = get_object_or_404(Tag, slug=tag_slug)
object_list = object_list.filter(tags__in=[tag])
paginator = Paginator(object_list,3)
page = request.GET.get('page')
try:
posts = paginator.page(page)
except PageNotAnInteger:
posts = paginator.page(1)
except EmptyPage:
posts = paginator.page(paginator.num_pages)
template = "blog/post/list.html"
context = {
"page":page,
"posts":posts,
"tag":tag,
}
return render(request,template,context)
- 视图接受一个可选的tag_slug参数,默认值为None。该参数将出现在URL中。
- 在视图中,构建初始QuerySet,检索所有已发布的文章,如果存在给定的标记段,我们使用get_object_or_404()获取带有给定段的tag对象。
- 然后,根据包含给定标记的帖子筛选帖子列表。由于这是一个多对多关系,我们必须根据给定列表中包含的标记进行筛选,在示例中,该列表只包含一个元素。
添加标签路径到urls.py
path("", views.post_list, name="post_list"),
path('tag/<slug:tag_slug>/',views.post_list, name='post_list_by_tag'),
两个路径都指向相同的视图,但是我们对它们的命名不同。
第一种模式将调用不带任何可选参数的post_list视图,而第二种模式将调用带tag_slug参数的视图。这里使用一个slug路径转换器将参数匹配为一个小写字符串,其中包含ASCII字母或数字,以及连字符和下划线字符。
更新list.html模版,为标签增加链接
{% if tag %}
<h2>Posts tagged with "{{ tag.name }}"</h2>
{% endif %}
如果tag不是None,显示正在过滤的tag名称。
<p class="tags">
Tags:
{% for tag in post.tags.all %}
<a href="{% url 'blog:post_list_by_tag' tag.slug %}">
{{ tag.name }}
</a>
{% if not forloop.last %},{% endif %}
{% endfor %}
</p>
遍历一个帖子的所有标记,显示一个自定义链接到URL,以根据该标记过滤帖子。我们使用{% URL "blog:post_list_by_tag"标签构建URL。使用URL的名称和slug标记作为其参数。我们用逗号分隔标签。