django multivaluefield & multiwidget - сделайте один необязательный

Это связано с моим более ранним вопросом.

Я хочу иметь MultiValueField, который содержит виджет Choice и TextInput. Если пользователь выбирает "ДРУГОЙ" из выбора, то значение TextInput должно быть сохранено. В противном случае значение выбора должно быть сохранено. Пока у меня есть следующий код:

custom_choices = [("one","one"),("two","two"),("OTHER","OTHER")]

class MyMultiWidget(forms.MultiWidget):
    def __init__(self,*args,**kwargs):
        widgets = (
            forms.Select(choices=custom_choices),
            forms.TextInput(),
        )
        super(MyMultiWidget, self).__init__(widgets,*args,**kwargs)

    def decompress(self, value):
        if value:
             return value.split("|")
        return ['', '']

class MyMultiValueField(forms.MultiValueField):
    def __init__(self, *args, **kwargs):
        fields = (
            forms.CharField(max_length=128,required=True),
            forms.CharField(max_length=128,required=False),
        )
        super(MyMultiValueField, self).__init__(fields, *args, **kwargs)
        self.widget = TestMultiWidget()

    def compress(self, data_list):
        if data_list:
            return '|'.join(data_list)

class MyTestField(models.CharField):
    def formfield(self, **kwargs):
        return super(MyTestField, self).formfield(form_class=MyMultiValueField)

class MyModel(models.Model):
    myField = MyTestField()

Однако всякий раз, когда я пытаюсь сохранить что-то, что не содержит значения в TextInput, я получаю ошибку проверки "Это поле обязательно для заполнения". Это несмотря на "требуемый = ложный" кварг выше.

Я попытался добавить немного логики в чистую функцию; игнорировать виджет TextInput, если значение виджета Choice равно "OTHER":

def clean(self,value):
    if value[0]!="OTHER":
        value[1]=u''
    else:
        if not value[1]:
            msg = "unspecified value"
            raise forms.ValidationError(msg)
        elif "|" in value[1]:
            msg = "bad value ('|' character is not allowed"
            raise forms.ValidationError(msg)
    super(TestFormField,self).clean(value)

Но это не имеет никакого эффекта.

Какие-либо предложения?

1 ответ

Решение

Правильная версия clean должна выглядеть так:

def clean(self,value):
    if value[0] != "OTHER":
        value[1] = u' '
    else:
        if value[1].strip() == u' ':
            msg = "unspecified value"
            raise forms.ValidationError(msg)
        elif "|" in value[1]:
            msg = "bad value ('|' character is not allowed)"
            raise forms.ValidationError(msg)
    return "|".join(value)
Другие вопросы по тегам