Как создать форму UserProfile в Django с изменениями first_name, last_name? - PullRequest
21 голосов
/ 13 ноября 2009

Если вы думаете, что мой вопрос довольно очевиден, и почти каждый разработчик, работающий с UserProfile, сможет ответить на него.

Однако я не смог найти никакой помощи в документации по Django или в Книге Django.

Если вы хотите заполнить форму UserProfile с помощью форм Django, вам нужно изменить поля профиля, а также некоторые поля User.

Но нет forms.UserProfileForm (пока?)!

Как ты это делаешь?

Ответы [ 7 ]

37 голосов
/ 06 февраля 2010

Я наткнулся на это сегодня и после некоторого поиска в Google нашел решение, которое, на мой взгляд, немного чище:

#in forms.py
class UserForm(forms.ModelForm):
    class Meta:
        model = User
        fields = ["username", "email"]

class UserProfileForm(forms.ModelForm):
    class Meta:
        model = UserProfile

#in views.py
def add_user(request):
    ...
    if request.method == "POST":
        uform = UserForm(data = request.POST)
        pform = UserProfileForm(data = request.POST)
        if uform.is_valid() and pform.is_valid():
            user = uform.save()
            profile = pform.save(commit = False)
            profile.user = user
            profile.save()
            ....
    ...

#in template
<form method="post">
    {{ uform.as_p }}
    {{ pform.as_p }}
    <input type="submit" ...>
</form>

Источник

35 голосов
/ 13 ноября 2009

Вот как я наконец сделал:

class UserProfileForm(forms.ModelForm):
    first_name = forms.CharField(label=_(u'Prénom'), max_length=30)
    last_name = forms.CharField(label=_(u'Nom'), max_length=30)

    def __init__(self, *args, **kw):
        super(UserProfileForm, self).__init__(*args, **kw)
        self.fields['first_name'].initial = self.instance.user.first_name
        self.fields['last_name'].initial = self.instance.user.last_name

        self.fields.keyOrder = [
            'first_name',
            'last_name',
            ...some_other...
            ]

    def save(self, *args, **kw):
        super(UserProfileForm, self).save(*args, **kw)
        self.instance.user.first_name = self.cleaned_data.get('first_name')
        self.instance.user.last_name = self.cleaned_data.get('last_name')
        self.instance.user.save()

    class Meta:
        model = UserProfile
5 голосов
/ 09 декабря 2009

Вот как я это сделал в текущем транке (Редакция: 11804). Решение Натима мне не помогло.

В admin.py:

class ProfileAdmin(admin.ModelAdmin):
    form = ProfileForm

    def save_model(self, request, obj, form, change):
        obj.user.first_name = form.cleaned_data['first_name']
        obj.user.last_name = form.cleaned_data['last_name']
        obj.user.save()
        obj.save()

В forms.py:

class ProfileForm(forms.ModelForm):
    first_name = forms.CharField(max_length=256)
    last_name = forms.CharField(max_length=256)

    def __init__(self, *args, **kwargs):
        super(ProfileForm, self).__init__(*args, **kwargs)
        try:
            self.fields['first_name'].initial = self.instance.user.first_name
            self.fields['last_name'].initial = self.instance.user.last_name
        except User.DoesNotExist:
            pass

    class Meta:
         fields = ['first_name', 'last_name', ...etc.]
1 голос
/ 05 июля 2014

Это то, что я использовал в одном из своих проектов давно, надеюсь, это поможет другим погуглить эту проблему.

class SignUpForm(forms.ModelForm):
    first_name = forms.CharField(max_length = 30)
    last_name = forms.CharField(max_length = 30)
    username = forms.CharField(max_length = 30)
    password = forms.CharField(widget = forms.PasswordInput)
    password1 = forms.CharField(widget = forms.PasswordInput)
    class Meta:
        model = UserProfile
        exclude = ['user']

    def clean_username(self): # check if username does not exist before
        try:
            User.objects.get(username=self.cleaned_data['username']) #get user from user model
        except User.DoesNotExist :
            return self.cleaned_data['username']

        raise forms.ValidationError("This user exist already choose an0ther username")



    def clean(self, *args , **kwargs):
        super(SignUpForm).clean(*args ,**kwargs) # check if password 1 and password2 match each other
        if 'password1' in self.cleaned_data and 'password2' in self.cleaned_data:#check if both pass first validation
            if self.cleaned_data['password1'] != self.cleaned_data['password2']: # check if they match each other
                raise forms.ValidationError("Passwords don't match each other")

        return self.cleaned_data
    def save(self): # create new user
        new_user = User.objects.create_user(username=self.cleaned_data['username'],password=self.cleaned_data['password1'],email=self.cleaned_data['email'])
        new_user.first_name = self.cleaned_data['first_name']
        new_user.last_name = self.cleaned_data['last_name']
        new_user.save()
        UserProf =  super(SignUpForm,self).save(commit = False)
        UserProf.user = new_user
        UserProf.save()
        return UserProf
1 голос
/ 03 февраля 2010

Вы также можете попробовать использовать проект django-basic-apps, в котором есть приложение с профилями:

https://github.com/nathanborror/django-basic-apps

0 голосов
/ 09 декабря 2009

Почему бы не иметь две формы модели на сервере и просто представить их как одну форму в шаблоне? Удалите поля имени из вашей UserProfileForm и создайте вторую форму модели для объекта пользователя?

0 голосов
/ 13 ноября 2009

Я считаю нормальным, что вы не найдете никакой информации в документах, поскольку вы объединяете две модели в одну форму.

В качестве альтернативы и, возможно, совершенно очевидно, что вы можете: Создать две модели моделей, одну для пользователя и другую для пользовательского профиля. Установите форму модели userprofile для отображения только имени и фамилии. Поместите обе формы в один шаблон в один тег <form>. Когда оно будет отправлено, вызовите методы сохранения каждой формы.

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...