4

我目前正在学习 Django,但我不知道如何自己解决这个问题。我正在阅读Developers Library - Python Web Development With Django一书,在一章中,您构建了一个简单的 CMS 系统,其中包含两个模型(故事和类别)、一些通用视图和自定义视图以及视图模板。

本书只包含列出故事、故事细节和搜索的代码。我想对此进行扩展并构建一个包含类别和故事的嵌套列表的页面。

- Category1
-- Story1
-- Story2
- Category2
- Story3 etc.

我设法弄清楚如何为类别列表添加我自己的通用 object_list 视图。我的问题是,如果 Story 是否公开,Story 模型具有 STATUS_CHOICES,并且默认情况下只会获取公共 Stories 的自定义管理器。我不知道如何告诉我的通用类别列表视图也使用自定义管理器并且只获取公共故事。除了那个小问题,一切正常。我可以为所有类别创建一个列表,并在单个页面上为该类别中的所有故事创建一个子列表,唯一的问题是该列表包含非公开故事。

我不知道我是否在正确的轨道上。我的 urls.py 包含一个通用视图,它获取所有 Category 对象,在我的模板中,我使用category.story_set.all来获取该类别的所有 Story 对象,然后循环遍历。

我认为可以在模板中添加一个 if 语句并使用我的模型文件中的 VIEWABLE_STATUS 来检查它是否应该被列出。该解决方案的问题在于它不是非常兼容 DRY。

是否也可以为 Category 模型添加某种管理器,仅在使用 category 上的 story_set 时才会获取公共 Story 对象?

或者这是攻击我的问题的错误方法?

相关代码

urls.py(仅类别列表视图):

urlpatterns += patterns('django.views.generic.list_detail',
    url(r'^categories/$', 'object_list', {'queryset': Category.objects.all(),
                                          'template_object_name': 'category'
                                         }, name='cms-categories'),

模型.py:

from markdown import markdown
import datetime
from django.db import models
from django.db.models import permalink
from django.contrib.auth.models import User

VIEWABLE_STATUS = [3, 4]

class ViewableManager(models.Manager):
    def get_query_set(self):
        default_queryset = super(ViewableManager, self).get_query_set()
        return default_queryset.filter(status__in=VIEWABLE_STATUS)

class Category(models.Model):
    """A content category"""
    label = models.CharField(blank=True, max_length=50)
    slug = models.SlugField()

    class Meta:
        verbose_name_plural = "categories"

    def __unicode__(self):
        return self.label

    @permalink
    def get_absolute_url(self):
        return ('cms-category', (), {'slug': self.slug})

class Story(models.Model):
    """A hunk of content for our site, generally corresponding to a page"""

    STATUS_CHOICES = (
        (1, "Needs Edit"),
        (2, "Needs Approval"),
        (3, "Published"),
        (4, "Archived"),
    )

    title = models.CharField(max_length=100)
    slug = models.SlugField()
    category = models.ForeignKey(Category)
    markdown_content = models.TextField()
    html_content = models.TextField(editable=False)
    owner = models.ForeignKey(User)
    status = models.IntegerField(choices=STATUS_CHOICES, default=1)
    created = models.DateTimeField(default=datetime.datetime.now)
    modified = models.DateTimeField(default=datetime.datetime.now)

    class Meta:
        ordering = ['modified']
        verbose_name_plural = "stories"

    def __unicode__(self):
        return self.title

    @permalink
    def get_absolute_url(self):
        return ("cms-story", (), {'slug': self.slug})

    def save(self):
        self.html_content = markdown(self.markdown_content)
        self.modified = datetime.datetime.now()
        super(Story, self).save()

    admin_objects = models.Manager()
    objects = ViewableManager()

category_list.html(相关模板):

{% extends "cms/base.html" %}
{% block content %}
    <h1>Categories</h1>
    {% if category_list %}
        <ul id="category-list">
        {% for category in category_list %}
            <li><a href="{{ category.get_absolute_url }}">{{ category.label }}</a></li>
            {% if category.story_set %}
                <ul>
                    {% for story in category.story_set.all %}
                        <li><a href="{{ story.get_absolute_url }}">{{ story.title }}</a></li>
                    {% endfor %}
                </ul>
            {% endif %}
        {% endfor %}
        </ul>
    {% else %}
        <p>
            Sorry, no categories at the moment.
        </p>
    {% endif %}
{% endblock %}
4

4 回答 4

11

我发现文档有点不清楚,但该属性可能是您正在寻找的。当您尝试它时,请注意此功能使用,它将始终指向在 Model 类中声明的第一个 Manager,无论它被赋予什么属性名称。您始终可以通过覆盖适当类中的方法来自定义管理站点中可编辑的项目。use_for_related_fields Manager_default_managerquerysetModelAdmin

做不到...

class Category(models.Model):
    def public_stories(self):
        return Story.objects.filter(category=self)

...和...

{% for story in category.public_stories %}

这至少避免了重复确定哪些故事是可见的逻辑。

于 2009-02-19T15:08:48.863 回答
4

如果您想要一个特定于模板的解决方案,您可以创建一个自定义过滤器。这样你就可以避免接触你的模型类

这将是您的过滤器:

from django import template

register = template.Library()

@register.filter
def viewable(queryset):
    return queryset.filter(status__in=VIEWABLE_STATUS)

你可以在你的模板中使用它

{% category.story_set|viewable %}
于 2009-10-10T18:29:44.877 回答
2

要详细说明@insin 的答案,您应该添加use_for_related_fields = TrueViewableManager课程中。这样,Category.story_set将返回自定义管理器的实例,该实例具有get_query_set您需要的方法。您的模板标签看起来像

{% category.story_set.get_query_set %}

编辑:@insin 的答案要好得多,我会使用它。

于 2009-02-19T15:21:12.910 回答
2

只需更改经理声明的位置即可。

admin_objects = models.Manager()
objects = ViewableManager()

objects = ViewableManager()
admin_objects = models.Manager()

根据文档

类的默认管理器是在该类上声明的第一个管理器(如果存在),或者是父层次结构中第一个抽象基类的默认管理器(如果存在)。如果没有显式声明默认管理器,则使用 Django 的正常默认管理器。

于 2010-03-13T01:04:55.287 回答