1

I've been scouring the internet for a couple days and couldn't find anything, so I'm hoping you guys can point me in the right direction. I'm trying to customize the django admin so that a button appears inline after a URL field. The button appears, and the javascript works, except despite marking it null=True and blank=True the admin validation keeps saying the field is required; I want the url field to be optional.

Is there any way to make this field optional? I'm assuming it's some combination of blank=True and null=True, but I've tried it in a handful of places with no luck.

Here are what I think to be the relevant code bits (also, I know inline CSS from the widget is a bad idea. It's only until I get everything working!). If you need to see anything else, please let me know.

models.py

class Team(models.Model):
    name = models.CharField(max_length=64)
    name_color = models.CharField(max_length=7, default='#000000')
    name_shadow_color = models.CharField(max_length=7, default='#ffffff')
    created = models.DateField(editable=True, default=datetime.now)
    retired = models.DateField(null=True, blank=True)
    url = models.URLField(null=True, blank=True, default=None)

admin.py

class TeamAdmin(admin.ModelAdmin):
    list_filter = ('created', 'retired',)
    list_select_related = True
    list_display = ('name', 'created',)
    search_fields = ('name', )
    ordering = ('name',)
    form = TeamAdminForm

admin_forms.py

class TeamAdminForm(forms.ModelForm):
    url = URLActionField()

    class Media:
        js = ('js/jquery-1.8.0.min.js', 'js/admin/teamform.js', )

    class Meta:
        model = Team

admin_widgets.py

class URLActionField(forms.TextInput):
    def render(self, name, value, attrs=None):
        if attrs is None:
            attrs = {}

        # TODO: not responsive!!
        if 'style' not in attrs.keys():
            attrs['style'] = 'width: 275px;'
        else:
            attrs['style'] = '%s width: 275px;' % attrs['style']

        attrs['required'] = False
        attrs['blank'] = True
        attrs['null'] = True

        output = []
        output.append(super(URLActionField, self).render(name, value, attrs))
        output.append(' <input type="button" value="%s" style="width: 200px; margin-left: 20px; height: 24px; line-height: 15px;" class="grp-button" id="url-scraper">' % unicode(_(u'Scrape URL for data')))
        return mark_safe(u''.join(output))

Thanks in advance.

leo
  • 339
  • 2
  • 13

1 Answers1

1

You need to make a custom widget and use it with the build-in URL field. You are dealing with a formfield and not a modelfield. So use 'required=False'. Avoid using null on string-based fields unless you have an excellent reason. If a string-based field has null=True, that means it has two possible values for “no data”: NULL, and the empty string.

In Model.py:

class Team(models.Model):
    ...
    url = models.URLField(blank=True)

In Admin.py append to the build-in AdminURLFieldWidget output (no js required):

from django.contrib.admin.widgets import AdminURLFieldWidget

class CustomAdminURLFieldWidget(AdminURLFieldWidget):
    def render(self, name, value, attrs=None):
        output = []
        output.append(super(CustomAdminURLFieldWidget, 
            self).render(name, value, attrs))
        if value:
            output.append('<p><a href="%s">%s</a></p>' %(value, value))
        return mark_safe(u''.join(output))

In Admin.py create a form:

from models import Team

class TeamAdminForm(forms.ModelForm):
        url = forms.URLField(required=False, widget=CustomAdminURLFieldWidget)
        class Meta:
            model = Team

In Admin.py create a ModelAdmin:

class TeamAdmin(admin.ModelAdmin):
    form = TeamAdminForm
allcaps
  • 10,945
  • 1
  • 33
  • 54