Django Admin 嵌套内联

Posted

技术标签:

【中文标题】Django Admin 嵌套内联【英文标题】:Django Admin nested inline 【发布时间】:2012-12-27 19:11:03 【问题描述】:

我需要一个嵌套的 django admin inline, 我可以在下面的其他内联中包含日期字段内联。

我有以下型号:

class Person(models.Model):
     name = models.CharField(max_length=200)
     id_no = models.IntegerField()

class Certificate(models.Model):
     cerfificate_no = models.CharField(max_length=200)
     certificate_date = models.DateField(max_length=100)
     person = models.ForeignKey(Person)
     training = models.CharField(max_length=200)

class Training_Date(models.Model):
      date = models.DateField()
      certificate = models.ForeignKey(Certificate)

和下面的管理员:

class CertificateInline(admin.StackedInline):
    model = Certificate

class PersonAdmin(admin.ModelAdmin):
     inlines = [CertificateInline,]
admin.site.register(Person,PersonAdmin)

但我需要将 Training_Date 模型作为内联包含在内,这是证书管理内联的一部分。

有什么想法吗?

【问题讨论】:

【参考方案1】:

更新的解决方案(2021 年 2 月)是使用 show_change_link 配置变量:https://docs.djangoproject.com/en/3.1/ref/contrib/admin/#django.contrib.admin.InlineModelAdmin.show_change_link

这与上述解决方案中提出的 EditLinkToInlineObject 完全相同,但代码更少,并且可能已经过 Django 开发人员的良好测试

您只需在每个内联中定义 show_change_link=True

更新(2022 年 1 月 25 日): 这是文档中的更新链接(Django 4.0):https://docs.djangoproject.com/en/4.0/ref/contrib/admin/#django.contrib.admin.InlineModelAdmin.show_change_link

【讨论】:

【参考方案2】:

https://code.djangoproject.com/ticket/9025最近有一些动静,但我不会屏住呼吸。

解决此问题的一种常见方法是通过为同一模型同时拥有 ModelAdmin 和内联来链接到第一和第二(或第二和第三)级别之间的管理员:

Certificate 一个ModelAdminTrainingDate 作为内联。为CertificateInline 提供一个附加字段“详细信息”,该字段是指向其ModelAdmin 更改表单的链接。

models.py:

from django.core import urlresolvers

class Certificate(models.Model):
    
    # ...
    
    def changeform_link(self):
        if self.id:
            # Replace "myapp" with the name of the app containing
            # your Certificate model:
            changeform_url = urlresolvers.reverse(
                'admin:myapp_certificate_change', args=(self.id,)
            )
            return u'<a href="%s" target="_blank">Details</a>' % changeform_url
        return u''
    changeform_link.allow_tags = True
    changeform_link.short_description = ''   # omit column header

admin.py:

# Certificate change form has training dates as inline

class TrainingDateInline(admin.StackedInline):
    model = TrainingDate

class CertificateAdmin(admin.ModelAdmin):
    inlines = [TrainingDateInline,]
admin.site.register(Certificate ,CertificateAdmin)

# Person has Certificates inline but rather
# than nesting inlines (not possible), shows a link to
# its own ModelAdmin's change form, for accessing TrainingDates:

class CertificateLinkInline(admin.TabularInline):
    model = Certificate
    # Whichever fields you want: (I usually use only a couple
    # needed to identify the entry)
    fields = ('cerfificate_no', 'certificate_date', 'changeform_link')
    readonly_fields = ('changeform_link', )

class PersonAdmin(admin.ModelAdmin):
    inlines = [CertificateLinkInline,]
admin.site.register(Person, PersonAdmin)

【讨论】:

