将Django表单中的所有CharField Form Field输入转换为小写



我使用Django表单的用户注册,其中用户可以输入优惠券代码。我希望在优惠券代码字段中输入的所有字符都转换为小写。我尝试过在保存方法、自定义清理方法和自定义验证器中使用.lower(),但我没有使用这些方法。下面是我的代码。

class StripeSubscriptionSignupForm(forms.Form):
    coupon = forms.CharField(max_length=30,
        required=False,
        validators=[validate_coupon],
        label=mark_safe("<p class='signup_label'>Promo Code</p>")
    def save(self, user):
        try:
            customer, created = Customer.get_or_create(user)
            customer.update_card(self.cleaned_data["stripe_token"])
            customer.subscribe(self.cleaned_data["plan"], self.cleaned_data["coupon"].lower())
        except stripe.StripeError as e:
            # handle error here
            raise e

如上所述,我还尝试了一种清理方法,但这也不起作用:

def clean_coupon(self):
    return self.cleaned_data['coupon'].lower()

解决方案是创建一个自定义表单字段,它允许您重写to_python方法,然后可以修改表单字段中的原始值。

class CouponField(forms.CharField):
    def to_python(self, value):
        return value.lower()

class StripeSubscriptionSignupForm(forms.Form):
    coupon = CouponField(max_length=30,
        required=False,
        validators=[validate_coupon],
        label=mark_safe("<p class='signup_label'>Promo Code</p>")
    )

尝试在表单中使用css文本转换工具,如下所示:

class StripeSubscriptionSignupForm(forms.Form):
    coupon = forms.CharField(max_length=30,
        required=False,
        validators=[validate_coupon],
        label=mark_safe("<p class='signup_label'>Promo Code</p>")
        widget=TextInput(attrs={'style': 'text-transform:lowercase;'})
        )

我自己在确保用户模型中的email字段仅保存为小写时遇到了这个问题。我在下面概述的方法的优点是,您可以控制表单中每个字段的格式—与上面选择的答案相反,无论您是否愿意,它都会将所有字段转换为小写。

对我来说,我相信上面的OP的问题是,清理后的值现在确实是小写的,但是HTML页面(在验证和清理后呈现的那个)显示了预先清理的值(即仍然是大写的),这会让用户感到困惑。发生的事情是,表单字段的值仍然是初始数据,即X@Y.com和清理后的数据实际上是x@y.com。

处理提交的表单后:

>>>user_form.cleaned_data['email']
'x@y.com'

>>>user_form['email'].value()
'X@Y.com'

模板使用user_form['email'].value()代替user_form提供的值。Cleaned_data ['email'],因此用户认为他的电子邮件是以大写形式保存的,而实际上是以小写形式保存的。

在这种情况下,将user_form连同模板中出现的已清理字段一起呈现给客户机的最简单方法是在保存后直接重新加载保存的表单。根据以下两个示例(一个保存到数据库,另一个不保存到数据库)

forms.py

from django.contrib.auth.models import User
class UserForm(forms.ModelForm):
    """
    UserForm is a simple form to allow a user to change his/her name and email.
    """
    class Meta:
        model = User 
        fields = ['first_name', 'last_name', 'email']
    def clean_email(self):
        """
        ensure that email is always lower case.
        """
        return self.cleaned_data['email'].lower()
在views.py

def post(self, request):
    user_form = UserForm(request.POST, instance=request.user)
    if user_form.is_valid():
        user_form.save()  # using the provided save from Modelform in this case
        user_form = UserForm(instance=request.user)  # reload the amended user data
    return render(request, 'myApp/user_details.html',{'user_form': user_form})

这里的关键行在views.py中,user_form = UserForm(instance=request.user),表单在这里被重新加载。这样做的效果是在将清理过的(在本例中是保存的)数据呈现给用户之前重新填充表单。现在,您可以通过对这些字段调用clean_fieldname,将表单中的每个字符字段更改为小写。

注意:如果您不与数据库交互(或者只是不希望从数据库中重新加载),您可以按照以下方式重新填充表单:

def post(self, request):
    user_form = UserForm(request.POST) #gather the post'ed data
    if user_form.is_valid():
        user_form.process()  # process the gathered cleaned data            
        user_form = UserForm(
            {'email': user_form.cleaned_data['email'],
            'first_name': user_form.cleaned_data['first_name'],
            'last_name': user_form.cleaned_data['last_name'],}
        ) # reload the form       
    return render(request, 'myApp/user_details.html',{'user_form': user_form})
作为一个小小的优化,您可以使用内置的检查:
 if user_form.has_changed():

在is_valid()检查之后(或与之结合)——如果表单上没有任何更改,通常不需要保存或处理表单。

相关内容

最新更新