Laravel 5 系列入门教程(二)【最适合中国人的 Laravel 教程】

本教程示例代码见:https://github.com/johnlui/Learn-Laravel-5  

大家在任何地方卡住,最快捷的解决方式就是去看我的示例代码。

我们将改变学习路线,不再像 Laravel 4 教程那样先构建登录系统。在本篇教程中,我们将一起构建 Pages 的管理功能,尝试 Laravel 的路由和 PHP 的命名空间

1. 路由

Laravel 中的路由,跟其他 PHP 框架一样,作用是把各种请求分流到各个控制器。

在 `learnlaravel5/app/Http/routes.php` 的末尾添加以下代码:

Route::group(['prefix' => 'admin', 'namespace' => 'Admin'], function()
{
  Route::get('/', 'AdminHomeController@index');
});

这表示创建了一个路由组。

1. `'prefix' => 'admin'` 表示这个路由组的 url 前缀是 /admin,也就是说中间那一行代码 `Route::get('/'` 对应的链接不是 http://fuck.io:88/ 而是 http://fuck.io:88/admin ,如果这段代码是 `Route::get('fuck'` 的话,那么 URL 就应该是 http://fuck.io:88/admin/fuck

2. `'namespace' => 'Admin'` 表示下面的 `AdminHomeController@index` 不是在 `\App\Http\Controllers\AdminHomeController@index` 而是在 `\App\Http\Controllers\Admin\AdminHomeController@index`,加上了一个命名空间的前缀。

如果你用过 Laravel 4,会发现 Laravel 5 的命名空间规划比较怪异,这其实是一个非常大的进步。Laravel 4 其实已经全面引入了命名空间这个强大的特性,但是为了“降低学习成本”,把 路由、控制器、模型 的默认命名空间全部设置成了顶级命名空间,这个举动反而让很多人比较轻易地“上手”了 Laravel,但是在用了一段时间以后,还需要翻越一堵高墙,那就是命名空间,而且有了前面的“容易上手”的印象作为铺垫,后期的学习会更加困难。Laravel 5 把命名空间全部隔开,控制器在 `\App\Http\Controllers`,模型在 `\App`,让我们在刚上手的时候就体验命名空间分离的感觉,总体上其实是会降低学习成本的。

2. 控制器

我们可以使用 Artisan 非常方便地构建控制器:

php artisan make:controller Admin/AdminHomeController

得到 `learnlaravel5/app/Http/Controllers/Admin/AdminHomeController.php` 文件。

在 `class AdminHomeController extends Controller {` 上面增加一行:

use App\Page;

修改 index() 的代码如下:

public function index()
{
  return view('AdminHome')->withPages(Page::all());
}

控制器中文文档:http://laravel-china.org/docs/5.0/controllers

控制器中涉及到了许多的命名空间知识,可以参考 PHP 命名空间 解惑

3. 视图

新建 `learnlaravel5/resources/views/AdminHome.blade.php`:

@extends('app')

@section('content')
<div class="container">
  <div class="row">
    <div class="col-md-10 col-md-offset-1">
      <div class="panel panel-default">
        <div class="panel-heading">后台首页</div>

        <div class="panel-body">

        <a href="{{ URL('admin/pages/create') }}" class="btn btn-lg btn-primary">新增</a>

          @foreach ($pages as $page)
            <hr>
            <div class="page">
              <h4>{{ $page->title }}</h4>
              <div class="content">
                <p>
                  {{ $page->body }}
                </p>
              </div>
            </div>
            <a href="{{ URL('admin/pages/'.$page->id.'/edit') }}" class="btn btn-success">编辑</a>

            <form action="{{ URL('admin/pages/'.$page->id) }}" method="POST" style="display: inline;">
              <input name="_method" type="hidden" value="DELETE">
              <input type="hidden" name="_token" value="{{ csrf_token() }}">
              <button type="submit" class="btn btn-danger">删除</button>
            </form>
          @endforeach

        </div>
      </div>
    </div>
  </div>
</div>
@endsection

视图的基本用法在此不再赘述,请阅读中文文档:http://laravel-china.org/docs/5.0/views

访问 http://fuck.io:88/admin 得到如下页面:

Image

至此,包含 路由 》 控制器 》 模型 》 视图 的整个流程都已经完成。

4. 完成 Pages 管理功能

接下来,我将记录下我实现 Pages 管理功能的过程,不再做过多的阐述。大家有问题可以直接在本文下面留言,我会及时回复。

4.1 修改路由 learnlaravel5/app/Http/routes.php

