Laravel重置密碼重構(gòu)

需要使用laravel搭建一個(gè)后臺(tái)內(nèi)容管理系統(tǒng),但是laravel默認(rèn)的登陸注冊(cè)不能滿足目前的需求

重置密碼的話因?yàn)槭怯迷诤笈_(tái),并且不需要發(fā)送郵件進(jìn)行重置,所以默認(rèn)的重置密碼肯定是不行的诚啃。

1. 首先確定重置密碼的路由

我們?cè)诎惭b好laravel的時(shí)候默認(rèn)生成的重置密碼是在用戶未登錄的情況下進(jìn)行的。所以使用原來(lái)的控制器是不可行的

并且原有的重置密碼私沮,并不需要查看原始密碼是否正確始赎,而是通過(guò)郵件來(lái)進(jìn)行直接更改密碼,所以控制器方法的話,我們也需要重新寫個(gè)

我們使用php artisan make:controller UserController創(chuàng)建一個(gè)控制器類

然后創(chuàng)建兩條路由Route::get('reset', 'UserController@getReset')Route::post('reset', 'UserController@postReset')

前者是顯示一個(gè)重置密碼的頁(yè)面get請(qǐng)求造垛,后面是重置密碼post請(qǐng)求魔招。

2. 顯示重置密碼頁(yè)

這個(gè)使用的是getReset這個(gè)方法,這個(gè)方法只需要顯示一個(gè)視圖所以并沒(méi)有特別的邏輯

public function getReset()
{
    return view('auth.reset');
}

3. 請(qǐng)求重置密碼

這個(gè)使用的是postReset這個(gè)方法

接收數(shù)據(jù)的話我們使用兩種方法接收傳過(guò)來(lái)的數(shù)據(jù)都可以:

一種是使用request的方法接收數(shù)據(jù)五辽,另外一種是使用Input::get的方法獲取數(shù)據(jù)办斑。

Request的話需要引入use Illuminate\Http\Request

Input的話需要引入use Input

這里我們選擇使用request來(lái)接收

4. 驗(yàn)證規(guī)則

驗(yàn)證的話,laravel為我們提供了一套驗(yàn)證的規(guī)則杆逗,使用validatorValidator::make()方法進(jìn)行驗(yàn)證

$data = $request->all(); //接收所有的數(shù)據(jù)
$rules = [
    'oldpassword'=>'required|between:6,20',
    'password'=>'required|between:6,20|confirmed',
];
$messages = [
    'required' => '密碼不能為空',
    'between' => '密碼必須是6~20位之間',
    'confirmed' => '新密碼和確認(rèn)密碼不匹配'
];
$validator = Validator::make($data, $rules, $messages);

$data 接收到從from傳過(guò)來(lái)的數(shù)據(jù)信息

rules 對(duì)接收到的值進(jìn)行判斷乡翅,其中數(shù)組前面的oldpasswordpassword是從前端from接收到的原始密碼和新密碼的name字段數(shù)據(jù)進(jìn)行驗(yàn)證

驗(yàn)證規(guī)則的話在手冊(cè)的驗(yàn)證章節(jié)都有,值得注意的是罪郊,使用confirmed的話是為了新密碼和確認(rèn)密碼進(jìn)行相同判斷蠕蚜,確認(rèn)密碼必須的name值必須是

新密碼的name值后面加上'_confirmation',比如新密碼的name值為newpassword的話,確認(rèn)密碼的name值則必須為newpassword_confirmation才可以進(jìn)行判斷

messages對(duì)驗(yàn)證的數(shù)據(jù)請(qǐng)求悔橄,顯示什么提示



然后通過(guò)上面的驗(yàn)證靶累,還有個(gè)情況是沒(méi)有驗(yàn)證的,那就是輸入的原始密碼是否和數(shù)據(jù)庫(kù)里的原始密碼相同癣疟。

這里我們可以先把這個(gè)用戶的信息從數(shù)據(jù)庫(kù)里給查出來(lái)挣柬,然后和輸入的原始密碼進(jìn)行比對(duì)。

這里我們使用Auth::user()來(lái)獲取用戶的信息睛挚,這個(gè)方法需要引入use Auth;

然后通過(guò)Hash::check()來(lái)進(jìn)行密碼判斷凛忿。

判斷完以后還有個(gè)問(wèn)題,那就是竞川,如何把錯(cuò)誤信息給壓入到validator的錯(cuò)誤信息里店溢,這里laravel為我們提供了after方法

$user = Auth::user();
$validator->after(function($validator) use ($oldpassword, $user) {
    if (!\Hash::check($oldpassword, $user->password)) { //原始密碼和數(shù)據(jù)庫(kù)里的密碼進(jìn)行比對(duì)
        $validator->errors()->add('oldpassword', '原密碼錯(cuò)誤'); //錯(cuò)誤的話顯示原始密碼錯(cuò)誤
    }
});
if ($validator->fails()) {      //判斷是否有錯(cuò)誤
    return back()->withErrors($validator);  //重定向頁(yè)面,并把錯(cuò)誤信息存入一次性session里
}
$user->password = bcrypt($password);       //使用bcrypt函數(shù)進(jìn)行新密碼加密
$user->save();      //成功后委乌,保存新密碼

