限制Django Admin内联显示的查询集条目
在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 个回答
我假设你在你的 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]
这样应该可以工作。
正如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()