Ошибка Django / Bootstrap: параметр "form" должен содержать действительную форму Django - PullRequest
4 голосов
/ 11 октября 2019

У меня есть приложение Django под названием «Учетные записи», которое расширяет модель пользователя Django по умолчанию. Я хочу, чтобы каждый пользователь мог обновлять свою учетную запись, которая представляет собой модель под названием UserProfile с использованием generic.UpdateView. Я получаю следующую ошибку при переходе по URL-адресу UpdateView:

BootstrapError в / accounts / user-profile / 5 / edit / Параметр «form» должен содержать действительную форму Django.

Iвключил мой код ниже. Спасибо!

models.py

from django.db import models
from django.contrib import auth

# Create your models here.

class User(auth.models.User,auth.models.PermissionsMixin):
    readonly_fields = ('id','pk')

    def __str__(self):
        return self.username

class UserProfile(models.Model):
    user = models.OneToOneField(auth.models.User,on_delete=models.CASCADE)
    join_date = models.DateTimeField(auto_now=True)
    profile_pic = models.ImageField(upload_to='profile_pics',default='media/block-m.png')
    skills = models.TextField()
    major = models.CharField(max_length=128)
    grad_year = models.CharField(max_length=4)
    clubs = models.TextField() #make FK to Clubs
    opt_in = models.BooleanField(default=True)

    def __str__(self):
        return self.user.username

forms.py

from django.contrib.auth import get_user_model
from django.contrib.auth.forms import UserCreationForm
from django.contrib.auth.models import User
from .models import UserProfile
from django import forms

class UserCreateForm(UserCreationForm):

    class Meta:
        model = get_user_model()
        fields = ('first_name','last_name','username','email','password1','password2')

    def __init__(self,*args,**kwargs):
        super().__init__(*args,**kwargs)
        # self.fields['username'].label = 'Username'
        # self.fields['email'].label = 'Email Address'
        # self.fields['password1'].label = 'Password'
        # self.fields['password2'].label = 'Confirm Password'

class UserProfileForm(forms.ModelForm):
    class Meta:
        model = UserProfile
        fields = ('profile_pic','grad_year','opt_in')

views.py

from django.shortcuts import render,redirect
from django.contrib import messages
from django.contrib.auth.mixins import(
    LoginRequiredMixin,
    PermissionRequiredMixin
)
from django.urls import reverse,reverse_lazy
from django.db import IntegrityError
from django.shortcuts import get_object_or_404
from django.views import generic
from .models import User,UserProfile
from .forms import UserCreateForm,UserProfileForm
import easygui
from . import models

from . import forms

# Create your views here.


class ListAccounts(generic.ListView):
    model = models.UserProfile

class DetailAccounts(generic.DetailView):
    model = models.UserProfile

class UpdateAccounts(generic.UpdateView):
    fields = ('grad_year',)
    model = models.UserProfile

def SignUp(request):

    registered=False

    if request.method == "POST":
        user_create_form = UserCreateForm(data=request.POST)
        user_profile_form = UserProfileForm(data=request.POST)

        if user_create_form.is_valid() and user_profile_form.is_valid():
            user = user_create_form.save()

            user_profile = user_profile_form.save(commit=False)
            user_profile.user = user


            if 'profile_pic' in request.FILES:
                user_profile.profile_pic = request.FILES['profile_pic']

            user_profile.save()

            registered = True

            return redirect('/accounts/login/')

        else:
            print(user_create_form.errors,user_profile_form.errors)
    else:
        user_create_form = UserCreateForm()
        user_profile_form = UserProfileForm()

    return render(request,'accounts/signup.html',
        {'user_create_form':user_create_form,
         'user_profile_form':user_profile_form,
         'registered':registered})

urls.py

from django.contrib.auth import views as auth_views
from . import views

app_name = 'accounts'

urlpatterns = [
    path('login/',auth_views.LoginView.as_view(template_name='accounts/login.html'),name='login'),
    path('logout',auth_views.LogoutView.as_view(),name='logout'),
    path('',views.ListAccounts.as_view(),name='all'),
    path('signup/',views.SignUp,name='signup'),
    path('user-profile/<int:pk>/',views.DetailAccounts.as_view(),name='detail'),
    path('user-profile/<int:pk>/edit/',views.UpdateAccounts.as_view(),name='user_update'),
    path('match/',views.Match.as_view(),name='match'),
]

userprofile_form.html

{% extends 'base.html' %}
{% load bootstrap4 %}

{% block content %}

<div class="container login">
    <h2>Update</h2>
    <form action="{% url 'accounts:all' %}" enctype="multipart/form-data" method="POST">
      {% csrf_token %}
      {% bootstrap_form user_profile_form layout='inline' %}
      <input type="submit" class="btn" style="background:#f86041; color:white;" value="Update">
    </form>
</div>

{% endblock %}

1 Ответ

2 голосов
/ 17 октября 2019

Я попытаюсь объяснить шаги, которые я предпринял, чтобы попытаться решить эту проблему, чтобы вы могли лучше понять, как решить эту проблему для себя:

Во-первых, вам нужно понять, почему возникает ошибкаподнял. Это BootstrapError не стандартная ошибка, поэтому вам нужно взглянуть на исходный код для django-bootstrap4, чтобы понять, почему он возникает. Глядя на исходный код (просто поиск), вы обнаружите следующее:

        if not isinstance(form, BaseForm):
            raise BootstrapError('Parameter "form" should contain a valid Django Form.')

Попробуйте и поработайте над кодом от того места, где вы вызываете тег шаблона, до места возникновения этой ошибки (если посмотреть на трассировку, возможно,также поможет вам в этом), чтобы понять, что все различные переменные равны.

В вашем случае form будет равен user_profile_form, и возникает ошибка, потому что user_profile_form не является экземпляромBaseForm (импортировано из django.forms.forms). Так что вы можете вернуться и посмотреть на user_profile_form. Ну, это экземпляр UserProfileForm, который, в свою очередь, является подклассом ModelForm. Но ModelForm является подклассом BaseForm. Хм ... это кажется немного странным. Почему user_profile_form не является экземпляром BaseForm, когда это должно быть.

На этом этапе сначала проверьте наличие ошибок, таких как орфографические ошибки? Может быть, проверить, если ваше промежуточное программное обеспечение может играть с формой вообще. Кроме того, попробуйте упростить вещи, просто визуализируйте форму {{ user_profile_form }} ... вы обнаружите, что это не работает. Таким образом, проблема, кажется, с user_profile_form?

Ага ... мы передали user_profile_form шаблону в представлении, SignUp. Но это не то представление, которое мы здесь просматриваем !!! Представление, которое мы рассматриваем, это представление UpdateAccounts. Это generic.UpdateView, и если мы посмотрим на документы django (всегда хорошая идея) и посмотрим на пример, мы увидим, что форма передается шаблону как просто form.

Заменитестрока:

      {% bootstrap_form user_profile_form layout='inline' %}

с

      {% bootstrap_form form layout='inline' %}

В шаблоне userprofile_form.html, и все должно работать плавно:)

Как видите, я спустилсянекоторые тупики, прежде чем я наткнулся на (возможно, более очевидное) правильное решение. Но я надеюсь, что это полезно.

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