36

众所周知,在 Django admin 中将方法返回值显示为布尔值很容易通过设置boolean属性来完成:

class MyModel(models.Model):
    def is_something(self):
        if self.something == 'something':
            return True
        return False
    is_something.boolean = True

您如何才能为属性实现相同的效果,例如以下情况?

class MyModel(models.Model):
    @property
    def is_something(self):
        if self.something == 'something':
            return True
        return False
4

5 回答 5

39

这是我找到的最简单的方法,直接在 ModelAdmin 中:

class MyModelAdmin(admin.ModelAdmin):
    def is_something(self, instance):
        return instance.something == "something"
    is_something.boolean = True
    is_something.short_description = u"Is something"

    list_display = ['is_something']
于 2016-12-14T15:40:43.813 回答
35

等待更好的解决方案出现,我通过以下方式解决了它:

class MyModel(models.Model):
    def _is_something(self):
        if self.something == 'something':
            return True
        return False
    _is_something.boolean = True
    is_something = property(_is_something)

然后我将_is_something在子类中引用该方法ModelAdmin

class MyModelAdmin(admin.ModelAdmin):
    list_display = ['_is_something']

is_something否则财产:

if my_model_instance.is_something:
    print("I'm something")
于 2012-10-11T14:42:10.773 回答
15

您需要为shadowing模型中的属性创建一个函数。我的意思是,您将需要在 ModelAdmin 类中重新创建一个与主模型中定义的属性同名的函数。

例子:

# Model
class Product(models.Model):

    @property  # you can omit this decorator if you will access this property as a method of the model instance
    def in_stock(self):
        # boolean check return
        return self.quantity > 0

...

# Django-modeladmin
class ProductAdmin(admin.ModelAdmin):
    list_display = ('in_stock', ...)
    def in_stock(self, instance):
        return instance.in_stock

    in_stock.boolean = True        
于 2018-06-26T23:21:42.720 回答
3

你可以像这样创建一个装饰器

from six.moves import reduce

def list_property(field_name, **kwargs):
    def _from_property(obj):
        rv = reduce(getattr, field_name.split("."), obj)
        return rv() if callable(rv) else rv

    for key, value in kwargs.items():
        setattr(_from_property, key, value)
    return _from_property

这是您的模型和管理员定义:

# model

class MyModel(models.Model):
    @property
    def is_something(self):
        if self.something == 'something':
            return True
        return False


# admin

class MyModelAdmin(admin.ModelAdmin):
    list_display = [list_property("is_something", boolean=True)]

对于 modeladmin 中的只读字段,您可以使用此装饰器:

def field_property(field_name, **kwargs):
    def _from_property(admin, obj=None):
        if not obj:
            return None
        rv = reduce(getattr, field_name.split("."), obj)
        return rv() if callable(rv) else rv

    for key, value in kwargs.items():
        setattr(_from_property, key, value)
    return _from_property

# admin
class MyModelAdmin(admin.ModelAdmin):
    readonly_fields = ["is_something"]

    is_something = field_property("is_something", boolean=True)
于 2019-03-19T05:42:44.353 回答
-1

如果定义is_something为属性,它将是不可变对象,而不是函数,但该对象包含对属性中修饰的 getter 的引用fget。我认为 Django 管理界面使用该属性的 getter,因此这可能有效

class MyModel(models.Model):
    @property
    def is_something(self):
        if self.something == 'something':
            return True
        return False
    is_something.fget.boolean = True
于 2012-10-11T15:44:36.047 回答