Route::group(['prefix' => 'admin', 'namespace' => 'Admin'], function()
{
  Route::get('/', 'AdminHomeController@index');
  Route::resource('pages', 'PagesController');
});

此处增加了一条“资源控制器”,中文文档地址:http://laravel-china.org/docs/5.0/controllers#restful-resource-controllers

4.2 创建 learnlaravel5/app/Http/Controllers/Admin/PagesController.php

运行:

php artisan make:controller Admin/PagesController

4.3 修改 learnlaravel5/app/Http/Controllers/Admin/PagesController.php 为:

<?php namespace App\Http\Controllers\Admin;

use App\Http\Requests;
use App\Http\Controllers\Controller;

use Illuminate\Http\Request;

use App\Page;

use Redirect, Input, Auth;

class PagesController extends Controller {

	/**
	 * Show the form for creating a new resource.
	 *
	 * @return Response
	 */
	public function create()
	{
		return view('admin.pages.create');
	}

	/**
	 * Store a newly created resource in storage.
	 *
	 * @return Response
	 */
	public function store(Request $request)
	{
		$this->validate($request, [
			'title' => 'required|unique:pages|max:255',
			'body' => 'required',
		]);

		$page = new Page;
		$page->title = Input::get('title');
		$page->body = Input::get('body');
		$page->user_id = 1;//Auth::user()->id;

		if ($page->save()) {
			return Redirect::to('admin');
		} else {
			return Redirect::back()->withInput()->withErrors('保存失败!');
		}

	}

	/**
	 * Show the form for editing the specified resource.
	 *
	 * @param  int  $id
	 * @return Response
	 */
	public function edit($id)
	{
		return view('admin.pages.edit')->withPage(Page::find($id));
	}

	/**
	 * Update the specified resource in storage.
	 *
	 * @param  int  $id
	 * @return Response
	 */
	public function update(Request $request,$id)
	{
		$this->validate($request, [
			'title' => 'required|unique:pages,title,'.$id.'|max:255',
			'body' => 'required',
		]);

		$page = Page::find($id);
		$page->title = Input::get('title');
		$page->body = Input::get('body');
		$page->user_id = 1;//Auth::user()->id;

		if ($page->save()) {
			return Redirect::to('admin');
		} else {
			return Redirect::back()->withInput()->withErrors('保存失败!');
		}
	}

	/**
	 * Remove the specified resource from storage.
	 *
	 * @param  int  $id
	 * @return Response
	 */
	public function destroy($id)
	{
		$page = Page::find($id);
		$page->delete();

		return Redirect::to('admin');
	}

}

4.4 创建视图文件

首先在 learnlaravel5/resources/views 下创建 admin/pages 两级文件夹。

然后创建 learnlaravel5/resources/views/admin/pages/create.blade.php:

@extends('app')

@section('content')
<div class="container">
  <div class="row">
    <div class="col-md-10 col-md-offset-1">
      <div class="panel panel-default">
        <div class="panel-heading">新增 Page</div>

        <div class="panel-body">

          @if (count($errors) > 0)
            <div class="alert alert-danger">
              <strong>Whoops!</strong> There were some problems with your input.<br><br>
              <ul>
                @foreach ($errors->all() as $error)
                  <li>{{ $error }}</li>
                @endforeach
              </ul>
            </div>
          @endif

          <form action="{{ URL('admin/pages') }}" method="POST">
            <input type="hidden" name="_token" value="{{ csrf_token() }}">
            <input type="text" name="title" class="form-control" required="required">
            <br>
            <textarea name="body" rows="10" class="form-control" required="required"></textarea>
            <br>
            <button class="btn btn-lg btn-info">新增 Page</button>
          </form>

        </div>
      </div>
    </div>
  </div>
</div>
@endsection

之后创建 learnlaravel5/resources/views/admin/pages/edit.blade.php:

@extends('app')

@section('content')
<div class="container">
  <div class="row">
    <div class="col-md-10 col-md-offset-1">
      <div class="panel panel-default">
        <div class="panel-heading">编辑 Page</div>

        <div class="panel-body">

          @if (count($errors) > 0)
            <div class="alert alert-danger">
              <strong>Whoops!</strong> There were some problems with your input.<br><br>
              <ul>
                @foreach ($errors->all() as $error)
                  <li>{{ $error }}</li>
                @endforeach
              </ul>
            </div>
          @endif

          <form action="{{ URL('admin/pages/'.$page->id) }}" method="POST">
            <input name="_method" type="hidden" value="PUT">
            <input type="hidden" name="_token" value="{{ csrf_token() }}">
            <input type="text" name="title" class="form-control" required="required" value="{{ $page->title }}">
            <br>
            <textarea name="body" rows="10" class="form-control" required="required">{{ $page->body }}</textarea>
            <br>
            <button class="btn btn-lg btn-info">编辑 Page</button>
          </form>

        </div>
      </div>
    </div>
  </div>
