Django ManyTomany Validação
-
27-09-2019 - |
Pergunta
Por favor veja o código abaixo. Basicamente, quando o usuário cria um objeto dessa classe, eles precisam especificar o value_type
. Se value_type==2
(porcentagem), então percentage_calculated_on
(que é um seleção de seleção, o lado do formulário/modelo precisa ter um ou mais itens verificados. A validação do modelo não está me permitindo validar como estou tentando - basicamente lança uma exceção que me diz que a instância precisa Para ter um valor de chave primária antes que um relacionamento muitos para muitos possa ser usado. Mas preciso primeiro validar o objeto antes de salvá-lo. Eu tentei essa validação no lado do formulário (Modelform) (usando o método limpo do formulário), Mas a mesma coisa acontece lá também.
Como faço para conseguir essa validação?
INHERENT_TYPE_CHOICES = ((1, 'Payable'), (2, 'Deductible'))
VALUE_TYPE_CHOICES = ((1, 'Amount'), (2, 'Percentage'))
class Payable(models.Model):
name = models.CharField()
short_name = models.CharField()
inherent_type = models.PositiveSmallIntegerField(choices=INHERENT_TYPE_CHOICES)
value = models.DecimalField(max_digits=12,decimal_places=2)
value_type = models.PositiveSmallIntegerField(choices=VALUE_TYPE_CHOICES)
percentage_calculated_on = models.ManyToManyField('self', symmetrical=False)
def clean(self):
from django.core.exceptions import ValidationError
if self.value_type == 2 and not self.percentage_calculated_on:
raise ValidationError("If this is a percentage, please specify on what payables/deductibles this percentage should be calculated on.")
Solução
Testei seu código em um dos meus projetos de administrador dos meus projetos. Consegui executar a validação necessária usando um personalizado ModelForm
. Veja abaixo.
# forms.py
class MyPayableForm(forms.ModelForm):
class Meta:
model = Payable
def clean(self):
super(MyPayableForm, self).clean() # Thanks, @chefsmart
value_type = self.cleaned_data.get('value_type', None)
percentage_calculated_on = self.cleaned_data.get(
'percentage_calculated_on', None)
if value_type == 2 and not percentage_calculated_on:
message = "Please specify on what payables/deductibles ..."
raise forms.ValidationError(message)
return self.cleaned_data
# admin.py
class PayableAdmin(admin.ModelAdmin):
form = MyPayableForm
admin.site.register(Payable, PayableAdmin)
O aplicativo admin usa o SelectMultiple
widget (em vez de CheckboxSelectMultiple
como você faz) para representar muitos para muitos relacionamentos. Eu acredito que isso não deveria importar.