跳至内容

03. 创建 Chirps

现在您已准备好开始构建您的新应用程序!让我们允许用户发布称为Chirps的简短消息。

模型、迁移和控制器

为了允许用户发布 Chirps,我们需要创建模型、迁移和控制器。让我们更深入地探讨每个概念。

  • 模型 为您提供了一个强大且愉快的界面,用于与数据库中的表进行交互。
  • 迁移 允许您轻松地创建和修改数据库中的表。它们确保您的应用程序在所有运行位置都具有相同的数据库结构。
  • 控制器 负责处理对您的应用程序发出的请求并返回响应。

几乎你构建的每个功能都将涉及所有这些部分协同工作,因此artisan make:model命令可以一次性为你创建所有这些部分。

让我们使用以下命令为我们的 Chirps 创建一个模型、迁移和控制器

php artisan make:model -mc Chirp

此命令将为你创建三个文件

  • app/Models/Chirp.php - Eloquent 模型。
  • database/migrations/<timestamp>_create_chirps_table.php - 将创建你的数据库表的数据库迁移。
  • app/Http/Controllers/ChirpController.php - 将接收传入请求并返回响应的 HTTP 控制器。

路由

我们还需要为我们的控制器创建 URL。我们可以通过添加“路由”来实现,这些路由在项目的routes目录中进行管理。

因为我们使用的是 Livewire,所以我们只需要定义一个Route::get路由来显示我们的 Chirp 创建表单和现有 Chirps 的列表。此外,我们将把此路由放在两个中间件后面

  • auth中间件确保只有登录用户才能访问该路由。
  • 如果你决定启用电子邮件验证,将使用verified中间件。
routes/web.php
<?php
 
+use App\Http\Controllers\ChirpController;
use Illuminate\Support\Facades\Route;
 
Route::view('/', 'welcome');
+ 
+Route::get('chirps', [ChirpController::class, 'index'])
+ ->middleware(['auth', 'verified'])
+ ->name('chirps');
 
Route::view('dashboard', 'dashboard')
->middleware(['auth', 'verified'])
->name('dashboard');
 
Route::view('profile', 'profile')
->middleware(['auth'])
->name('profile');
 
require __DIR__.'/auth.php';

让我们通过从新ChirpController类的index方法返回测试消息来测试我们的路由和控制器

app/Http/Controllers/ChirpController.php
<?php
 
namespace App\Http\Controllers;
 
-use Illuminate\Http\Request;
+use Illuminate\Http\Response;
 
class ChirpController extends Controller
{
+ /**
+ * Display a listing of the resource.
+ */
+ public function index(): Response
+ {
+ return response('Hello, World!');
+ }
}

如果你之前仍然登录,则在导航到https://127.0.0.1:8000/chirpshttps://127.0.0.1/chirps(如果你使用的是 Sail)时,你应该会看到你的消息。

Livewire

还没有印象深刻?让我们更新ChirpController类的index方法以渲染 Blade 视图

app/Http/Controllers/ChirpController.php
<?php
 
namespace App\Http\Controllers;
 
-use Illuminate\Http\Response;
+use Illuminate\View\View;
 
class ChirpController extends Controller
{
/**
* Display a listing of the resource.
*/
- public function index(): Response
+ public function index(): View
{
- return response('Hello, World!');
+ return view('chirps', [
+ //
+ ]);
}
}

接下来,我们可以创建 Blade 模板并包含一个 Livewire 组件,该组件将渲染用于创建新 Chirps 的表单。

resources/views/chirps.blade.php
<x-app-layout>
<div class="max-w-2xl mx-auto p-4 sm:p-6 lg:p-8">
<livewire:chirps.create />
</div>
</x-app-layout>

接下来,让我们创建 Livewire 组件来渲染表单。为此,您可以使用 make:volt Artisan 命令。

请注意,以下代码片段提供了两种创建组件的不同方法:一种使用 Class API,另一种使用 Functional API。您将在本教程中看到这两种 API,您可以选择您喜欢的 Livewire 开发风格。

php artisan make:volt chirps/create --class
php artisan make:volt chirps/create

此命令将在 resources/views/livewire/chirps/create.blade.php 中创建一个新的 Livewire 组件。

让我们更新 Livewire 组件以显示表单。

resources/views/livewire/chirps/create.blade.php
<?php
 
use Livewire\Volt\Component;
 
new class extends Component
{
+ public string $message = '';
}; ?>
 
<div>
- //
+ <form wire:submit="store">
+ <textarea
+ wire:model="message"
+ placeholder="{{ __('What\'s on your mind?') }}"
+ class="block w-full border-gray-300 focus:border-indigo-300 focus:ring focus:ring-indigo-200 focus:ring-opacity-50 rounded-md shadow-sm"
+ ></textarea>
+ 
+ <x-input-error :messages="$errors->get('message')" class="mt-2" />
+ <x-primary-button class="mt-4">{{ __('Chirp') }}</x-primary-button>
+ </form>
</div>
resources/views/livewire/chirps/create.blade.php
<?php
 