这是一个不错的解决方案。我想指出您可以将changeform_link 放在CertificateLinkInline 中。那可能是一个更好的地方,因为它是 django-admin 特定的。请注意,当您这样做时,您应该使用instance.id 而不是self.id 来访问模型的实例。见docs.djangoproject.com/en/dev/ref/contrib/admin/… 如果 changeform_link 方法产生异常,那么 django 将吃掉它并继续,将字段留空。我永远无法找到 django 将回溯放在哪里,或者它是否对它做了任何事情。我建议将其包装在 try/except 中,以确保在某处记录异常。为简化起见,创建一个装饰器来处理所有这些可能会很有用。 看起来在 Django 2+ 上,您只需将 show_change_link = True 添加到您的 admin.TabularInline 类,我认为只要您也有一个常规的 admin.ModelAdmin ,它就会显示“更改” 内联中的链接。 哇?,完美的解决方案。 虽然使用show_change_link = True 要简单得多。【参考方案3】:

更通用的解决方案

from django.utils.safestring import mark_safe
from django.urls import reverse

class EditLinkToInlineObject(object):
    def edit_link(self, instance):
        url = reverse('admin:%s_%s_change' % (
            instance._meta.app_label,  instance._meta.model_name),  args=[instance.pk] )
        if instance.pk:
            return mark_safe(u'<a href="u">edit</a>'.format(u=url))
        else:
            return ''

class MyModelInline(EditLinkToInlineObject, admin.TabularInline):
    model = MyModel
    readonly_fields = ('edit_link', )

class MySecondModelAdmin(admin.ModelAdmin):
    inlines = (MyModelInline, )

admin.site.register(MyModel)
admin.site.register(MySecondModel, MySecondModelAdmin)

【讨论】:

简单灵活。 +1 本文中更详细地描述了一个非常相似的解决方案:djangotricks.blogspot.com/2016/12/…【参考方案4】:

我使用了@bigzbig 提供的解决方案(谢谢)。

我还想在保存更改后返回第一个列表页面,因此添加了:

class MyModelInline(EditLinkToInlineObject, admin.TabularInline):
    model = MyModel
    readonly_fields = ('edit_link', )

    def response_post_save_change(self, request, obj):
        my_second_model_id = MyModel.objects.get(pk=obj.pk).my_second_model_id
        return redirect("/admin/mysite/mysecondmodel/%s/change/" % (my_second_model_id))

【讨论】:

感谢sn-p!我只是想补充一点,这仅在 MyModelAdmin 中包含 post_save_change 函数后才对我有用。【参考方案5】:

嵌套内联在以下位置提供: https://github.com/BertrandBordage/django-super-inlines/

pip install django-super-inlines

【讨论】:

我试过了。它不允许我添加或删除一些内联,并迫使我制作 3 个新的子对象。在这个开发阶段它不可用。【参考方案6】:
pip install django-nested-inline

这个包应该能满足你的需要。

【讨论】:

django-nested-inline 在最新的 django 版本上(还没有?)支持。但是你可以考虑使用github.com/theatlantic/django-nested-admin,这几乎是一样的。 @vmonteco 你能给我一个文档链接吗,其中描述了如何在没有包的情况下使用它? @OleksandrDashkov django-nested-admin.readthedocs.io/en/latest,在 github 页面的顶部。但可能仍然需要安装软件包。 警告,我开始使用 django-nested-inline 并且遇到了多个错误,这些错误需要很长时间才能弄清楚。在决定使用哪个模块之前,您可能需要评估其他选项。 @pspahn 你有没有找到任何合理的替代方案【参考方案7】:

AFAIK,您不能在默认的 Django 管理员中拥有第二级内联。

Django admin 只是一个普通的 Django 应用程序,所以没有什么能阻止您实现第二级嵌套表单,但恕我直言,这将是一种复杂的实现设计。也许这就是为什么没有提供它的原因。

【讨论】:

我面临着类似的情况。我想我将覆盖内联模板并添加一些链接到第二级。

以上是关于Django Admin 嵌套内联的主要内容,如果未能解决你的问题,请参考以下文章

Django Admin Cookbook-23如何在Django admin中添加嵌套的内联

Django 嵌套内联 TemplateDoesNotExist

在 django admin 中为内联项目添加完整更改表单的链接?

Django Admin 中的内联

Django admin - 内联内联(或者,一次编辑三个模型)

如何在 django admin 上添加只读内联