首页 > 其他分享 >BBS项目 前期准备及注册、登录部分

BBS项目 前期准备及注册、登录部分

时间:2023-01-02 21:36:38浏览次数:35  
标签:obj name 登录 models True length 注册 文章 BBS

项目开发基本流程

1.需求分析
2.架构设计
3.分组开发
4.提交测试
5.交付上线

创建项目配置

环境配置
TEMPLATES = [
    {
        'BACKEND': 'django.template.backends.django.DjangoTemplates',
        'DIRS': [os.path.join(BASE_DIR, 'templates'), ]
        ,
        'APP_DIRS': True,
        'OPTIONS': {
            'context_processors': [
                'django.template.context_processors.debug',
                'django.template.context_processors.request',
                'django.contrib.auth.context_processors.auth',
                'django.contrib.messages.context_processors.messages',
            ],
        },
    },
]
数据库配置 提前在终端创建好空数据库
DATABASES = {
    'default': {
        'ENGINE': 'django.db.backends.mysql',
        'NAME': 'bbs',
        'USER': 'root',
        'PASSWORD': '222',
        'HOST': '127.0.0.1',
        'PORT': 3306,
        'CHARSET': 'utf8',
    }
}

静态文件配置

STATICFILES_DIRS = [
    os.path.join(BASE_DIR, 'static')
]
AUTH_USER_MODEL = 'app01.UserInfo'  # 前端校验forms

路由代码

from django.contrib import admin
from django.urls import path
from app01 import views

urlpatterns = [
    path('admin/', admin.site.urls),
    # 注册功能
    path('register/', views.register_func, name='register_view'),
    # 登录功能
    path('login/', views.login_func, name='login_view'),
    # 图片验证码相关功能
    path('get_code/', views.get_code_func),

]

项目流程

仿造博客园项目
     核心:文章的增删改查

表分析
     先确定表的数量 再确定表的基础字段 最后确定表的外键字段
     1.用户表
     2.个人站点表
     3.文章表
     4.文章分类表
     5.文章标签表
     6.点赞点踩表
     7.文章评论表

基础字段分析
    '''下列表字段设计仅供参考 你可以有更多的想法'''
   用户表
       替换auth_user表并扩展额外的字段
           电话号码、头像、注册时间
   个人站点表
       站点名称(jason\lili\kevin)
       站点标题(努力奋斗哈哈哈哈)
       站点样式(css文件)
   文章表
       文章标题
       文章简介
       文章内容
       发布时间
   文章分类表
       分类名称
   文章标签表
       标签名称
   点赞点踩表:记录哪个用户给哪篇文章点了推荐(赞)还是反对(踩)
        用户字段(用户主键)>>>:外键字段
        文章字段(文章主键)>>>:外键字段
        点赞点踩
   文章评论表:记录哪个用户给哪篇文章评论了什么内容
        用户字段(用户主键)>>>:外键字段
        文章字段(文章主键)>>>:外键字段
        外键字段(自关联)
   """
        id  user_id  article_id   content       parent_id
        1    1         1            哈哈哈         null
        2    2         1            哈尼没          1
        3    3         1            讲文明          2
   """
 外键字段
       用户表
            用户与个人站点是一对一外键关系
       个人站点表

       文章表
          文章表与个人站点表是一对多外键关系
          文章表与文章分类表是一对多外键关系
          文章表与文章标签表是多对多外键关系
 '''
     数据库字段优化设计:我们想统计文章的评论数 点赞数
         通过文章数据跨表查询到文章评论表中对应的数据统计即可
     但是文章需要频繁的展示 每次都跨表查询的话效率极低
         我们在文章表中再创建三个普通字段
     之后只需要确保每次操作评论表或者点赞点踩表时同步修改上述三个普通字段即可

 '''
     文章评论数
     文章点赞数
     文章点踩数

  文章分类表
      文章分类与个人站点是一对多外键关系
  文章标签表
      文章标签与个人站点是一对多外键关系

image
创建表字段

from django.db import models

# Create your models here.
from django.contrib.auth.models import AbstractUser