use function Livewire\Volt\{state};
 
+state(['message' => '']);
 
?>
 
<div>
- //
+ <form wire:submit="store">
+ <textarea
+ wire:model="message"
+ placeholder="{{ __('What\'s on your mind?') }}"
+ class="block w-full border-gray-300 focus:border-indigo-300 focus:ring focus:ring-indigo-200 focus:ring-opacity-50 rounded-md shadow-sm"
+ ></textarea>
+ 
+ <x-input-error :messages="$errors->get('message')" class="mt-2" />
+ <x-primary-button class="mt-4">{{ __('Chirp') }}</x-primary-button>
+ </form>
</div>

就是这样!刷新浏览器中的页面,您将在 Breeze 提供的默认布局中看到新渲染的表单!

Chirp form

如果您的屏幕截图看起来与上面的不同,您可能需要停止并启动 Vite 开发服务器,以便 Tailwind 检测到我们刚刚创建的新文件中的 CSS 类。

从现在开始,我们对 Blade 模板所做的任何更改都将在 Vite 开发服务器通过 npm run dev 运行时自动在浏览器中刷新。

导航菜单

让我们花点时间在 Breeze 提供的导航菜单中添加一个链接。

更新 Breeze 提供的 navigation.blade.php 组件,为桌面屏幕添加一个菜单项。

resources/views/livewire/layout/navigation.blade.php
<div class="hidden space-x-8 sm:-my-px sm:ml-10 sm:flex">
<x-nav-link :href="route('dashboard')" :active="request()->routeIs('dashboard')" wire:navigate>
{{ __('Dashboard') }}
</x-nav-link>
+ <x-nav-link :href="route('chirps')" :active="request()->routeIs('chirps')" wire:navigate>
+ {{ __('Chirps') }}
+ </x-nav-link>
</div>

以及移动屏幕。

resources/views/livewire/layout/navigation.blade.php
<div class="pt-2 pb-3 space-y-1">
<x-responsive-nav-link :href="route('dashboard')" :active="request()->routeIs('dashboard')" wire:navigate>
{{ __('Dashboard') }}
</x-responsive-nav-link>
+ <x-responsive-nav-link :href="route('chirps')" :active="request()->routeIs('chirps')" wire:navigate>
+ {{ __('Chirps') }}
+ </x-responsive-nav-link>
</div>

保存 Chirp

我们的表单已配置为在单击 Chirp 按钮时调用 store 操作。让我们在 chirp.create 组件中添加一个 store 操作来验证数据并创建一个新的 Chirp。

resources/views/livewire/chirps/create.blade.php
<?php
 
+use Livewire\Attributes\Validate;
use Livewire\Volt\Component;
 
new class extends Component
{
+ #[Validate('required|string|max:255')]
public string $message = '';
+ 
+ public function store(): void
+ {
+ $validated = $this->validate();
+ 
+ auth()->user()->chirps()->create($validated);
+ 
+ $this->message = '';
+ }
}; ?>
 
<div>
<form wire:submit="store">
<textarea
wire:model="message"
placeholder="{{ __('What\'s on your mind?') }}"
class="block w-full border-gray-300 focus:border-indigo-300 focus:ring focus:ring-indigo-200 focus:ring-opacity-50 rounded-md shadow-sm"
></textarea>
<x-input-error :messages="$errors->get('message')" class="mt-2" />
<x-primary-button class="mt-4">{{ __('Chirp') }}</x-primary-button>
</form>
</div>
resources/views/livewire/chirps/create.blade.php
<?php
 
-use function Livewire\Volt\{state};
+use function Livewire\Volt\{rules, state};
 
state(['message' => '']);
+ 
+rules(['message' => 'required|string|max:255']);
+ 
+$store = function () {
+ $validated = $this->validate();
+ 
+ auth()->user()->chirps()->create($validated);
+ 
+ $this->message = '';
+};
 
?>
 
<div>
<form wire:submit="store">
<textarea
wire:model="message"
placeholder="{{ __('What\'s on your mind?') }}"
class="block w-full border-gray-300 focus:border-indigo-300 focus:ring focus:ring-indigo-200 focus:ring-opacity-50 rounded-md shadow-sm"
></textarea>
 
<x-input-error :messages="$errors->get('message')" class="mt-2" />
<x-primary-button class="mt-4">{{ __('Chirp') }}</x-primary-button>
</form>
</div>

使用 Livewire 的 Validate 属性,我们利用 Laravel 强大的验证功能来确保用户提供的消息不超过我们将要创建的数据库列的 255 个字符限制。

然后,我们创建一个记录,该记录将属于登录用户,方法是利用 chirps 关系。我们很快就会定义这种关系。

