django-allauth 教程(二) 一鼓作气

会飞的鱼 等级 732 0 0
标签: Python

前言:上一篇:django-allauth(一)小试牛刀 介绍了django-allauth的安装及基本使用(如用户的注册,登录,邮箱验证和密码重置),然而allauth并没有提供展示和修改用户资料的功能,也没有对用户资料进行扩展。那么本篇就来介绍如何拓展用户个人资料和修改个人资料。一个在用户登录后跳转到个人信息页面(/accounts/profile/),一个允许登录用户编辑个人资料/accounts/profile/update/)。

1.创建一个APP,叫做myaccount

这里教大家一个便捷使用 python manage.py shell 的方法。 首先打开manage.py文件,然后在pycharm中找到菜单栏的工具,如图:

django-allauth 教程(二) 一鼓作气 点击后,会出现 django-allauth 教程(二) 一鼓作气 这样就可以不用每次在terminal中输入python manage.py ... 比如createsuperuser, startapp, migrate,makemigrations 其次呢我创建一个源码目录apps/,用来放自己的APP, django-allauth 教程(二) 一鼓作气 再创建一个源码目录extra_apps/,用来存放额外添加的APP,比如百度的富文本编辑器UEditor,xadmin等 django-allauth 教程(二) 一鼓作气 将其加入到settings.py配置文件INSTALLED_APP里去,同时把urls也加入到项目的urls里去,如下图所示。 House_website/settings.py

INSTALLED_APPS = [
    'django.contrib.admin',
    'django.contrib.auth',
    'django.contrib.contenttypes',
    'django.contrib.sessions',
    'django.contrib.messages',
    'django.contrib.staticfiles',
    'django.contrib.sites',
    'allauth',
    'allauth.account',
    'allauth.socialaccount',
    'allauth.socialaccount.providers.github',
    'myaccount',
]

为了方便国内开发者,我建议在settings.py里添加

LANGUAGE_CODE = 'zh-hans'  # 中文支持,时区为中国上海

TIME_ZONE = 'Asia/Shanghai'

USE_I18N = True

USE_L10N = True

USE_TZ = False

2.编写模型

由于Django自带的User模型字段邮箱,所以我们需要对其扩展,最便捷的方式就是创建UserProfile的模型,如下所示。我们添加了需要拓展的字段。

myaccount/models.py

from django.db import models
from django.contrib.auth.models import User
from allauth.account.models import EmailAddress
# Create your models here.


class UserProfile(models.Model):
    """用户"""
    user = models.OneToOneField(User, on_delete=models.CASCADE, related_name='profile')
    org = models.CharField('Organization', max_length=128, blank=True)
    birthday = models.DateField(null=True, blank=True, verbose_name='出生日期')
    gender = models.CharField(max_length=6, choices=(('male', u'男'), ('female', u'女')), default='female',
                              verbose_name='性别')
    age = models.IntegerField(verbose_name='年龄', null=True)
    QQ = models.CharField(max_length=20, null=True, blank=True, verbose_name='QQ', default='')

    telephone = models.CharField(max_length=50, null=True, blank=True, verbose_name='电话', default='')

    signature = models.TextField(max_length=500, verbose_name='个性签名',default='',null=True)

    mod_date = models.DateTimeField('Last modified', auto_now=True, )

    is_delete = models.BooleanField(default=False, verbose_name='是否删除')

    class Meta:
        verbose_name = 'User Profile'

    def __str__(self):
        return "{}'s profile".format(self.user.__str__())

# models.py中新定义一个account_verified方法,来提醒邮箱是否验证
    def account_verified(self):
        if self.user.is_authenticated:
            result = EmailAddress.objects.filter(email=self.user.email)
            if len(result):
                return result[0].verified
            else:
                return False
        else:
            return False

3.编写urls

House_website/House_website/urls.py

from django.contrib import admin
from django.urls import path, include

urlpatterns = [
    path('admin/', admin.site.urls),
    path('accounts/', include('allauth.urls')),
    path('accounts/', include('myaccount.urls')),
]