class UserInfo(AbstractUser):
    """用户表"""
    phone = models.BigIntegerField(verbose_name='手机号', null=True)
    avatar = models.FileField(upload_to='avatar/', default='avatar/default.jpg', verbose_name='用户头像')
    register_time = models.DateTimeField(verbose_name='注册时间', auto_now_add=True)

    """用户表与个人站点一对一外键"""
    site = models.OneToOneField(to='Site', on_delete=models.CASCADE, null=True)


class Site(models.Model):
    """个人站点表"""
    site_name = models.CharField(verbose_name='站点名称', max_length=32)
    site_title = models.CharField(verbose_name='站点标题', max_length=32)
    site_theme = models.CharField(verbose_name='站点样式', max_length=32, null=True)  # 简单模拟样式文件


class Article(models.Model):
    """文章表"""
    title = models.CharField(verbose_name='文章标题', max_length=32)
    desc = models.CharField(verbose_name='文章简介', max_length=255)
    content = models.TextField(verbose_name='文章内容')
    create_time = models.DateTimeField(verbose_name='创建时间', auto_now_add=True)
    #  三个优化字段
    comment_num = models.IntegerField(verbose_name='评论数', default=0)
    up_num = models.IntegerField(verbose_name='点赞数', default=0)
    down_num = models.IntegerField(verbose_name='点踩数', default=0)

    """文章与个人站点一对多外键"""
    site = models.ForeignKey(to='Site', on_delete=models.CASCADE, null=True)
    """文章表与分类表外键"""
    category = models.ForeignKey(to='Category', on_delete=models.CASCADE, null=True)
    """文章表与标签表是多对多  半自动创建"""
    tags = models.ManyToManyField(to='Tag',
                                  through='Article2Tag',
                                  through_fields=('article', 'tag'),
                                  )


class Category(models.Model):
    """文章分类表"""
    name = models.CharField(verbose_name='分类名称', max_length=32)

    """个人站点和文章分类的外键 """
    site = models.ForeignKey(to='Site', on_delete=models.CASCADE, null=True)


class Tag(models.Model):
    """文章标签表"""
    name = models.CharField(verbose_name='标签名称', max_length=32)
    """个人站点和文章标签的外键 """
    site = models.ForeignKey(to='Site', on_delete=models.CASCADE, null=True)



"""文章表与标签表是多对多  半自动创建"""

class Article2Tag(models.Model):
    article = models.ForeignKey(to='Article', on_delete=models.CASCADE, null=True)
    tag = models.ForeignKey(to='Tag', on_delete=models.CASCADE, null=True)


class UpAndDown(models.Model):
    """文章点赞点踩表"""
    user = models.ForeignKey(to='UserInfo', on_delete=models.CASCADE, null=True)
    article = models.ForeignKey(to='Article', on_delete=models.CASCADE, null=True)
    is_up = models.BooleanField(verbose_name='点赞点踩')  # 传布尔值存 0或者1


class Comment(models.Model):
    """文章评论表"""
    user = models.ForeignKey(to='UserInfo', on_delete=models.CASCADE, null=True)
    article = models.ForeignKey(to='Article', on_delete=models.CASCADE, null=True)
    content = models.TextField(verbose_name='评论内容')
    comment_time = models.DateTimeField(auto_now_add=True, verbose_name='评论时间')
    parent = models.ForeignKey(to='self', on_delete=models.CASCADE, null=True)


注册功能

用户注册 
     1.渲染前端标签
     2.校验用户数据
     3.展示错误提示
ps:forms组件、modelform组件

单独开设py文件编写 解耦合!!!
渲染标签 前端校验
from django import forms
from app01 import models


