Web Programming with Yii2 – Creating Costume Templates for Gii
In this article, I will create a series on programming with the Yii framework. This series will be structured to create a simple program using common features that can be used in many cases. This article will use the basic Yii2 model (which in my opinion is a bit more difficult to use than the advanced Yii2 model).
In general (and perhaps it will increase depending on possible needs), the plan for this series of articles will be as follows.
- Main installation
- Installing YiiFramework2.0
- Log in
- Connect with database on Yii2
- Working with Gii
- Modules in Yii2
- Basic Layout and Manipulation in Yii2
- Custom Assets and Asset Groups in Yii2
- Alias Yii2
- Working with forms
- Timestamps, Blameable Behavior and Sluggable on Yii2
- Using Rich Text Input and CkEditor Alternatives in Yii2
- Grid view and list view
- SEO friendly URL with slug
- Scenarios in the Yii2 model
- Recover Data on Yii2
- Database Relations in Yii2
- Use Bootstrap4
- Kartik Widgets and Kartik Gridview
- Select2 with Kartik extension
- Dependent dropdown on Yii2
- Working with modals
- Create costume designs for Gii
- Uploading files
- Upload files with Kartik widget
- Gridview or Datatables?
- Session and cookies in Yii2
- Use AdminLTE on Yii2
- Create themes on Yii2
- Use the GoogleMaps API on Yii2
- Using Socket.io on Yii2
- Management error
- Working with AuthClient
- Use Amazon S3 on Yii2
- Sending emails with Swiftmailer on Yii2
- Middleware on Yii2
- Getting to know RBAC on Yii2
- Implementing RBAC on Yii2
- Real-time notifications with Socket.io on Yii2
You can download and track the progress of this series via the Belajararief-yii2series Github repository. Please check to be able to follow the development of the repository more easily.
Contents
Requirement
To make these instructions easier to understand, you should pay attention to several things:
- The operating system I’m using is Windows 10 64-bit with a terminal using the Powershell terminal (some commands such as cd may be different from the usual command prompt), but I will try as much as possible to illustrate the commands on other operating systems if possible.
- The stack used is Wamp Server (Apache 2.2, MySQL, PHP 7).
- The Yii used is Yii 2.0 with a note that the jquery used is not jquery3. Some jquery commands will be different in jquery3, and some extensions I use don’t seem to support jquery3
Gii
Yii2 is equipped with a GUI-based code generator which, in my opinion, is what sets Yii2 apart from other frameworks. With Gii, the prototyping process can be carried out faster and manage changes when making a demo becomes easier. But is Gii just for prototyping? Of course not. You can easily extend Gii to create your own templates if your application has the same display template across many menus. How ? Let’s try.
In this article we will create Gii so that it can provide AJAX Crud as we created it previously. We will create a menu for managing blog categories via modified Gii.
Duplicating Yii2 Default Generator CRUD Model
In this example, we will modify Yii2’s built-in CRUD generator to have a nicer Ajax CRUD function. For this reason we will copy all the default templates to the directory vendor/yiisoft/yii2-gii/src/generators/crud/default Then we copy it to the folder templates/modalcrud. We will name this new model Gii modalcrud.
Next, we will modify the necessary files to be able to run Ajax CRUD. The first document we will modify is templates/modalcrud/controller.php until it becomes like below.
<?php
/**
* This is the template for generating a CRUD controller class file.
*/
use yii\db\ActiveRecordInterface;
use yii\helpers\StringHelper;
/* @var $this yii\web\View */
/* @var $generator yii\gii\generators\crud\Generator */
$controllerClass = StringHelper::basename($generator->controllerClass);
$modelClass = StringHelper::basename($generator->modelClass);
$searchModelClass = StringHelper::basename($generator->searchModelClass);
if ($modelClass === $searchModelClass) {
$searchModelAlias = $searchModelClass . 'Search';
}
/* @var $class ActiveRecordInterface */
$class = $generator->modelClass;
$pks = $class::primaryKey();
$urlParams = $generator->generateUrlParams();
$actionParams = $generator->generateActionParams();
$actionParamComments = $generator->generateActionParamComments();
echo "<?php\n";
?>
namespace <?= StringHelper::dirname(ltrim($generator->controllerClass, '\\')) ?>;
use Yii;
use <?= ltrim($generator->modelClass, '\\') ?>;
<?php if (!empty($generator->searchModelClass)): ?>
use <?= ltrim($generator->searchModelClass, '\\') . (isset($searchModelAlias) ? " as $searchModelAlias" : "") ?>;
<?php else: ?>
use yii\data\ActiveDataProvider;
<?php endif; ?>
use <?= ltrim($generator->baseControllerClass, '\\') ?>;
use yii\web\NotFoundHttpException;
use yii\filters\VerbFilter;
/**
* <?= $controllerClass ?> implements the CRUD actions for <?= $modelClass ?> model.
*/
class <?= $controllerClass ?> extends <?= StringHelper::basename($generator->baseControllerClass) . "\n" ?>
{
/**
* @inheritdoc
*/
public function behaviors()
{
return [
'verbs' => [
'class' => VerbFilter::className(),
'actions' => [
'delete' => ['POST'],
],
],
];
}
/**
* Lists all <?= $modelClass ?> models.
* @return mixed
*/
public function actionIndex()
{
<?php if (!empty($generator->searchModelClass)): ?>
$searchModel = new <?= isset($searchModelAlias) ? $searchModelAlias : $searchModelClass ?>();
$dataProvider = $searchModel->search(Yii::$app->request->queryParams);
return $this->render('index', [
'searchModel' => $searchModel,
'dataProvider' => $dataProvider,
]);
<?php else: ?>
$dataProvider = new ActiveDataProvider([
'query' => <?= $modelClass ?>::find(),
]);
return $this->render('index', [
'dataProvider' => $dataProvider,
]);
<?php endif; ?>
}
/**
* Displays a single <?= $modelClass ?> model.
* <?= implode("\n * ", $actionParamComments) . "\n" ?>
* @return mixed
*/
public function actionView(<?= $actionParams ?>)
{
return $this->renderAjax('view', [
'model' => $this->findModel(<?= $actionParams ?>),
]);
}
/**
* Creates a new <?= $modelClass ?> model.
* If creation is successful, the browser will be redirected to the 'view' page.
* @return mixed
*/
public function actionCreate()
{
$model = new <?= $modelClass ?>();
if ($model->load(Yii::$app->request->post())) {
if ($model->save()){
return 1;
} else {
return 0;
}
} else {
return $this->renderAjax('_form', [
'model' => $model,
]);
}
}
/**
* Updates an existing <?= $modelClass ?> model.
* If update is successful, the browser will be redirected to the 'view' page.
* <?= implode("\n * ", $actionParamComments) . "\n" ?>
* @return mixed
*/
public function actionUpdate(<?= $actionParams ?>)
{
$model = $this->findModel(<?= $actionParams ?>);
if ($model->load(Yii::$app->request->post())) {
if ($model->save()){
return 1;
} else {
return 0;
}
} else {
return $this->renderAjax('_form', [
'model' => $model,
]);
}
}
/**
* Deletes an existing <?= $modelClass ?> model.
* If deletion is successful, the browser will be redirected to the 'index' page.
* <?= implode("\n * ", $actionParamComments) . "\n" ?>
* @return mixed
*/
public function actionDelete(<?= $actionParams ?>)
{
$this->findModel(<?= $actionParams ?>)->delete();
return $this->redirect(Yii::$app->request->referrer);
}
/**
* Finds the <?= $modelClass ?> model based on its primary key value.
* If the model is not found, a 404 HTTP exception will be thrown.
* <?= implode("\n * ", $actionParamComments) . "\n" ?>
* @return <?= $modelClass ?> the loaded model
* @throws NotFoundHttpException if the model cannot be found
*/
protected function findModel(<?= $actionParams ?>)
{
<?php
if (count($pks) === 1) {
$condition = '$id';
} else {
$condition = [];
foreach ($pks as $pk) {
$condition[] = "'$pk' => \$$pk";
}
$condition = '[' . implode(', ', $condition) . ']';
}
?>
if (($model = <?= $modelClass ?>::findOne(<?= $condition ?>)) !== null) {
return $model;
} else {
throw new NotFoundHttpException('The requested page does not exist.');
}
}
}
Next, we will modify the index file on templates/modalcrud/views/index.php until it becomes like below.
<?php
use yii\helpers\Inflector;
use yii\helpers\StringHelper;
/* @var $this yii\web\View */
/* @var $generator yii\gii\generators\crud\Generator */
$urlParams = $generator->generateUrlParams();
$nameAttribute = $generator->getNameAttribute();
echo "<?php\n";
?>
use yii\helpers\Html;
use <?= $generator->indexWidgetType === 'grid' ? "kartik\\grid\\GridView" : "yii\\widgets\\ListView" ?>;
<?= $generator->enablePjax ? 'use yii\widgets\Pjax;' : '' ?>
use yii\bootstrap4\Modal;
/* @var $this yii\web\View */
<?= !empty($generator->searchModelClass) ? "/* @var \$searchModel " . ltrim($generator->searchModelClass, '\\') . " */\n" : '' ?>
/* @var $dataProvider yii\data\ActiveDataProvider */
$this->title = <?= $generator->generateString(Inflector::pluralize(Inflector::camel2words(StringHelper::basename($generator->modelClass)))) ?>;
$this->params['breadcrumbs'][] = $this->title;
?>
<div class="<?= Inflector::camel2id(StringHelper::basename($generator->modelClass)) ?>-index">
<h1><?= "<?= " ?>Html::encode($this->title) ?></h1>
<?php if(!empty($generator->searchModelClass)): ?>
<?= " <?php " . ($generator->indexWidgetType === 'grid' ? "// " : "") ?>echo $this->render('_search', ['model' => $searchModel]); ?>
<?php endif; ?>
<p>
<?= "<?= " ?>Html::a(<?= $generator->generateString('Tambah ' . Inflector::camel2words(StringHelper::basename($generator->modelClass))) ?>, ['create'], [
'class' => 'btn btn-xs btn-success',
'data-toggle'=>"modal",
'data-target'=>"#myModal",
'data-title'=> <?= $generator->generateString("Tambah " . Inflector::camel2words(StringHelper::basename($generator->modelClass))) ?>,
]) ?>
</p>
<?php if ($generator->indexWidgetType === 'grid'): ?>
<?= "<?= " ?>GridView::widget([
'id' => '<?= Inflector::camel2id(StringHelper::basename($generator->modelClass)) ?>',
'dataProvider' => $dataProvider,
'export' => true,
'responsive'=>true,
'hover'=>true,
'resizableColumns'=>true,
'panel'=>['type'=>'primary', 'heading'=>$this->title],
'responsiveWrap' => false,
'toolbar' => [
[
'content' => '{export}{toggleData}' // $this->render('_search', ['model' => $searchModel]),
],
],
// set export properties
'export' => [
'fontAwesome' => true
],
'pager' => [
'firstPageLabel' => 'Awal',
'lastPageLabel' => 'Akhir'
],
'pjax'=>true,
'pjaxSettings'=>[
'options' => ['id' => '<?= Inflector::camel2id(StringHelper::basename($generator->modelClass)) ?>-pjax', 'timeout' => 5000],
],
<?= !empty($generator->searchModelClass) ? "'filterModel' => \$searchModel,\n 'columns' => [\n" : "'columns' => [\n"; ?>
['class' => 'yii\grid\SerialColumn'],
<?php
$count = 0;
if (($tableSchema = $generator->getTableSchema()) === false) {
foreach ($generator->getColumnNames() as $name) {
if (++$count < 6) {
echo " '" . $name . "',\n";
} else {
echo " // '" . $name . "',\n";
}
}
} else {
foreach ($tableSchema->columns as $column) {
$format = $generator->generateColumnFormat($column);
if (++$count < 6) {
echo " '" . $column->name . ($format === 'text' ? "" : ":" . $format) . "',\n";
} else {
echo " // '" . $column->name . ($format === 'text' ? "" : ":" . $format) . "',\n";
}
}
}
?>
[
'class' => 'kartik\grid\ActionColumn',
'template' => '{view} {update} {delete}',
'noWrap' => true,
'vAlign'=>'top',
'buttons' => [
'update' => function ($url, $model) {
return Html::a('<span class="fas fa-edit"></span>', $url,
[
'title' => Yii::t('yii', 'ubah'),
'data-toggle'=>"modal",
'data-target'=>"#myModal",
'data-title'=> "Ubah",
// 'data-confirm' => "Yakin menghapus ini?",
// 'data-method' => 'POST',
// 'data-pjax' => 1
]);
},
'view' => function ($url, $model) {
return Html::a('<span class="fas fa-eye"></span>', $url,
[
'title' => Yii::t('yii', 'lihat'),
'data-toggle'=>"modal",
'data-target'=>"#myModal",
'data-title'=> "Lihat",
]);
},
]
],
],
]); ?>
<?php else: ?>
<?= "<?= " ?>ListView::widget([
'dataProvider' => $dataProvider,
'itemOptions' => ['class' => 'item'],
'itemView' => function ($model, $key, $index, $widget) {
return Html::a(Html::encode($model-><?= $nameAttribute ?>), ['view', <?= $urlParams ?>]);
},
]) ?>
<?php endif; ?>
</div>
<?= "<?php " ?>
Modal::begin([
'id' => 'myModal',
'title' => 'Lihat lebih...',
'options' => [
'tabindex' => false // important for Select2 to work properly
],
]);
echo '...';
Modal::end();
$this->registerJs("
$('#myModal').on('show.bs.modal', function (event) {
var button = $(event.relatedTarget)
var modal = $(this)
var title = button.data('title')
var href = button.attr('href')
modal.find('.modal-title').html(title)
modal.find('.modal-body').html('<i class=\"fa fa-spinner fa-spin\"></i>')
$.post(href)
.done(function( data ) {
modal.find('.modal-body').html(data)
});
})
");
?>
Next, we will edit the form file on templates/modalcrud/views/_form.php until it becomes like below.
<?php
use yii\helpers\Inflector;
use yii\helpers\StringHelper;
/* @var $this yii\web\View */
/* @var $generator yii\gii\generators\crud\Generator */
/* @var $model \yii\db\ActiveRecord */
$model = new $generator->modelClass();
$safeAttributes = $model->safeAttributes();
if (empty($safeAttributes)) {
$safeAttributes = $model->attributes();
}
echo "<?php\n";
?>
use yii\helpers\Html;
use yii\widgets\ActiveForm;
/* @var $this yii\web\View */
/* @var $model <?= ltrim($generator->modelClass, '\\') ?> */
/* @var $form yii\widgets\ActiveForm */
?>
<div class="<?= Inflector::camel2id(StringHelper::basename($generator->modelClass)) ?>-form">
<?= "<?php " ?>$form = ActiveForm::begin(['id' => $model->formName()]); ?>
<?php foreach ($generator->getColumnNames() as $attribute) {
if (in_array($attribute, $safeAttributes)) {
echo " <?= " . $generator->generateActiveField($attribute) . " ?>\n\n";
}
} ?>
<div class="form-group">
<?= "<?= " ?>Html::submitButton($model->isNewRecord ? <?= $generator->generateString('Create') ?> : <?= $generator->generateString('Update') ?>, ['class' => $model->isNewRecord ? 'btn btn-success' : 'btn btn-primary']) ?>
</div>
<?= "<?php " ?>ActiveForm::end(); ?>
</div>
<?= "<?php " ?>
$script = <<<JS
$('form#{$model->formName()}').on('beforeSubmit',function(e)
{
var \$form = $(this);
$.post(
\$form.attr("action"), //serialize Yii2 form
\$form.serialize()
)
.done(function(result){
if(result == 1)
{
$("#myModal").modal('hide'); //hide modal after submit
//$(\$form).trigger("reset"); //reset form to reuse it to input
$.pjax.reload({container:'#<?= Inflector::camel2id(StringHelper::basename($generator->modelClass)) ?>-pjax'});
}else
{
$("#message").html(result);
}
}).fail(function(){
console.log("server error");
});
return false;
});
JS;
$this->registerJs($script);
?>
Modify the configuration by adding additional templates
Next, we will modify the Gii configuration to add modalcrud to our Gii CRUD template options. File Editing config/web.php so that the Gii section becomes like below.
if (YII_ENV_DEV) {
// configuration adjustments for 'dev' environment
$config['bootstrap'][] = 'debug';
$config['modules']['debug'] = [
'class' => 'yii\debug\Module',
// uncomment the following to add your IP if you are not connecting from localhost.
//'allowedIPs' => ['127.0.0.1', '::1'],
];
$config['bootstrap'][] = 'gii';
$config['modules']['gii'] = [
'class' => 'yii\gii\Module',
// uncomment the following to add your IP if you are not connecting from localhost.
//'allowedIPs' => ['127.0.0.1', '::1'],
'generators' => [
'crud' => [
'class' => 'yii\gii\generators\crud\Generator',
'templates' => ['modalcrud' => '@app/templates/modalcrud']
]
]
];
}
Once the template and parameters are created, it’s time to create an Ajax CRUD template for the Category table using the modified Gii. Go to Gii then select Crud Generator. Then fill out the information below.
In the example above, we are using the modalcrud code template that we created earlier. Press Preview after that press Generate.
After creating the CRUD, don’t forget to add a menu to access the category. File Editing views/layouts/main.php and modify parts Nav::widget() until it becomes like below.
echo Nav::widget([
'options' => ['class' => 'navbar-nav ml-auto'],
'items' => [
['label' => 'Home', 'url' => ['/site/index']],
['label' => 'About', 'url' => ['/site/about']],
['label' => 'Contact', 'url' => ['/site/contact']],
['label' => 'Blog', 'url' => ['/administrator/blog'], 'visible' => !Yii::$app->user->isGuest, 'items' => [
['label' => 'Manajemen', 'url' => ['/administrator/blog']],
['label' => 'Statistik', 'url' => ['/administrator/statistik']],
['label' => 'Kategori', 'url' => ['/administrator/category']],
]],
['label' => 'Register', 'url' => ['/site/signup'], 'visible' => Yii::$app->user->isGuest],
Yii::$app->user->isGuest ? (
['label' => 'Login', 'url' => ['/site/login']]
) : (
'<li>'
. Html::beginForm(['/site/logout'], 'post')
. Html::submitButton(
'Logout (' . Yii::$app->user->identity->username . ')',
['class' => 'btn btn-outline-success my-2 my-sm-0'] // logout btn-link
)
. Html::endForm()
. '</li>'
)
],
]);
That’s all for this article, I hope it’s useful and Happy Coding!
PakarPBN
A Private Blog Network (PBN) is a collection of websites that are controlled by a single individual or organization and used primarily to build backlinks to a “money site” in order to influence its ranking in search engines such as Google. The core idea behind a PBN is based on the importance of backlinks in Google’s ranking algorithm. Since Google views backlinks as signals of authority and trust, some website owners attempt to artificially create these signals through a controlled network of sites.
In a typical PBN setup, the owner acquires expired or aged domains that already have existing authority, backlinks, and history. These domains are rebuilt with new content and hosted separately, often using different IP addresses, hosting providers, themes, and ownership details to make them appear unrelated. Within the content published on these sites, links are strategically placed that point to the main website the owner wants to rank higher. By doing this, the owner attempts to pass link equity (also known as “link juice”) from the PBN sites to the target website.
The purpose of a PBN is to give the impression that the target website is naturally earning links from multiple independent sources. If done effectively, this can temporarily improve keyword rankings, increase organic visibility, and drive more traffic from search results.