這里因?yàn)?code>after 引入了一個(gè)PHP的匿名函數(shù)床牧,所以我們需要使用use 關(guān)鍵字把外部數(shù)據(jù)給傳入到匿名函數(shù)里(PS:php新特性,閉包和匿名函數(shù))

在匿名函數(shù)里我們引入了一個(gè)全局函數(shù)所以我們需要在函數(shù)前面加\(PS:php新特性遭贸,命名空間章節(jié)戈咳,全局命名空間)

5. 前端顯示錯(cuò)誤信息

前端顯示的話,我們使用$errors變量來(lái)顯示錯(cuò)誤壕吹,根據(jù)官方文檔說(shuō)明著蛙,調(diào)用的是Illuminate\Support\MessageBag的示例,有興趣的話耳贬,可以看下

我們使用count($errors) > 0來(lái)判斷是否有錯(cuò)誤踏堡,使用 $errors->first()顯示一條錯(cuò)誤信息

@if(count($errors) > 0)
    <div class="alert alert-danger display-hide" style="display: block;">
        <button class="close" data-close="alert"></button>
        <span> {{$errors->first()}}  </span>
    </div>
@endif

可能會(huì)有人問(wèn),如果我的錯(cuò)誤不是顯示在固定的一個(gè)地方咒劲,而是在每個(gè)表單的后面顯示錯(cuò)誤信息的話顷蟆,這樣我們?cè)撛趺磁袛嗪惋@示呢诫隅?
答案是使用$errors->has('oldpassword')來(lái)判斷有沒(méi)有這個(gè)名稱的錯(cuò)誤,如果有的話帐偎,使用 $errors->first('oldpassword') 顯示這條錯(cuò)誤

@if( $errors->has('oldpassword') )
    <div class="alert alert-danger display-hide" style="display: block;">
        <button class="close" data-close="alert"></button>
        <span> {{$errors->first('oldpassword')}}  </span>
    </div>
@endif

其中oldpassword是每個(gè)表單的里的name值逐纬,所以在使用after方法添加自定義錯(cuò)誤的時(shí)候
$validator->errors()->add('oldpassword', '原密碼錯(cuò)誤');中,oldpassword一定要寫對(duì)是在哪個(gè)表單的錯(cuò)誤削樊,這樣才能正確的顯示豁生。

8. 完成后的示例

UserController

public function getReset()
{
    return view('auth.reset');
}

public function postReset(Request $request)
{
    $oldpassword = $request->input('oldpassword');
    $password = $request->input('password');
    $data = $request->all();
    $rules = [
        'oldpassword'=>'required|between:6,20',
        'password'=>'required|between:6,20|confirmed',
    ];
    $messages = [
        'required' => '密碼不能為空',
        'between' => '密碼必須是6~20位之間',
        'confirmed' => '新密碼和確認(rèn)密碼不匹配'
    ];
    $validator = Validator::make($data, $rules, $messages);
    $user = Auth::user();
    $validator->after(function($validator) use ($oldpassword, $user) {
        if (!\Hash::check($oldpassword, $user->password)) {
            $validator->errors()->add('oldpassword', '原密碼錯(cuò)誤');
        }
    });
    if ($validator->fails()) {
        return back()->withErrors($validator);  //返回一次性錯(cuò)誤
    }
    $user->password = bcrypt($password);
    $user->save();
    Auth::logout();  //更改完這次密碼后,退出這個(gè)用戶
    return redirect('/login');
}

reset.blade

<form class="login-form" action="{{ url('/reset') }}" method="post">
        <h3 class="font-green">修改密碼</h3>
        @if($errors->first())
            <div class="alert alert-danger display-hide" style="display: block;">
                <button class="close" data-close="alert"></button>
                <span> {{$errors->first()}}  </span>
            </div>
        @endif
        {!! csrf_field() !!}

        <div class="form-group">
            <label class="control-label visible-ie8 visible-ie9">原始密碼</label>
            <input class="form-control placeholder-no-fix" type="password" autocomplete="off" placeholder="Old Password" name="oldpassword"> </div>
        <div class="form-group">
            <label class="control-label visible-ie8 visible-ie9">新密碼</label>
            <input class="form-control placeholder-no-fix" type="password" autocomplete="off" id="register_password" placeholder="New password" name="password"> </div>
        <div class="form-group">
            <label class="control-label visible-ie8 visible-ie9">重復(fù)密碼</label>
            <input class="form-control placeholder-no-fix" type="password" autocomplete="off" placeholder="Repeat password" name="password_confirmation"> </div>
        <div class="form-actions">
            <button type="submit" id="register-submit-btn" class="btn btn-success uppercase pull-right">確定</button>
        </div>
    </form>

