ModelForm с OneToOneField в Django
У меня есть две модели в Django, которые связаны с OneToOneField
(PrinterProfile
и PrinterAdress
). Я пытаюсь создать форму с помощью PrinterProfileForm
, но по какой-то причине она НЕ передает поля PrinterAddress
в форму (она не отображается в магии Django в шаблоне).
Что я должен сделать, чтобы мой PrinterProfileForm
включал в себя также поля из PrinterAddress
(связанного с OneToOneField
)?
большое спасибо
class PrinterProfile(TimeStampedModel):
user = models.OneToOneField(User)
phone_number = models.CharField(max_length=120, null=False, blank=False)
additional_notes = models.TextField()
delivery = models.BooleanField(default=False)
pickup = models.BooleanField(default=True)
# The main address of the profile, it will be where are located all the printers.
class PrinterAddress(TimeStampedModel):
printer_profile = models.OneToOneField(PrinterProfile)
formatted_address = models.CharField(max_length=200, null=True)
latitude = models.DecimalField(max_digits=25, decimal_places=20) # NEED TO CHECK HERE THE PRECISION NEEDED.
longitude = models.DecimalField(max_digits=25, decimal_places=20) # NEED TO CHECK HERE THE PRECISION NEEDED.
point = models.PointField(srid=4326)
def __unicode__(self, ):
return self.user.username
class PrinterProfileForm(forms.ModelForm):
class Meta:
model = PrinterProfile
exclude = ['user']
Ответы
Ответ 1
Вам нужно создать вторую форму для PrinterAddress
и обработать обе формы в вашем представлении:
if all((profile_form.is_valid(), address_form.is_valid())):
profile = profile_form.save()
address = address_form.save(commit=False)
address.printer_profile = profile
address.save()
Конечно, в шаблоне вам нужно показать обе формы под одним тегом <form>
: -)
<form action="" method="post">
{% csrf_token %}
{{ profile_form }}
{{ address_form }}
</form>
Ответ 2
В дополнение к принятому ответу :
Если у вас есть пользовательские методы очистки, вам нужно добавить случай try/except
. В представленном примере, если address
имел метод clean()
для проверки того, что вам нужно было изменить, чтобы:
def clean(self):
try:
printer_profile = self.printer_profile
except ObjectDoesNotExist:
pass
else:
...code to validate address...