Создание сайта на Yii2 - Часть 2

Просмотров: 20375Комментарии: 9
Yii2 Framework
Создание сайта на Yii2 - Часть 2 В первой части статьи мы рассмотрели как выполнить установку Yii Framework 2.0 и выполнить первоначальную настройку приложения. Во второй части мы посмотрим на создание каркаса приложения, настройку аутентификации пользователей и начальную доработку созданного каркаса. Данный пример призван помочь в понимании того, как строится приложении в Yii2.

Создание моделей для всех таблиц

Переходим к созданию моделей для наших таблиц, для этого мы воспользуемся генератором кода GII. Открываем http://myblog.loc/gii/ и выбираем Model Generator. В поле Table_Name для использования сразу всех таблиц указываем звездочку с заданным в настройках префиксом таблиц.
blog_*
Неймспейс и базовый класс оставляем по умолчанию. Обязательно ставим галочку напротив Use Table Prefix, если у нас используется префикс для таблиц. При создании снимаем галочку с Migration, для этой таблицы модель нам не нужна. Также не нужно перезаписывать модель User.

Создание CRUD

После того, как модели для наших таблиц созданы переходим к CRUD генератору. Нам необходимо сгенерировать CRUD для трех моделей Post, Comment и Category. В "Model Class" вписываем имя модели (с указанием namespace) для которой будем генерировать CRUD. В "Controller Class" вписываем имя генерируемого контроллера (с указанием namespace).
"Model Class": app\models\Comment
"Controller Class": app\controllers\CommentController
Проверить работу сгенерированного кода можно используя следующий url:
http://myblog.loc/comment
Аналогично вписываем для моделей Post и Category.

Настройка аутентификации

Для того, чтобы в нашем приложении работала аутентификацию, нам необходимо изменить нашу модель User. Изменяем наследование на AciveRecord, так как мы работаем с базой.
class User extends \yii\db\ActiveRecord implements \yii\web\IdentityInterface
Также удаляем все объявления переменных в начале класса, они нам не понадобятся. Добавляем use
use yii\base\NotSupportedException;
use Yii;
Теперь остается переделать существующие функции на работу с БД.
public static function findIdentity($id)
{
    return static::findOne(['id' => $id]);
}
public static function findIdentityByAccessToken($token, $type = null)
{
    throw new NotSupportedException('"findIdentityByAccessToken" is not implemented.');
}
public static function findByUsername($username)
{
    return static::findOne(['username' => $username]);
}
public function getId()
{
    return $this->getPrimaryKey();
}
public function getAuthKey()
{
    return $this->authKey;
}    
public function validateAuthKey($authKey)
{
    return $this->getAuthKey() === $authKey;
}
public function validatePassword($password)
{
    return Yii::$app->security->validatePassword($password, $this->password);
}

Изменяем меню сайта

Сделаем выпадающее меню для быстрого доступа к изменению постов, категорий и комментариев, если пользователь зашел под своим именем. Открываем файл ./views/layouts/main.php и меняем код:
['label' => 'Logout (' . Yii::$app->user->identity->username . ')',
                            'url' => ['/site/logout'],
                            'linkOptions' => ['data-method' => 'post']],
на следующий блок:
['label' => Yii::$app->user->identity->username, 'items' => [
   ['label' => 'Category', 'url' => ['/category']],
   ['label' => 'Posts', 'url' => ['/post']],
   ['label' => 'Comments', 'url' => ['/comment']],
   [
       'label' => 'Logout',
       'url' => ['/site/logout'],
       'linkOptions' => ['data-method' => 'post']]
   ]
]

Доработка форм модели

Созданные модели и формы с помощью Gii генератора кода, необходимо доработать, для удобного использования. В форме создания статьи присутствует поле с именем категории, для удобства лучше вставить туда выпадающий список существующих категорий, а не вводить ее каждый раз вручную. Открываем файл ./views/post/_form.php Добавляем подключение класса:
use yii\helpers\ArrayHelper;
И заменяем сгенерированный код для элемента category_id
field($model, 'category_id')->dropDownList(
  ArrayHelper::map($category, 'id', 'name')
) ?>
Также сразу заменим код для поля status
field($model, 'status')->dropDownList([
    '0' => 'Черновик',
    '1' => 'Опубликован'
]) ?>
Удаляем следующие две строки:
field($model, 'created_at')->textInput() ?>
field($model, 'updated_at')->textInput() ?>
Но для того, чтобы данный код работал, нам придется передать список категорий из контроллера в переменную $category. Открываем файл ./controller/PostController.php Вверху файла добавляем подключение классов:
use app\models\Category;
use yii\helpers\ArrayHelper;
И меняем функцию actionCreate():
public function actionCreate()
{
    $model = new Post();

    if ($model->load(Yii::$app->request->post()) && $model->save()) {
        return $this->redirect(['view', 'id' => $model->id]);
    } else {
        return $this->render('create', [
            'model' => $model,
            'category' => Category::find()->all()
        ]);
    }
}
Аналогично добавляем категории для функции actionUpdate(). Также в файлы ./views/post/create.php и ./views/post/update.php нужно добавить переменную категорий в передачу рендера формы
render('_form', [
    'model' => $model,
    'category' => $category,
]) ?>
Так как мы удалили поля Created At и Updated At из формы, нам необходимо позаботиться о их заполнении в моделе ./models/Post.php, дописав функцию:
use yii\behaviors\TimestampBehavior;

