Laravel 基础学习 第三天

用户注册功能的实现

  • 现在使用资源控制器 UserController 来完成注册
public function create()
{
    return view('user.create');
}
  • 完善 /resources/views/user/create.blade.php 在form中添加 @csrf 防止伪造表单攻击我们的服务器
@extends('layouts.master') 

@section('title', '用户注册') 

@section('content')
<div class="container">
    <h1 class="text-center">注册</h1>
    <form method="POST" action=" {{ route('user.store') }} ">
        {{--  这里是为了防止 csrf(跨站请求伪造) 攻击而配置的隐藏保护令牌 --}}
        @csrf
        <div class="form-group">
            <label for="name">用户名</label>
            <input type="text" class="form-control" id="name" placeholder="请输入用户名" name="name">
        </div>
        <div class="form-group">
            <label for="email">邮箱</label>
            <input type="email" class="form-control" id="email" placeholder="请输入邮箱" name="email">
        </div>
        <div class="form-group">
            <label for="password">密码</label>
            <input type="password" class="form-control" id="password" placeholder="请输入密码" name="password">
            <input type="password" class="form-control" placeholder="请在此输入密码以确认" name="password_confirmation">
        </div>
        <button type="rest" class="btn btn-secondary">重置</button>
        <button type="submit" class="btn btn-primary">注册</button>
    </form>
</div>
@endsection
  • 验证用户输入的数据
public function store(Request $request)
{   
    // 注意:$request 即通过 依赖注入 自动生成的用户提交的数据的临时存储变量
    // 我们需要先验证用户的数据
    $this->validate($request, [
        'name' => 'required|min:8|max:32|unique:users',
        'email' => 'required|email|unique:users',
        'password' => 'required|min:8|confirmed' //注意 confirmed => 其实是验证的页面上的 password_confirmation 是否等于 password
    ]);
    echo '没有错误!';
}

注意此时如果用户输入正确,则会显示 “没有错误!” ,但是如果不能通过验证,则会弹回 create.blade.php,但是用户输入的东西也都没有了。

  • 在视图上使用 old()方法 读取上次用户输入的但是没有通过验证的数据提升用户体验 <input type="text" ... name="name" value="{{ old('name') }}">

old('字段名') => 这是存在Session中的,只生效一次的临时保存的用户输入的信息(不会保存 type = password 类型的数据)

  • 在视图上显示错误提示信息

一旦出错, laravel 框架会将错误信息存放在一个叫 $errors 的变量中并传递给视图模板

{{--  @if(判断是否有错误)   --}}
@if(count($errors) > 0)
    <ul class="alert alert-danger">
        {{--  @foreach( $datas as $data )  --}}
        {{--  $errors->all()获取所有信息  --}}
        @foreach ($errors->all() as $error)
            <li> {{ $error }} </li>
        @endforeach
        {{--  记得结束循环@endforeach  --}}
    </ul>
@endif
{{--  记得结束判断@endif  --}}
  • 优化: 将这个错误用于提示的 html代码片段 放置到组件视图 /resources/views/components/ 下 取名叫做 _error.blade.php , 然后在 create.blade.php 中 @include('components._error') 引用

这时的错误提示信息是英文的!

i18n(国际化) 的 laravel

  • 语言包都放在 /resources/lang/ 下, 默认只有 en/ 英文语言包
  • 使用哪个语言包的配置文件是 /config/app.php 中的 'locale' => 'en',
  • 下载语言包,确保composer可用且位于laravel框架下 composer require caouecs/laravel-lang
  • 下载好的语言包位于 /vendor/caouecs/laravel-lang/src/ 中文叫 zh-CN/ 将它复制到 /resources/lang/ 下,并配置 /config/app.php locale => 'zh-CN'
  • 用于验证和错误提示的语言包叫 validation.php

自定义错误提示信息

  • 在 UserController 中自定义错误提示信息
