自定义 Django 管理界面中的多对多内联模型

1. 问题背景

在 Django 管理界面中,用户可以使用内联模型来管理一对多的关系。但是,当一对多关系是多对多时,Django 提供的默认内联模型可能并不适合。例如,如果存在一个产品模型和一个发票模型,并且产品和发票之间是多对多的关系,那么在发票的管理界面中,Django 会显示一个表格,其中包含所有产品及其对应的复选框。这种形式的内联模型对于管理少量产品还可以接受,但是如果产品数量很多,那么这种内联模型就会非常不美观和难以使用。

2. 解决方案

为了解决这个问题,我们可以自定义多对多内联模型的显示方式。具体步骤如下:

  1. 创建一个新的内联模型类。这个类继承自 admin.TabularInlineadmin.StackedInline

  2. 在新的内联模型类中,重写 get_formset() 方法。这个方法负责返回一个表单集,表单集中的每个表单对应于内联模型中的一个对象。

  3. get_formset() 方法中,使用 formset_factory() 函数创建表单集。在 formset_factory() 函数中,指定 model 参数为内联模型的模型类,并指定 fields 参数为内联模型中需要显示的字段。

  4. 在新的内联模型类中,重写 has_add_permission()has_change_permission() 方法。这两个方法分别负责判断用户是否有添加和修改内联模型对象

  5. 将新的内联模型类添加到 ModelAdmin 类中。在 ModelAdmin 类的 inlines 属性中,添加新的内联模型类。

下面是一个示例代码,演示了如何自定义多对多内联模型的显示方式:

python 复制代码
from django.contrib import admin
from django.contrib.admin.utils import NestedObjects

class Product(models.Model):
    name = models.TextField()
    price = models.DecimalField(max_digits=10, decimal_places=2)

class Invoice(models.Model):
    company = models.ForeignKey(Company)
    customer = models.ForeignKey(Customer)
    products = models.ManyToManyField(Product)

class InvoiceAdmin(admin.ModelAdmin):

    # 使用 formset_factory() 函数创建表单集
    def get_formset(self, request, obj=None, **kwargs):
        formset_class = super().get_formset(request, obj, **kwargs)
        formset_class = formset_factory(InvoiceProductFormset, extra=0,
                                        fields=('product',))
        return formset_class

    # 重写 has_add_permission() 和 has_change_permission() 方法
    def has_add_permission(self, request, obj=None):
        return False

    def has_change_permission(self, request, obj=None):
        return False

class InvoiceProductFormset(admin.BaseInlineFormSet):
    def __init__(self, *args, **kwargs):
        super().__init__(*args, **kwargs)
        NestedObjects(self)

    # 重写 get_queryset() 方法
    def get_queryset(self):
        qs = super().get_queryset()
        # 过滤掉已经被删除的对象
        return qs.filter(is_deleted=False)

# 将自定义的内联模型类添加到 ModelAdmin 类中
admin.site.register(Invoice, InvoiceAdmin)

在上面的代码中,我们首先创建了一个新的内联模型类 InvoiceProductFormset。这个类继承自 admin.BaseInlineFormSet。然后,我们在 InvoiceProductFormset 类中重写了 get_formset() 方法、has_add_permission() 方法和 has_change_permission() 方法。最后,我们将 InvoiceProductFormset 类添加到 InvoiceAdmin 类中。

相关推荐
C***11502 小时前
Spring TransactionTemplate 深入解析与高级用法
java·数据库·spring
+VX:Fegn08952 小时前
计算机毕业设计|基于springboot + vue建筑材料管理系统(源码+数据库+文档)
数据库·vue.js·spring boot·后端·课程设计
2301_800256112 小时前
B+树:数据库的基石 R树:空间数据的索引专家 四叉树:空间划分的网格大师
数据结构·数据库·b树·机器学习·postgresql·r-tree
大厂技术总监下海2 小时前
用户行为分析怎么做?ClickHouse + 嵌套数据结构,轻松处理复杂事件
大数据·数据结构·数据库
alonewolf_992 小时前
深入理解MySQL事务与锁机制:从原理到实践
android·数据库·mysql
朝依飞3 小时前
fastapi+SQLModel + SQLAlchemy2.x+mysql
数据库·mysql·fastapi
3***g2053 小时前
redis连接服务
数据库·redis·bootstrap
m0_598177233 小时前
SQL 方法函数(1)
数据库
oMcLin3 小时前
如何在Oracle Linux 8.4上通过配置Oracle RAC集群,确保企业级数据库的高可用性与负载均衡?
linux·数据库·oracle
信创天地3 小时前
核心系统去 “O” 攻坚:信创数据库迁移的双轨运行与数据一致性保障方案
java·大数据·数据库·金融·架构·政务