public function behaviors()
{
    return [
        TimestampBehavior::className(),
    ];
}
Также сюда добавляем связь с таблицей категорий, для вывода названия категории в представлениях.
public function getCategory()
{
   return $this->hasOne(Category::className(), ['id' => 'category_id']);
}
Также нужно исправить отображение ./views/post/view.php, а именно виджет DetailView
 $model,
    'attributes' => [
        'id',
        'title',
        'content:ntext',
        'category.name',
        [
            'attribute' => 'status',
            'value' => ($model->status==1)?'Опубликован':'Черновик',
        ],
        'created_at:datetime',
        'updated_at:datetime'
    ],
]) ?>
В списке ./views/post/index.php используется виджет GridView, тут мы применяем конструкцию вида:
 $dataProvider,
    'columns' => [
        ['class' => 'yii\grid\SerialColumn'],
        'id',
        'title',
        'content:ntext',
        'category.name',
        [
            'attribute' => 'status',
            'value' => function ($data) { return ($data->status==1)?'Опубликован':'Черновик'; }
        ],
        ['class' => 'yii\grid\ActionColumn'],
    ],
]); ?>
В модели категории ./models/Category.php меняем метку для поля названия категории:
public function attributeLabels()
{
    return [
        'id' => 'ID',
        'name' => 'Category',
        'description' => 'Description',
    ];
}

Комментариев: 9 RSS

1 Евгений 19-10-2014 05:38

Форматировать можно гораздо компактней:

Вместо:

[
    'attribute' => 'created_at',
    'value' => Yii::$app->formatter->asDatetime($model->created_at),
],

Можно писать так:

[
    'created_at:datetime'
]

2 Евгений 19-10-2014 05:46

Да и не помешало бы наверно использовать поведение для установки даты создания и изменения.

Добавить в модели где используются даты создания/обновления:

/*
 * Указываем путь до поведения
 */
use yii\behaviors\TimestampBehavior;
/*
 * И добавляем метод
 */
public function behaviors()
{
    return [
        TimestampBehavior::className(),
    ];
}

4 Владимир 10-11-2014 04:21

Доброго дня

Не забываем подключить классы "use app\models\Category;" и "use yii\helpers\ArrayHelper;" в ./controller/PostController.php и в ./views/post/_form.php соответственно, т.к. в первом случае будет ругаться на Category::find()->all();, а во втором случае на ArrayHelper::map($category, 'id', 'title')

7 Вячеслав 24-06-2015 23:59

В Моделе User есть ошибка:

//return $this->authKey;

return $this->auth_key;

8 Gluck 09-08-2015 08:58

У вас created_at updated_at как integer, Это нормально?

И что то там с выставлением этих значений не так, по этому примеру не работает.

Я только изучаю этот фреймворк и php в целом, так что пока не докопался до истины, но точно знаю собака зарыта в public function behaviors(), то ли ему имена полей не нравятся(не совпадают с оф документацией), то ли формат. У меня этот пример заработал только тогда когда я в првилах указал нулевые значения этих полей, и в бд указал их автозаполненее по текущей дате, но это не решение(

9 Web Design Dubai 03-09-2015 15:29

Gluck, TimestampBehavior по умолчанию пишет результат функции time(), .т.е timestamp (int). Вы можете переопределить присваемые значения через свойства поведения.

Оставьте комментарий!

grin LOL cheese smile wink smirk rolleyes confused surprised big surprise tongue laugh tongue rolleye tongue wink raspberry blank stare long face ohh grrr gulp oh oh downer red face sick shut eye hmmm mad angry zipper kiss shock cool smile cool smirk cool grin cool hmm cool mad cool cheese vampire snake excaim question

Используйте нормальные имена. Ваш комментарий будет опубликован после проверки.

     

  

Если вы уже зарегистрированы как комментатор или хотите зарегистрироваться, укажите пароль и свой действующий email. При регистрации на указанный адрес придет письмо с кодом активации и ссылкой на ваш персональный аккаунт, где вы сможете изменить свои данные, включая адрес сайта, ник, описание, контакты и т.д., а также подписку на новые комментарии.

(обязательно)