微信公众号搜"智元新知"关注
微信扫一扫可直接关注哦!

如何在 Django 3.1 ModelForm 中动态设置内置 MinValueValidator 的 limit_value

如何解决如何在 Django 3.1 ModelForm 中动态设置内置 MinValueValidator 的 limit_value

我正在尝试在 Django 3.1 ModelForm 中动态设置内置 MinValueValidator 的 limit_value。以下代码适用于固定的 limit_value 为 10(请参阅 views.py 中的第 21 行)。

models.py

from django.contrib.auth.models import AbstractUser
from django.db import models

class Bid(models.Model):
    listing = models.ForeignKey(Listing,on_delete=models.CASCADE,related_name="bids")
    user = models.ForeignKey(User,related_name="bids")
    bid = models.DecimalField(decimal_places=2,max_digits=9)

views.py

from django.contrib.auth import authenticate,login,logout
from django.db import IntegrityError
from django.http import HttpResponse,HttpResponseRedirect
from django.shortcuts import render
from django.urls import reverse
from django import forms
from .models import User,Listing,Category,Bid
from django.db.models import Max
from decimal import Decimal,DecimalException
from django.core.validators import MaxValueValidator,MinValueValidator
from django.core.exceptions import ValidationError

class NewBidForm(forms.ModelForm):
    class Meta:
        model = Bid
        fields = '__all__'
        widgets = {
            'user': forms.Hiddeninput(),'listing': forms.Hiddeninput(),}

    def __init__(self,*args,**kwargs):
        super(NewBidForm,self).__init__(*args,**kwargs)
        self.fields['user'].show_hidden_initial=True
        self.fields['listing'].show_hidden_initial=True
        self.fields['bid'].validators=[MinValueValidator(10)]

    def clean(self):
        if 'user' in self.changed_data or 'listing' in self.changed_data:
            raise forms.ValidationError('Non editable field have changed!') 
        return self.cleaned_data

def index(request):
    listings = Listing.objects.all()
    return render(request,"auctions/index.html",{
        "listings" : listings,})

def listing(request,listing_id):
    if request.method == 'POST':
        data = request.POST
        form = NewBidForm(data)
        if form.is_valid():
            form.save()
            return HttpResponseRedirect(reverse("index"))
        else:
            listing = Listing.objects.get(pk=listing_id)
            bids = Bid.objects.filter(listing=listing)
            if bids:  
                highest_bid = bids.aggregate(Max('bid'))['bid__max']
            else:  
                highest_bid = listing.starting_bid 
            return render(request,"auctions/listing.html",{
                "listing" : listing,"highest_bid" : highest_bid,"form" : form
                })
    else:
        listing = Listing.objects.get(pk=listing_id)
        bids = Bid.objects.filter(listing=listing)
        if bids:  
            highest_bid = bids.aggregate(Max('bid'))['bid__max']
        else:  
            highest_bid = listing.starting_bid
        form = NewBidForm(initial={'listing':listing,'user':request.user})
        return render(request,{
            "listing" : listing,"form" : form
            })

但是,当我在 ModelForm 实例化期间尝试通过“my_arg”将变量传递给 NewBidForm 的 init 方法时,我收到以下错误消息:

  • 密钥错误:'my_arg'
  • 列表项 AttributeError: 'NewBidForm' 对象没有属性 '_errors'

以下是views.py中修改后的代码

views.py

class NewBidForm(forms.ModelForm):
    class Meta:
        model = Bid
        fields = '__all__'
        widgets = {
            'user': forms.Hiddeninput(),**kwargs):
        my_arg = kwargs.pop('my_arg')
        super(NewBidForm,**kwargs)
        self.fields['user'].show_hidden_initial=True
        self.fields['listing'].show_hidden_initial=True
        self.fields['bid'].validators=[MinValueValidator(my_arg)]

    def clean(self):
        if 'user' in self.changed_data or 'listing' in self.changed_data:
            raise forms.ValidationError('Non editable field have changed!') 
        return self.cleaned_data


def index(request):
    listings = Listing.objects.all()
    return render(request,'user':request.user},my_arg=12)
        return render(request,"form" : form
            })

谁能告诉我如何在实例化期间将变量传递给 ModelForm 内的 init 方法? 在运行时更改 MinValueValidator 中构建的 limit_value 的替代解决方案也是可以接受的。但是,我不喜欢在 ModelForm 中重新定义 fromfields。

BR,康拉德

解决方法

下面的代码示例显示了我的问题的答案。

模型表单类

class NewBidForm(forms.ModelForm):
    class Meta:
        model = Bid
        fields = '__all__'

    def __init__(self,*args,**kwargs):
        my_arg = kwargs.pop('my_arg')
        super(NewBidForm,self).__init__(*args,**kwargs)
        self.fields['bid'].validators=[MinValueValidator(my_arg)]

然后每次实例化表单对象时,请确保像这样传入 my_variable

form = NewBidForm(my_arg=my_variable)

我的错误是在我的代码中的两个位置实例化表单,但只在其中一个实例中传递参数。 我

版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 dio@foxmail.com 举报,一经查实,本站将立刻删除。