class RegisterForm(forms.Form):
    """用户注册form类"""
    username = forms.CharField(min_length=3, max_length=8, label='用户注册',
                               error_messages={
                                   'min_length': '用户名最短三位',
                                   'max_length': '用户名最长八位',
                                   'required': '用户名不能为空',
                               }, widget=forms.widgets.TextInput(attrs={'class': 'form-control'})
                               )
    password = forms.CharField(min_length=3, max_length=8, label='密码',
                               error_messages={
                                   'min_length': '密码最短三位',
                                   'max_length': '密码最长八位',
                                   'required': '密码不能为空',
                               },
                               widget=forms.widgets.PasswordInput(attrs={'class': 'form-control'})
                               )
    confirm_password = forms.CharField(min_length=3, max_length=8, label='确认密码',
                                       error_messages={
                                           'min_length': '密码最短三位',
                                           'max_length': '密码最长八位',
                                           'required': '密码不能为空',
                                       },
                                       widget=forms.widgets.PasswordInput(attrs={'class': 'form-control'})
                                       )
    email = forms.EmailField(label='邮箱',
                             error_messages={
                                 'required': '邮箱不能为空',
                                 'invalid': '邮箱格式不正确',
                             },
                             widget=forms.widgets.EmailInput(attrs={'class': 'form-control'})
                             )

    # 用户头像单独校验  不使用校验类  其他字段自己看加校验
    # 钩子函数
    # 局部钩子校验用户名是否已存在
    def clean_username(self):
        username = self.cleaned_data.get('username')
        res = models.UserInfo.objects.filter(username=username)
        if res:
            self.add_error('username', '用户名已存在')
        return username

    # 全局钩子校验两次密码是否一致
    def clean(self):
        password = self.cleaned_data.get('password')
        confirm_password = self.cleaned_data.get('confirm_password')
        if not password == confirm_password:
            self.add_error('confirm_password', '两次密码不一致')
        return self.cleaned_data

代码

前端代码
<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>Title</title>
    <script src="https://cdn.bootcdn.net/ajax/libs/jquery/3.6.1/jquery.js"></script>
    {% load static %}
    <link rel="stylesheet" href="{% static 'bootstrap-3.4.1-dist/css/bootstrap.min.css' %}">
    <script src="{% static 'bootstrap-3.4.1-dist/js/bootstrap.min.js' %}"></script>
