Django集成第三方标签功能

django-taggit模块是一个可重用的应用程序,它主要提供一个标签模型和一个管理器,可以轻松地向任意模型添加标签。
https://github.com/alex/django-taggit

目录

安装django-taggit

添加taggit到setting.py中的INSTALLED_APPS

编辑models.py,将标签应用到post模型

更新数据库

重构blog应用程序的views.py文件

添加标签路径到urls.py

更新list.html模版,为标签增加链接


安装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标记作为其参数。我们用逗号分隔标签。