4.编写视图函数

myaccount/views.py

from django.shortcuts import render, get_object_or_404
from .models import UserProfile
from .forms import ProfileForm
from django.http import HttpResponseRedirect
from django.urls import reverse
from django.contrib.auth.decorators import login_required


@login_required
def profile(request):
    user = request.user
    return render(request, 'account/profile.html', {'user': user})


@login_required
def profile_update(request):
    user = request.user
    user_profile = get_object_or_404(UserProfile, user=user)

    if request.method == "POST":
        form = ProfileForm(request.POST)

        if form.is_valid():
            user.first_name = form.cleaned_data['first_name']
            user.last_name = form.cleaned_data['last_name']
            user.save()

            user_profile.org = form.cleaned_data['org']
            user_profile.birthday = form.cleaned_data['birthday']
            user_profile.age = form.cleaned_data['age']
            user_profile.gender = form.cleaned_data['gender']
            user_profile.QQ = form.cleaned_data['QQ']
            user_profile.telephone = form.cleaned_data['telephone']
            user_profile.signature = form.cleaned_data['signature']
            user_profile.save()

            return HttpResponseRedirect(reverse('myaccount:profile'))
    else:
        default_data = {'first_name': user.first_name, 'last_name': user.last_name, 'org': user_profile.org,
                        'telephone': user_profile.telephone, }
        form = ProfileForm(default_data)

    return render(request, 'account/profile_update.html', {'form': form, 'user': user})

5.编写表单

在myaccount/下新建一个forms.py 我们用户更新资料需要用到表单,所以我们把表单单独放在forms.py, 代码如下所示。我们创建了两个表单:一个是更新用户资料时使用,一个是重写用户注册表单。

from django import forms
from .models import UserProfile


class ProfileForm(forms.Form):
    first_name = forms.CharField(label='First Name', max_length=50, required=False)
    last_name = forms.CharField(label='Last Name', max_length=50, required=False)
    org = forms.CharField(label='Organization', max_length=50, required=False)
    telephone = forms.CharField(label='Telephone', max_length=50, required=False)
    birthday = forms.DateField(label="birthday", required=False)
    age = forms.IntegerField(label='age', required=False)
    gender = forms.CharField(label="gender", widget=forms.RadioSelect(
        choices=(('female', '女'), ('male', '男'))), initial=('female', '女'), required=False)
    QQ = forms.CharField(label='QQ', required=False, max_length=20)
    signature = forms.CharField(label='signature', required=False, max_length=500)


class SignupForm(forms.Form):

    def signup(self, request, user):
        user_profile = UserProfile()

        user_profile.user = user
        user.save()
        user_profile.save()

为什么我们需要重写用户注册表单?因为django-allauth在用户注册只会创建User对象,不会创建与之关联的UserProfile对象,我们希望用户在注册时两个对象一起被创建,并存储到数据库中。这点非常重要。通过重写表单,你还可以很容易添加其它字段。

要告诉django-allauth使用我们自定义的注册表单,我们只需要在settings.py里加入一行。

ACCOUNT_SIGNUP_FORM_CLASS = 'myaccount.forms.SignupForm'

6.编写模板

因为django-allauth默认会在templates/account/文件夹下寻找模板文件,为方便后续集中美化模板,我们也把模板文件放在这个文件夹中。 templates/account/profile.html

{% load account %}
{% block content %}
{% if user.is_authenticated %}
<a href="{% url 'myaccount:profile_update' %}">Update Profile</a> | <a href="{% url 'account_email' %}">Manage Email</a>  | <a href="{% url 'account_change_password' %}">Change Password</a> |
<a href="{% url 'account_logout' %}">Logout</a>
{% endif %}
<p>Welcome, {{ user.username }}.
    {% if not user.profile.account_verified %}
    (Unverified email.)
    {% endif %}
</p>