原文鏈接:Dennis`s blog

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末漫贞,一起剝皮案震驚了整個(gè)濱河市沛硅,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌绕辖,老刑警劉巖摇肌,帶你破解...
    沈念sama閱讀 217,826評(píng)論 6 506
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場(chǎng)離奇詭異仪际,居然都是意外死亡围小,警方通過(guò)查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,968評(píng)論 3 395
  • 文/潘曉璐 我一進(jìn)店門树碱,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái)肯适,“玉大人,你說(shuō)我怎么就攤上這事成榜】蛱颍” “怎么了?”我有些...
    開封第一講書人閱讀 164,234評(píng)論 0 354
  • 文/不壞的土叔 我叫張陵赎婚,是天一觀的道長(zhǎng)刘绣。 經(jīng)常有香客問(wèn)我,道長(zhǎng)挣输,這世上最難降的妖魔是什么纬凤? 我笑而不...
    開封第一講書人閱讀 58,562評(píng)論 1 293
  • 正文 為了忘掉前任,我火速辦了婚禮撩嚼,結(jié)果婚禮上停士,老公的妹妹穿的比我還像新娘。我一直安慰自己完丽,他們只是感情好恋技,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,611評(píng)論 6 392
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著逻族,像睡著了一般蜻底。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上瓷耙,一...
    開封第一講書人閱讀 51,482評(píng)論 1 302
  • 那天朱躺,我揣著相機(jī)與錄音,去河邊找鬼搁痛。 笑死长搀,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的鸡典。 我是一名探鬼主播源请,決...
    沈念sama閱讀 40,271評(píng)論 3 418
  • 文/蒼蘭香墨 我猛地睜開眼,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼彻况!你這毒婦竟也來(lái)了谁尸?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 39,166評(píng)論 0 276
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤纽甘,失蹤者是張志新(化名)和其女友劉穎良蛮,沒(méi)想到半個(gè)月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體悍赢,經(jīng)...
    沈念sama閱讀 45,608評(píng)論 1 314
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡决瞳,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,814評(píng)論 3 336
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了左权。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片皮胡。...
    茶點(diǎn)故事閱讀 39,926評(píng)論 1 348
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖赏迟,靈堂內(nèi)的尸體忽然破棺而出屡贺,到底是詐尸還是另有隱情,我是刑警寧澤锌杀,帶...
    沈念sama閱讀 35,644評(píng)論 5 346
  • 正文 年R本政府宣布甩栈,位于F島的核電站,受9級(jí)特大地震影響糕再,放射性物質(zhì)發(fā)生泄漏谤职。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,249評(píng)論 3 329
  • 文/蒙蒙 一亿鲜、第九天 我趴在偏房一處隱蔽的房頂上張望允蜈。 院中可真熱鬧,春花似錦蒿柳、人聲如沸饶套。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,866評(píng)論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)妓蛮。三九已至,卻和暖如春圾叼,著一層夾襖步出監(jiān)牢的瞬間蛤克,已是汗流浹背捺癞。 一陣腳步聲響...
    開封第一講書人閱讀 32,991評(píng)論 1 269
  • 我被黑心中介騙來(lái)泰國(guó)打工, 沒(méi)想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留构挤,地道東北人髓介。 一個(gè)月前我還...
    沈念sama閱讀 48,063評(píng)論 3 370
  • 正文 我出身青樓,卻偏偏與公主長(zhǎng)得像筋现,于是被迫代替她去往敵國(guó)和親唐础。 傳聞我的和親對(duì)象是個(gè)殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,871評(píng)論 2 354

推薦閱讀更多精彩內(nèi)容

  • 1矾飞、簡(jiǎn)介 Laravel 提供了多種方法來(lái)驗(yàn)證應(yīng)用輸入數(shù)據(jù)一膨。默認(rèn)情況下,Laravel 的控制器基類使用Valid...
    伊Summer閱讀 1,525評(píng)論 0 3
  • 驗(yàn)證 簡(jiǎn)介 Laravel 對(duì)驗(yàn)證應(yīng)用的輸入數(shù)據(jù)提供了多中途徑的實(shí)現(xiàn)洒沦。默認(rèn)的豹绪,Laravel 的基礎(chǔ)控制器類使用了...
    Dearmadman閱讀 10,208評(píng)論 5 8
  • 原文鏈接 必備品 文檔:Documentation API:API Reference 視頻:Laracasts ...
    layjoy閱讀 8,607評(píng)論 0 121
  • HTML表單 在HTML中,表單是 ... 之間元素的集合申眼,它們?cè)试S訪問(wèn)者輸入文本森篷、選擇選項(xiàng)、操作對(duì)象等等豺型,然后將...
    蘭山小亭閱讀 3,417評(píng)論 2 14
  • 22年12月更新:個(gè)人網(wǎng)站關(guān)停仲智,如果仍舊對(duì)舊教程有興趣參考 Github 的markdown內(nèi)容[https://...
    tangyefei閱讀 35,182評(píng)論 22 257