...
$this->validate($request, [
    'name' => 'required|min:8|max:32|unique:users',
    'email' => 'required|email|unique:users',
    'password' => 'required|min:8|confirmed' //注意 confirmed => 其实是验证的页面上的 password_confirmation 是否等于 password
], [
    // 如果你在这里定义错误提示信息, 则会覆盖由语言包提供的默认的提示信息
    'name.required' => '用户名不可能不填,这辈子都不可能',
]);
// $this->validate($要验证的数据, [验证规则...], [错误提示信息...]);
...

创建一个专门用于验证的 Request 类

  • 创建一个 request 类 php artisan make:request UserRequest
  • 生成的文件位于 /app/Http/Requests/
  • 编辑 /app/Http/Requests/UserRequest
<?php

namespace App\Http\Requests;

use Illuminate\Foundation\Http\FormRequest;

class UserRequest extends FormRequest
{
    // 权限认证
    public function authorize()
    {
        return true; //这里先改为true跳过验证
    }

    // rules() 定义规则
    public function rules()
    {
        return [
            'name' => 'required|min:8|max:32|unique:users',
            'email' => 'required|email|unique:users',
            'password' => 'required|min:8|confirmed'
        ];
    }
    // messages() 定义提示信息
    public function messages()
    {
        return [
            'name.required' => '名字是不可以不写的,这辈子都不可能不写。',
        ];
    }
}
  • 在 UserController 中使用 UserRequest :
// 1、 引用 UserRequest
...
use App\Http\Requests\UserRequest;

// 2、 在参数列表中使用依赖注入的形式,构造$request
...
public function store(UserRequest $request)
{   
    // 这里面的验证我们不再需要了
    echo '没有错误!';
}
...
  • 完成注册功能
// 引用 Hash 类 
...
use Illuminate\Support\Facades\Hash;

// 完成注册逻辑
public function store(UserRequest $request)
{   
    // 获取所有post数据
    $data = $request->post();
    
    // 处理数据
    unset($data['_token']); //干掉 @csrf
    unset($data['password_confirmation']); //干掉 确认密码
    $data['password'] = Hash::make($data['password']); //使用 Hash::make() 加密密码
        // dd($data);
    
    // 插入数据
    $user = User::create($data); //插入成功后返回值为新增成功后的对象
    return redirect()->route('user.show', [$user]); //如果将对象作为 route() 的第二参数,laravel则会自动解析$user中的id,将主键id作为真正的参数
}

可以使用 User::insert() 插入数据,但是 created_time 和 updated_time 字段不会更新时间
所以最好使用 User::create

页面重定向 redirect()

  • 页面之间的跳转可以使用 return redirect() 实现
return redirect('/'); //跳转到 '/' 路由即 根网页
return redirect()->route('user.create'); //跳转到某命名路由
return redirect()->route('user.show', [参数列表]); //跳转到名字叫 'user.show' 的路由,并且传递参数

闪存 session()->flash()

  • 其实 old() 就是laravel偷偷帮我们使用这种方式实现的
// 控制器中设置闪存 session()->flash('key', 'value');
session()->flash('success', '成功');

// 视图中判断和读取闪存
{{--  判断是否有闪存 success 存在 has()  --}}
@if (session()->has('success'))
    {{--  如果存在则显示 get() --}}
    <p class="bg-success">{{ session()->get('success') }}</p>
@endif

方便我们部署提示消息

部署一个公共提示消息的视图组件

  • 创建一个闪存组件的视图 /resources/views/components/_message.blade.php
{{--  公共提示消息视图模板  --}}
<div class="container">
    {{--  定义3中情况然后循环  --}}
    @foreach (['success', 'danger', 'info'] as $msg)
        {{--  如果闪存中存有3种情况中的一种  --}}
        @if (session()->has($msg))
            {{--  则显示一个对应颜色的提示框  --}}
            <div class="alert alert-{{$msg}}">
                {{--  显示提示信息  --}}
                {{session()->get($msg)}}
            </div>
        @endif
    @endforeach
