I want to create registration form without use any external app but I do not manage to save foregin and OneToOne relation with a formset. Here my models.py :
class UserProfile(models.Model):
user = models.OneToOneField(User)
avatar = models.ImageField(null=True, blank=True, upload_to='avatar/')
website = models.URLField(null=True, blank=True)
birthday = models.DateField()
country = CountryField()
GENDER_CHOICES = (
('M', 'Male'),
('F', 'Female'),
)
gender = models.CharField(max_length=1, choices=GENDER_CHOICES)
def __str__(self):
return "UserProfile(models)"
class UserLanguage(models.Model):
user = models.ForeignKey(User)
language = models.CharField(max_length=7, choices=LANGUAGES)
level = models.CharField(max_length=1)
# http://stackoverflow.com/questions/3201018/django-language-codes
def __str__(self):
return "UserLanguage(models)"
Here my forms.py :
class RegisterForm(forms.ModelForm):
class Meta:
model = User
fields = ('username', 'password',)
exclude = ('last_login',)
widgets = {'password': forms.PasswordInput(), }
UserProfileFormSet = inlineformset_factory(User, UserProfile, can_delete=False)
UserLanguageFormSet = inlineformset_factory(User, UserLanguage, can_delete=False, extra=1)
And my views.py :
def registration(request):
if request.method == "POST":
user = RegisterForm(request.POST)
user_profile = UserProfileFormSet(request.POST)
user_language = UserLanguageFormSet(request.POST)
if user.is_valid() and user_profile.is_valid() and user_language.is_valid():
u = user.save()
user_profile.user = u
user_profile.save()
user_language.user = u
user_language.save()
else:
user = RegisterForm()
user_profile = UserProfileFormSet()
user_language = UserLanguageFormSet()
return render(request, 'user/registration.html', locals())
And I get this error :
Column 'user_id' cannot be null
What can I do for resolve my problem ? Have you another remarks ? Thank you.
You've confusingly named your formsets user_profile
and user_language
. But they are not the profile or language objects, they are formsets, so setting their user attribute will have no effect.
This works:
for profile_form in user_profile.forms:
up = profile_form.save(commit=False)
up.user = u
up.save()
for language_form in user_language.forms:
ul = language_form.save(commit=False)
ul.user = u
ul.save()
but really you should name things what they are.