Страница отдельной записи
Давайте добавим в наш блог еще одну страницу, на которой будет отображаться содержимое одной конкретной записи блога.
Нам нужно создать новый метод render, который будет получать одну
конкретную запись блога и передавать её в шаблон. Иметь это
представление в HomePresenter
не очень приятно, потому что речь идёт о
записи в блоге, а не о главной странице. Итак, давайте создадим новый
класс PostPresenter
и поместим его в app/UI/Post/
. Ему потребуется
соединение с базой данных, поэтому снова поместите туда код
внедрения зависимости.
PostPresenter
должен выглядеть следующим образом:
<?php
namespace App\UI\Post;
use Nette;
use Nette\Application\UI\Form;
final class PostPresenter extends Nette\Application\UI\Presenter
{
public function __construct(
private Nette\Database\Explorer $database,
) {
}
public function renderShow(int $id): void
{
$this->template->post = $this->database
->table('posts')
->get($id);
}
}
Мы должны установить правильное пространство имен App\UI\Post
для
нашего презентера. Это зависит от presenter mapping.
Метод renderShow
требует один аргумент — ID отображаемого поста.
Затем он загружает этот пост из базы данных и передает результат в
шаблон.
В шаблоне Home/default.latte
мы добавляем ссылку на действие
Post:show
:
...
<h2><a href="{link Post:show $post->id}">{$post->title}</a></h2>
...
Тег {link}
генерирует адрес URL, который указывает на действие
Post:show
. Этот тег также передает ID поста в качестве аргумента.
То же самое мы можем написать коротко, используя n:attribute:
...
<h2><a n:href="Post:show $post->id">{$post->title}</a></h2>
...
Атрибут n:href
аналогичен тегу {link}
.
Шаблон для действия Post:show
ещё не существует. Мы можем открыть
ссылку на этот пост. Tracy покажет ошибку о том, что
app/UI/Post/show.latte
не существует. Если вы видите любой другой отчёт об
ошибке, вероятно, вам нужно включить mod_rewrite в вашем веб-сервере.
Поэтому мы создадим Post/show.latte
с таким содержанием:
{block content}
<p><a n:href="Home:default">← вернуться к списку постов</a></p>
<div class="date">{$post->created_at|date:'j.m.Y'}</div>
<h1 n:block="title">{$post->title}</h1>
<div class="post">{$post->content}</div>
Рассмотрим некоторые моменты.
Первая строка начинает определение именованного блока под
названием content
, которые мы видели ранее. Он будет отображаться в
шаблоне макета.
Вторая строка содержит обратную ссылку на список постов блога, чтобы
пользователь мог плавно перемещаться по нашему блогу вперед и назад.
Мы снова используем атрибут n:href
, поэтому Nette позаботится о
генерации URL для нас. Ссылка указывает на действие default
презентера Home
(можно просто написать n:href="Home:"
, так как
действие default
может быть опущено).
Третья строка форматирует временную метку публикации с помощью
фильтра date
, как мы уже знаем.
Четвертая строка отображает заголовок записи блога в виде
заголовка <h1>
. Есть часть, с которой вы, возможно, не знакомы,
это n:block="title"
. Можете ли вы догадаться, что она делает? Если вы
внимательно читали предыдущие части, мы упоминали n: атрибуты
.
Вот ещё один пример. Это эквивалентно:
{block title}<h1>{$post->title}</h1>{/block}
Проще говоря, он переопределяет блок под названием title
.
Блок определен в шаблоне макета (/app/UI/@layout.latte:11
) и, как и в
случае с переопределением ООП, он переопределяется здесь. Поэтому
<title>
страницы будет содержать заголовок отображаемого
поста. Мы переопределили заголовок страницы, и всё, что нам было нужно,
это n:block="title"
. Здорово, да?
Пятая и последняя строка шаблона отображает полное содержание вашего поста.
Проверка идентификатора поста
Что произойдет, если кто-то изменит URL и вставит несуществующий
postId
? Мы должны предоставить пользователю красивую страницу
ошибки «страница не найдена». Давайте обновим метод render
в файле
PostPresenter.php
:
public function renderShow(int $id): void
{
$post = $this->database
->table('posts')
->get($id);
if (!$post) {
$this->error('Страница не найдена!');
}
$this->template->post = $post;
}
Если пост не может быть найден, вызов $this->error(...)
покажет
страницу 404 с красивым и понятным сообщением. Обратите внимание, что в
режиме разработки вы не увидите страницу ошибки. Вместо этого Tracy
покажет исключение с полной информацией, что довольно удобно для
разработки. Вы можете проверить оба режима, просто изменив значение,
передаваемое в setDebugMode
в Bootstrap.php
.
Подведём итог
У нас есть база данных с записями блога и веб-приложение с двумя представлениями: первое отображает сводку всех последних записей, а второе — одну конкретную запись.