Django: Forcing admin users to enter at least one

2019-03-16 06:12发布

In my admin for an object Chair I have a TabularInline for an arbitrary number of Desk objects. I want every Chair to always have at least one Desk object associated with it. Is there a way to make the admin interface force the user to enter at least one Desk? Like show an error if no Desk is entered?

标签: django admin
3条回答
手持菜刀,她持情操
2楼-- · 2019-03-16 06:20

Matthew Flanagan has a great example of how to require one valid form in a formset: http://code.google.com/p/wadofstuff/wiki/WadOfStuffDjangoForms and http://wadofstuff.blogspot.com/2009/08/requiring-at-least-one-inline-formset.html Hope that helps you out.

查看更多
仙女界的扛把子
3楼-- · 2019-03-16 06:29

A generic FormSet clean() method for requiring at least one item:

    def clean(self):
        """Check that at least one service has been entered."""
        super(MyFormSet, self).clean()
        if any(self.errors):
            return
        if not any(cleaned_data and not cleaned_data.get('DELETE', False)
                   for cleaned_data in self.cleaned_data):
            raise forms.ValidationError('At least one item required.')

This should work for plain formsets, model formsets and in-line model formsets.

查看更多
叼着烟拽天下
4楼-- · 2019-03-16 06:35

Using akaihola's answer, here is a more complete example:

Add this to your forms.py:

from django import forms
from django.forms.models import BaseInlineFormSet

class AtLeastOneRequiredInlineFormSet(BaseInlineFormSet):

    def clean(self):
        """Check that at least one service has been entered."""
        super(AtLeastOneRequiredInlineFormSet, self).clean()
        if any(self.errors):
            return
        if not any(cleaned_data and not cleaned_data.get('DELETE', False)
            for cleaned_data in self.cleaned_data):
            raise forms.ValidationError('At least one item required.')

And then, in your admin.py:

class DeskInline(admin.TabularInline):
    model = Desk
    formset = AtLeastOneRequiredInlineFormSet

class ChairAdmin(admin.ModelAdmin):
    inlines = [DeskInline,]

admin.site.register(Chair, ChairAdmin)
查看更多
登录 后发表回答