首页 > 解决方案 > Django 验证错误引发但未显示在购物车应用程序中

问题描述

我正在运行 Django 2.2 并编写了一个简单的购物车。我希望同时验证两个字段,这样两个字段不能同时为空。在我的 forms.py 中,

from django import forms

class CartAddProductForm(forms.Form):
    cc_handle = forms.CharField(required=False, label='CC Handle', empty_value='')
    lc_handle = forms.CharField(required=False, label='LC Handle', empty_value='')

    def clean(self):
        cleaned_data = super().clean()
        if cleaned_data.get('cc_handle') == '' and cleaned_data.get('lc_handle') == '':
            print("issue detected")
            raise forms.ValidationError('Either cc or lc handle is required.')
        return cleaned_data

这是遵循官方Django 文档关于清理和验证相互依赖的字段的。上面的 print() 语句让我知道问题已被检测到,即两个字段都是空的。运行 Django 服务器,我看到确实检测到了问题,但原始页面顶部没有显示验证错误消息。原始页面是包含产品和将产品添加到购物车的链接的产品页面。通常验证错误消息显示在页面顶部。

根据文档,验证是在调用 is_valid() 时完成的。所以我把我的views.py的诊断打印

from django.shortcuts import render, redirect, get_object_or_404
from django.views.decorators.http import require_POST
from shop.models import Product
from .cart import Cart
from .forms import CartAddProductForm

@require_POST
def cart_add(request, product_id):
    cart = Cart(request)
    product = get_object_or_404(Product, id=product_id)
    form = CartAddProductForm(request.POST)
    if form.is_valid():
        cd = form.cleaned_data
        cart.add(product=product,
                cc_handle=cd['cc_handle'],
                lc_handle=cd['lc_handle'])
    else:
        print('invalid form')
    return redirect('cart:cart_detail')

确实弹出了“无效形式”一词。然后代码将我带到购物车。相反,我想要的是在产品页面上显示验证错误,通知读者这两个字段不能为空。有没有简单的方法呢?

对于表单中的 required=True 字段,如果我将其留空,则会弹出一条消息说我需要填写它。所以我想做类似的事情,除了验证要求两个字段不能为空。

这与这个Stackoverflow 答案不同,因为那是一个注册表单。您可以将其重定向到相同的表单,而对于这种情况,CartAddProductForm 嵌入在站点上的所有产品页面中。如果可能,我希望验证与 required=True 选项的字段在同一阶段进行。

product/detail.html 模板如下所示。

{% extends "shop/base.html" %}
{% load static %}

{% block title %}
{{ product.name }}
{% endblock %}

{% block content %}
<div class="product-detail">
    <img src="{% if product.image %}{{ product.image.url }}{% else %}{% static "img/no_image.png" %}{% endif %}">
    <h1>{{ product.name }}</h1>
    <h2><a href="{{ product.category.get_absolute_url }}">{{ product.category }}</a></h2>
    <p class="price">${{ product.price }}</p>
    <form action="{% url "cart:cart_add" product.id %}" method="post">
    {{ cart_product_form }}
    {% csrf_token %}
    <input type="submit" value="Add to cart">
    </form>
    {{ product.description|linebreaks }}
</div>
{% endblock %}

标签: pythondjangodjango-forms

解决方案


在表单模板中添加此行已解决您的问题。

{{ cart_product_form.non_field_errors }}

产品/detail.html:

{% extends "shop/base.html" %}
{% load static %}

{% block title %}
{{ product.name }}
{% endblock %}

{% block content %}
<div class="product-detail">
    <img src="{% if product.image %}{{ product.image.url }}{% else %}{% static "img/no_image.png" %}{% endif %}">
    <h1>{{ product.name }}</h1>
    <h2><a href="{{ product.category.get_absolute_url }}">{{ product.category }}</a></h2>
    <p class="price">${{ product.price }}</p>
    <form action="{% url "cart:cart_add" product.id %}" method="post">
    {{ cart_product_form }}
    {% csrf_token %}
    {{ cart_product_form.non_field_errors }} // This line will raise validation errors
    <input type="submit" value="Add to cart">
    </form>
    {{ product.description|linebreaks }}
</div>
{% endblock %}

文档:(从官方文档复制)

请注意,您的 Form.clean() 覆盖引发的任何错误都不会与任何特定字段相关联。它们进入一个特殊的“字段”(称为all),如果需要,您可以通过 non_field_errors() 方法访问该字段。如果要将错误附加到表单中的特定字段,则需要调用 add_error()。


推荐阅读