Django Forms: если недействителен, показать форму с сообщением об ошибке

112

I Django forms, он может проверить, действительна ли форма:

if form.is_valid(): 
    return HttpResponseRedirect('/thanks/')

Но мне не хватает, что делать, если это недействительно? Как мне вернуть форму с сообщениями об ошибках? Я не вижу «еще» ни в одном из примеров.

пользователь984003
источник

Ответы:

242

Если вы визуализируете то же представление, когда форма недействительна, тогда в шаблоне вы можете получить доступ к ошибкам формы, используяform.errors .

{% if form.errors %}
    {% for field in form %}
        {% for error in field.errors %}
            <div class="alert alert-danger">
                <strong>{{ error|escape }}</strong>
            </div>
        {% endfor %}
    {% endfor %}
    {% for error in form.non_field_errors %}
        <div class="alert alert-danger">
            <strong>{{ error|escape }}</strong>
        </div>
    {% endfor %}
{% endif %}

Пример:

def myView(request):
    form = myForm(request.POST or None, request.FILES or None)
    if request.method == 'POST':
        if form.is_valid():
            return HttpResponseRedirect('/thanks/')
    return render(request, 'my_template.html', {'form': form})
Аамир Аднан
источник
Я добавил простой пример. Убедитесь, что вы придерживаетесь того же подхода, о котором я упоминал.
Аамир Аднан
1
Понимаю. Я возвращаю ту же форму, в которой получил. Сообщения об ошибках автоматически добавлялись в нее функцией is_valid ().
user984003 01
да, вы получили это прямо сейчас. Если вы не отображали форму вручную, ошибки будут автоматически отображаться для каждого поля.
Аамир Аднан,
@AlexanderSupertramp myForm- это экземпляр forms.Formили forms.ModelForm, прочтите о Django Forms
Аамир Аднан,
Что делать, если у меня нет представления ... например, используя стандартную форму администратора в CMS. Например по UNIQUE constraint failed:исключению?
geoidesic 05
19

views.py

from django.contrib import messages 

def view_name(request):
    if request.method == 'POST':
        form = form_class(request.POST)
        if form.is_valid():
            return HttpResponseRedirect('/thanks'/)
        else:
            messages.error(request, "Error")
return render(request, 'page.html', {'form':form_class()})

Если вы хотите показать ошибки формы, отличные от недействительной, просто введите {{form.as_p}}, как я сделал ниже

page.html

<html>
    <head>
        <script>
            {% if messages %}
                {% for message in messages %}
                    alert(message);
                {% endfor %}
            {% endif %}
        </script>
    </head>
    <body>
        {{form.as_p}}
    </body>
</html> 
Екатерина
источник
И что тогда мне вернуть? Как это попадает в мой шаблон?
user984003 01
Я обновляю свой код. Вы также можете поместить сообщение цикла в свой шаблон вместо скрипта, если хотите.
Екатерина
1
это крутой подход, но он должен быть внимательным ('{{message}}');
amchugh89
Как бы вы поместили что-то более информативное в сообщение об ошибке из представления, чем «Ошибка», как вы это сделали messages.error(request, "Error")?
cbuch1800 04
3
def some_view(request):
    if request.method == 'POST':
        form = SomeForm(request.POST)
        if form.is_valid():
            return HttpResponseRedirect('/thanks'/)
    else:
        form = SomeForm()
    return render(request, 'some_form.html', {'form': form})
Лукаш Козиара
источник
3

ОБНОВЛЕНИЕ: Добавлено более подробное описание ошибок набора форм.


Form.errors объединяет все поля и non_field_errors. Поэтому вы можете упростить html до следующего:

шаблон

    {% load form_tags %}

    {% if form.errors %}
    <div class="alert alert-danger alert-dismissible col-12 mx-1" role="alert">
        <div id="form_errors">
            {% for key, value in form.errors.items %}
                <span class="fieldWrapper">
                    {{ key }}:{{ value }}
                </span>
            {% endfor %}
        </div>
        <button type="button" class="close" data-dismiss="alert" aria-label="Close">
            <span aria-hidden="true">&times;</span>
        </button>
    </div>
    {% endif %}


If you want to generalise it you can create a list_errors.html which you include in every form template. It handles form and formset errors:

    {% if form.errors %}
    <div class="alert alert-danger alert-dismissible col-12 mx-1" role="alert">
        <div id="form_errors">

            {% for key, value in form.errors.items %}
                <span class="fieldWrapper">
                    {{ key }}:{{ value }}
                </span>
            {% endfor %}
        </div>
        <button type="button" class="close" data-dismiss="alert" aria-label="Close">
            <span aria-hidden="true">&times;</span>
        </button>
    </div>
    {% elif formset.total_error_count %}
    <div class="alert alert-danger alert-dismissible col-12 mx-1" role="alert">
        <div id="form_errors">
            {% if formset.non_form_errors %}
                {{ formset.non_form_errors }}
            {% endif %}
            {% for form in formset.forms %}
                {% if form.errors %}
                    Form number {{ forloop.counter }}:
                    <ul class="errorlist">
                    {% for key, error in form.errors.items %}
                        <li>{{form.fields|get_label:key}}
                            <ul class="errorlist">
                                <li>{{error}}</li>
                            </ul>
                        </li>
                    {% endfor %}
                    </ul>
                {% endif %}
            {% endfor %}

        </div>
    </div>

    {% endif %}

form_tags.py

from django import template

register = template.Library()


def get_label(a_dict, key):
    return getattr(a_dict.get(key), 'label', 'No label')


register.filter("get_label", get_label)

Одно предостережение: в отличие от форм Formset.errors не включает non_field_errors.

П. Майно
источник
0

просто вы можете сделать это, потому что когда вы инициализировали форму, она также содержит данные формы и недопустимые данные:

def some_func(request):
    form = MyForm(request.POST)
    if form.is_valid():
         //other stuff
    return render(request,template_name,{'form':form})

if вызовет ошибку в шаблоне, если она есть, но данные формы все равно останутся такими:

error_demo_here

Динеш Кс
источник
0

@ Ответ Аамира Аднана отсутствует field.label; другой способ показать ошибки в нескольких строках.

{% if form.errors %}
    <!-- Error messaging -->
    <div id="errors">
        <div class="inner">
            <p>There were some errors in the information you entered. Please correct the following:</p>
            <ul>
                {% for field in form %}
                    {% if field.errors %}<li>{{ field.label }}: {{ field.errors|striptags }}</li>{% endif %}
                {% endfor %}
            </ul>
        </div>
    </div>
    <!-- /Error messaging -->
{% endif %}
Лакшмикант Ратнапархи
источник
-1

Вы можете просто указать переменную флага, в данном случае is_successed .

def preorder_view(request, pk, template_name='preorders/preorder_form.html'):
    is_successed=0
    formset = PreorderHasProductsForm(request.POST)
    client= get_object_or_404(Client, pk=pk)
    if request.method=='POST':
        #populate the form with data from the request
       # formset = PreorderHasProductsForm(request.POST)
        if formset.is_valid():
            is_successed=1
            preorder_date=formset.cleaned_data['preorder_date']
            product=formset.cleaned_data['preorder_has_products']
            return render(request, template_name, {'preorder_date':preorder_date,'product':product,'is_successed':is_successed,'formset':formset})



    return render(request, template_name, {'object':client,'formset':formset})

после этого в свой шаблон вы можете просто поместить код ниже

{%if is_successed == 1 %}
<h1>{{preorder_date}}</h1>
<h2> {{product}}</h2>
{%endif %}
gtopal
источник