最后,我们还清除 message 表单字段值。

创建关系

您可能在前面的步骤中注意到,我们在 auth()->user() 对象上调用了 chirps 方法。我们需要在 User 模型上创建此方法来定义一个 "一对多" 关系。

app/Models/User.php
<?php
 ...
namespace App\Models;
 
// use Illuminate\Contracts\Auth\MustVerifyEmail;
use Illuminate\Database\Eloquent\Factories\HasFactory;
+use Illuminate\Database\Eloquent\Relations\HasMany;
use Illuminate\Foundation\Auth\User as Authenticatable;
use Illuminate\Notifications\Notifiable;
use Laravel\Sanctum\HasApiTokens;
 
class User extends Authenticatable
{
 ...
use HasApiTokens, HasFactory, Notifiable;
 
/**
* The attributes that are mass assignable.
*
* @var array<int, string>
*/
protected $fillable = [
'name',
'email',
'password',
];
 
/**
* The attributes that should be hidden for serialization.
*
* @var array<int, string>
*/
protected $hidden = [
'password',
'remember_token',
];
 
/**
* Get the attributes that should be cast.
*
* @return array<string, string>
*/
protected function casts(): array
{
return [
'email_verified_at' => 'datetime',
'password' => 'hashed',
];
}
 
+ public function chirps(): HasMany
+ {
+ return $this->hasMany(Chirp::class);
+ }
}

批量赋值保护

将来自请求的所有数据传递给您的模型可能存在风险。想象一下,您有一个页面,用户可以在其中编辑他们的个人资料。如果您要将整个请求传递给模型,那么用户可以编辑他们喜欢的任何列,例如 is_admin 列。这被称为 批量赋值漏洞

Laravel 默认情况下会阻止批量赋值,从而防止您意外执行此操作。批量赋值非常方便,因为它可以防止您逐个分配每个属性。我们可以通过将安全属性标记为“fillable”来启用对它们的批量赋值。

让我们将 `$fillable` 属性添加到我们的 `Chirp` 模型中,以启用对 `message` 属性的批量赋值。

app/Models/Chirp.php
<?php
 ...
namespace App\Models;
 
use Illuminate\Database\Eloquent\Factories\HasFactory;
use Illuminate\Database\Eloquent\Model;
 
class Chirp extends Model
{
 ...
use HasFactory;
 
+ protected $fillable = [
+ 'message',
+ ];
}

您可以在 文档 中了解更多关于 Laravel 的批量赋值保护的信息。

更新迁移

在应用程序创建期间,Laravel 已经应用了包含在 `database/migrations` 目录中的默认迁移。您可以使用 `php artisan db:show` 和 `php artisan db:table` 命令检查当前数据库结构。

php artisan db:show
php artisan db:table users

因此,唯一缺少的是数据库中的额外列,用于存储 `Chirp` 与其 `User` 之间的关联关系以及消息本身。还记得我们之前创建的数据库迁移吗?现在是打开该文件添加一些额外列的时候了。

database/migrations/<timestamp>_create_chirps_table.php
<?php
 ...
use Illuminate\Database\Migrations\Migration;
use Illuminate\Database\Schema\Blueprint;
use Illuminate\Support\Facades\Schema;
 
return new class extends Migration
{
/**
* Run the migrations.
*/
public function up(): void
{
Schema::create('chirps', function (Blueprint $table) {
$table->id();
+ $table->foreignId('user_id')->constrained()->cascadeOnDelete();
+ $table->string('message');
$table->timestamps();
});
}
 ...
/**
* Reverse the migrations.
*/
public function down(): void
{
Schema::dropIfExists('chirps');
}
 
};

自从我们添加了这个迁移以来,我们还没有迁移数据库,所以现在就来做吧。

php artisan migrate

测试一下

我们现在可以使用刚刚创建的表单发送 Chirp 了!我们还无法看到结果,因为我们还没有在页面上显示现有的 Chirp。

Chirp form

如果您将消息字段留空,或输入超过 255 个字符,那么您将看到验证生效。

Artisan Tinker

现在是学习 Artisan Tinker 的好时机,它是一个 REPL (Read-eval-print loop),您可以在其中执行 Laravel 应用程序中的任意 PHP 代码。

在您的控制台中,启动一个新的 tinker 会话。

php artisan tinker

接下来,执行以下代码以显示数据库中的 Chirps

App\Models\Chirp::all();
=> Illuminate\Database\Eloquent\Collection {#4512
all: [
App\Models\Chirp {#4514
id: 1,
user_id: 1,
message: "I'm building Chirper with Laravel!",
created_at: "2022-08-24 13:37:00",
updated_at: "2022-08-24 13:37:00",
},
],
}

您可以使用 `exit` 命令退出 Tinker,或按 Ctrl + c

继续开始显示 Chirps...