Laravel 4 初級教程之視圖、命名空間、路由
1. 視圖分離與嵌套
在 learnlaravel 文件夾下運行命令:
php artisan generate:view admin._layouts.default
這時候generator插件幫我們創(chuàng)建了app/views/admin/_layouts/default.blade.php 文件,將內(nèi)容修改為:
<!doctype html>
<html>
<head>
<meta charset="utf-8">
<title>Learn Laravel 4</title>
@include('admin._partials.assets')
</head>
<body>
<div class="container">
<div class="navbar navbar-inverse navbar-fixed-top">
<div class="navbar-inner">
<div class="container">
<a class="brand" href="{{ URL::route('admin.pages.index') }}">Learn Laravel 4</a>
@include('admin._partials.navigation')
</div>
</div>
</div>
<hr>
@yield('main')
</div>
</body>
</html>
這就是視圖文件,MVC中的V。視圖需要仔細講一下。
views文件夾為視圖文件夾,視圖文件夾可以嵌套,就像我上面一樣創(chuàng)建了admin/_layout嵌套文件夾,在里面創(chuàng)建了一個叫default.blade.php的文件,那么以后我們在Laravel內(nèi)任何地方要用到這個視圖的時候,他就叫admin._layouts.default。
我們看到,上面代碼的第七行是“@include('admin._partials.assets')”,根據(jù)上面我們剛剛了解的知識,這表示載入了另外一個文件。blade是Laravel的模板引擎,此處的 @include 表示直接把那個文件的所有代碼帶入進來放到這里,變成當前視圖的一部分。
注意看第25行“@yield('main')”,這表示什么呢?這個有點復(fù)雜,我們稍后再講。
2. 權(quán)限驗證
Laravel支持標準HTTP認證,但是在此處我們需要構(gòu)建blog系統(tǒng),所以我們將編寫完善的管理員登陸系統(tǒng),從頁面登錄。
用命令行創(chuàng)建app/views/admin/auth/login.blade.php文件,代碼如下:
@extends('admin._layouts.default')
@section('main')
<div id="login" class="login">
{{ Form::open() }}
@if ($errors->has('login'))
<div class="alert alert-error">{{ $errors->first('login', ':message') }}</div>
@endif
<div class="control-group">
{{ Form::label('email', 'Email') }}
<div class="controls">
{{ Form::text('email') }}
</div>
</div>
<div class="control-group">
{{ Form::label('password', 'Password') }}
<div class="controls">
{{ Form::password('password') }}
</div>
</div>
<div class="form-actions">
{{ Form::submit('Login', array('class' => 'btn btn-inverse btn-login')) }}
</div>
{{ Form::close() }}
</div>
@stop
大家應(yīng)該注意到了前兩行:
@extends('admin._layouts.default')@section('main')
這代表什么?實際上,以后我們會了解到,在controller中調(diào)用view的時候,調(diào)用的只是這個login.blade.php文件,第一行表示,此視圖是admin._layouts.default的子視圖,這時blade引擎會把這個視圖也載入進來,怎么組裝呢?這時候下面那個@section('main')就該出場了,被它包裹的代碼將會直接放到admin._layouts.default中的@yield('main')中。section和yield可以任意搭配,只要兩個視圖之間有調(diào)用關(guān)系,他們就可以這樣用,非常靈活。
寫到這里大家可能有個疑問,為什么示例代碼里空行那么多?這一點就是個人經(jīng)驗了。blade引擎的所有標簽都會在視圖編譯時用正則處理,引擎本身有一個問題,算不上bug,就是換行符會被處理掉,導(dǎo)致前后行和這一行都緊緊地擠在一起,在前端瀏覽器中“查看源代碼”時,比較不清晰,前后加上空行可以解決這個問題。當然這可能是一個自動的“壓縮”特性,不再深入討論。
增加控制器文件app/controllers/admin/AuthController.php,這時候有人就說了,這我知道,哈哈,運行
“php artisan generate:controller admin.AuthController”
這個想法是對的,但你運行一下試試?會直接在app/controllers目錄下創(chuàng)建一個“admin.AuthController.php”文件,有人又說,那我用“admin/AuthController”總行了吧,你試一下?也不行。所以我們要先在app/controllers 下手動創(chuàng)建 admin 文件夾,這時候,再命令行輸入:
php artisan generate:controller admin/AuthController
這樣就可以了。接下來改寫AuthController.php 的內(nèi)容為:
<?php
namespace App\Controllers\Admin;
use Auth, BaseController, Form, Input, Redirect, Sentry, View;
class AuthController extends BaseController {
/**
* 顯示登錄頁面
* @return View
*/
public function getLogin()
{
return View::make('admin.auth.login');
}
/**
* POST 登錄驗證
* @return Redirect
*/
public function postLogin()
{
$credentials = array(
'email' => Input::get('email'),
'password' => Input::get('password')
);
try
{
$user = Sentry::authenticate($credentials, false);
if ($user)
{
return Redirect::route('admin.pages.index');
}
}
catch(\Exception $e)
{
return Redirect::route('admin.login')->withErrors(array('login' => $e->getMessage()));
}
}
/**
* 注銷
* @return Redirect
*/
public function getLogout()
{
Sentry::logout();
return Redirect::route('admin.login');
}
}
這就是我們登錄、注銷的控制器,MVC中的C。接下來我將講解命名空間,這是Laravel的基礎(chǔ),或者說是composer的基礎(chǔ),是整個Laravel教程中的重點、難點,希望大家錙銖必較,任何不懂都不要放過??梢缘絧hphub論壇或者golaravel論壇相應(yīng)帖子下面提問,或者直接發(fā)帖提問。
我們首先觀察這個文件的位置,它位于 app/controllers/admin 目錄下,這有什么不同呢?在其他框架如 CI 中,子文件夾直接加上文件夾名就可以直接調(diào)用到了,雖然最多只能有一層。而Laravel沒有這么簡單,涉及到了PHP的命名空間。
1. composer 支持 PSR-0 及 PSR-4 標準,標準規(guī)定 PHP 包以命名空間為區(qū)分,向外提供服務(wù),所有暴露出來的類都應(yīng)該在 \作者名\包名 命名空間下,例如 \lui\MFFC\Mail 類。這樣,哪怕是名稱一樣的包只要是不同作者也可以在https://packagist.org/上共存,供大家使用。
2. 命名空間可以類比成 Linux 系統(tǒng)中的 目錄,在任何目錄下都可以直接使用文件名打開當前目錄下的所有文件和可執(zhí)行程序,如果需要打開其他目錄下的文件,就需要使用絕對路徑或者相對路徑。
3. 大家可能在許多其他教程中見到過controller頭部沒有 namesapce 申明,更沒有那一堆的 use xxx,像這個文件https://github.com/cecoo/laravel4demo/blob/master/app/controllers /BlogController.php。這個文件在第8行直接使用了 Blog 這個類,這是為什么呢?
因為他們都已經(jīng)在 learnlaravel 這個 composer 應(yīng)用的配置文件中聲明為自動加載了,而他們沒有在頂部聲明他們所在的命名空間,這樣就會被自動加為頂級命名空間。這個配置文件是 composer.json,對象配置項為autoload 下的classmap 項。這個聲明會讓 Composer 在生成自動載入文件的時候,自動掃描該文件下所有的類以及所有子文件夾中的類,只要沒有聲明特定的命名空間,將會被自動加載為頂級空間?!局氨硎鲇姓`,特此更正!】
關(guān)于命名空間更多詳情,可以參考 【PHP 命名空間 入門】。
OK,到目前為止我們的MVC三元素已經(jīng)集齊了,那接下來該做什么了呢?配置路由。這里的路由并不是家里用的無線路由 :-D,而是 用戶請求的URL到控制器某個方法的轉(zhuǎn)換,function是PHP中代碼段的最小單位,所以用戶請求的一個路徑,如 http://ooxx.com/fuck/me ,這條URL打給路由之后,路由就會去解析,應(yīng)該調(diào)用哪個function,最終返回結(jié)果給用戶。
Laravel的路由采用閉包的方式返回結(jié)果,在app/routes.php 中增加下列內(nèi)容:
Route::get('admin/logout', array('as' => 'admin.logout', 'uses' => 'App\Controllers\Admin\AuthController@getLogout'));
Route::get('admin/login', array('as' => 'admin.login', 'uses' => 'App\Controllers\Admin\AuthController@getLogin'));
Route::post('admin/login', array('as' => 'admin.login.post', 'uses' => 'App\Controllers\Admin\AuthController@postLogin'));
Route::group(array('prefix' => 'admin', 'before' => 'auth.admin'), function()
{
Route::any('/', 'App\Controllers\Admin\PagesController@index');
Route::resource('articles', 'App\Controllers\Admin\ArticlesController');
Route::resource('pages', 'App\Controllers\Admin\PagesController');
});
前三條的意思是hold住兩個get請求和一個post請求,下面是一個路由組,規(guī)定了一個前綴admin,增加了一個過濾器,auth.admin,內(nèi)部有一個能同時適應(yīng)get和post請求的‘/'路徑,其完整路徑是 http://ooxx.com/admin/。剩下的兩個資源控制器本質(zhì)上只是一種簡寫,URL和控制器類中的方法名的對應(yīng)表見 資源控制器。
上面說的那個過濾器 auth.admin,是Laravel提供的一個請求過濾器,這個文件就在路由文件的旁邊,app/filters.php,在文件末尾增加:
Route::filter('auth.admin', function()
{
if ( ! Sentry::check()) {
return Redirect::route('admin.login');
}
});
這樣我們的權(quán)限驗證就完成了。上面的代碼意思是,在進入這個路由組中的任何一條路由之前,會先過一遍 auth.admin這個filter,這個filter會調(diào)用Sentry::check(),如果為false,將會進入if代碼塊,將用戶的請求跳轉(zhuǎn)到 命名路由‘a(chǎn)dmin.login',命名路由文檔。從這個命名路由的名稱大家也能看出來,就是跟訪客說:傻逼,干啥呢,登錄去~
這里的“命名路由”功能是為了模仿 Ruby On Rails 的 “l(fā)ink_to”到對象 的路由綁定功能,無奈PHP上傳即部署無守護進程的特性,使得我們沒法維護一個全量代碼的路由表,沒法像Rails那樣實現(xiàn) 資源路由-資源對象-路由調(diào)用 三者綁定的功能,只能搞出一個半成品命名路由,人為地解決了當調(diào)整 /people 到 /human 時,要求名稱改變而功能不變,同時要求代碼自適應(yīng)的需求。
這時候,我們就可以嘗試訪問我們的項目了。推薦配置Apache將一個端口指向learnlaravel這個項目的public目錄下,即項目通過 http://127.0.0.1:8080 這樣的地址訪問,十分不建議從子文件夾訪問。如果你不會,可以運行
php artisan serve
啟動PHP5.4的內(nèi)建HTTP服務(wù)器。地址將會是http://localhost:8000,注意此處 127.0.0.1 不可以訪問。
下面,我們在瀏覽器中訪問 /admin,注意URL會自動跳轉(zhuǎn)到 /admin/login,這說明我們的filter起作用了,但你可能得到以下頁面
這說明代碼出錯了。接下來我們修改 app/config/app.php 第一項為:
'debug' => true,
刷新頁面,錯誤提示出來了!有沒有感覺Laravel4.2的錯誤提示很好看啊,確實不錯,但我覺得沒有4.1之前的好看 :-D。我得到了如下錯誤:
說“App\Controllers\Admin\AuthController”這個類未找到,這是為什么呢?這個文件明明有啊。
這就涉及到了另一個問題,Laravel中的autoload問題。Laravel基于命名空間,它只會自動加載所有頂級命名空間的類,就是說我們新增的這個控制器類不是在頂級命名空間下,所以就需要告訴Laravel,我這個類是存在的,怎么告訴它呢?運行
composer dump-autoload
可以了,刷新頁面,他告訴我
View [admin._partials.assets] not found.
這個確實是,我們還沒建立這個文件呢。建立一個空文件即可,如果是用generator建的話,別忘了把里面默認的內(nèi)容刪掉哦。再刷新頁面,如果還有問題,我相信這個問題你可以自己解決。
OK,一個丑的一逼的頁面出現(xiàn)了,為什么它這么丑?(鴿子為什么這么大?)因為我們沒有引入任何css和js文件,甚至連導(dǎo)航欄的html都不完整。這不要緊,來,按照我github上的代碼,自己復(fù)制到相應(yīng)文件中吧。另外,非常重要的一點,把我的項目中的public文件下的 js 和 css 兩個文件夾完全復(fù)制到你們的public文件夾中。
再刷新,如果你看到以下頁面,說明你成功了!
3. 嘗試登錄
用seed新增一名管理員,順便新增一個管理員組。新建app/database/seeds/SentrySeeder.php,內(nèi)容為:
<?php
class SentrySeeder extends Seeder {
public function run()
{
DB::table('users')->delete();
DB::table('groups')->delete();
DB::table('users_groups')->delete();
Sentry::getUserProvider()->create(array(
'email' => 'oo@xx.com',
'password' => "ooxx",
'first_name' => 'OO',
'last_name' => 'XX',
'activated' => 1,
));
Sentry::getGroupProvider()->create(array(
'name' => 'Admin',
'permissions' => ['admin' => 1],
));
// 將用戶加入用戶組
$adminUser = Sentry::getUserProvider()->findByLogin('oo@xx.com');
$adminGroup = Sentry::getGroupProvider()->findByName('Admin');
$adminUser->addGroup($adminGroup);
}
}
給app/database/seeds/DatabaseSeeder.php 新增一行:
$this->call('SentrySeeder');
然后運行:
php artisan db:seed
成功以后,進數(shù)據(jù)庫就會發(fā)現(xiàn),users、groups、users_groups表均新增了一行。但是,articles和pages表也分別新增了10行,對,seed就是這么蠢萌^_^
讓我們來嘗試登錄!如果你得到:
Class App\Controllers\Admin\PagesController does not exist
這說明你成功了!
- Laravel5.1框架路由分組用法實例分析
- laravel框架分組控制器和分組路由實現(xiàn)方法示例
- laravel中命名路由的使用方法
- Laravel 5框架學(xué)習(xí)之路由、控制器和視圖簡介
- Laravel中間件實現(xiàn)原理詳解
- Laravel框架實現(xiàn)利用中間件進行操作日志記錄功能
- Laravel獲取當前請求的控制器和方法以及中間件的例子
- Laravel 6 將新增為指定隊列任務(wù)設(shè)置中間件的功能
- 淺談Laravel中的三種中間件的作用
- 解決Laravel自定義類引入和命名空間的問題
- 修改Laravel自帶的認證系統(tǒng)的User類的命名空間的步驟
- laravel框架路由分組,中間件,命名空間,子域名,路由前綴實例分析
相關(guān)文章
php中使用array_filter()函數(shù)過濾空數(shù)組的實現(xiàn)代碼
這篇文章主要介紹了php中使用array_filter()函數(shù)過濾空數(shù)組的實現(xiàn)代碼,這是瀏覽PHP手冊時無意發(fā)意的一個有意思的array_filter()函數(shù)用法,需要的朋友可以參考下2014-08-08淺析PHP中call user func()函數(shù)及如何使用call user func調(diào)用自定義函數(shù)
使用call_user_func函數(shù),通過傳入字符串函數(shù),可以調(diào)用自定義函數(shù),并且支持引用。該函數(shù)允許用戶調(diào)用直接寫的函數(shù)并傳入一定的參數(shù),下面總結(jié)下這個函數(shù)的使用方法,需要的朋友參考下2015-11-11yii2-GridView在開發(fā)中常用的功能及技巧總結(jié)
本篇文章主要介紹了yii2-GridView在開發(fā)中常用的功能及技巧總結(jié),數(shù)據(jù)網(wǎng)格或者說 GridView 小部件是Yii中最強大的部件之一。有興趣的可以了解一下。2017-01-01Yii凈化器CHtmlPurifier用法示例(過濾不良代碼)
這篇文章主要介紹了Yii凈化器CHtmlPurifier用法,可實現(xiàn)過濾不良代碼的功能,涉及在控制器、模型、過濾器及視圖中的相關(guān)使用技巧,需要的朋友可以參考下2016-07-07