Не удается обновить информацию профиля пользователя - PullRequest
0 голосов
/ 08 января 2020

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

Проблема, с которой я сталкиваюсь в настоящее время, заключается в том, что когда пользователь заполняет форму и сохраняет это, информация в их профиле не обновляется на основе значений, которые они заполнили.

Я пытался прочитать документацию по формам в Django, но все еще не могу понять это.

Это код, который у меня есть в forms.py:

#mentor sign up form
class TeacherSignUpForm(UserCreationForm):
    email = forms.EmailField(max_length=100)
    first_name = forms.CharField(max_length=100)
    last_name = forms.CharField(max_length=100)
    linkedin = forms.URLField(max_length=200)

    class Meta(UserCreationForm.Meta):
        model = User
        fields = ('email', 'username', 'first_name', 'last_name')

    def save(self, commit=True):
        self.instance.is_teacher = True
        user = super(UserCreationForm, self).save(commit=False)
        user.email = self.cleaned_data['email']
        user.first_name = self.cleaned_data['first_name']
        user.last_name = self.cleaned_data['last_name']
        user.save()
        mentor = Mentor.objects.create(
            user=user,
            linkedin=self.cleaned_data['linkedin']
        )
        return user


# edit mentor profile
class MentorProfileForm(forms.Form):
    class Meta(TeacherSignUpForm.Meta):
        model = User
        exclude = ('password')

models.py

class User(AbstractUser):
    is_student = models.BooleanField(default=False)
    is_teacher = models.BooleanField(default=False)
...
class Mentor(models.Model):
    user = models.OneToOneField(User,on_delete=models.CASCADE,primary_key=True)
    linkedin = models.URLField(max_length=200,null=True,blank=True)
    photo = models.ImageField(null=True,blank=True)

    def __str__(self):
        return "Profile of user {}".format(self.user.username)

@receiver(post_save,sender=User)
def create_or_update(sender, instance,created, **kwargs):
    if created:
        Mentor.objects.get_or_create(user=instance)
        post_save.connect(create_or_update, sender=User)

teacher.py - это эквивалент views.py - конкретизируется c в профиль наставника / учителя:

# edit mentor profile
def edit_user(request):
    user = request.user
    form = MentorProfileForm(initial={'first_name': user.first_name, 'last_name': user.last_name, 'email': user.email})
    if request.method == 'POST':
        if form.is_valid():
            user.first_name = MentorProfileForm(request.POST['first_name'])
            user.last_name = MentorProfileForm(request.POST['last_name'])
            user.email = MentorProfileForm(request.POST['email'])
            # user.password = request.POST['password']
            form.save(commit=True)
            return HttpResponseRedirect('%s' % (reverse('profile')))
        else:
            form = MentorProfileForm()
    return render(request, 'classroom/teachers/app-instructor-profile.html', {'form': form})

это шаблон, который я использую с формой

<form id="edit-mentor-profile" class="form-horizontal" method="post" >
                        {% csrf_token %}
                      <div class="form-group">
                        <label for="photo" class="col-sm-2 control-label">Avatar</label>
                        <div class="col-md-6">
                          <div class="media v-middle">
                            <div class="media-left">
                              <div class="icon-block width-100 bg-grey-100">
                                <i class="fa fa-photo text-light"></i>
                              </div>
                            </div>
                            <div class="media-body">
                              <a href="#" class="btn btn-white btn-sm paper-shadow relative" data-z="0.5" data-hover-z="1" data-animated> Add Image<i class="fa fa-upl"></i></a>
                            </div>
                          </div>
                        </div>
                      </div>
                      <div class="form-group">
                        <label for="inputEmail3" class="col-md-2 control-label">Full Name</label>
                        <div class="col-md-8">
                          <div class="row">
                            <div class="col-md-6">
                              <div class="form-control-material">
                                {{ form.first_name }}
                                <label for="edit-mentor-profile-first_name"></label>
                              </div>
                            </div>
                            <div class="col-md-6">
                              <div class="form-control-material">
                                {{ form.last_name }}
                                <label for="edit-mentor-profile-last_name"></label>
                              </div>
                            </div>
                          </div>
                        </div>
                      </div>
                      <div class="form-group">
                        <label for="email" class="col-md-2 control-label">Email</label>
                        <div class="col-md-6">
                          <div class="form-control-material">
                            <div class="input-group">
                              <span class="input-group-addon"><i class="fa fa-envelope"></i></span>
                              {{ form.email }}
                              <label for="edit-mentor-profile-email"></label>
                            </div>
                          </div>
                        </div>
                      </div>
                      <div class="form-group">
                        <div class="col-md-offset-2 col-md-6">
                          <div class="checkbox checkbox-success">
                            <input id="checkbox3" type="checkbox" checked="">
                            <label for="checkbox3">Subscribe to our Newsletter</label>
                          </div>
                        </div>
                      </div>
                      <div class="form-group margin-none">
                        <div class="col-md-offset-2 col-md-10">
                          <button type="submit" class="btn btn-primary paper-shadow relative" data-z="0.5" data-hover-z="1" data-animated>Save Changes</button>
                        </div>
                      </div>
                    </form>