</div>
@endsection

4.5 查看结果

后台首页 http://fuck.io:88/admin

Image

新增 Page http://fuck.io:88/admin/pages/create

Image

编辑 Page http://fuck.io:88/admin/pages/1/edit

Image

页面上的新增、编辑、删除的功能均已经完成,并且加入了表单验证,Pages 管理功能完成!



教程(二)代码快照:https://github.com/johnlui/Learn-Laravel-5/archive/tutorial_2.zip


下一步:Laravel 5 系列入门教程(三)【最适合中国人的 Laravel 教程】

WRITTEN BY

avatar
2015.3.8   /   热度:106691   /   分类: Laravel

评论:

leo
2016-05-26 10:35
博主你好,刚才的问题解决了,我找到的一个国外的论坛说routes.php文件的命名空间要这样写:use Illuminate\Support\Facades\Route;,而不是按照原先的:use Illuminate\Routing\Route;但我还是有点不太理解为什么要这样写呢?
刚才运行的语句是:php artisan make:controller Admin/AdminHomeController
刚才的报错是: Call to undefined method Illuminate\Routing\Route::get()
Bob
2016-08-09 17:21
@leo:看看这个吧
http://www.tuicool.com/articles/Bfi2Y3f
leo
2016-05-26 10:10
运行php artisan make:controller Admin/AdminHomeController的时候报错
:[Symfony\Component\Debug\Exception\FatalErrorException]
  Call to undefined method Illuminate\Routing\Route::get()
这是怎么回事呢?
yanzi
2016-05-13 18:22
<form action="{{ URL('admin/pages/'.$page->id) }}" method="POST" style="display: inline;">
                                <input name="_method" type="hidden" value="DELETE">
                                <input type="hidden" name="_token" value="{{ csrf_token() }}">
                                <button type="submit" class="btn btn-danger">删除</button>
</form> 这是如何删除的呀
轻歌
2016-04-17 10:19
如果我想用Model管理多个MySQL数据库,要怎么做?
梦遥奇缘
2016-03-31 00:15
博主,请问下关于访问admin出现404时,怎样解决nginx的伪静态问题?在官网上看到有句如何优雅的使用伪静态,但是配置后直接nginx不能重启了,求贴nginx配置。
强有力的实力丶
2016-03-30 17:47
为什么我按照楼主的教程做完以后,没有显示bootstarp的样式,点击新增、编辑、删除后跳转报错。下面是提示信息:
Sorry, the page you are looking for could not be found.
NotFoundHttpException in RouteCollection.php line 145:

求解这是什么情况~~~~~
mozhe
2016-03-27 21:57
博主你好,我想问下,为什么我用return Redirect::back()->withInput()->withErrors('保存失败!');页面上得不到$errors,也没有保留之前填写的数据呢?
JohnLui
2016-03-27 23:40
@mozhe:Laravel 5.2 已经改变了很多系统组件的工作方式,使用 redirect()->back() 试试吧
单独
2016-03-23 11:26
create.blade.php
中的(count($errors) > 0)  errors在哪定义的?提示变量未定义
东方财富
2016-01-22 19:39
楼主 这个框架怎么写form提交地址?
无限上网
2016-01-17 20:04
FatalErrorException in AdminHomeController.php
line 16: Class 'app\Page' not found

我把use app\Page;改成小写也不行,
我的文件夹名是小写的
Carrie
2016-01-28 13:30
@无限上网:同问,文件目录是小写app,程序中的namespace全是App...
类似use App\Http\Requests;在app下这个路径还是存在的,但是App\Page没有这个路径...
霜落夜白
2016-04-05 13:26
@无限上网:这个应该是model吧。。。有建page 模型么
5959
2015-12-02 16:34
报错:Class 'App\Page' not found
yanhws
2016-01-08 17:33
@5959:你命名空间的 app 可能是小写的 改成小写应该就好了
motoon
2015-12-01 14:26
老师你好,
@extends('app')
中app文件在哪里啊,为什么找不到呢?
JohnLui
2015-12-01 15:01
@motoon:app.blade.php
tju
2015-11-24 00:12
你是大神!初学者的福音啊!
李传奇
2015-11-18 16:36
@extends('app')

