这几天又在折腾admin这颗Django皇冠上的明珠... 这次折腾的还是跟附件相关,不过主要是图片,也就是在表单里显示上传的图片。

查看了一下Django admin的模板,发现对于表单的处理,只用了几个简单的条件判断和循环:

{% if fieldset.name %}

{{ fieldset.name }}

{% endif %} {% if fieldset.description %}
{{ fieldset.description|safe }}
{% endif %} {% for line in fieldset %}
{{ line.errors }} {% for field in line %} {% if field.is_checkbox %} {{ field.field }}{{ field.label_tag }} {% else %} {{ field.label_tag }}{{ field.field }} {% endif %} {% if field.field.field.help_text %} {{ field.field.field.help_text|safe }} {% endif %}
{% endfor %} {% endfor %}

并没有出现对表单元素的相关处理,也就是说相关表单元素的已经在{{ field.label_tag }}{{ field.field }}中包装好,模板只负责显示,遂转移目标。

翻文档,在ModelAdmin Options里发现formfield_overrides,用法如下:

from django.db import models
from django.contrib import admin

Import our custom widget and our model from where they're defined

from myapp.widgets import RichTextEditorWidget from myapp.models import MyModel

class MyModelAdmin(admin.ModelAdmin): formfield_overrides = { models.TextField: {'widget': RichTextEditorWidget}, }

也就是说可以用widget的方式来进行包装,继续翻Django的代码,发现奥秘:在Django/contrib/admin/widgets.py中,有对各种input的包装!哈,有戏,于是自己写了一个widget,其实也就是原本照搬Django自带的那个AdminFileWidget,将其中的a改成img:
def render(self, name, value, attrs=None):
        output = []
        if value and hasattr(value, "url"):
            output.append('%s 
%s ' % 
                (_('Currently:'), value.url,  _('Change:')))
        output.append(super(AdminFileWidget, self).render(name, value, attrs))
        return mark_safe(u''.join(output))
然后在formfield_overrides中填写自己写的这个widget即可。

但是这样做有个问题,就是使用formfield_overrides处理的话,表单中所有的同类型input(比如上面的TextField)都会被这么包装一次,这显然是不行的,那怎么解决呢? 别着急,在在ModelAdmin Options里还有一个参数,就是ModelAdmin.form,可以自定义一个form,而不是Django admin默认调用model生成表单,那就好办了,参照这里就可以很简单的重写指定的一个input,避免上面的情况~~~

应为Django的资料相对来说比较少(特别是中文资料),加上自己对Django了解也比较肤浅,实现上面的这个效果费了小周折,不过通过这次,对Django的代码结构有了进一步的了解,相信以后应用起来要轻松很多了~~