Django Forms: se não for válido, mostra o formulário com mensagem de erro

112

No formulário Django, ele pode verificar se o formulário é válido:

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

Mas estou perdendo o que fazer se não for válido? Como faço para retornar o formulário com as mensagens de erro? Não estou vendo o "outro" em nenhum dos exemplos.

user984003
fonte

Respostas:

242

Se você renderizar a mesma visualização quando o formulário não for válido, no modelo, você pode acessar os erros do formulário usandoform.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 %}

Um exemplo:

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})
Aamir Adnan
fonte
Eu adicionei um exemplo simples. Certifique-se de seguir a mesma abordagem que mencionei.
Aamir Adnan
1
Entendo. Retorno o mesmo formulário que recebi. As mensagens de erro foram adicionadas automaticamente a ele pela função is_valid ().
user984003 01 de
sim, você entendeu agora. Se você não renderizou o formulário manualmente, os erros serão mostrados automaticamente para cada campo.
Aamir Adnan
@AlexanderSupertramp myFormé uma instância de forms.Formou forms.ModelForm, leia sobre Django Forms
Aamir Adnan
E se eu não tiver uma visão .. por exemplo, usando um formulário de administração padrão dentro do CMS. Por exemplo, em uma UNIQUE constraint failed:exceção?
geoidésico
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()})

Se quiser mostrar os erros do formulário diferente daquele inválido basta colocar {{form.as_p}} como fiz abaixo

page.html

<html>
    <head>
        <script>
            {% if messages %}
                {% for message in messages %}
                    alert(message);
                {% endfor %}
            {% endif %}
        </script>
    </head>
    <body>
        {{form.as_p}}
    </body>
</html> 
Catarina
fonte
E então o que eu retorno? Como isso chega ao meu modelo?
user984003 01 de
Eu atualizo meu código. Você também pode colocar a mensagem de loop for em seu modelo em vez de no script, se desejar.
catherine
1
esta é uma abordagem legal - mas precisa estar alerta ('{{message}}');
amchugh89
Como você colocaria algo mais descritivo na mensagem de erro do modo de exibição do que 'Erro', como você fez messages.error(request, "Error")?
cbuch1800
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})
Lukasz Koziara
fonte
3

ATUALIZAÇÃO: Adicionada uma descrição mais detalhada dos erros do formset.


Form.errors combina todos os campos e non_field_errors. Portanto, você pode simplificar o html para isso:

modelo

    {% 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)

Uma advertência: Em contraste com os formulários, Formset.errors não inclui non_field_errors.

P. Maino
fonte
0

simplesmente você pode fazer assim porque, quando inicializou o formulário, em contém dados do formulário e dados inválidos também:

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

if irá gerar o erro no modelo se houver algum, mas os dados do formulário ainda permanecerão como:

error_demo_here

Dinesh Kc
fonte
-1

Você pode colocar simplesmente uma variável de sinalizador, neste caso 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})

depois em seu modelo você pode apenas colocar o código abaixo

{%if is_successed == 1 %}
<h1>{{preorder_date}}</h1>
<h2> {{product}}</h2>
{%endif %}
gtopal
fonte