Django: Как сделать форму условной?
У меня есть форма с двумя полями. От пользователя требуется выбрать только одно из двух. Не оба и не ни одного. Я попытался решить эту проблему, переписав чистый метод, как описано в Django Doc:
forms.py
class ConfigureWorkout(forms.Form):
first_exercise = forms.ModelChoiceField(empty_label="select", label="Choose First Exercise", queryset=FirstExercise.objects.all(), required=False)
sec_exercise = forms.ModelChoiceField(empty_label="select", label="Choose Sec Exercise", queryset=SecExercise.objects.all(), required=False)
def clean(self):
first_exercise = self.cleaned_data.get("first_exercise")
sec_exercise = self.cleaned_data.get("sec_exercise")
if first_exercise and sec_exercise:
raise forms.ValidationError("Enter either a First Exercise or a Secondary Exercise.")
else:
return self.cleaned_data
views.py
def configure(request):
configure_workout = ConfigureWorkout()
if request.method == "GET":
return render(request, "userprofile/some.html", configure_workout)
else:
return render(request, "app/other.html")
шаблон
<form action="{% url 'configure' %}" method="POST">
{% csrf_token %}
{{ configure_workout }}
<input type="submit" name="configuration_completed">
</form>
Однако, если я проверю это, выбрав оба поля в форме, ошибка не отображается/выдается. Я успешно прохожу форму и попадаю на страницу "other.html".
Что я упускаю? Заранее большое спасибо за любую помощь :)
Похоже, что вы не передаете фактические данные в форму. Возможно, это поможет:
def configure(request):
configure_workout = ConfigureWorkout()
if request.method == "GET":
return render(request, "userprofile/some.html", configure_workout)
else:
configure_workout = ConfigureWorkout(request.POST)
configure_workout.clean()
return render(request, "app/other.html")
Основываясь на ответе @trafalinos и изучив документацию по Forms, я бы предложил сделать следующее:
В forms.py
:
class ConfigureWorkout(forms.Form):
first_exercise = forms.ModelChoiceField(empty_label="select", label="Choose First Exercise", queryset=FirstExercise.objects.all(), required=False)
sec_exercise = forms.ModelChoiceField(empty_label="select", label="Choose Sec Exercise", queryset=SecExercise.objects.all(), required=False)
def clean(self):
cleaned_data = super().clean() # compare documentation
first_exercise = cleaned_data.get("first_exercise")
sec_exercise = cleaned_data.get("sec_exercise")
if first_exercise and sec_exercise:
raise forms.ValidationError("Enter either a First Exercise or a Secondary Exercise.")
else:
return self.cleaned_data
И (кредиты Трафалино) в views.py
:
def configure(request):
if request.method == "GET":
configure_workout = ConfigureWorkout()
return render(request, "userprofile/some.html", configure_workout)
else:
configure_workout = ConfigureWorkout(request.POST)
configure_workout.clean()
return render(request, "app/other.html")