上一节课咱们学习了从数据库拿数据,下面咱们来学习下经过建立一条记录到数据库,经常使用的操做就是表单的提交了.css
咱们先看下: http://localhost:8000/posts/1html
好的,下面咱们再这个页面添加一个评论框,为了让页面稍微好看点,咱们把bootstrap引入吧~laravel
咱们打开layout.blade.php
, 修改里面的代码以下:数据库
<!DOCTYPE html> <html> <head> <meta charset="utf-8"> <title>从零开始学Laravel - 周继平</title> <link rel="stylesheet" href="http://cdn.bootcss.com/bootstrap/3.3.0/css/bootstrap.min.css"> {{-- <link rel="stylesheet" href="{{ elixir('css/app.css') }}"> 这行注释的你们删除掉--}} @yield('header') </head> <body> @yield('content') @yield('footer') </body> </html>
上面注释的这行你们本身删除掉,正式项目的代码不要有没有用的注释的代码,不干净。bootstrap
打开resources/pages/show.blade.php
, 修改下其中的代码,让页面变的漂亮点:数组
@extends('layout') @section('content') <div class="row"> <div class="col-md-6 col-md-offset-3"> <h1>{{ $post->title }}</h1> <ul class="list-group"> @foreach ($post->comments as $comment) <li class="list-group-item">{{ $comment->content }}</li> @endforeach </ul> </div> </div> @stop
从零开始学Laravel目录bash
好,在看下咱们的页面:http://localhost:8000/posts/1restful
下面给该页面添加一个能够发表评论的评论框,代码以下:app
@extends('layout') @section('content') <div class="row"> <div class="col-md-6 col-md-offset-3"> <h1>{{ $post->title }}</h1> <ul class="list-group"> @foreach ($post->comments as $comment) <li class="list-group-item">{{ $comment->content }}</li> @endforeach </ul> <h3>Add a New Comment</h3> <form method="POST" action="/posts/{{$post->id}}/comments"> {{ csrf_field() }} <div class="form-group"> {{-- 这里的name的取名一般和数据库表的字段名同样 --}} <textarea name="content" class="form-control"></textarea> </div> <div class="form-group"> <button type="submit" class="btn btn-primary">Add Comment</button> </div> </form> </div> </div> @stop
提交表单咱们能够用POST
方法,action路由为/posts/1/comments
,固然也能够直接写成/comments
,这个看具体状况本身定义,路径不要太深就行。上面还有这样一句代码{{ csrf_field() }}
, Larvel为了防止CSRF攻击(跨站请求伪造攻击),Laravel 会自动生成一个 CSRF token 给每一个用户的 Session。该 token 用来验证用户是否为实际发出请求的用户。可使用 csrf_field()
辅助函数来生成一个包含 CSRF token 的 _token 隐藏表单字段,像下面这样的:
<input type="hidden" name="_token" value="8akoksagpytDpWY5ivK53B6uhD2GNJ5XCSj2B6iZ">
咱们回到routes.php
,建立这条方法为post
的路由:
Route::post('posts/{post}/comments', 'PostsController@storeComment');
对于咱们如今的列子,像上面这样的写法是能够的,不过若是正式项目中这么写,每每会让控制器变的很大,咱们能够这么写:
Route::post('posts/{post}/comments', 'PostCommentsController@store');
咱们也能够这么写:
Route::post('posts/{post}/comments', 'CommentsController@store');
只要不要把控制器写的过大就行,咱们就用上面这条路由吧,下面去创建控制器层
php artisan make:controller CommentsController
编写下store
函数
<?php namespace App\Http\Controllers; use Illuminate\Http\Request; use App\Http\Requests; use App\Comment; use App\Post; class CommentsController extends Controller { public function store(Request $request, Post $post) { $comment = New Comment; $comment->content = $request->content; $post->comments()->save($comment); return redirect('posts/' . $post->id); } }
咱们先看Request
类,表单提交的时候会把传递来的值都放到$request
对象中,咱们以前设置了<texrarea>
的name=content,这里咱们只要从$request对象中拿到这个content便可,若是要拿到全部的值,能够用$request->all()
,你们也能够dd($request)
,看看这个对象中还有别的什么东东.
最后一句话的redirect()
是laravel的一个帮助函数,这里咱们让其跳转到类式posts/1
这样的帖子页,如今咱们的页面以下,已经能够建立帖子的评论了:
咱们再来试试别的方式来写这个store
方法:
public function store(Request $request, Post $post) { $comment = New Comment(['content' => $request->content]); $post->comments()->save($comment); // 跳转到前一个路由 return back(); }
若是使用New Comment(['content' => $request->content]);
, 必定要保证AppComment
类中的$fillable
属性中已经设置了容许的字段.
固然,在简单点,能够这样:(若是要传入的参数不少,那这样的写法就不是很合适了。)
public function store(Request $request, Post $post) { $post->comments()->save( new Comment(['content' => $request->content]) ); return back(); }
再简单点:
public function store(Request $request, Post $post) { $post->comments()->create( ['content' => $request->content] ); return back(); }
再再简单点:
public function store(Request $request, Post $post) { $post->comments()->create($request->all()); return back(); }
使用$request->all()
的风险很大,由于它会将对应模型设置的$fillable容许的全部字段的值都插入到数据库,若是你必定要这么作,那么必定要想好模型层的$fillable
数组内你容许哪些值能够批量插入。
上面的这些方法均可以用,你们看本身的喜欢,不过像上面这样的$post->comments()->create()
链式调用的方法对于代码的可读性来讲很差,尤为是控制器中的代码,都是用来处理业务逻辑的,因此最好是能够读通的英文句子,如,给贴子加评论:Add comment to the post
,那么代码咱们这么写:
$post->addComment($comment);
先去App/Post.php
编写addComment($comment)
函数,以下:
public function addComment(Comment $comment) { return $this->comments()->save($comment); }
控制器的代码:
public function store(Request $request, Post $post) { $post->addComment( new Comment($request->all()) ); return back(); }
嗯,这样易读性好多了,关于写控制器个人建议是:
控制器的类命名一般都是复数的
每一个控制器内方法不能过多,方法对应restful路由的方法
控制器每一个方法中的代码一般都是调用方法(重模型,轻控制器,模型若是太乱,模型在分层,或者使用trait),只考虑业务逻辑,不要将实现方法的流程写在这里,若是你的控制器让不懂代码的人能看懂业务逻辑,那说明写的还不错了。