Django - 如何使用CreateView扩展自定义用户

Posted

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了Django - 如何使用CreateView扩展自定义用户相关的知识,希望对你有一定的参考价值。

我想使用CreateView创建用户注册表单。但是,我不知道如何在view.py中将用户模型与Profile Model相结合

我的环境·Django2.2

这是我写的代码:

▪️model.py

class User(AbstractBaseUser, PermissionsMixin):
    email = models.EmailField(_('email address'), unique=True)
    first_name = models.CharField(_('first name'), max_length=30, blank=True)
    last_name = models.CharField(_('last name'), max_length=30, blank=True)

class Profile(models.Model):
    user = models.OneToOneField(settings.AUTH_USER_MODEL, on_delete=models.CASCADE)
    gender = models.CharField(max_length=20, blank=True)

▪️form.py

class UserCreateForm(UserCreationForm):

    class Meta:
        model = User
        if User.USERNAME_FIELD == 'email':
            fields = ('email',)
        else:
            fields = ('username', 'email')

    def __init__(self, *args, **kwargs):
        super().__init__(*args, **kwargs)
        for field in self.fields.values():
            field.widget.attrs['class'] = 'form-control'

class ProfileForm(forms.ModelForm):
    CHOICES = (
        ('female', 0,),
        ('male', 1,),
        ('not_applicable', 2,)
    )
    gender = forms.ChoiceField(widget=forms.Radioselect, choices=CHOICES, required=False)

▪️View.py

from django.views import generic
from .forms import UserCreateForm

class UserCreate(generic.CreateView):
    template_name = 'accounts/create.html'
    form_class = UserCreateForm

    def form_valid(self, form):
        user = form.save(commit=False)
        user.is_active = False
        user.save()

        current_site = get_current_site(self.request)
        domain = current_site.domain
        context = {
            'protocol': 'https' if self.request.is_secure() else 'http',
            'domain': domain,
            'token': dumps(user.pk),
            'user': user,
        }

        subject_template = get_template('accounts/mail/create/subject.txt')
        subject = subject_template.render(context)

        message_template = get_template('accounts/mail/create/message.txt')
        message = message_template.render(context)

        user.email_user(subject, message)
        return redirect('accounts:user_create_done')

▪️create.html

<form action="" method="POST">
    {{ form.non_field_errors }}
            {% for field in form %}
                    <p>{{ field }} {{ field.errors }}</p>
            {% endfor %}
    {% csrf_token %}
    <button type="submit" class="button-submit">submit</button>
</form>

但是,这只能显示用户列(仅限电子邮件,密码)。但我想显示用户列和配置文件列(电子邮件,密码,first_name,last_name,性别)。

这可能吗?

答案

您应该在自己的AbstractBaseUser中继承CustomUserModel,添加所需的其他字段,然后在AUTH_USER_MODEL=yourapp.CustomUserModel文件中覆盖settings.py

可能值得一看this

另一答案

fieldsMeta中的UserCreationForm列表应包含您需要显示的所有字段。您还应该在UserCreationFrom中添加配置文件字段并手动更新配置文件。另外,请更改您的表单名称,以区别于django的UserCreateForm

此外,请确保无论何时调用user.profile,都会为用户创建配置文件。

# models.py
# add the below line in the end of the file.
User.profile = property(lambda u: Profile.objects.get_or_create(user=u)[0])

更新您的表单。

# forms.py

class CustomUserCreateForm(UserCreationForm):
    CHOICES = (
        ('female', 0,),
        ('male', 1,),
        ('not_applicable', 2,)
    )
    gender = forms.ChoiceField(widget=forms.RadioSelect, choices=CHOICES, required=False)

    class Meta:
        model = User
        if User.USERNAME_FIELD == 'email':
            fields = ('email',)
        else:
            fields = ('username', 'email')
        fields += ('password', 'first_name', 'last_name', 'gender')

    def __init__(self, *args, **kwargs):
        super().__init__(*args, **kwargs)
        for field in self.fields.values():
            field.widget.attrs['class'] = 'form-control'

在您查看中,您可以在gender中保存user.profile的值。

class UserCreate(generic.CreateView):
    template_name = 'accounts/create.html'
    form_class = CustomUserCreateForm

    def form_valid(self, form):
        user = form.save(commit=False)
        user.is_active = False
        user.save()

        # get the value of gender and save to user.profile
        gender = form.cleaned_data.get("gender")
        profile = user.profile  # this will get_or_create the profile instance
        profile.gender = gender
        profile.save()

        current_site = get_current_site(self.request)
        domain = current_site.domain
        context = {
            'protocol': 'https' if self.request.is_secure() else 'http',
            'domain': domain,
            'token': dumps(user.pk),
            'user': user,
        }

        subject_template = get_template('accounts/mail/create/subject.txt')
        subject = subject_template.render(context)

        message_template = get_template('accounts/mail/create/message.txt')
        message = message_template.render(context)

        user.email_user(subject, message)
        return redirect('accounts:user_create_done')

以上是关于Django - 如何使用CreateView扩展自定义用户的主要内容,如果未能解决你的问题,请参考以下文章

如何使用请求数据在基于 Django 类的通用 createview 上设置初始数据

在 Django 中,如何在提交 CreateView 时重定向到 UpdateView?

如何使用测试客户端测试 django (3.0) CreateView 的“form_valid”方法和参数?

Django:如何在新模板中的另一个模型上使用 CreateView 的 form_valid() 返回的响应?

Django 1.11 CreateView 传递 URL 参数

如何将引导类添加到模板中的 Django CreateView 表单字段?