@section('content')
<div class="container">
  <div class="row">
    <div class="col-md-10 col-md-offset-1">
      <div class="panel panel-default">
        <div class="panel-heading">后台首页</div>

        <div class="panel-body">

        <a href="{{ URL('admin/pages/create') }}" class="btn btn-lg btn-primary">新增</a>

          @foreach ($pages as $page)
            <hr>
            <div class="page">
              <h4>{{ $page->title }}</h4>
              <div class="content">
                <p>
                  {{ $page->body }}
                </p>
              </div>
            </div>

请问@foreach ($pages as $page)中pages的定义在哪里?急切求解答
sha
2015-11-22 10:05
@李传奇:视图由AdminHomeController的index方法传送过去的
view('AdminHome')->withPages(Page::all())

page模型继承model模型
class Page extends Model

model模型中有一个all方法,查询数据库所有字段

withPages是指查询Pages表

相关文件:
page模型:app/Page.php
model模型: vendor/laravel/framework/src/Illuminate/Database/Eloquent/Model.php
AdminHomeController控制器:app/Http/Controllers/Admin/AdminHomeController.php
个人理解,如有误请更正。
ying
2016-08-04 10:26
@sha:withPages是指查询Pages表  我觉得这句话可能不对,如果把这句改成:withP ,视图文件里 @foreach ($p as $page) 也是可以运行的,难道这是指的p表?
123
2016-11-09 10:47
@ying:改了并不可以运行啊
laravel初学者
2015-11-17 17:09
楼主,form要加上上传文件的功能,加了这句
<form action="{{ URL('admin/pages') }}" method="POST" enctype="multipart/form-data">
   <input type="file" name="myfile" multiple="true">,在点击提交的时候报“There were some problems with your input.

The myfile format is invalid.”,向您请教下这个是哪里出了问题?
JohnLui
2015-11-17 17:14
@laravel初学者:你上传的文件类型有问题
nashJG
2015-11-13 21:47
老师,我新安装的laravel5,用POST方法提交表单总是出现 TokenMismatchException错误
我的页面表单代码:
<form action="p1" method="POST">
    <input type="hidden" name="_token" value="<?php echo csrf_token() ?>">
    <input type="submit" vlaue="Test">
</form>
路由中添加了:
Route::post('p1',function(){
    return "hello laravel by POST!";
});

我把csrf禁用之后可以正常显示,开始csrf总是token mismatchexception,请您帮忙解答一下
JohnLui
2015-11-13 23:46
@nashJG:对比一下你的和我的视图文件,你就会明白的
nashJG
2015-11-14 00:10
@JohnLui:对比了一下您的表单,您用的是
<input type="hidden" name="_token" value="{{ csrf_token() }}">
我试了这种方法,还是提示TokenMismatchException in VerifyCsrfToken.php line 46:
hans
2015-11-07 09:33
GET                     /admin                             索引     AdminHomeController::index
GET                     /admin/pages/create     创建     PagesController::create
POST             /admin/pages                     保存     PagesController::store
GET                     /admin/pages/{id}             显示            PagesController::show
GET                     /admin/pages/{id}/edit       编辑    PagesController::edit
PUT/PATCH     /admin/pages/{id}             更新            PagesController::update
DELETE             /admin/pages/{pages}     删除      PagesController::destroy
把这个对应关系加进去,可以看的更清楚
yanzi
2016-05-13 18:18
@hans:GET                     /admin                             索引     AdminHomeController::index
GET                     /admin/pages/create     创建     PagesController::create
POST             /admin/pages                     保存     PagesController::store
GET                     /admin/pages/{id}             显示            PagesController::show
GET                     /admin/pages/{id}/edit       编辑    PagesController::edit
PUT/PATCH     /admin/pages/{id}             更新            PagesController::update
DELETE             /admin/pages/{pages}     删除      PagesController::destroy
这个需要在什么地方配置吗
hans
2015-11-07 09:24
HTTP请求 -> 路由 -> 控制器 -> 模型 -> 视图  

2015-11-05 15:42
楼主大大 请教一下 我按照教程操作,页面显示都正常,就是执行新增和编辑的时候 PagesController报错:
Undefined variable: request
页面里也引入了use Illuminate\Http\Request;
$this->validate($request, []); 就是这个$request找不到 。是什么原因呢?
JohnLui
2015-11-06 00:40
@牛:因为 public function store(Request $request) 括号里的参数你可能没写
Yingdynasty
2015-11-03 11:43
问题已经解决了
’admin\xxxx\‘.$xxxx->id  
其中这个反斜杠是不能省略的

发表评论:

© 2011-2017 岁寒  |  Powered by Emlog