<h2>My Profile</h2>
<ul>
    <li>First Name: {{ user.first_name }} </li>
    <li>Last Name: {{ user.last_name }} </li>
    <li>Organization: {{ user.profile.org }} </li>
    <li>Telephone: {{ user.profile.telephone }} </li>
        <li>birthday: {{ user.profile.birthday }} </li>
        <li>age: {{ user.profile.age }} </li>
        <li>gender: {{ user.profile.gender }} </li>
        <li>QQ: {{ user.profile.QQ }} </li>
        <li>signature: {{ user.profile.signature }}</li>
</ul>


{% endblock %}

templates/account/profile_update.html

{% block content %}
{% if user.is_authenticated %}
<a href="{% url 'myaccount:profile_update' %}">Update Profile</a> | <a href="{% url 'account_email' %}">Manage Email</a>  | <a href="{% url 'account_change_password' %}">Change Password</a> |
<a href="{% url 'account_logout' %}">Logout</a>
{% endif %}
<h2>Update My Profile</h2>

<div class="form-wrapper">
   <form method="post" action="" enctype="multipart/form-data">
      {% csrf_token %}
      {% for field in form %}
           <div class="fieldWrapper">
        {{ field.errors }}
        {{ field.label_tag }} {{ field }}
        {% if field.help_text %}
             <p class="help">{{ field.help_text|safe }}</p>
        {% endif %}
           </div>
        {% endfor %}
      <div class="button-wrapper submit">
         <input type="submit" value="Update" />
      </div>
   </form>
</div>


{% endblock %}

7.查看效果

在Terminal输入以下命令:

python manage.py makemigrations  # 生成映射文件
# 如果之前生成过映射文件,那就把之前的映射文件删除
# \House_website\apps\myaccount\migrations\0001_initial.py
python manage.py migrate  # 执行映射文件,创建数据表
python manage.py runserver   # 运行服务

下面是django_allauth所有内置的URLs,均可以访问的。

/accounts/login/(URL名account_login): 登录

/accounts/signup/ (URL名account_signup): 注册

/accounts/password/reset/(URL名: account_reset_password) :重置密码

/accounts/logout/ (URL名account_logout): 退出登录

/accounts/password/set/ (URL名:account_set_password): 设置密码

/accounts/password/change/ (URL名: account_change_password): 改变密码(需登录)

/accounts/email/(URL名: account_email) 用户可以添加和移除email,并验证

/accounts/social/connections/(URL名:socialaccount_connections): 管理第三方账户

如果没账号,先注册一个,我就不演示了,

登录之后会进入profile页面:

django-allauth 教程(二) 一鼓作气

点击上方Update Profile,进入个人信息修改页面:


django-allauth 教程(二) 一鼓作气


输入信息后点击下方的Update 按钮就可以完后修改,重定向到/accounts/profile/ ,个人信息就修改完成了!

到这里本篇久写完了,希望大家点个赞支持一下!

特别鸣谢:大江狗前辈 声明:我写博客只是记录自己的学习进度和总结并分享给大家,并不保证原创,如有引用您的博文,请您理解!

收藏
评论区

相关推荐