Чтобы повторить, я могу просматривать форму, редактировать ее, но когда я пытаюсь и сохранить значения, ничего не сохраняется, и страница обновляется

Обновление

После прочтения о ModelForm я попытался внести некоторые изменения в мой forms.py в меру лучших моя способность

forms.py

class MentorProfileForm(forms.ModelForm):
    class Meta:
        model = User
        exclude = ('password',)

На основе одного из приведенных ответов я также внес изменения в models.py

@receiver(post_save,sender=User)
def create_or_update(sender, instance,created, **kwargs):
    if created:
        post_save.connect(create_or_update, sender=User)

и teacher.py (просмотров):

# edit mentor profile
def edit_user(request):
    user = request.user
    form = MentorProfileForm(instance=user)
    if request.method == 'POST':
        form = MentorProfileForm(request.POST or None, instance=user)
        if form.is_valid():
            user.first_name = form.cleaned_data['first_name']
            user.last_name = form.cleaned_data['last_name']
            user.email = form.cleaned_data['email']
            # user.password = request.POST['password']
            form.save(commit=True)
            return HttpResponseRedirect('%s' % (reverse('profile')))
        else:
            return render(request, 'classroom/teachers/app-instructor-profile.html', {'form': form,
                                                                                      'errors': form.errors})
    return render(request, 'classroom/teachers/app-instructor-profile.html', {'form': form})

Ответы [ 2 ]

2 голосов
/ 08 января 2020

Вероятно, проблема здесь:

class MentorProfileForm(forms.Form):  #<-- Here
    class Meta(TeacherSignUpForm.Meta):
        model = User
        exclude = ('password')

Нет функции сохранения для forms.Form, для этого вам нужно создать подкласс MentorProfileForm из forms.ModelForm. Или, в вашем случае, TeacherSignUpForm (поскольку вы переопределяете его мета-класс).

Кроме того, я также запутался в этой части:

def save(self, commit=True):
      ....
      # this section
      mentor = Mentor.objects.create(
            user=user,
            linkedin=self.cleaned_data['linkedin']
      )
      return user

Поскольку у вас есть сообщение сохранить сигнал для создания Mentor, тогда зачем вы его используете? Лучше, если вы уберете сигнал сохранения записи и используете здесь get_or_create, ie:

  _, mentor = Mentor.objects.get_or_create(
        user=user,
        linkedin=self.cleaned_data.get('linkedin')
  )

Наконец, обновите представление также:

form = MentorProfileForm(request.POST or None, instance=user)
if request.method == 'POST':
    if form.is_valid():
        user = form.save()
1 голос
/ 08 января 2020

Вам необходимо передать «экземпляр» при обновлении любого экземпляра модели. Измените свой взгляд на это:

# edit mentor profile
def edit_user(request):
    user = request.user
    form = MentorProfileForm(instance=user)
    if request.method == 'POST':
        form  = MentorProfileForm(request.POST, instance=user)
        if form.is_valid():
            user.first_name = form.cleaned_data['first_name']
            user.last_name = form.cleaned_data['last_name']
            user.email = form.cleaned_data['email']
            # user.password = request.POST['password']
            form.save(commit=True)
            return HttpResponseRedirect('%s' % (reverse('profile')))
        else:
            return render(request, 'classroom/teachers/app-instructor-profile.html', {'form': form, 'errors': form.errors})
    return render(request, 'classroom/teachers/app-instructor-profile.html', {'form': form})
...