</head>
<body>
    <div class="container">
        <div class="col-md-8 col-md-offset-2">
            <h2 class="text-center">用户注册</h2>
            <form id="form"> <!--不使用form表单提交数据 但是用一下form标签 它有一个序列化功能-->
                {% csrf_token %}
                {% for form in form_obj %}
                    <div class="form-group"> <!--目的是让多个获取用户数据的标签上下间距更大一些-->
                        <label for="{{ form.auto_id }}">{{ form.label }}</label> <!--form.auto_id 自动获取渲染的标签id值-->
                    {{ form }}
                    <span style="color: red" class="pull-right"></span>

                    </div>
            {% endfor %}
                <!--用户头像自己编写相关标签获取-->
                <div class="form-group">
                    <label for="myfile">头像
                        <img src="/static/img/default.jpg" alt="" id="myimg" width="120">
                    </label>
                    <input type="file" id="myfile" style="display: none" >

                </div>
                <input type="button" id="subBtn" class="btn btn-primary btn-block" value="用户注册">

            </form>


        </div>
    </div>
    <script>
        //1.用户头像的实时展示
        $('#myfile').change(function (){
            //1.产生一个文件阅读器对象
            let myFilReaderObj = new FileReader();
            // 2.获取用户上传的头像文件
            let fileObj = this.files[0];
            //3.将文件对象交给阅读器对象读取
            myFilReaderObj.readAsDataURL(fileObj); //异步
            // 等待文件阅读器对象加载完毕之后再修改src
            myFilReaderObj.onload = function (){
             //4.修改img标签的src 属性展示图片
            $('#myimg').attr('src',myFilReaderObj.result)
            }

        })

        //2.给注册按钮绑定点击事件 发送ajax 携带了文件数据
        $('#subBtn').click(function (){
            //1.先产生一个内置对象
            let myFormDataObj = new FormData();
            //2.添加普通数据(单个单个的编写效率极低)
            {#console.log($('#form').serializeArray())  // 可以一次性获取form标签内所有普通字段数据 [{数组},{数组},{数组}]#}
            $.each($('#form').serializeArray(),function (index,dataObj){  // 对结果for循环 然后交给后面的函数处理
                myFormDataObj.append(dataObj.name,dataObj.value) // {'name':'','value':'',}
            })
            //3.添加文件数据
            myFormDataObj.append('avatar',$('#myfile')[0].files[0])
            //4.发送ajax请求
            $.ajax({
                url:'',
                type: 'post',
                data:myFormDataObj,

                contentType:false,
                processData: false,
                success: function (args){
                    if(args.code === 10000){
                      window.location.href = args.url
                    }else{
                        {#console.log(args.msg)#}
                        let dataObj = args.msg;
                        // 如何针对性的渲染错误提示 {'username'}  id_username
                        $.each(dataObj,function (k,msgArray){
                            // 拼接标签的id值
                            let eleId = '#id_' + k
                            // 根据id查找标签 修改下面span标签的内容 并给父标签添加错误样式
                            $(eleId).next().text(msgArray[0]).parent().addClass('has-error')
                        })
                    }
                }
            })
        })

        //3.给所有的input标签绑定获取焦点事件 移除错误样式
        $('input').focus(function (){
            $(this).next().text('').parent().removeClass('has-error')
        })
    </script>

</body>
</html>

后端代码
from django.shortcuts import render, HttpResponse, redirect
from app01 import models
# Create your views here.
from app01 import myforms
from django.http import JsonResponse


def register_func(request):
    # 前后段ajax交互 通常采用字段作为交互对象
    back_dict = {'code': 10000, 'msg': ''}

    # 1.先产生一个空的form_obj
    form_obj = myforms.RegisterForm()
    if request.method == 'POST':
        form_obj = myforms.RegisterForm(request.POST)  # username password confirm_password email csrfmiddlewaretoken
        if form_obj.is_valid():
            clean_data = form_obj.cleaned_data  # 存储符合校验的数据  {username password confirm_password email}
            #  将confirm_password键值对移除
            clean_data.pop('confirm_password')  # {username password email}
            # 获取用户上传的头像文件
            avatar_obj = request.FILES.get('avatar')  # 用户有可能没有上传
            if avatar_obj:
                clean_data['avatar'] = avatar_obj  # {username password email avatar}
            # 创建用户数据
            models.UserInfo.objects.create_user(**clean_data)  # 上述处理字典的目的就是为了创建数据省事
            # return HttpResponse('注册成功!!!!')
            back_dict['msg'] = '注册功能'
            back_dict['url'] = '/login/'
        else:
            back_dict['code'] = 10001
            back_dict['msg'] = form_obj.errors
        return JsonResponse(back_dict)
    return render(request, 'registerPage.html', locals())


登录功能

img标签的src属性
    1.可以直接填写图片地址
    2.还可以填写一个路由 会自动朝该路由发送get请求
    如果结果是图片的二进制数据 那么自动渲染图片

报错终端执行
pip3.8 install pillow -i http://mirrors.aliyun.com/pypi/simple/ --trusted-host mirrors.aliyun.com

代码(不全 明天改)

前端代码
<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <title>Title</title>
    <script src="https://cdn.bootcdn.net/ajax/libs/jquery/3.6.1/jquery.js"></script>
    {% load static %}
    <link rel="stylesheet" href="{% static 'bootstrap-3.4.1-dist/css/bootstrap.min.css' %}">
    <script src="{% static 'bootstrap-3.4.1-dist/js/bootstrap.min.js' %}"></script>

</head>
<body>
    <div class="container">
        <div class="col-md-8 col-md-offset-2">
            <h2 class="text-center">用户登录</h2>
            <div class="form-group">
                <label for="name">用户名</label>
                <input type="text" id="name" class="form-control">
            </div>
            <div class="form-group">
                <label for="password">密码</label>
                <input type="password" id="password" class="form-control">
            </div>
            <div class="form-group">
                <label for="code">验证码</label>
                <div class="row">
                    <div class="col-md-6">
                         <input type="text" id="code" class="form-control" >
                    </div>
                    <div class="col-md-6">
                        <img src="/get_code/" alt="" width="350" height="35">
                    </div>
                </div>

            </div>
            <input type="button" class="btn btn-success btn-block" value="用户登录">
        </div>

    </div>

</body>
</html>
后端代码
from django.shortcuts import render, HttpResponse, redirect
from app01 import models

def login_func(request):
    return render(request, 'loginPage.html')


from PIL import Image, ImageFont, ImageDraw

"""
Image:       产生图片
ImageFont:   字体样式
ImageDraw:   画笔对象
"""
from io import BytesIO, StringIO

"""
BytesIO:    在内存中临时存储 读取的时候以bytes格式为准
StringIO:   在内存中临时存储 读取的时候以字符串格式为准

"""
import random


def get_random():
    return random.randint(0, 255), random.randint(0, 255), random.randint(0, 255)


def get_code_func(request):
    # 1.推导步骤1:直接读取图片文件返回
    # with open(r'D:\djangoProject\BBS\app01\avatar\555.jpg', 'rb') as f:
    #     data = f.read()
    # return HttpResponse(data)
    # # 2.推导步骤2:随机产生图片动态返回  pillow模块
    # img_obj = Image.new('RGB',(350,35),'yellow')
    # with open(r'xxx.png','wb')as f:
    #     img_obj.save(f,'png')
    # with open(r'xxx.png','rb')as f:
    #     data = f.read()
    # return HttpResponse(data)
    # 3.推导步骤3:针对图片的保存与读取 做优化 内存管理器
    # img_obj = Image.new('RGB', (350, 35), 'yellow')
    # io_obj = BytesIO()
    # img_obj.save(io_obj, 'png')
    # return HttpResponse(io_obj.getvalue())
    # 4.推导步骤4:图片颜色可以随机变换
    img_obj = Image.new('RGB', (350, 35), get_random())
    io_obj = BytesIO()
    img_obj.save(io_obj, 'png')
    return HttpResponse(io_obj.getvalue())

标签:obj,name,登录,models,True,length,注册,文章,BBS
From: https://www.cnblogs.com/xm15/p/17020558.html

相关文章

  • BBS项目练习
    BBS项目练习此项目是对博客园软件的简单模仿,旨在整合django的知识点。七张信息表用户表(与站点一对一)站点表文章表(与站点一对多)文章分类表文章标签表点赞点踩表......
  • Spring Security 源码分析(四):Spring Social实现微信社交登录
    前言在上一章Spring-Security源码分析三-Spring-Social社交登录过程中,我们已经实现了使用​​SpringSocial​​​+​​Security​​的QQ社交登录。本章我们将实现微信的社......
  • BBS项目
    项目开发基本流程1.需求分析2.架构设计3.分组开发4.提交测试5.交付上线项目流程仿博客园项目 核心:文章的增删改查表分析 先确定表的数量,在确定表的基础字段,最......
  • BBS(仿博客园项目)-基于django框架详解(包含图文)
    BBS项目(仿博客园项目)项目简介使用python中django框架开发类似博客园基本功能的小项目技术:django框架、编程语言(python3.8)、前端基础(bootstarp3.4.1,jQuery3.5.1)、......
  • bbs项目(部分讲解)
    项目开发基本流程需求分析架构设计分组开发提交测试交付上线项目分析流程仿BBS项目:仿造博客园项目核心:文章的增删改查技术栈Django、MySQL功能注册......
  • redis之登录模块的开发
    UserHolder此类封装了ThreadLocal保证了线程安全packagecom.hmdp.utils;importcom.hmdp.dto.UserDTO;publicclassUserHolder{privatestaticfinalThre......
  • bbs项目day01---注册登录
    今日内容概要主题:仿BBS项目项目开发基本流程项目分析(表)项目注册功能项目登录功能今日内容详细项目开发基本流程1.需求分析2.架构设计3.分组开发4.提交测......
  • BBS项目
    目录表之间关系表之间关系先确定表的数量再确定表的基础字段最后确定表的外键字段用户表替换auth_user表并扩展额外的字段(电话号码、头像、注册时间)classUserInfo......
  • BBS注册登入
    主题:仿BBS项目项目开发基本流程项目分析(表)项目注册功能项目登录功能今日内容详细项目开发基本流程1.需求分析2.架构设计3.分组开发4.提交测试5.交付上线......
  • Linux Debian11服务器安装SSH,创建新用户并允许远程SSH远程登录,并禁止root用户远程SSH
    一、Debian添加sudo用户1.创建新用户首先,要创建用户,当前用户必须是root用户或者sudo用户。使用下面adduser命令创建一个用户名为test的sudo用户,按照提示输入密码,......