14. Python 与数据库那点事儿,滚雪球学 Python
本篇文章将给大家介绍 Python 如何操作 SQLite 数据库,本文将是滚雪球学 Python 第一阶段的倒数第二篇文章。 <center<font color
Java的synchronized 能防止指令重排序吗?
**Python实战社群** **Java实战社群** ****长按识别下方二维码,******按需求添加** ![](https://imgconvert.csdnimg.cn/aHR0cHM6Ly9tbWJpei5xcGljLmNuL21tYml6X3BuZy9nU2dLV0Z3TjFkQ0xpYzNJeDZEc1JyZlh3aWFndExYTDl
java向python ,text文件动态传参或传值问题完美解决
由于业务需要对python文件进行参数传递,通过下面两个java方法完美解决此问题,我的思路是:首先我要先把上次写的参数删除,第二我要新的参数写到python文件中。 第一个方法解决了删除上次传递的参数问题。 第二个方法解决了参数传递到python文件 /** * @param file python文件的路径
VSCode 配置 Python 开发环境
### 一、环境准备 首先需要先安装好 Python 和 VSCode, 下载地址如下 * [VSCode](https://www.oschina.net/action/GoToLink?url=https%3A%2F%2Fcode.visualstudio.com%2F) * [Python](https://www.oschina.net/
22款好用的CLI工具
**Python实战社群** **Java实战社群** ****长按识别下方二维码,******按需求添加** ![](https://imgconvert.csdnimg.cn/aHR0cHM6Ly9tbWJpei5xcGljLmNuL21tYml6X3BuZy9nU2dLV0Z3TjFkQ0xpYzNJeDZEc1JyZlh3aWFndExYTDl
Ubuntu安装配置 Shadowsocks客户端
1. 安装python-pip 参考[pip-Installation](https://www.oschina.net/action/GoToLink?url=https%3A%2F%2Fpip.pypa.io%2Fen%2Fstable%2Finstalling) wget -c "https://bootstra
055 Python第三方库安装
\[TOC\] 一、概述 ==== * 看见更大的Python世界 * 第三方库的pip安装方法 * 第三方库的集成安装方法 * 第三方库的文件安装方法 二、看见更大的Python世界 =============== 2.1 Python社区 ------------ \>13万个第三方库:<a target='\_blank' h
055 Python第三方库安装
\[TOC\] 一、概述 ==== * 看见更大的Python世界 * 第三方库的pip安装方法 * 第三方库的集成安装方法 * 第三方库的文件安装方法 二、看见更大的Python世界 =============== 2.1 Python社区 ------------ \>13万个第三方库:<a target='\_blank' h
Django 多进程多线程的一些个人理解
### 一、Django 开发环境与生产环境 ##### 开发/调试 python manage.py runserver 0.0.0.0:8080 ##### 生产环境 Nginx + uWSGI + supervisor > 生产环境不用 `python manage.py runserver 0.0.0.0:8080` 的原
GitHub 上 25 个 Python 学习资源,墙裂推荐!
**Python实战社群** **Java实战社群** ****长按识别下方二维码,******按需求添加** ![](https://imgconvert.csdnimg.cn/aHR0cHM6Ly9tbWJpei5xcGljLmNuL21tYml6X3BuZy9nU2dLV0Z3TjFkQ0xpYzNJeDZEc1JyZlh3aWFndExYTDl
GitHub网页端更新,新增黑暗模式
**Python实战社群** **Java实战社群** **长按识别下方二维码,****按需求添加** ![](https://imgconvert.csdnimg.cn/aHR0cHM6Ly9tbWJpei5xcGljLmNuL21tYml6X3BuZy9nU2dLV0Z3TjFkQ0xpYzNJeDZEc1JyZlh3aWFndExYTDlLOEF
Python List数据的遍历
方式一: <!-- lang: python --> app_list = [1234, 5677, 8899] <!-- lang: python --> for app_id in app_list: <!-- lang: python --> print app_id 输出: 123
Python函数(二)
* ### 位置参数 根据位置顺序来传递参数 # -*- coding:utf-8 -*- __author__ = "MuT6 Sch01aR" def test(a,b): #a和b为形参 print(a) print(b) return 0
Python基础(二)
1.赋值语句 ====== (1)作用:定义+赋值 ----------- (2)复合与链式赋值 ---------- a = b = 2 a *= 2 (3)特殊类型的赋值 ---------- 序列的赋值: a,b,c = (1,2,3) print(a,b,c) 变量交换: a
Python并发(二)
并发是指一次处理多件事,而并行是指一次做多件事。二者不同,但互相有联系。打个比方:像Python的多线程,就是并发,因为Python的解释器GIL是线程不安全的,一次只允许执行一个线程的Python字节码,我们在使用多线程时,看上去像很多个任务同时进行,但实际上但一个线程在执行的时候,其他线程是处于休眠状态的。而在多CPU的服务器上,Java或Go的多线程,

热门文章

django-allauth 教程(一)小试牛刀

最新文章

django-allauth 教程(一)小试牛刀