限制Django Admin内联显示的查询集条目

13 投票
2 回答
11239 浏览
提问于 2025-04-16 21:35

在Django的管理后台中,我正在使用Django 1.2,想要添加一个InlineModelAdmin,以便在每次修改条目时添加评论(每次编辑都需要有一个“修改评论”)。

我不想显示之前的条目,所以我试图让ChangeCommentInline的表单集合(formset)为空。我创建了一个NoCommentsInlineFormset并把它分配给我的ChangeCommentInline,但仍然显示了已有的条目。

https://docs.djangoproject.com/en/1.3/topics/forms/modelforms/#changing-the-queryset

注意 - 在上面的链接中,他们使用的是BaseModelFormset,而我使用的是BaseInlineFormset,我觉得这可能是问题所在。如果我把BaseInlineFormset换成BaseModelFormset,就会出现“实例”不存在的错误。

admin.py

class NoCommentsInlineFormset(models.BaseInlineFormset):
    def __init__(self, *args, **kwargs):
        super(NoCommentsInlineFormset, self).__init__(*args, **kwargs)
        self.queryset = ChangeComment.objects.none()


class ChangeCommentInline(admin.StackedInline):
    model = ChangeComment
    extra = 1
    exclude = ("user", )
    formset = NoCommentsInlineFormset

    def save_model(self, request, obj, form, change):
        """auto-assign logined in user to comment"""
        if not change:
            obj.user = request.user
        obj.save()    


class EntryAdmin(admin.ModelAdmin):   
    inlines = (ChangeCommentInline, )

有没有办法限制在Inline中显示的ChangeComment条目,或者有没有更好的处理方法?

2 个回答

3

我假设你在你的 ChangeComment 模型中使用了一个

models.ForeignKey(EntryAdmin)

如果你想要每个 EntryAdmin 只对应一个评论,那你应该用一个:

models.OneToOneField(EntryAdmin)

这样的话,你就不需要 NoCommentsInlineFormset 这个东西,也不需要你的内联类了。这是我会这么做的。

编辑

好的,如果你想保留评论的历史记录,你可以在 NoCommentsInlineFormset 中重写查询集,像这样:

def __init__(self, *args, **kwargs):
    super(NoCommentsInlineFormset, self).__init__(*args, **kwargs)
    self.queryset = ChangeComment.objects.order_by('-created_at')[:1]

这样应该可以工作。

14

正如benjaoming在评论中提到的,我们需要重写get_queryset()这个方法,来修改InlineModelAdmin的行为。其实并不需要像我最开始想的那样,去重写并附加一个新的表单集合到InlineModelAdmin的定义上。

下面是最终的实现代码:

class ChangeCommentInline(admin.StackedInline):
    """For allowing logged in user to add change comment"""
    model = ChangeComment
    extra = 1
    exclude = ("user", ) # auto-update user field in save_formset method of parent modeladmin.


    def get_queryset(self, request):
        """Alter the queryset to return no existing entries"""
        # get the existing query set, then empty it.
        qs = super(ChangeCommentInline, self).get_queryset(request)
        return qs.none()  

撰写回答