</div>
  • 在注册逻辑中,添加数据成功后,页面重定向之前添加一条闪存
public function store(UserRequest $request)
{   
    // 获取所有post数据
    ...
    
    // 处理数据
    ...
    
    // 插入数据
    $user = User::create($data); 
    // 增加闪存
    session()->flash('success', '注册成功,以下是您的账号信息!');
    // 跳转页面
    return redirect()->route('user.show', [$user]); 
}

完善登陆功能

  • 创建一个控制器专门用于登陆操作 php artisan make:controller LoginController
<?php

namespace App\Http\Controllers;

use Illuminate\Http\Request;

class LoginController extends Controller
{
    /**
     * 登陆页面
     */
    public function login() {
        return view('login.login');
    }

    /**
     * 存储登陆信息
     */
    public function store() {

    }

    /**
     * 注销
     */
    public function logout() {

    }
}
  • 配置路由 /routes/web.php
Route::get('login', 'LoginController@login')->name('login');
// 可以定义两个同名,同uri 的路由,只要他们的action 不一样即是两条不同的路由
Route::post('login', 'LoginController@store')->name('login'); 
Route::get('logout', 'LoginController@logout')->name('logout');
  • 登陆页面视图 /resources/views/login/login.blade.php => 直接复制 create.blade.php ,不要邮箱和密码确认即可。
  • 使用 Auth 验证登陆
// 1 引用Auth
...
use Auth;

// 2 调用 Auth::attempt(用户信息) 验证
public function store(Request $request) {
// 验证数据: 因为只有2个字段,只需要验证他们是否填写了即可
$user = $this->validate($request, [
    'name' => 'required',
    'password' => 'required',
]); //$this->validate() 方法如果通过验证 会将表单数据变成一个数组当作返回值
    //dd($user);

// 使用 Auth 验证用户名密码是否正确
$res = Auth::attempt($user); // Auth::attempt(要求参数是一个数组) 返回值是boolean 验证通过是true 否则是false
if(!$res) {
    session()->flash('danger', '用户名或密码错误!'); //使用增加一条提示“密码错误”的闪存信息
    return redirect()->back()->withInput(); //back() 返回前一页面 withInput() 将表单数据也还回去
}
session()->flash('success', '欢迎');
return redirect('/');    
}
  • 在视图层使用 Auth 提供的方法判断用户是否登陆 Auth::check() ,如果登陆,使用 Auth::user() 提取用户数据并显示
{{--  如果登陆 Auth::check() == true  --}}
@if (Auth::check())
    {{--  Auth::user() 提取用户信息为对象 ->name 访问用户名  --}}
    <a href="#" class="btn btn-outline-success"> {{ Auth::user()->name }} </a>
    <a href="{{ route('logout') }}" class="btn btn-outline-danger"> 注销 </a>
    @else
    <a href="{{ route('user.create') }}" class="btn btn-outline-success">注册</a>
    <a href="{{ route('login') }}" class="btn btn-outline-primary">登陆</a>
@endif
  • 注册逻辑中,让用户注册成功后自动登陆 UserController
// 1 使用 Auth
use Auth;

// 2 调用 Auth::login() 将注册信息作为参数,实现直接登陆
public function store(UserRequest $request)
{   
    // 获取所有post数据
    ...
    // 处理数据
    ...
    // 插入数据
    ...
    // 增加闪存
    ...
    // 注册成功后自动登陆
    Auth::login($user); //Auth::login(直接将$user数据信息传入)
    // 跳转页面
    ...
}
  • 完成退出功能
public function logout() {
    Auth::logout(); // Auth::logout() 会清空已登陆的用户信息
    session()->flash('info', '您已成功退出!'); //发送一条提示信息
    return redirect()->route('login'); //跳转到登陆页面
}
  • 完成 “记住我” 功能

