Django 管理员覆盖模型字段验证

Django admin override model field validation

我的模型中有一个 fee DecimalField,如下所示:

class CardTypes(models.Model):
    class Meta:
        db_table = "card_types"
        verbose_name = 'Card Type'
        verbose_name_plural = 'Card Types'

    name = models.CharField("Name", max_length=255, null=True, blank=True)
    fee = models.DecimalField("Fee", max_digits=65, decimal_places=0, default=0)
    image_url = models.CharField("Image Url", max_length=255, null=True, blank=True)

    deleted_at = models.DateTimeField(null=True, blank=True)

    def card_type_image(self):
        if self.image_url is not None:
            return mark_safe('<img src="%s" width="130" height="90" />' % (self.image_url))
        else:
            return "No image yet"
    card_type_image.short_description = 'image'

    def fee_convert(self):
        if self.fee is not None:
            currency = Currency.objects.get(pk=1)
            return Decimal(self.fee / pow(10, currency.decimals))
        else:
            return None
    fee_convert.short_description = 'Fee'

我设置 DecimalField("Fee", max_digits=65, decimal_places=0, default=0) 是因为我想在我的数据库中保存它时不带小数位,但在管理站点上我想与其转换后的方法进行交互。

例如,如果数据库中的费用是 0.00005,那么在管理员中它只会显示并保存值 5(因为 currency.decimals = 5)

我设法在管理端显示转换后的值并保存为 5,如下所示:

class CardTypeAdminForm(forms.ModelForm):
    def __init__(self, *args, **kwargs):
        super(CardTypeAdminForm, self).__init__(*args, **kwargs)
        if self.instance.fee is not None:
            try:
                self.initial['fee'] = self.instance.fee_convert()
            except:
                pass

    def clean_fee(self):
        if self.cleaned_data['fee'] is not None:
            currency = Currency.objects.get(pk=1)
            data = Decimal(self.cleaned_data['fee'] * pow(10, currency.decimals))
        else:
            data = None

        return data

class CardTypesAdmin(admin.ModelAdmin):
    list_display = ('name', 'card_type_image',)
    list_display_links = ('name', 'card_type_image', )

    form = CardTypeAdminForm
    fields = (
        'id',
        'name',
        'fee',
        'image_url',
        'card_type_image',
        'deleted_at',
    )

    readonly_fields = (
        'id',
        'card_type_image',
        'deleted_at'
    )

admin.site.register(CardTypes, CardTypesAdmin)

但是如果我想在管理员中输入值为 0.005 或小数位的费用字段(应该可以,因为在数据库中它将保存为 500),但是管理员验证 return 以下验证错误:

Ensure that there are no more than 0 decimal places.

因为管理表单字段继承了模型字段的验证,我是否可以覆盖它?

我找到了一种解决方法,方法是在费用表单字段的表单替换中创建新字段:

class CardTypeAdminForm(forms.ModelForm):
    currency = Currency.objects.get(pk=1)
    fee = forms.DecimalField(required=False, decimal_places=currency.decimals)
    def __init__(self, *args, **kwargs):
        super(CardTypeAdminForm, self).__init__(*args, **kwargs)
        if self.instance.fee is not None:
            try:
                self.initial['fee'] = self.instance.fee_convert()
            except:
                pass

    def clean_fee(self):
        if self.cleaned_data['fee'] is not None:
            currency = Currency.objects.get(pk=1)
            data = Decimal(self.cleaned_data['fee'] * pow(10, currency.decimals))
        else:
            data = None

        return data
    class Meta:
        model = CardTypes
        exclude = ['fee']

class CardTypesAdmin(admin.ModelAdmin):
    list_display = ('name', 'card_type_image',)
    list_display_links = ('name', 'card_type_image', )

    form = CardTypeAdminForm
    fields = (
        'id',
        'name',
        'fee',
        'image_url',
        'card_type_image',
        'deleted_at',
    )

    readonly_fields = (
        'id',
        'card_type_image',
        'deleted_at'
    )

    def save_model(self, request, obj, form, change):
        obj.fee = form.cleaned_data['fee']
        return super(CardTypesAdmin, self).save_model(request, obj, form, change)