是否可以构建自定义模型字段/窗口小部件组合,该组合显示值但从不将任何内容写回数据库?我会在管理员的表单中专门使用这个小部件。
我编写了自己的字段,它覆盖了formfield()
方法以声明自己的窗口小部件类。它显示得很好,但只要在管理员中单击“保存”按钮,我就会收到验证错误:
This field is required.
这是有道理的,考虑到我的小部件没有呈现出表单字段。但是,我想要做的是基本上从更新过程中删除此字段:无论何时在管理员中使用,都不应该在SQL UPDATE中提及它。
这可能吗?
这是我到目前为止的代码草图:
class MyWidget(Widget):
def render(self, name, value, attrs=None):
if value is None:
value = ""
else:
# pretty print the contents of value here
return '<table>' + ''.join(rows) + '</table>'
class MyField(JSONField):
def __init__(self, *args, **kwargs):
kwargs['null'] = False
kwargs['default'] = list
super(MyField, self).__init__(*args, **kwargs)
def formfield(self, **kwargs):
defaults = {
'form_class': JSONFormField,
'widget': MyWidget,
}
defaults.update(**kwargs)
return super(MyField, self).formfield(**defaults)
更新1:用例是该字段代表审计日志。在内部,它将定期写入。然而,管理员永远不需要写入它,它只需要以非常易读的格式呈现它。
我没有在应用程序中使用任何其他ModelForms,因此admin是唯一的表单用户。我不想在管理类本身上实现这种行为,因为这个字段将在各种模型中重用,并且总是应该以相同的方式运行。
答案 0 :(得分:8)
有多种方法可以在管理页面中创建只读字段。您对数据库存储的要求有点模糊,所以我会查看选项。
您必须先在admin.py
注册一个AdminModel:
from django.contrib import admin
from yourapp.models import YourModel
class YourAdmin(admin.ModelAdmin):
pass
admin.site.register(YourModel, YourAdmin)
现在您可以为其添加不同的行为。例如,您可以添加编辑/添加页面中显示的字段列表:
class YourAdmin(admin.ModelAdmin):
fields = ['field1', 'field2']
这可以是模型字段,模型属性或模型方法的名称。方法以只读方式显示。
如果你想让一个字段只读,请明确添加:
class YourAdmin(admin.ModelAdmin):
fields = ['field1', 'field2']
readonly_fields = ['field2']
然后,您可以选择通过添加具有相同名称的方法来完全覆盖字段的显示。您甚至不需要具有该名称的模型字段/方法,然后:
class YourAdmin(admin.ModelAdmin):
fields = ['field1', 'field2']
readonly_fields = ['field2']
def field2(self, obj):
return '*** CLASSIFIED *** {}'.format(obj.field2)
使用django.utils.safestring.mark_safe
,您也可以返回HTML代码。
管理员的所有其他选项均可用,但小部件配置除外,因为它仅适用于可写字段。
答案 1 :(得分:1)
我可能会对你想要的东西感到困惑,但你可能想看看模型属性。这是我当前项目的一个例子。
模型中的代码:
class Textbook(models.Model):
#other fields
@property
def NumWishes(self):
return self.wishlist_set.count()
然后你可以在管理页面上显示它。
class Textbook_table(admin.ModelAdmin):
fields = ["""attributes that are saved in the model"""]
list_display = ("""attributes that are saved in the model""", 'NumWishes'')
现在我可以在管理页面中显示NumWishes,但不需要使用该模型创建。