Auth::attmept() 其实是将登陆信息写入 Cookies 中
可以看到普通登陆后,laravel_session的有效时间就是一天以内(具体多久我没算)

* 首先登陆页面上的表单更新一下
```
<div class="form-check">
    {{--  勾选为true 不勾选没有 remember_me 字段  --}}
    <input type="checkbox" class="form-check-input" id="remember_me" name="remember_me" value="true">
    <label class="form-check-label" for="remember_me">记住我</label>
</div>
```

* 然后在登陆逻辑中
```
// 使用 Auth 验证 Auth::attempt(用户信息, 是否记住用户true记住)
$res = Auth::attempt($user, $request->has('remember_me')); // $request->has('某个字段') 如果字段存在则返回true 否则返回false
```

可以看到这样做会在 Cookies 里生成一个 remember_web_xxxxxxxxxxxx 的字段,有效期大约5年。它会存一个和 users表中 remember_token 字段有关的一个值。通过比对两个值来确定用户是否勾选过长期登陆。

总结一下

  • post表单需要添加 @csrf 来告诉laravel,这是内部自己的表单,可以信赖
  • 验证数据可以在控制器内部验证,也可以新建一个 Request 验证
    • 自己内部验证
    $this->validate($data, [验证规则], [提示信息]);
    
    • 建一个 Request 验证 php artisan make:request XxRequest => 然后在构建 $request 的时候使用 XxRequest 为规范类
    // rules() 定义规则
    public function rules()
    {
        return [
            'name' => 'required|min:8|max:32|unique:users',
            'email' => 'required|email|unique:users',
            'password' => 'required|min:8|confirmed'
        ];
    }
    // messages() 定义提示信息
    public function messages()
    {
        return [
            'name.required' => '名字是不可以不写的,这辈子都不可能不写。',
        ];
    }
    
  • 在视图层表单项使用 value="{{ old('表单字段') }}" 使出错后,除 password 字段的其他字段可以依然存在,提升用户体验
  • 一旦验证出错,错误都存放在一个叫 $errors 的对象中,可以通过遍历 $errors->all() 来显示错误。
  • session()->flash('key', 'value') 显示临时的提示信息
  • Auth::login($用户信息) => 注册的时候我们将用户信息直接作为参数给login()方法实现了登陆
  • Auth::logout() => 注销登陆的账号
  • Auth::attempt($用户信息, 是否记住账号) => 登陆验证用户名密码, 第二参数默认false不记住
  • Auth::check() => 判断用户是否登陆
  • Auth::user() => 获取用户的信息,可以通过 Auth::user()->属性 来展示一些信息,比如 ...->name 显示用户名
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,539评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,911评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,337评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,723评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,795评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,762评论 1 294
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,742评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,508评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,954评论 1 308
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,247评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,404评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,104评论 5 340
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,736评论 3 324
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,352评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,557评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,371评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,292评论 2 352

推荐阅读更多精彩内容

  • 一、框架概述 课程概述 laravel 4天(之前TP框架还是很大的区别)(国外框架) 在线教育平台 6天(lar...
    大大的世界小小的梦想_97ef阅读 1,352评论 0 2
  • Laravel 学习交流 QQ 群:375462817 本文档前言Laravel 文档写的很好,只是新手看起来会有...
    Leonzai阅读 7,880评论 2 12
  • # 一、框架概述 # 课程概述 1. laravel 4天(之前TP框架还是很大的区别)(国外框架) 2. 在线教...
    关进一阅读 367评论 0 0
  • 1. 一、框架概述 2. 课程概述 laravel 4天(之前TP框架还是很大的区别)(国外框架) 在线教育平台 ...
    jim1999阅读 327评论 0 0
  • 一、框架概述 1、什么是框架? 框架是许多代码的集合,为项目开发提供了很多基础性的代码。例如常见的参数过滤,DB类...
    神经波阅读 5,115评论 0 3