Browse Source
Merge branch 'master' of https://git.dev.tencent.com/linyaostalker/Antbasic
wechat_public_accounts
Merge branch 'master' of https://git.dev.tencent.com/linyaostalker/Antbasic
wechat_public_accounts
root
5 years ago
20 changed files with 2704 additions and 5 deletions
-
3common/config/main.php
-
9environments/skeleton/kcadmin/config/main-local.php
-
371vendor/blobt/generators/crud/Generator.php
-
2vendor/blobt/generators/crud/default/views/create.php
-
2vendor/blobt/generators/crud/default/views/update.php
-
17vendor/blobt/generators/crud/form.php
-
555vendor/linyao/generators/crud/Generator.php
-
179vendor/linyao/generators/crud/default/controller.php
-
86vendor/linyao/generators/crud/default/search.php
-
43vendor/linyao/generators/crud/default/views/_form.php
-
45vendor/linyao/generators/crud/default/views/_search.php
-
27vendor/linyao/generators/crud/default/views/create.php
-
61vendor/linyao/generators/crud/default/views/index.php
-
38vendor/linyao/generators/crud/default/views/update.php
-
49vendor/linyao/generators/crud/default/views/view.php
-
17vendor/linyao/generators/crud/form.php
-
995vendor/linyao/generators/model/Generator.php
-
124vendor/linyao/generators/model/default/model.php
-
56vendor/linyao/generators/model/default/query.php
-
30vendor/linyao/generators/model/form.php
@ -0,0 +1,17 @@ |
|||||
|
<?php |
||||
|
/* @var $this yii\web\View */ |
||||
|
/* @var $form yii\widgets\ActiveForm */ |
||||
|
/* @var $generator yii\gii\generators\crud\Generator */ |
||||
|
|
||||
|
echo $form->field($generator, 'modelClass'); |
||||
|
echo $form->field($generator, 'searchModelClass'); |
||||
|
echo $form->field($generator, 'controllerClass'); |
||||
|
echo $form->field($generator, 'viewPath'); |
||||
|
echo $form->field($generator, 'baseControllerClass'); |
||||
|
echo $form->field($generator, 'indexWidgetType')->dropDownList([ |
||||
|
'grid' => 'GridView', |
||||
|
'list' => 'ListView', |
||||
|
]); |
||||
|
echo $form->field($generator, 'enableI18N')->checkbox(); |
||||
|
echo $form->field($generator, 'enablePjax')->checkbox(); |
||||
|
echo $form->field($generator, 'messageCategory'); |
@ -0,0 +1,555 @@ |
|||||
|
<?php |
||||
|
|
||||
|
/* |
||||
|
* The MIT License |
||||
|
* |
||||
|
* Copyright 2019 Blobt. |
||||
|
* |
||||
|
* Permission is hereby granted, free of charge, to any person obtaining a copy |
||||
|
* of this software and associated documentation files (the "Software"), to deal |
||||
|
* in the Software without restriction, including without limitation the rights |
||||
|
* to use, copy, modify, merge, publish, distribute, sublicense, and/or sell |
||||
|
* copies of the Software, and to permit persons to whom the Software is |
||||
|
* furnished to do so, subject to the following conditions: |
||||
|
* |
||||
|
* The above copyright notice and this permission notice shall be included in |
||||
|
* all copies or substantial portions of the Software. |
||||
|
* |
||||
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR |
||||
|
* IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, |
||||
|
* FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE |
||||
|
* AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER |
||||
|
* LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, |
||||
|
* OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN |
||||
|
* THE SOFTWARE. |
||||
|
*/ |
||||
|
|
||||
|
namespace blobt\generators\crud; |
||||
|
|
||||
|
use Yii; |
||||
|
use yii\db\ActiveRecord; |
||||
|
use yii\db\BaseActiveRecord; |
||||
|
use yii\db\Schema; |
||||
|
use yii\gii\CodeFile; |
||||
|
use yii\helpers\Inflector; |
||||
|
use yii\helpers\VarDumper; |
||||
|
use yii\web\Controller; |
||||
|
|
||||
|
/** |
||||
|
* Description of Generator |
||||
|
* |
||||
|
* @author Blobt |
||||
|
* @email 380255922@qq.com |
||||
|
* @created Oct 6, 2019 |
||||
|
*/ |
||||
|
class Generator extends \yii\gii\Generator { |
||||
|
|
||||
|
public $modelClass; |
||||
|
public $controllerClass; |
||||
|
public $viewPath; |
||||
|
public $baseControllerClass = 'yii\web\Controller'; |
||||
|
public $indexWidgetType = 'grid'; |
||||
|
public $searchModelClass = ''; |
||||
|
|
||||
|
public $enablePjax = false; |
||||
|
public $strictInflector = true; |
||||
|
|
||||
|
/** |
||||
|
* {@inheritdoc} |
||||
|
*/ |
||||
|
public function getName() { |
||||
|
return 'CRUD Generator'; |
||||
|
} |
||||
|
|
||||
|
/** |
||||
|
* {@inheritdoc} |
||||
|
*/ |
||||
|
public function getDescription() { |
||||
|
return 'This generator generates a controller and views that implement CRUD (Create, Read, Update, Delete) |
||||
|
operations for the specified data model.'; |
||||
|
} |
||||
|
|
||||
|
/** |
||||
|
* {@inheritdoc} |
||||
|
*/ |
||||
|
public function rules() { |
||||
|
return array_merge(parent::rules(), [ |
||||
|
[['controllerClass', 'modelClass', 'searchModelClass', 'baseControllerClass'], 'filter', 'filter' => 'trim'], |
||||
|
[['modelClass', 'controllerClass', 'baseControllerClass', 'indexWidgetType'], 'required'], |
||||
|
[['searchModelClass'], 'compare', 'compareAttribute' => 'modelClass', 'operator' => '!==', 'message' => 'Search Model Class must not be equal to Model Class.'], |
||||
|
[['modelClass', 'controllerClass', 'baseControllerClass', 'searchModelClass'], 'match', 'pattern' => '/^[\w\\\\]*$/', 'message' => 'Only word characters and backslashes are allowed.'], |
||||
|
[['modelClass'], 'validateClass', 'params' => ['extends' => BaseActiveRecord::className()]], |
||||
|
[['baseControllerClass'], 'validateClass', 'params' => ['extends' => Controller::className()]], |
||||
|
[['controllerClass'], 'match', 'pattern' => '/Controller$/', 'message' => 'Controller class name must be suffixed with "Controller".'], |
||||
|
[['controllerClass'], 'match', 'pattern' => '/(^|\\\\)[A-Z][^\\\\]+Controller$/', 'message' => 'Controller class name must start with an uppercase letter.'], |
||||
|
[['controllerClass', 'searchModelClass'], 'validateNewClass'], |
||||
|
[['indexWidgetType'], 'in', 'range' => ['grid', 'list']], |
||||
|
[['modelClass'], 'validateModelClass'], |
||||
|
[['enableI18N', 'enablePjax'], 'boolean'], |
||||
|
[['messageCategory'], 'validateMessageCategory', 'skipOnEmpty' => false], |
||||
|
['viewPath', 'safe'], |
||||
|
]); |
||||
|
} |
||||
|
|
||||
|
/** |
||||
|
* {@inheritdoc} |
||||
|
*/ |
||||
|
public function attributeLabels() { |
||||
|
return array_merge(parent::attributeLabels(), [ |
||||
|
'modelClass' => 'Model Class', |
||||
|
'controllerClass' => 'Controller Class', |
||||
|
'viewPath' => 'View Path', |
||||
|
'baseControllerClass' => 'Base Controller Class', |
||||
|
'indexWidgetType' => 'Widget Used in Index Page', |
||||
|
'searchModelClass' => 'Search Model Class', |
||||
|
'enablePjax' => 'Enable Pjax', |
||||
|
]); |
||||
|
} |
||||
|
|
||||
|
/** |
||||
|
* {@inheritdoc} |
||||
|
*/ |
||||
|
public function hints() { |
||||
|
return array_merge(parent::hints(), [ |
||||
|
'modelClass' => 'This is the ActiveRecord class associated with the table that CRUD will be built upon. |
||||
|
You should provide a fully qualified class name, e.g., <code>app\models\Post</code>.', |
||||
|
'controllerClass' => 'This is the name of the controller class to be generated. You should |
||||
|
provide a fully qualified namespaced class (e.g. <code>app\controllers\PostController</code>), |
||||
|
and class name should be in CamelCase with an uppercase first letter. Make sure the class |
||||
|
is using the same namespace as specified by your application\'s controllerNamespace property.', |
||||
|
'viewPath' => 'Specify the directory for storing the view scripts for the controller. You may use path alias here, e.g., |
||||
|
<code>/var/www/basic/controllers/views/post</code>, <code>@app/views/post</code>. If not set, it will default |
||||
|
to <code>@app/views/ControllerID</code>', |
||||
|
'baseControllerClass' => 'This is the class that the new CRUD controller class will extend from. |
||||
|
You should provide a fully qualified class name, e.g., <code>yii\web\Controller</code>.', |
||||
|
'indexWidgetType' => 'This is the widget type to be used in the index page to display list of the models. |
||||
|
You may choose either <code>GridView</code> or <code>ListView</code>', |
||||
|
'searchModelClass' => 'This is the name of the search model class to be generated. You should provide a fully |
||||
|
qualified namespaced class name, e.g., <code>app\models\PostSearch</code>.', |
||||
|
'enablePjax' => 'This indicates whether the generator should wrap the <code>GridView</code> or <code>ListView</code> |
||||
|
widget on the index page with <code>yii\widgets\Pjax</code> widget. Set this to <code>true</code> if you want to get |
||||
|
sorting, filtering and pagination without page refreshing.', |
||||
|
]); |
||||
|
} |
||||
|
|
||||
|
/** |
||||
|
* {@inheritdoc} |
||||
|
*/ |
||||
|
public function requiredTemplates() { |
||||
|
return ['controller.php']; |
||||
|
} |
||||
|
|
||||
|
/** |
||||
|
* {@inheritdoc} |
||||
|
*/ |
||||
|
public function stickyAttributes() { |
||||
|
return array_merge(parent::stickyAttributes(), ['baseControllerClass', 'indexWidgetType']); |
||||
|
} |
||||
|
|
||||
|
/** |
||||
|
* Checks if model class is valid |
||||
|
*/ |
||||
|
public function validateModelClass() { |
||||
|
/* @var $class ActiveRecord */ |
||||
|
$class = $this->modelClass; |
||||
|
$pk = $class::primaryKey(); |
||||
|
if (empty($pk)) { |
||||
|
$this->addError('modelClass', "The table associated with $class must have primary key(s)."); |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
/** |
||||
|
* {@inheritdoc} |
||||
|
*/ |
||||
|
public function generate() { |
||||
|
$controllerFile = Yii::getAlias('@' . str_replace('\\', '/', ltrim($this->controllerClass, '\\')) . '.php'); |
||||
|
|
||||
|
$files = [ |
||||
|
new CodeFile($controllerFile, $this->render('controller.php')), |
||||
|
]; |
||||
|
|
||||
|
if (!empty($this->searchModelClass)) { |
||||
|
$searchModel = Yii::getAlias('@' . str_replace('\\', '/', ltrim($this->searchModelClass, '\\') . '.php')); |
||||
|
$files[] = new CodeFile($searchModel, $this->render('search.php')); |
||||
|
} |
||||
|
|
||||
|
$viewPath = $this->getViewPath(); |
||||
|
$templatePath = $this->getTemplatePath() . '/views'; |
||||
|
foreach (scandir($templatePath) as $file) { |
||||
|
if (empty($this->searchModelClass) && $file === '_search.php') { |
||||
|
continue; |
||||
|
} |
||||
|
if (is_file($templatePath . '/' . $file) && pathinfo($file, PATHINFO_EXTENSION) === 'php') { |
||||
|
$files[] = new CodeFile("$viewPath/$file", $this->render("views/$file")); |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
return $files; |
||||
|
} |
||||
|
|
||||
|
/** |
||||
|
* @return string the controller ID (without the module ID prefix) |
||||
|
*/ |
||||
|
public function getControllerID() { |
||||
|
$pos = strrpos($this->controllerClass, '\\'); |
||||
|
$class = substr(substr($this->controllerClass, $pos + 1), 0, -10); |
||||
|
|
||||
|
return Inflector::camel2id($class, '-', $this->strictInflector); |
||||
|
} |
||||
|
|
||||
|
/** |
||||
|
* @return string the controller view path |
||||
|
*/ |
||||
|
public function getViewPath() { |
||||
|
if (empty($this->viewPath)) { |
||||
|
return Yii::getAlias('@app/views/' . $this->getControllerID()); |
||||
|
} |
||||
|
|
||||
|
return Yii::getAlias(str_replace('\\', '/', $this->viewPath)); |
||||
|
} |
||||
|
|
||||
|
/** |
||||
|
* @return string |
||||
|
*/ |
||||
|
public function getNameAttribute() { |
||||
|
foreach ($this->getColumnNames() as $name) { |
||||
|
if (!strcasecmp($name, 'name') || !strcasecmp($name, 'title')) { |
||||
|
return $name; |
||||
|
} |
||||
|
} |
||||
|
/* @var $class \yii\db\ActiveRecord */ |
||||
|
$class = $this->modelClass; |
||||
|
$pk = $class::primaryKey(); |
||||
|
|
||||
|
return $pk[0]; |
||||
|
} |
||||
|
|
||||
|
/** |
||||
|
* Generates code for active field |
||||
|
* @param string $attribute |
||||
|
* @return string |
||||
|
*/ |
||||
|
public function generateActiveField($attribute) { |
||||
|
$tableSchema = $this->getTableSchema(); |
||||
|
if ($tableSchema === false || !isset($tableSchema->columns[$attribute])) { |
||||
|
if (preg_match('/^(password|pass|passwd|passcode)$/i', $attribute)) { |
||||
|
return "\$form->field(\$model, '$attribute')->passwordInput()"; |
||||
|
} |
||||
|
|
||||
|
return "\$form->field(\$model, '$attribute')"; |
||||
|
} |
||||
|
$column = $tableSchema->columns[$attribute]; |
||||
|
if ($column->phpType === 'boolean') { |
||||
|
return "\$form->field(\$model, '$attribute')->checkbox()"; |
||||
|
} |
||||
|
|
||||
|
if ($column->type === 'text') { |
||||
|
return "\$form->field(\$model, '$attribute')->textarea(['rows' => 6])"; |
||||
|
} |
||||
|
|
||||
|
if (preg_match('/^(password|pass|passwd|passcode)$/i', $column->name)) { |
||||
|
$input = 'passwordInput'; |
||||
|
} else { |
||||
|
$input = 'textInput'; |
||||
|
} |
||||
|
|
||||
|
if (is_array($column->enumValues) && count($column->enumValues) > 0) { |
||||
|
$dropDownOptions = []; |
||||
|
foreach ($column->enumValues as $enumValue) { |
||||
|
$dropDownOptions[$enumValue] = Inflector::humanize($enumValue); |
||||
|
} |
||||
|
return "\$form->field(\$model, '$attribute')->dropDownList(" |
||||
|
. preg_replace("/\n\s*/", ' ', VarDumper::export($dropDownOptions)) . ", ['prompt' => ''])"; |
||||
|
} |
||||
|
|
||||
|
if ($column->phpType !== 'string' || $column->size === null) { |
||||
|
return "\$form->field(\$model, '$attribute')->$input()"; |
||||
|
} |
||||
|
|
||||
|
return "\$form->field(\$model, '$attribute')->$input(['maxlength' => true])"; |
||||
|
} |
||||
|
|
||||
|
/** |
||||
|
* Generates column format |
||||
|
* @param \yii\db\ColumnSchema $column |
||||
|
* @return string |
||||
|
*/ |
||||
|
public function generateColumnFormat($column) { |
||||
|
if ($column->phpType === 'boolean') { |
||||
|
return 'boolean'; |
||||
|
} |
||||
|
|
||||
|
if ($column->type === 'text') { |
||||
|
return 'ntext'; |
||||
|
} |
||||
|
|
||||
|
if (stripos($column->name, 'time') !== false && $column->phpType === 'integer') { |
||||
|
return 'datetime'; |
||||
|
} |
||||
|
|
||||
|
if (stripos($column->name, 'email') !== false) { |
||||
|
return 'email'; |
||||
|
} |
||||
|
|
||||
|
if (preg_match('/(\b|[_-])url(\b|[_-])/i', $column->name)) { |
||||
|
return 'url'; |
||||
|
} |
||||
|
|
||||
|
return 'text'; |
||||
|
} |
||||
|
|
||||
|
/** |
||||
|
* Generates validation rules for the search model. |
||||
|
* @return array the generated validation rules |
||||
|
*/ |
||||
|
public function generateSearchRules() |
||||
|
{ |
||||
|
if (($table = $this->getTableSchema()) === false) { |
||||
|
return ["[['" . implode("', '", $this->getColumnNames()) . "'], 'safe']"]; |
||||
|
} |
||||
|
$types = []; |
||||
|
foreach ($table->columns as $column) { |
||||
|
switch ($column->type) { |
||||
|
case Schema::TYPE_TINYINT: |
||||
|
case Schema::TYPE_SMALLINT: |
||||
|
case Schema::TYPE_INTEGER: |
||||
|
case Schema::TYPE_BIGINT: |
||||
|
$types['integer'][] = $column->name; |
||||
|
break; |
||||
|
case Schema::TYPE_BOOLEAN: |
||||
|
$types['boolean'][] = $column->name; |
||||
|
break; |
||||
|
case Schema::TYPE_FLOAT: |
||||
|
case Schema::TYPE_DOUBLE: |
||||
|
case Schema::TYPE_DECIMAL: |
||||
|
case Schema::TYPE_MONEY: |
||||
|
$types['number'][] = $column->name; |
||||
|
break; |
||||
|
case Schema::TYPE_DATE: |
||||
|
case Schema::TYPE_TIME: |
||||
|
case Schema::TYPE_DATETIME: |
||||
|
case Schema::TYPE_TIMESTAMP: |
||||
|
default: |
||||
|
$types['safe'][] = $column->name; |
||||
|
break; |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
$rules = []; |
||||
|
foreach ($types as $type => $columns) { |
||||
|
$rules[] = "[['" . implode("', '", $columns) . "'], '$type']"; |
||||
|
} |
||||
|
|
||||
|
return $rules; |
||||
|
} |
||||
|
|
||||
|
/** |
||||
|
* @return array searchable attributes |
||||
|
*/ |
||||
|
public function getSearchAttributes() |
||||
|
{ |
||||
|
return $this->getColumnNames(); |
||||
|
} |
||||
|
|
||||
|
/** |
||||
|
* Generates the attribute labels for the search model. |
||||
|
* @return array the generated attribute labels (name => label) |
||||
|
*/ |
||||
|
public function generateSearchLabels() |
||||
|
{ |
||||
|
/* @var $model \yii\base\Model */ |
||||
|
$model = new $this->modelClass(); |
||||
|
$attributeLabels = $model->attributeLabels(); |
||||
|
$labels = []; |
||||
|
foreach ($this->getColumnNames() as $name) { |
||||
|
if (isset($attributeLabels[$name])) { |
||||
|
$labels[$name] = $attributeLabels[$name]; |
||||
|
} else { |
||||
|
if (!strcasecmp($name, 'id')) { |
||||
|
$labels[$name] = 'ID'; |
||||
|
} else { |
||||
|
$label = Inflector::camel2words($name); |
||||
|
if (!empty($label) && substr_compare($label, ' id', -3, 3, true) === 0) { |
||||
|
$label = substr($label, 0, -3) . ' ID'; |
||||
|
} |
||||
|
$labels[$name] = $label; |
||||
|
} |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
return $labels; |
||||
|
} |
||||
|
|
||||
|
/** |
||||
|
* Generates search conditions |
||||
|
* @return array |
||||
|
*/ |
||||
|
public function generateSearchConditions() |
||||
|
{ |
||||
|
$columns = []; |
||||
|
if (($table = $this->getTableSchema()) === false) { |
||||
|
$class = $this->modelClass; |
||||
|
/* @var $model \yii\base\Model */ |
||||
|
$model = new $class(); |
||||
|
foreach ($model->attributes() as $attribute) { |
||||
|
$columns[$attribute] = 'unknown'; |
||||
|
} |
||||
|
} else { |
||||
|
foreach ($table->columns as $column) { |
||||
|
$columns[$column->name] = $column->type; |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
$likeConditions = []; |
||||
|
$hashConditions = []; |
||||
|
foreach ($columns as $column => $type) { |
||||
|
switch ($type) { |
||||
|
case Schema::TYPE_TINYINT: |
||||
|
case Schema::TYPE_SMALLINT: |
||||
|
case Schema::TYPE_INTEGER: |
||||
|
case Schema::TYPE_BIGINT: |
||||
|
case Schema::TYPE_BOOLEAN: |
||||
|
case Schema::TYPE_FLOAT: |
||||
|
case Schema::TYPE_DOUBLE: |
||||
|
case Schema::TYPE_DECIMAL: |
||||
|
case Schema::TYPE_MONEY: |
||||
|
case Schema::TYPE_DATE: |
||||
|
case Schema::TYPE_TIME: |
||||
|
case Schema::TYPE_DATETIME: |
||||
|
case Schema::TYPE_TIMESTAMP: |
||||
|
$hashConditions[] = "'{$column}' => \$this->{$column},"; |
||||
|
break; |
||||
|
default: |
||||
|
$likeKeyword = $this->getClassDbDriverName() === 'pgsql' ? 'ilike' : 'like'; |
||||
|
$likeConditions[] = "->andFilterWhere(['{$likeKeyword}', '{$column}', \$this->{$column}])"; |
||||
|
break; |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
$conditions = []; |
||||
|
if (!empty($hashConditions)) { |
||||
|
$conditions[] = "\$query->andFilterWhere([\n" |
||||
|
. str_repeat(' ', 12) . implode("\n" . str_repeat(' ', 12), $hashConditions) |
||||
|
. "\n" . str_repeat(' ', 8) . "]);\n"; |
||||
|
} |
||||
|
if (!empty($likeConditions)) { |
||||
|
$conditions[] = "\$query" . implode("\n" . str_repeat(' ', 12), $likeConditions) . ";\n"; |
||||
|
} |
||||
|
|
||||
|
return $conditions; |
||||
|
} |
||||
|
|
||||
|
/** |
||||
|
* Generates URL parameters |
||||
|
* @return string |
||||
|
*/ |
||||
|
public function generateUrlParams() { |
||||
|
/* @var $class ActiveRecord */ |
||||
|
$class = $this->modelClass; |
||||
|
$pks = $class::primaryKey(); |
||||
|
if (count($pks) === 1) { |
||||
|
if (is_subclass_of($class, 'yii\mongodb\ActiveRecord')) { |
||||
|
return "'id' => (string)\$model->{$pks[0]}"; |
||||
|
} |
||||
|
|
||||
|
return "'id' => \$model->{$pks[0]}"; |
||||
|
} |
||||
|
|
||||
|
$params = []; |
||||
|
foreach ($pks as $pk) { |
||||
|
if (is_subclass_of($class, 'yii\mongodb\ActiveRecord')) { |
||||
|
$params[] = "'$pk' => (string)\$model->$pk"; |
||||
|
} else { |
||||
|
$params[] = "'$pk' => \$model->$pk"; |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
return implode(', ', $params); |
||||
|
} |
||||
|
|
||||
|
/** |
||||
|
* Generates action parameters |
||||
|
* @return string |
||||
|
*/ |
||||
|
public function generateActionParams() { |
||||
|
/* @var $class ActiveRecord */ |
||||
|
$class = $this->modelClass; |
||||
|
$pks = $class::primaryKey(); |
||||
|
if (count($pks) === 1) { |
||||
|
return '$id'; |
||||
|
} |
||||
|
|
||||
|
return '$' . implode(', $', $pks); |
||||
|
} |
||||
|
|
||||
|
/** |
||||
|
* Generates parameter tags for phpdoc |
||||
|
* @return array parameter tags for phpdoc |
||||
|
*/ |
||||
|
public function generateActionParamComments() { |
||||
|
/* @var $class ActiveRecord */ |
||||
|
$class = $this->modelClass; |
||||
|
$pks = $class::primaryKey(); |
||||
|
if (($table = $this->getTableSchema()) === false) { |
||||
|
$params = []; |
||||
|
foreach ($pks as $pk) { |
||||
|
$params[] = '@param ' . (strtolower(substr($pk, -2)) === 'id' ? 'integer' : 'string') . ' $' . $pk; |
||||
|
} |
||||
|
|
||||
|
return $params; |
||||
|
} |
||||
|
if (count($pks) === 1) { |
||||
|
return ['@param ' . $table->columns[$pks[0]]->phpType . ' $id']; |
||||
|
} |
||||
|
|
||||
|
$params = []; |
||||
|
foreach ($pks as $pk) { |
||||
|
$params[] = '@param ' . $table->columns[$pk]->phpType . ' $' . $pk; |
||||
|
} |
||||
|
|
||||
|
return $params; |
||||
|
} |
||||
|
|
||||
|
/** |
||||
|
* Returns table schema for current model class or false if it is not an active record |
||||
|
* @return bool|\yii\db\TableSchema |
||||
|
*/ |
||||
|
public function getTableSchema() { |
||||
|
/* @var $class ActiveRecord */ |
||||
|
$class = $this->modelClass; |
||||
|
if (is_subclass_of($class, 'yii\db\ActiveRecord')) { |
||||
|
return $class::getTableSchema(); |
||||
|
} |
||||
|
|
||||
|
return false; |
||||
|
} |
||||
|
|
||||
|
/** |
||||
|
* @return array model column names |
||||
|
*/ |
||||
|
public function getColumnNames() { |
||||
|
/* @var $class ActiveRecord */ |
||||
|
$class = $this->modelClass; |
||||
|
if (is_subclass_of($class, 'yii\db\ActiveRecord')) { |
||||
|
return $class::getTableSchema()->getColumnNames(); |
||||
|
} |
||||
|
|
||||
|
/* @var $model \yii\base\Model */ |
||||
|
$model = new $class(); |
||||
|
|
||||
|
return $model->attributes(); |
||||
|
} |
||||
|
|
||||
|
/** |
||||
|
* @return string|null driver name of modelClass db connection. |
||||
|
* In case db is not instance of \yii\db\Connection null will be returned. |
||||
|
* @since 2.0.6 |
||||
|
*/ |
||||
|
protected function getClassDbDriverName() { |
||||
|
/* @var $class ActiveRecord */ |
||||
|
$class = $this->modelClass; |
||||
|
$db = $class::getDb(); |
||||
|
return $db instanceof \yii\db\Connection ? $db->driverName : null; |
||||
|
} |
||||
|
|
||||
|
} |
@ -0,0 +1,179 @@ |
|||||
|
<?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 |
||||
|
* @throws NotFoundHttpException if the model cannot be found |
||||
|
*/ |
||||
|
public function actionView(<?= $actionParams ?>)
|
||||
|
{ |
||||
|
return $this->render('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()) && $model->save()) { |
||||
|
return $this->redirect(['view', <?= $urlParams ?>]);
|
||||
|
} |
||||
|
|
||||
|
return $this->render('create', [ |
||||
|
'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 |
||||
|
* @throws NotFoundHttpException if the model cannot be found |
||||
|
*/ |
||||
|
public function actionUpdate(<?= $actionParams ?>)
|
||||
|
{ |
||||
|
$model = $this->findModel(<?= $actionParams ?>);
|
||||
|
|
||||
|
if ($model->load(Yii::$app->request->post()) && $model->save()) { |
||||
|
return $this->redirect(['view', <?= $urlParams ?>]);
|
||||
|
} |
||||
|
|
||||
|
return $this->render('update', [ |
||||
|
'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 |
||||
|
* @throws NotFoundHttpException if the model cannot be found |
||||
|
*/ |
||||
|
public function actionDelete(<?= $actionParams ?>)
|
||||
|
{ |
||||
|
$this->findModel(<?= $actionParams ?>)->delete();
|
||||
|
|
||||
|
return $this->redirect(['index']); |
||||
|
} |
||||
|
|
||||
|
/** |
||||
|
* 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; |
||||
|
} |
||||
|
|
||||
|
throw new NotFoundHttpException(<?= $generator->generateString('The requested page does not exist.') ?>);
|
||||
|
} |
||||
|
} |
@ -0,0 +1,86 @@ |
|||||
|
<?php |
||||
|
/** |
||||
|
* This is the template for generating CRUD search class of the specified model. |
||||
|
*/ |
||||
|
|
||||
|
use yii\helpers\StringHelper; |
||||
|
|
||||
|
|
||||
|
/* @var $this yii\web\View */ |
||||
|
/* @var $generator yii\gii\generators\crud\Generator */ |
||||
|
|
||||
|
$modelClass = StringHelper::basename($generator->modelClass); |
||||
|
$searchModelClass = StringHelper::basename($generator->searchModelClass); |
||||
|
if ($modelClass === $searchModelClass) { |
||||
|
$modelAlias = $modelClass . 'Model'; |
||||
|
} |
||||
|
$rules = $generator->generateSearchRules(); |
||||
|
$labels = $generator->generateSearchLabels(); |
||||
|
$searchAttributes = $generator->getSearchAttributes(); |
||||
|
$searchConditions = $generator->generateSearchConditions(); |
||||
|
|
||||
|
echo "<?php\n"; |
||||
|
?>
|
||||
|
|
||||
|
namespace <?= StringHelper::dirname(ltrim($generator->searchModelClass, '\\')) ?>;
|
||||
|
|
||||
|
use yii\base\Model; |
||||
|
use yii\data\ActiveDataProvider; |
||||
|
use <?= ltrim($generator->modelClass, '\\') . (isset($modelAlias) ? " as $modelAlias" : "") ?>;
|
||||
|
|
||||
|
/** |
||||
|
* <?= $searchModelClass ?> represents the model behind the search form of `<?= $generator->modelClass ?>`.
|
||||
|
*/ |
||||
|
class <?= $searchModelClass ?> extends <?= isset($modelAlias) ? $modelAlias : $modelClass ?>
|
||||
|
|
||||
|
{ |
||||
|
/** |
||||
|
* {@inheritdoc} |
||||
|
*/ |
||||
|
public function rules() |
||||
|
{ |
||||
|
return [ |
||||
|
<?= implode(",\n ", $rules) ?>,
|
||||
|
]; |
||||
|
} |
||||
|
|
||||
|
/** |
||||
|
* {@inheritdoc} |
||||
|
*/ |
||||
|
public function scenarios() |
||||
|
{ |
||||
|
// bypass scenarios() implementation in the parent class
|
||||
|
return Model::scenarios(); |
||||
|
} |
||||
|
|
||||
|
/** |
||||
|
* Creates data provider instance with search query applied |
||||
|
* |
||||
|
* @param array $params |
||||
|
* |
||||
|
* @return ActiveDataProvider |
||||
|
*/ |
||||
|
public function search($params) |
||||
|
{ |
||||
|
$query = <?= isset($modelAlias) ? $modelAlias : $modelClass ?>::find();
|
||||
|
|
||||
|
// add conditions that should always apply here
|
||||
|
|
||||
|
$dataProvider = new ActiveDataProvider([ |
||||
|
'query' => $query, |
||||
|
]); |
||||
|
|
||||
|
$this->load($params); |
||||
|
|
||||
|
if (!$this->validate()) { |
||||
|
// uncomment the following line if you do not want to return any records when validation fails
|
||||
|
// $query->where('0=1');
|
||||
|
return $dataProvider; |
||||
|
} |
||||
|
|
||||
|
// grid filtering conditions
|
||||
|
<?= implode("\n ", $searchConditions) ?>
|
||||
|
|
||||
|
return $dataProvider; |
||||
|
} |
||||
|
} |
@ -0,0 +1,43 @@ |
|||||
|
<?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(); ?>
|
||||
|
|
||||
|
<?php foreach ($generator->getColumnNames() as $attribute) { |
||||
|
if (in_array($attribute, $safeAttributes)) { |
||||
|
echo " <?= " . $generator->generateActiveField($attribute) . " ?>\n\n"; |
||||
|
} |
||||
|
} ?>
|
||||
|
<div class="form-group"> |
||||
|
<?= "<?= " ?>Html::submitButton(<?= $generator->generateString('保存') ?>, ['class' => 'btn btn-success']) ?>
|
||||
|
<?= "<?= " ?>Html::a(<?= $generator->generateString('返回') ?>, ['index'], ['class' => 'btn btn-info']) ?>
|
||||
|
</div> |
||||
|
|
||||
|
<?= "<?php " ?>ActiveForm::end(); ?>
|
||||
|
|
||||
|
</div> |
@ -0,0 +1,45 @@ |
|||||
|
<?php |
||||
|
|
||||
|
use yii\helpers\Inflector; |
||||
|
use yii\helpers\StringHelper; |
||||
|
|
||||
|
/* @var $this yii\web\View */ |
||||
|
/* @var $generator yii\gii\generators\crud\Generator */ |
||||
|
|
||||
|
echo "<?php\n"; |
||||
|
?>
|
||||
|
|
||||
|
use yii\helpers\Html; |
||||
|
use yii\widgets\ActiveForm; |
||||
|
|
||||
|
/* @var $this yii\web\View */ |
||||
|
/* @var $model <?= ltrim($generator->searchModelClass, '\\') ?> */ |
||||
|
/* @var $form yii\widgets\ActiveForm */ |
||||
|
?>
|
||||
|
|
||||
|
<?= "<?php " ?>$form = ActiveForm::begin([
|
||||
|
'action' => ['index'], |
||||
|
'method' => 'get', |
||||
|
'validateOnType' => true, |
||||
|
]); |
||||
|
?>
|
||||
|
<div class="col-sm-12"> |
||||
|
<div class="dataTables_filter"> |
||||
|
<?="<?= "?>$form->field($model, 'id', [
|
||||
|
"template" => "{input}{error}", |
||||
|
"inputOptions" => [ |
||||
|
"placeholder" => "检索ID", |
||||
|
"class" => "form-control", |
||||
|
], |
||||
|
"errorOptions" => [ |
||||
|
"class" => "error-tips" |
||||
|
] |
||||
|
]) |
||||
|
?>
|
||||
|
<div class="form-group"> |
||||
|
<?= "<?= " ?>Html::submitButton('<i class="fa fa-filter"></i>', ['class' => 'btn btn-default']) ?>
|
||||
|
<?= "<?= " ?>Html::resetButton('<i class="fa fa-eraser"></i>', ['class' => 'btn btn-default']) ?>
|
||||
|
</div> |
||||
|
</div> |
||||
|
</div> |
||||
|
<?= "<?php " ?>ActiveForm::end(); ?>
|
@ -0,0 +1,27 @@ |
|||||
|
<?php |
||||
|
|
||||
|
use yii\helpers\Inflector; |
||||
|
use yii\helpers\StringHelper; |
||||
|
|
||||
|
/* @var $this yii\web\View */ |
||||
|
/* @var $generator yii\gii\generators\crud\Generator */ |
||||
|
|
||||
|
echo "<?php\n"; |
||||
|
?>
|
||||
|
|
||||
|
use yii\helpers\Html; |
||||
|
|
||||
|
/* @var $this yii\web\View */ |
||||
|
/* @var $model <?= ltrim($generator->modelClass, '\\') ?> */ |
||||
|
|
||||
|
$this->title = <?= $generator->generateString('创建 ' . Inflector::camel2words(StringHelper::basename($generator->modelClass))) ?>;
|
||||
|
$this->params['breadcrumbs'][] = ['label' => <?= $generator->generateString(Inflector::pluralize(Inflector::camel2words(StringHelper::basename($generator->modelClass)))) ?>, 'url' => ['index']];
|
||||
|
$this->params['breadcrumbs'][] = $this->title; |
||||
|
?>
|
||||
|
<div class="<?= Inflector::camel2id(StringHelper::basename($generator->modelClass)) ?>-create"> |
||||
|
|
||||
|
<?= "<?= " ?>$this->render('_form', [
|
||||
|
'model' => $model, |
||||
|
]) ?>
|
||||
|
|
||||
|
</div> |
@ -0,0 +1,61 @@ |
|||||
|
<?php |
||||
|
/* blobt */ |
||||
|
|
||||
|
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 blobt\grid\GridView; |
||||
|
|
||||
|
/* @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="row"> |
||||
|
<div class="col-xs-12"> |
||||
|
<?= "<?= " ?>GridView::widget([
|
||||
|
'dataProvider' => $dataProvider, |
||||
|
'filter' => $this->render("_search", ['model' => $searchModel]), |
||||
|
'batch' => [ |
||||
|
[ |
||||
|
"label" => "删除", |
||||
|
"url" => "<?=$generator->controllerID?>/deletes" |
||||
|
], |
||||
|
], |
||||
|
'columns' => [ |
||||
|
[ |
||||
|
'class' => 'blobt\grid\CheckboxColumn', |
||||
|
'width' => '2%', |
||||
|
'align' => 'center' |
||||
|
], |
||||
|
<?php |
||||
|
$count = 0; |
||||
|
foreach ($generator->getColumnNames() as $name) { |
||||
|
if (++$count < 6) { |
||||
|
echo " '" . $name . "',\n"; |
||||
|
} else { |
||||
|
echo " //'" . $name . "',\n"; |
||||
|
} |
||||
|
} |
||||
|
?>
|
||||
|
[ |
||||
|
'class' => 'blobt\grid\ActionColumn', |
||||
|
'align' => 'center' |
||||
|
], |
||||
|
] |
||||
|
]); |
||||
|
<?= "?>" ?>
|
||||
|
</div> |
||||
|
</div> |
@ -0,0 +1,38 @@ |
|||||
|
<?php |
||||
|
|
||||
|
use yii\helpers\Inflector; |
||||
|
use yii\helpers\StringHelper; |
||||
|
|
||||
|
/* @var $this yii\web\View */ |
||||
|
/* @var $generator yii\gii\generators\crud\Generator */ |
||||
|
|
||||
|
$urlParams = $generator->generateUrlParams(); |
||||
|
$modelClassName = Inflector::camel2words(StringHelper::basename($generator->modelClass)); |
||||
|
$nameAttributeTemplate = '$model->' . $generator->getNameAttribute(); |
||||
|
$titleTemplate = $generator->generateString('编辑 ' . $modelClassName . ': {name}', ['name' => '{nameAttribute}']); |
||||
|
if ($generator->enableI18N) { |
||||
|
$title = strtr($titleTemplate, ['\'{nameAttribute}\'' => $nameAttributeTemplate]); |
||||
|
} else { |
||||
|
$title = strtr($titleTemplate, ['{nameAttribute}\'' => '\' . ' . $nameAttributeTemplate]); |
||||
|
} |
||||
|
|
||||
|
echo "<?php\n"; |
||||
|
?>
|
||||
|
|
||||
|
use yii\helpers\Html; |
||||
|
|
||||
|
/* @var $this yii\web\View */ |
||||
|
/* @var $model <?= ltrim($generator->modelClass, '\\') ?> */ |
||||
|
|
||||
|
$this->title = <?= $title ?>;
|
||||
|
$this->params['breadcrumbs'][] = ['label' => <?= $generator->generateString(Inflector::pluralize(Inflector::camel2words(StringHelper::basename($generator->modelClass)))) ?>, 'url' => ['index']];
|
||||
|
$this->params['breadcrumbs'][] = ['label' => $model-><?= $generator->getNameAttribute() ?>, 'url' => ['view', <?= $urlParams ?>]];
|
||||
|
$this->params['breadcrumbs'][] = <?= $generator->generateString('Update ') ?>;
|
||||
|
?>
|
||||
|
<div class="<?= Inflector::camel2id(StringHelper::basename($generator->modelClass)) ?>-update"> |
||||
|
|
||||
|
<?= '<?= ' ?>$this->render('_form', [
|
||||
|
'model' => $model, |
||||
|
]) ?>
|
||||
|
|
||||
|
</div> |
@ -0,0 +1,49 @@ |
|||||
|
<?php |
||||
|
|
||||
|
use yii\helpers\Inflector; |
||||
|
use yii\helpers\StringHelper; |
||||
|
|
||||
|
/* @var $this yii\web\View */ |
||||
|
/* @var $generator yii\gii\generators\crud\Generator */ |
||||
|
|
||||
|
$urlParams = $generator->generateUrlParams(); |
||||
|
|
||||
|
echo "<?php\n"; |
||||
|
?>
|
||||
|
|
||||
|
use yii\helpers\Html; |
||||
|
use yii\widgets\DetailView; |
||||
|
|
||||
|
/* @var $this yii\web\View */ |
||||
|
/* @var $model <?= ltrim($generator->modelClass, '\\') ?> */ |
||||
|
|
||||
|
$this->title = $model-><?= $generator->getNameAttribute() ?>;
|
||||
|
$this->params['breadcrumbs'][] = ['label' => <?= $generator->generateString(Inflector::pluralize(Inflector::camel2words(StringHelper::basename($generator->modelClass)))) ?>, 'url' => ['index']];
|
||||
|
$this->params['breadcrumbs'][] = $this->title; |
||||
|
\yii\web\YiiAsset::register($this); |
||||
|
?>
|
||||
|
<div class="<?= Inflector::camel2id(StringHelper::basename($generator->modelClass)) ?>-view"> |
||||
|
|
||||
|
<p> |
||||
|
<?= "<?= " ?>Html::a(<?= $generator->generateString('返回列表') ?>, ['index'], ['class' => 'btn btn-success']) ?>
|
||||
|
</p> |
||||
|
|
||||
|
<?= "<?= " ?>DetailView::widget([
|
||||
|
'model' => $model, |
||||
|
'attributes' => [ |
||||
|
<?php |
||||
|
if (($tableSchema = $generator->getTableSchema()) === false) { |
||||
|
foreach ($generator->getColumnNames() as $name) { |
||||
|
echo " '" . $name . "',\n"; |
||||
|
} |
||||
|
} else { |
||||
|
foreach ($generator->getTableSchema()->columns as $column) { |
||||
|
$format = $generator->generateColumnFormat($column); |
||||
|
echo " '" . $column->name . ($format === 'text' ? "" : ":" . $format) . "',\n"; |
||||
|
} |
||||
|
} |
||||
|
?>
|
||||
|
], |
||||
|
]) ?>
|
||||
|
|
||||
|
</div> |
@ -0,0 +1,17 @@ |
|||||
|
<?php |
||||
|
/* @var $this yii\web\View */ |
||||
|
/* @var $form yii\widgets\ActiveForm */ |
||||
|
/* @var $generator yii\gii\generators\crud\Generator */ |
||||
|
|
||||
|
echo $form->field($generator, 'modelClass'); |
||||
|
echo $form->field($generator, 'searchModelClass'); |
||||
|
echo $form->field($generator, 'controllerClass'); |
||||
|
echo $form->field($generator, 'viewPath'); |
||||
|
echo $form->field($generator, 'baseControllerClass'); |
||||
|
echo $form->field($generator, 'indexWidgetType')->dropDownList([ |
||||
|
'grid' => 'GridView', |
||||
|
'list' => 'ListView', |
||||
|
]); |
||||
|
echo $form->field($generator, 'enableI18N')->checkbox(); |
||||
|
echo $form->field($generator, 'enablePjax')->checkbox(); |
||||
|
echo $form->field($generator, 'messageCategory'); |
@ -0,0 +1,995 @@ |
|||||
|
<?php |
||||
|
/** |
||||
|
* @link http://www.yiiframework.com/ |
||||
|
* @copyright Copyright (c) 2008 Yii Software LLC |
||||
|
* @license http://www.yiiframework.com/license/ |
||||
|
*/ |
||||
|
|
||||
|
namespace linyao\generators\model; |
||||
|
|
||||
|
use Yii; |
||||
|
use yii\base\NotSupportedException; |
||||
|
use yii\db\ActiveQuery; |
||||
|
use yii\db\ActiveRecord; |
||||
|
use yii\db\Connection; |
||||
|
use yii\db\Schema; |
||||
|
use yii\db\TableSchema; |
||||
|
use yii\gii\CodeFile; |
||||
|
use yii\helpers\Inflector; |
||||
|
|
||||
|
/** |
||||
|
* This generator will generate one or multiple ActiveRecord classes for the specified database table. |
||||
|
* |
||||
|
* @author Qiang Xue <qiang.xue@gmail.com> |
||||
|
* @since 2.0 |
||||
|
*/ |
||||
|
class Generator extends \yii\gii\Generator |
||||
|
{ |
||||
|
const RELATIONS_NONE = 'none'; |
||||
|
const RELATIONS_ALL = 'all'; |
||||
|
const RELATIONS_ALL_INVERSE = 'all-inverse'; |
||||
|
|
||||
|
public $db = 'db'; |
||||
|
public $ns = 'app\models'; |
||||
|
public $tableName; |
||||
|
public $modelClass; |
||||
|
public $baseClass = 'yii\db\ActiveRecord'; |
||||
|
public $generateRelations = self::RELATIONS_ALL; |
||||
|
public $generateRelationsFromCurrentSchema = true; |
||||
|
public $generateLabelsFromComments = false; |
||||
|
public $useTablePrefix = false; |
||||
|
public $standardizeCapitals = false; |
||||
|
public $singularize = false; |
||||
|
public $useSchemaName = true; |
||||
|
public $generateQuery = false; |
||||
|
public $queryNs = 'app\models'; |
||||
|
public $queryClass; |
||||
|
public $queryBaseClass = 'yii\db\ActiveQuery'; |
||||
|
|
||||
|
|
||||
|
/** |
||||
|
* {@inheritdoc} |
||||
|
*/ |
||||
|
public function getName() |
||||
|
{ |
||||
|
return 'Model Generator'; |
||||
|
} |
||||
|
|
||||
|
/** |
||||
|
* {@inheritdoc} |
||||
|
*/ |
||||
|
public function getDescription() |
||||
|
{ |
||||
|
return 'This generator generates an ActiveRecord class for the specified database table.'; |
||||
|
} |
||||
|
|
||||
|
/** |
||||
|
* {@inheritdoc} |
||||
|
*/ |
||||
|
public function rules() |
||||
|
{ |
||||
|
return array_merge(parent::rules(), [ |
||||
|
[['db', 'ns', 'tableName', 'modelClass', 'baseClass', 'queryNs', 'queryClass', 'queryBaseClass'], 'filter', 'filter' => 'trim'], |
||||
|
[['ns', 'queryNs'], 'filter', 'filter' => function ($value) { return trim($value, '\\'); }], |
||||
|
|
||||
|
[['db', 'ns', 'tableName', 'baseClass', 'queryNs', 'queryBaseClass'], 'required'], |
||||
|
[['db', 'modelClass', 'queryClass'], 'match', 'pattern' => '/^\w+$/', 'message' => 'Only word characters are allowed.'], |
||||
|
[['ns', 'baseClass', 'queryNs', 'queryBaseClass'], 'match', 'pattern' => '/^[\w\\\\]+$/', 'message' => 'Only word characters and backslashes are allowed.'], |
||||
|
[['tableName'], 'match', 'pattern' => '/^([\w ]+\.)?([\w\* ]+)$/', 'message' => 'Only word characters, and optionally spaces, an asterisk and/or a dot are allowed.'], |
||||
|
[['db'], 'validateDb'], |
||||
|
[['ns', 'queryNs'], 'validateNamespace'], |
||||
|
[['tableName'], 'validateTableName'], |
||||
|
[['modelClass'], 'validateModelClass', 'skipOnEmpty' => false], |
||||
|
[['baseClass'], 'validateClass', 'params' => ['extends' => ActiveRecord::className()]], |
||||
|
[['queryBaseClass'], 'validateClass', 'params' => ['extends' => ActiveQuery::className()]], |
||||
|
[['generateRelations'], 'in', 'range' => [self::RELATIONS_NONE, self::RELATIONS_ALL, self::RELATIONS_ALL_INVERSE]], |
||||
|
[['generateLabelsFromComments', 'useTablePrefix', 'useSchemaName', 'generateQuery', 'generateRelationsFromCurrentSchema'], 'boolean'], |
||||
|
[['enableI18N', 'standardizeCapitals', 'singularize'], 'boolean'], |
||||
|
[['messageCategory'], 'validateMessageCategory', 'skipOnEmpty' => false], |
||||
|
]); |
||||
|
} |
||||
|
|
||||
|
/** |
||||
|
* {@inheritdoc} |
||||
|
*/ |
||||
|
public function attributeLabels() |
||||
|
{ |
||||
|
return array_merge(parent::attributeLabels(), [ |
||||
|
'ns' => 'Namespace', |
||||
|
'db' => 'Database Connection ID', |
||||
|
'tableName' => 'Table Name', |
||||
|
'standardizeCapitals' => 'Standardize Capitals', |
||||
|
'singularize' => 'Singularize', |
||||
|
'modelClass' => 'Model Class Name', |
||||
|
'baseClass' => 'Base Class', |
||||
|
'generateRelations' => 'Generate Relations', |
||||
|
'generateRelationsFromCurrentSchema' => 'Generate Relations from Current Schema', |
||||
|
'generateLabelsFromComments' => 'Generate Labels from DB Comments', |
||||
|
'generateQuery' => 'Generate ActiveQuery', |
||||
|
'queryNs' => 'ActiveQuery Namespace', |
||||
|
'queryClass' => 'ActiveQuery Class', |
||||
|
'queryBaseClass' => 'ActiveQuery Base Class', |
||||
|
'useSchemaName' => 'Use Schema Name', |
||||
|
]); |
||||
|
} |
||||
|
|
||||
|
/** |
||||
|
* {@inheritdoc} |
||||
|
*/ |
||||
|
public function hints() |
||||
|
{ |
||||
|
return array_merge(parent::hints(), [ |
||||
|
'ns' => 'This is the namespace of the ActiveRecord class to be generated, e.g., <code>app\models</code>', |
||||
|
'db' => 'This is the ID of the DB application component.', |
||||
|
'tableName' => 'This is the name of the DB table that the new ActiveRecord class is associated with, e.g. <code>post</code>. |
||||
|
The table name may consist of the DB schema part if needed, e.g. <code>public.post</code>. |
||||
|
The table name may end with asterisk to match multiple table names, e.g. <code>tbl_*</code> |
||||
|
will match tables who name starts with <code>tbl_</code>. In this case, multiple ActiveRecord classes |
||||
|
will be generated, one for each matching table name; and the class names will be generated from |
||||
|
the matching characters. For example, table <code>tbl_post</code> will generate <code>Post</code> |
||||
|
class.', |
||||
|
'modelClass' => 'This is the name of the ActiveRecord class to be generated. The class name should not contain |
||||
|
the namespace part as it is specified in "Namespace". You do not need to specify the class name |
||||
|
if "Table Name" ends with asterisk, in which case multiple ActiveRecord classes will be generated.', |
||||
|
'standardizeCapitals' => 'This indicates whether the generated class names should have standardized capitals. For example, |
||||
|
table names like <code>SOME_TABLE</code> or <code>Other_Table</code> will have class names <code>SomeTable</code> |
||||
|
and <code>OtherTable</code>, respectively. If not checked, the same tables will have class names <code>SOMETABLE</code> |
||||
|
and <code>OtherTable</code> instead.', |
||||
|
'singularize' => 'This indicates whether the generated class names should be singularized. For example, |
||||
|
table names like <code>some_tables</code> will have class names <code>SomeTable</code>.', |
||||
|
'baseClass' => 'This is the base class of the new ActiveRecord class. It should be a fully qualified namespaced class name.', |
||||
|
'generateRelations' => 'This indicates whether the generator should generate relations based on |
||||
|
foreign key constraints it detects in the database. Note that if your database contains too many tables, |
||||
|
you may want to uncheck this option to accelerate the code generation process.', |
||||
|
'generateRelationsFromCurrentSchema' => 'This indicates whether the generator should generate relations from current schema or from all available schemas.', |
||||
|
'generateLabelsFromComments' => 'This indicates whether the generator should generate attribute labels |
||||
|
by using the comments of the corresponding DB columns.', |
||||
|
'useTablePrefix' => 'This indicates whether the table name returned by the generated ActiveRecord class |
||||
|
should consider the <code>tablePrefix</code> setting of the DB connection. For example, if the |
||||
|
table name is <code>tbl_post</code> and <code>tablePrefix=tbl_</code>, the ActiveRecord class |
||||
|
will return the table name as <code>{{%post}}</code>.', |
||||
|
'useSchemaName' => 'This indicates whether to include the schema name in the ActiveRecord class |
||||
|
when it\'s auto generated. Only non default schema would be used.', |
||||
|
'generateQuery' => 'This indicates whether to generate ActiveQuery for the ActiveRecord class.', |
||||
|
'queryNs' => 'This is the namespace of the ActiveQuery class to be generated, e.g., <code>app\models</code>', |
||||
|
'queryClass' => 'This is the name of the ActiveQuery class to be generated. The class name should not contain |
||||
|
the namespace part as it is specified in "ActiveQuery Namespace". You do not need to specify the class name |
||||
|
if "Table Name" ends with asterisk, in which case multiple ActiveQuery classes will be generated.', |
||||
|
'queryBaseClass' => 'This is the base class of the new ActiveQuery class. It should be a fully qualified namespaced class name.', |
||||
|
]); |
||||
|
} |
||||
|
|
||||
|
/** |
||||
|
* {@inheritdoc} |
||||
|
*/ |
||||
|
public function autoCompleteData() |
||||
|
{ |
||||
|
$db = $this->getDbConnection(); |
||||
|
if ($db !== null) { |
||||
|
return [ |
||||
|
'tableName' => function () use ($db) { |
||||
|
return $db->getSchema()->getTableNames(); |
||||
|
}, |
||||
|
]; |
||||
|
} |
||||
|
|
||||
|
return []; |
||||
|
} |
||||
|
|
||||
|
/** |
||||
|
* {@inheritdoc} |
||||
|
*/ |
||||
|
public function requiredTemplates() |
||||
|
{ |
||||
|
// @todo make 'query.php' to be required before 2.1 release
|
||||
|
return ['model.php'/*, 'query.php'*/]; |
||||
|
} |
||||
|
|
||||
|
/** |
||||
|
* {@inheritdoc} |
||||
|
*/ |
||||
|
public function stickyAttributes() |
||||
|
{ |
||||
|
return array_merge(parent::stickyAttributes(), ['ns', 'db', 'baseClass', 'generateRelations', 'generateLabelsFromComments', 'queryNs', 'queryBaseClass', 'useTablePrefix', 'generateQuery']); |
||||
|
} |
||||
|
|
||||
|
/** |
||||
|
* Returns the `tablePrefix` property of the DB connection as specified |
||||
|
* |
||||
|
* @return string |
||||
|
* @since 2.0.5 |
||||
|
* @see getDbConnection |
||||
|
*/ |
||||
|
public function getTablePrefix() |
||||
|
{ |
||||
|
$db = $this->getDbConnection(); |
||||
|
if ($db !== null) { |
||||
|
return $db->tablePrefix; |
||||
|
} |
||||
|
|
||||
|
return ''; |
||||
|
} |
||||
|
|
||||
|
/** |
||||
|
* {@inheritdoc} |
||||
|
*/ |
||||
|
public function generate() |
||||
|
{ |
||||
|
$files = []; |
||||
|
$relations = $this->generateRelations(); |
||||
|
$db = $this->getDbConnection(); |
||||
|
foreach ($this->getTableNames() as $tableName) { |
||||
|
// model :
|
||||
|
$modelClassName = $this->generateClassName($tableName); |
||||
|
$queryClassName = ($this->generateQuery) ? $this->generateQueryClassName($modelClassName) : false; |
||||
|
$tableSchema = $db->getTableSchema($tableName); |
||||
|
$params = [ |
||||
|
'tableName' => $tableName, |
||||
|
'className' => $modelClassName, |
||||
|
'queryClassName' => $queryClassName, |
||||
|
'tableSchema' => $tableSchema, |
||||
|
'properties' => $this->generateProperties($tableSchema), |
||||
|
'labels' => $this->generateLabels($tableSchema), |
||||
|
'rules' => $this->generateRules($tableSchema), |
||||
|
'relations' => isset($relations[$tableName]) ? $relations[$tableName] : [], |
||||
|
]; |
||||
|
$files[] = new CodeFile( |
||||
|
Yii::getAlias('@' . str_replace('\\', '/', $this->ns)) . '/' . $modelClassName . '.php', |
||||
|
$this->render('model.php', $params) |
||||
|
); |
||||
|
|
||||
|
// query :
|
||||
|
if ($queryClassName) { |
||||
|
$params['className'] = $queryClassName; |
||||
|
$params['modelClassName'] = $modelClassName; |
||||
|
$files[] = new CodeFile( |
||||
|
Yii::getAlias('@' . str_replace('\\', '/', $this->queryNs)) . '/' . $queryClassName . '.php', |
||||
|
$this->render('query.php', $params) |
||||
|
); |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
return $files; |
||||
|
} |
||||
|
|
||||
|
/** |
||||
|
* Generates the properties for the specified table. |
||||
|
* @param \yii\db\TableSchema $table the table schema |
||||
|
* @return array the generated properties (property => type) |
||||
|
* @since 2.0.6 |
||||
|
*/ |
||||
|
protected function generateProperties($table) |
||||
|
{ |
||||
|
$properties = []; |
||||
|
foreach ($table->columns as $column) { |
||||
|
$columnPhpType = $column->phpType; |
||||
|
if ($columnPhpType === 'integer') { |
||||
|
$type = 'int'; |
||||
|
} elseif ($columnPhpType === 'boolean') { |
||||
|
$type = 'bool'; |
||||
|
} else { |
||||
|
$type = $columnPhpType; |
||||
|
} |
||||
|
$properties[$column->name] = [ |
||||
|
'type' => $type, |
||||
|
'name' => $column->name, |
||||
|
'comment' => $column->comment, |
||||
|
]; |
||||
|
} |
||||
|
|
||||
|
return $properties; |
||||
|
} |
||||
|
|
||||
|
/** |
||||
|
* Generates the attribute labels for the specified table. |
||||
|
* @param \yii\db\TableSchema $table the table schema |
||||
|
* @return array the generated attribute labels (name => label) |
||||
|
*/ |
||||
|
public function generateLabels($table) |
||||
|
{ |
||||
|
// $labels = [];
|
||||
|
// foreach ($table->columns as $column) {
|
||||
|
// if ($this->generateLabelsFromComments && !empty($column->comment)) {
|
||||
|
// $labels[$column->name] = $column->comment;
|
||||
|
// } elseif (!strcasecmp($column->name, 'id')) {
|
||||
|
// $labels[$column->name] = 'ID';
|
||||
|
// } else {
|
||||
|
// $label = Inflector::camel2words($column->name);
|
||||
|
// if (!empty($label) && substr_compare($label, ' id', -3, 3, true) === 0) {
|
||||
|
// $label = substr($label, 0, -3) . ' ID';
|
||||
|
// }
|
||||
|
// $labels[$column->name] = $label;
|
||||
|
// }
|
||||
|
// }
|
||||
|
//
|
||||
|
// return $labels;
|
||||
|
|
||||
|
$labels=array(); |
||||
|
|
||||
|
$sql ="SELECT COLUMN_NAME, COLUMN_COMMENT FROM INFORMATION_SCHEMA.COLUMNS WHERE table_name = '$table->name'"; |
||||
|
|
||||
|
$res = Yii::$app->getDb()->createCommand($sql)->query(); |
||||
|
|
||||
|
foreach ($res as $column){ |
||||
|
|
||||
|
if (!empty($column['COLUMN_COMMENT'])) |
||||
|
|
||||
|
$labels[$column['COLUMN_NAME']]= $column['COLUMN_COMMENT']; |
||||
|
|
||||
|
else |
||||
|
|
||||
|
$labels[$column['COLUMN_NAME']]= $column['COLUMN_NAME']; |
||||
|
|
||||
|
} |
||||
|
|
||||
|
return $labels; |
||||
|
} |
||||
|
|
||||
|
/** |
||||
|
* Generates validation rules for the specified table. |
||||
|
* @param \yii\db\TableSchema $table the table schema |
||||
|
* @return array the generated validation rules |
||||
|
*/ |
||||
|
public function generateRules($table) |
||||
|
{ |
||||
|
$types = []; |
||||
|
$lengths = []; |
||||
|
foreach ($table->columns as $column) { |
||||
|
if($column->name == 'created_at' || $column->name == 'updated_at') { |
||||
|
continue; |
||||
|
} |
||||
|
if ($column->autoIncrement) { |
||||
|
continue; |
||||
|
} |
||||
|
if (!$column->allowNull && $column->defaultValue === null) { |
||||
|
$types['required'][] = $column->name; |
||||
|
} |
||||
|
switch ($column->type) { |
||||
|
case Schema::TYPE_SMALLINT: |
||||
|
case Schema::TYPE_INTEGER: |
||||
|
case Schema::TYPE_BIGINT: |
||||
|
case Schema::TYPE_TINYINT: |
||||
|
$types['integer'][] = $column->name; |
||||
|
break; |
||||
|
case Schema::TYPE_BOOLEAN: |
||||
|
$types['boolean'][] = $column->name; |
||||
|
break; |
||||
|
case Schema::TYPE_FLOAT: |
||||
|
case Schema::TYPE_DOUBLE: |
||||
|
case Schema::TYPE_DECIMAL: |
||||
|
case Schema::TYPE_MONEY: |
||||
|
$types['number'][] = $column->name; |
||||
|
break; |
||||
|
case Schema::TYPE_DATE: |
||||
|
case Schema::TYPE_TIME: |
||||
|
case Schema::TYPE_DATETIME: |
||||
|
case Schema::TYPE_TIMESTAMP: |
||||
|
case Schema::TYPE_JSON: |
||||
|
$types['safe'][] = $column->name; |
||||
|
break; |
||||
|
default: // strings
|
||||
|
if ($column->size > 0) { |
||||
|
$lengths[$column->size][] = $column->name; |
||||
|
} else { |
||||
|
$types['string'][] = $column->name; |
||||
|
} |
||||
|
} |
||||
|
} |
||||
|
$rules = []; |
||||
|
$driverName = $this->getDbDriverName(); |
||||
|
foreach ($types as $type => $columns) { |
||||
|
if ($driverName === 'pgsql' && $type === 'integer') { |
||||
|
$rules[] = "[['" . implode("', '", $columns) . "'], 'default', 'value' => null]"; |
||||
|
} |
||||
|
$rules[] = "[['" . implode("', '", $columns) . "'], '$type']"; |
||||
|
} |
||||
|
foreach ($lengths as $length => $columns) { |
||||
|
$rules[] = "[['" . implode("', '", $columns) . "'], 'string', 'max' => $length]"; |
||||
|
} |
||||
|
|
||||
|
$db = $this->getDbConnection(); |
||||
|
|
||||
|
// Unique indexes rules
|
||||
|
try { |
||||
|
$uniqueIndexes = array_merge($db->getSchema()->findUniqueIndexes($table), [$table->primaryKey]); |
||||
|
$uniqueIndexes = array_unique($uniqueIndexes, SORT_REGULAR); |
||||
|
foreach ($uniqueIndexes as $uniqueColumns) { |
||||
|
// Avoid validating auto incremental columns
|
||||
|
if (!$this->isColumnAutoIncremental($table, $uniqueColumns)) { |
||||
|
$attributesCount = count($uniqueColumns); |
||||
|
|
||||
|
if ($attributesCount === 1) { |
||||
|
$rules[] = "[['" . $uniqueColumns[0] . "'], 'unique']"; |
||||
|
} elseif ($attributesCount > 1) { |
||||
|
$columnsList = implode("', '", $uniqueColumns); |
||||
|
$rules[] = "[['$columnsList'], 'unique', 'targetAttribute' => ['$columnsList']]"; |
||||
|
} |
||||
|
} |
||||
|
} |
||||
|
} catch (NotSupportedException $e) { |
||||
|
// doesn't support unique indexes information...do nothing
|
||||
|
} |
||||
|
|
||||
|
// Exist rules for foreign keys
|
||||
|
foreach ($table->foreignKeys as $refs) { |
||||
|
$refTable = $refs[0]; |
||||
|
$refTableSchema = $db->getTableSchema($refTable); |
||||
|
if ($refTableSchema === null) { |
||||
|
// Foreign key could point to non-existing table: https://github.com/yiisoft/yii2-gii/issues/34
|
||||
|
continue; |
||||
|
} |
||||
|
$refClassName = $this->generateClassName($refTable); |
||||
|
unset($refs[0]); |
||||
|
$attributes = implode("', '", array_keys($refs)); |
||||
|
$targetAttributes = []; |
||||
|
foreach ($refs as $key => $value) { |
||||
|
$targetAttributes[] = "'$key' => '$value'"; |
||||
|
} |
||||
|
$targetAttributes = implode(', ', $targetAttributes); |
||||
|
$rules[] = "[['$attributes'], 'exist', 'skipOnError' => true, 'targetClass' => $refClassName::className(), 'targetAttribute' => [$targetAttributes]]"; |
||||
|
} |
||||
|
|
||||
|
return $rules; |
||||
|
} |
||||
|
|
||||
|
/** |
||||
|
* Generates relations using a junction table by adding an extra viaTable(). |
||||
|
* @param \yii\db\TableSchema the table being checked |
||||
|
* @param array $fks obtained from the checkJunctionTable() method |
||||
|
* @param array $relations |
||||
|
* @return array modified $relations |
||||
|
*/ |
||||
|
private function generateManyManyRelations($table, $fks, $relations) |
||||
|
{ |
||||
|
$db = $this->getDbConnection(); |
||||
|
|
||||
|
foreach ($fks as $pair) { |
||||
|
list($firstKey, $secondKey) = $pair; |
||||
|
$table0 = $firstKey[0]; |
||||
|
$table1 = $secondKey[0]; |
||||
|
unset($firstKey[0], $secondKey[0]); |
||||
|
$className0 = $this->generateClassName($table0); |
||||
|
$className1 = $this->generateClassName($table1); |
||||
|
$table0Schema = $db->getTableSchema($table0); |
||||
|
$table1Schema = $db->getTableSchema($table1); |
||||
|
|
||||
|
// @see https://github.com/yiisoft/yii2-gii/issues/166
|
||||
|
if ($table0Schema === null || $table1Schema === null) { |
||||
|
continue; |
||||
|
} |
||||
|
|
||||
|
$link = $this->generateRelationLink(array_flip($secondKey)); |
||||
|
$viaLink = $this->generateRelationLink($firstKey); |
||||
|
$relationName = $this->generateRelationName($relations, $table0Schema, key($secondKey), true); |
||||
|
$relations[$table0Schema->fullName][$relationName] = [ |
||||
|
"return \$this->hasMany($className1::className(), $link)->viaTable('" |
||||
|
. $this->generateTableName($table->name) . "', $viaLink);", |
||||
|
$className1, |
||||
|
true, |
||||
|
]; |
||||
|
|
||||
|
$link = $this->generateRelationLink(array_flip($firstKey)); |
||||
|
$viaLink = $this->generateRelationLink($secondKey); |
||||
|
$relationName = $this->generateRelationName($relations, $table1Schema, key($firstKey), true); |
||||
|
$relations[$table1Schema->fullName][$relationName] = [ |
||||
|
"return \$this->hasMany($className0::className(), $link)->viaTable('" |
||||
|
. $this->generateTableName($table->name) . "', $viaLink);", |
||||
|
$className0, |
||||
|
true, |
||||
|
]; |
||||
|
} |
||||
|
|
||||
|
return $relations; |
||||
|
} |
||||
|
|
||||
|
/** |
||||
|
* @return string[] all db schema names or an array with a single empty string |
||||
|
* @throws NotSupportedException |
||||
|
* @since 2.0.5 |
||||
|
*/ |
||||
|
protected function getSchemaNames() |
||||
|
{ |
||||
|
$db = $this->getDbConnection(); |
||||
|
|
||||
|
if ($this->generateRelationsFromCurrentSchema) { |
||||
|
if ($db->schema->defaultSchema !== null) { |
||||
|
return [$db->schema->defaultSchema]; |
||||
|
} |
||||
|
return ['']; |
||||
|
} |
||||
|
|
||||
|
$schema = $db->getSchema(); |
||||
|
if ($schema->hasMethod('getSchemaNames')) { // keep BC to Yii versions < 2.0.4
|
||||
|
try { |
||||
|
$schemaNames = $schema->getSchemaNames(); |
||||
|
} catch (NotSupportedException $e) { |
||||
|
// schema names are not supported by schema
|
||||
|
} |
||||
|
} |
||||
|
if (!isset($schemaNames)) { |
||||
|
if (($pos = strpos($this->tableName, '.')) !== false) { |
||||
|
$schemaNames = [substr($this->tableName, 0, $pos)]; |
||||
|
} else { |
||||
|
$schemaNames = ['']; |
||||
|
} |
||||
|
} |
||||
|
return $schemaNames; |
||||
|
} |
||||
|
|
||||
|
/** |
||||
|
* @return array the generated relation declarations |
||||
|
*/ |
||||
|
protected function generateRelations() |
||||
|
{ |
||||
|
if ($this->generateRelations === self::RELATIONS_NONE) { |
||||
|
return []; |
||||
|
} |
||||
|
|
||||
|
$db = $this->getDbConnection(); |
||||
|
$relations = []; |
||||
|
$schemaNames = $this->getSchemaNames(); |
||||
|
foreach ($schemaNames as $schemaName) { |
||||
|
foreach ($db->getSchema()->getTableSchemas($schemaName) as $table) { |
||||
|
$className = $this->generateClassName($table->fullName); |
||||
|
foreach ($table->foreignKeys as $refs) { |
||||
|
$refTable = $refs[0]; |
||||
|
$refTableSchema = $db->getTableSchema($refTable); |
||||
|
if ($refTableSchema === null) { |
||||
|
// Foreign key could point to non-existing table: https://github.com/yiisoft/yii2-gii/issues/34
|
||||
|
continue; |
||||
|
} |
||||
|
unset($refs[0]); |
||||
|
$fks = array_keys($refs); |
||||
|
$refClassName = $this->generateClassName($refTable); |
||||
|
|
||||
|
// Add relation for this table
|
||||
|
$link = $this->generateRelationLink(array_flip($refs)); |
||||
|
$relationName = $this->generateRelationName($relations, $table, $fks[0], false); |
||||
|
$relations[$table->fullName][$relationName] = [ |
||||
|
"return \$this->hasOne($refClassName::className(), $link);", |
||||
|
$refClassName, |
||||
|
false, |
||||
|
]; |
||||
|
|
||||
|
// Add relation for the referenced table
|
||||
|
$hasMany = $this->isHasManyRelation($table, $fks); |
||||
|
$link = $this->generateRelationLink($refs); |
||||
|
$relationName = $this->generateRelationName($relations, $refTableSchema, $className, $hasMany); |
||||
|
$relations[$refTableSchema->fullName][$relationName] = [ |
||||
|
"return \$this->" . ($hasMany ? 'hasMany' : 'hasOne') . "($className::className(), $link);", |
||||
|
$className, |
||||
|
$hasMany, |
||||
|
]; |
||||
|
} |
||||
|
|
||||
|
if (($junctionFks = $this->checkJunctionTable($table)) === false) { |
||||
|
continue; |
||||
|
} |
||||
|
|
||||
|
$relations = $this->generateManyManyRelations($table, $junctionFks, $relations); |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
if ($this->generateRelations === self::RELATIONS_ALL_INVERSE) { |
||||
|
return $this->addInverseRelations($relations); |
||||
|
} |
||||
|
|
||||
|
return $relations; |
||||
|
} |
||||
|
|
||||
|
/** |
||||
|
* Adds inverse relations |
||||
|
* |
||||
|
* @param array $relations relation declarations |
||||
|
* @return array relation declarations extended with inverse relation names |
||||
|
* @since 2.0.5 |
||||
|
*/ |
||||
|
protected function addInverseRelations($relations) |
||||
|
{ |
||||
|
$db = $this->getDbConnection(); |
||||
|
$relationNames = []; |
||||
|
|
||||
|
$schemaNames = $this->getSchemaNames(); |
||||
|
foreach ($schemaNames as $schemaName) { |
||||
|
foreach ($db->schema->getTableSchemas($schemaName) as $table) { |
||||
|
$className = $this->generateClassName($table->fullName); |
||||
|
foreach ($table->foreignKeys as $refs) { |
||||
|
$refTable = $refs[0]; |
||||
|
$refTableSchema = $db->getTableSchema($refTable); |
||||
|
if ($refTableSchema === null) { |
||||
|
// Foreign key could point to non-existing table: https://github.com/yiisoft/yii2-gii/issues/34
|
||||
|
continue; |
||||
|
} |
||||
|
unset($refs[0]); |
||||
|
$fks = array_keys($refs); |
||||
|
|
||||
|
$leftRelationName = $this->generateRelationName($relationNames, $table, $fks[0], false); |
||||
|
$relationNames[$table->fullName][$leftRelationName] = true; |
||||
|
$hasMany = $this->isHasManyRelation($table, $fks); |
||||
|
$rightRelationName = $this->generateRelationName( |
||||
|
$relationNames, |
||||
|
$refTableSchema, |
||||
|
$className, |
||||
|
$hasMany |
||||
|
); |
||||
|
$relationNames[$refTableSchema->fullName][$rightRelationName] = true; |
||||
|
|
||||
|
$relations[$table->fullName][$leftRelationName][0] = |
||||
|
rtrim($relations[$table->fullName][$leftRelationName][0], ';') |
||||
|
. "->inverseOf('".lcfirst($rightRelationName)."');"; |
||||
|
$relations[$refTableSchema->fullName][$rightRelationName][0] = |
||||
|
rtrim($relations[$refTableSchema->fullName][$rightRelationName][0], ';') |
||||
|
. "->inverseOf('".lcfirst($leftRelationName)."');"; |
||||
|
} |
||||
|
} |
||||
|
} |
||||
|
return $relations; |
||||
|
} |
||||
|
|
||||
|
/** |
||||
|
* Determines if relation is of has many type |
||||
|
* |
||||
|
* @param TableSchema $table |
||||
|
* @param array $fks |
||||
|
* @return bool |
||||
|
* @since 2.0.5 |
||||
|
*/ |
||||
|
protected function isHasManyRelation($table, $fks) |
||||
|
{ |
||||
|
$uniqueKeys = [$table->primaryKey]; |
||||
|
try { |
||||
|
$uniqueKeys = array_merge($uniqueKeys, $this->getDbConnection()->getSchema()->findUniqueIndexes($table)); |
||||
|
} catch (NotSupportedException $e) { |
||||
|
// ignore
|
||||
|
} |
||||
|
foreach ($uniqueKeys as $uniqueKey) { |
||||
|
if (count(array_diff(array_merge($uniqueKey, $fks), array_intersect($uniqueKey, $fks))) === 0) { |
||||
|
return false; |
||||
|
} |
||||
|
} |
||||
|
return true; |
||||
|
} |
||||
|
|
||||
|
/** |
||||
|
* Generates the link parameter to be used in generating the relation declaration. |
||||
|
* @param array $refs reference constraint |
||||
|
* @return string the generated link parameter. |
||||
|
*/ |
||||
|
protected function generateRelationLink($refs) |
||||
|
{ |
||||
|
$pairs = []; |
||||
|
foreach ($refs as $a => $b) { |
||||
|
$pairs[] = "'$a' => '$b'"; |
||||
|
} |
||||
|
|
||||
|
return '[' . implode(', ', $pairs) . ']'; |
||||
|
} |
||||
|
|
||||
|
/** |
||||
|
* Checks if the given table is a junction table, that is it has at least one pair of unique foreign keys. |
||||
|
* @param \yii\db\TableSchema the table being checked |
||||
|
* @return array|bool all unique foreign key pairs if the table is a junction table, |
||||
|
* or false if the table is not a junction table. |
||||
|
*/ |
||||
|
protected function checkJunctionTable($table) |
||||
|
{ |
||||
|
if (count($table->foreignKeys) < 2) { |
||||
|
return false; |
||||
|
} |
||||
|
$uniqueKeys = [$table->primaryKey]; |
||||
|
try { |
||||
|
$uniqueKeys = array_merge($uniqueKeys, $this->getDbConnection()->getSchema()->findUniqueIndexes($table)); |
||||
|
} catch (NotSupportedException $e) { |
||||
|
// ignore
|
||||
|
} |
||||
|
$result = []; |
||||
|
// find all foreign key pairs that have all columns in an unique constraint
|
||||
|
$foreignKeys = array_values($table->foreignKeys); |
||||
|
$foreignKeysCount = count($foreignKeys); |
||||
|
|
||||
|
for ($i = 0; $i < $foreignKeysCount; $i++) { |
||||
|
$firstColumns = $foreignKeys[$i]; |
||||
|
unset($firstColumns[0]); |
||||
|
|
||||
|
for ($j = $i + 1; $j < $foreignKeysCount; $j++) { |
||||
|
$secondColumns = $foreignKeys[$j]; |
||||
|
unset($secondColumns[0]); |
||||
|
|
||||
|
$fks = array_merge(array_keys($firstColumns), array_keys($secondColumns)); |
||||
|
foreach ($uniqueKeys as $uniqueKey) { |
||||
|
if (count(array_diff(array_merge($uniqueKey, $fks), array_intersect($uniqueKey, $fks))) === 0) { |
||||
|
// save the foreign key pair
|
||||
|
$result[] = [$foreignKeys[$i], $foreignKeys[$j]]; |
||||
|
break; |
||||
|
} |
||||
|
} |
||||
|
} |
||||
|
} |
||||
|
return empty($result) ? false : $result; |
||||
|
} |
||||
|
|
||||
|
/** |
||||
|
* Generate a relation name for the specified table and a base name. |
||||
|
* @param array $relations the relations being generated currently. |
||||
|
* @param \yii\db\TableSchema $table the table schema |
||||
|
* @param string $key a base name that the relation name may be generated from |
||||
|
* @param bool $multiple whether this is a has-many relation |
||||
|
* @return string the relation name |
||||
|
*/ |
||||
|
protected function generateRelationName($relations, $table, $key, $multiple) |
||||
|
{ |
||||
|
static $baseModel; |
||||
|
/* @var $baseModel \yii\db\ActiveRecord */ |
||||
|
if ($baseModel === null) { |
||||
|
$baseClass = $this->baseClass; |
||||
|
$baseClassReflector = new \ReflectionClass($baseClass); |
||||
|
if ($baseClassReflector->isAbstract()) { |
||||
|
$baseClassWrapper = |
||||
|
'namespace ' . __NAMESPACE__ . ';'. |
||||
|
'class GiiBaseClassWrapper extends \\' . $baseClass . ' {' . |
||||
|
'public static function tableName(){' . |
||||
|
'return "' . addslashes($table->fullName) . '";' . |
||||
|
'}' . |
||||
|
'};' . |
||||
|
'return new GiiBaseClassWrapper();'; |
||||
|
$baseModel = eval($baseClassWrapper); |
||||
|
} else { |
||||
|
$baseModel = new $baseClass(); |
||||
|
} |
||||
|
$baseModel->setAttributes([]); |
||||
|
} |
||||
|
|
||||
|
if (!empty($key) && strcasecmp($key, 'id')) { |
||||
|
if (substr_compare($key, 'id', -2, 2, true) === 0) { |
||||
|
$key = rtrim(substr($key, 0, -2), '_'); |
||||
|
} elseif (substr_compare($key, 'id', 0, 2, true) === 0) { |
||||
|
$key = ltrim(substr($key, 2, strlen($key)), '_'); |
||||
|
} |
||||
|
} |
||||
|
if ($multiple) { |
||||
|
$key = Inflector::pluralize($key); |
||||
|
} |
||||
|
$name = $rawName = Inflector::id2camel($key, '_'); |
||||
|
$i = 0; |
||||
|
while ($baseModel->hasProperty(lcfirst($name))) { |
||||
|
$name = $rawName . ($i++); |
||||
|
} |
||||
|
while (isset($table->columns[lcfirst($name)])) { |
||||
|
$name = $rawName . ($i++); |
||||
|
} |
||||
|
while (isset($relations[$table->fullName][$name])) { |
||||
|
$name = $rawName . ($i++); |
||||
|
} |
||||
|
|
||||
|
return $name; |
||||
|
} |
||||
|
|
||||
|
/** |
||||
|
* Validates the [[db]] attribute. |
||||
|
*/ |
||||
|
public function validateDb() |
||||
|
{ |
||||
|
if (!Yii::$app->has($this->db)) { |
||||
|
$this->addError('db', 'There is no application component named "db".'); |
||||
|
} elseif (!Yii::$app->get($this->db) instanceof Connection) { |
||||
|
$this->addError('db', 'The "db" application component must be a DB connection instance.'); |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
/** |
||||
|
* Validates the namespace. |
||||
|
* |
||||
|
* @param string $attribute Namespace variable. |
||||
|
*/ |
||||
|
public function validateNamespace($attribute) |
||||
|
{ |
||||
|
$value = $this->$attribute; |
||||
|
$value = ltrim($value, '\\'); |
||||
|
$path = Yii::getAlias('@' . str_replace('\\', '/', $value), false); |
||||
|
if ($path === false) { |
||||
|
$this->addError($attribute, 'Namespace must be associated with an existing directory.'); |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
/** |
||||
|
* Validates the [[modelClass]] attribute. |
||||
|
*/ |
||||
|
public function validateModelClass() |
||||
|
{ |
||||
|
if ($this->isReservedKeyword($this->modelClass)) { |
||||
|
$this->addError('modelClass', 'Class name cannot be a reserved PHP keyword.'); |
||||
|
} |
||||
|
if ((empty($this->tableName) || substr_compare($this->tableName, '*', -1, 1)) && $this->modelClass == '') { |
||||
|
$this->addError('modelClass', 'Model Class cannot be blank if table name does not end with asterisk.'); |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
/** |
||||
|
* Validates the [[tableName]] attribute. |
||||
|
*/ |
||||
|
public function validateTableName() |
||||
|
{ |
||||
|
if (strpos($this->tableName, '*') !== false && substr_compare($this->tableName, '*', -1, 1)) { |
||||
|
$this->addError('tableName', 'Asterisk is only allowed as the last character.'); |
||||
|
|
||||
|
return; |
||||
|
} |
||||
|
$tables = $this->getTableNames(); |
||||
|
if (empty($tables)) { |
||||
|
$this->addError('tableName', "Table '{$this->tableName}' does not exist."); |
||||
|
} else { |
||||
|
foreach ($tables as $table) { |
||||
|
$class = $this->generateClassName($table); |
||||
|
if ($this->isReservedKeyword($class)) { |
||||
|
$this->addError('tableName', "Table '$table' will generate a class which is a reserved PHP keyword."); |
||||
|
break; |
||||
|
} |
||||
|
} |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
protected $tableNames; |
||||
|
protected $classNames; |
||||
|
|
||||
|
/** |
||||
|
* @return array the table names that match the pattern specified by [[tableName]]. |
||||
|
*/ |
||||
|
protected function getTableNames() |
||||
|
{ |
||||
|
if ($this->tableNames !== null) { |
||||
|
return $this->tableNames; |
||||
|
} |
||||
|
$db = $this->getDbConnection(); |
||||
|
if ($db === null) { |
||||
|
return []; |
||||
|
} |
||||
|
$tableNames = []; |
||||
|
if (strpos($this->tableName, '*') !== false) { |
||||
|
if (($pos = strrpos($this->tableName, '.')) !== false) { |
||||
|
$schema = substr($this->tableName, 0, $pos); |
||||
|
$pattern = '/^' . str_replace('*', '\w+', substr($this->tableName, $pos + 1)) . '$/'; |
||||
|
} else { |
||||
|
$schema = ''; |
||||
|
$pattern = '/^' . str_replace('*', '\w+', $this->tableName) . '$/'; |
||||
|
} |
||||
|
|
||||
|
foreach ($db->schema->getTableNames($schema) as $table) { |
||||
|
if (preg_match($pattern, $table)) { |
||||
|
$tableNames[] = $schema === '' ? $table : ($schema . '.' . $table); |
||||
|
} |
||||
|
} |
||||
|
} elseif (($table = $db->getTableSchema($this->tableName, true)) !== null) { |
||||
|
$tableNames[] = $this->tableName; |
||||
|
$this->classNames[$this->tableName] = $this->modelClass; |
||||
|
} |
||||
|
|
||||
|
return $this->tableNames = $tableNames; |
||||
|
} |
||||
|
|
||||
|
/** |
||||
|
* Generates the table name by considering table prefix. |
||||
|
* If [[useTablePrefix]] is false, the table name will be returned without change. |
||||
|
* @param string $tableName the table name (which may contain schema prefix) |
||||
|
* @return string the generated table name |
||||
|
*/ |
||||
|
public function generateTableName($tableName) |
||||
|
{ |
||||
|
if (!$this->useTablePrefix) { |
||||
|
return $tableName; |
||||
|
} |
||||
|
|
||||
|
$db = $this->getDbConnection(); |
||||
|
if (preg_match("/^{$db->tablePrefix}(.*?)$/", $tableName, $matches)) { |
||||
|
$tableName = '{{%' . $matches[1] . '}}'; |
||||
|
} elseif (preg_match("/^(.*?){$db->tablePrefix}$/", $tableName, $matches)) { |
||||
|
$tableName = '{{' . $matches[1] . '%}}'; |
||||
|
} |
||||
|
return $tableName; |
||||
|
} |
||||
|
|
||||
|
/** |
||||
|
* Generates a class name from the specified table name. |
||||
|
* @param string $tableName the table name (which may contain schema prefix) |
||||
|
* @param bool $useSchemaName should schema name be included in the class name, if present |
||||
|
* @return string the generated class name |
||||
|
*/ |
||||
|
protected function generateClassName($tableName, $useSchemaName = null) |
||||
|
{ |
||||
|
if (isset($this->classNames[$tableName])) { |
||||
|
return $this->classNames[$tableName]; |
||||
|
} |
||||
|
|
||||
|
$schemaName = ''; |
||||
|
$fullTableName = $tableName; |
||||
|
if (($pos = strrpos($tableName, '.')) !== false) { |
||||
|
if (($useSchemaName === null && $this->useSchemaName) || $useSchemaName) { |
||||
|
$schemaName = substr($tableName, 0, $pos) . '_'; |
||||
|
} |
||||
|
$tableName = substr($tableName, $pos + 1); |
||||
|
} |
||||
|
|
||||
|
$db = $this->getDbConnection(); |
||||
|
$patterns = []; |
||||
|
$patterns[] = "/^{$db->tablePrefix}(.*?)$/"; |
||||
|
$patterns[] = "/^(.*?){$db->tablePrefix}$/"; |
||||
|
if (strpos($this->tableName, '*') !== false) { |
||||
|
$pattern = $this->tableName; |
||||
|
if (($pos = strrpos($pattern, '.')) !== false) { |
||||
|
$pattern = substr($pattern, $pos + 1); |
||||
|
} |
||||
|
$patterns[] = '/^' . str_replace('*', '(\w+)', $pattern) . '$/'; |
||||
|
} |
||||
|
$className = $tableName; |
||||
|
foreach ($patterns as $pattern) { |
||||
|
if (preg_match($pattern, $tableName, $matches)) { |
||||
|
$className = $matches[1]; |
||||
|
break; |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
if ($this->standardizeCapitals) { |
||||
|
$schemaName = ctype_upper(preg_replace('/[_-]/', '', $schemaName)) ? strtolower($schemaName) : $schemaName; |
||||
|
$className = ctype_upper(preg_replace('/[_-]/', '', $className)) ? strtolower($className) : $className; |
||||
|
$this->classNames[$fullTableName] = Inflector::camelize(Inflector::camel2words($schemaName.$className)); |
||||
|
} else { |
||||
|
$this->classNames[$fullTableName] = Inflector::id2camel($schemaName.$className, '_'); |
||||
|
} |
||||
|
|
||||
|
if ($this->singularize) { |
||||
|
$this->classNames[$fullTableName] = Inflector::singularize($this->classNames[$fullTableName]); |
||||
|
} |
||||
|
|
||||
|
return $this->classNames[$fullTableName]; |
||||
|
} |
||||
|
|
||||
|
/** |
||||
|
* Generates a query class name from the specified model class name. |
||||
|
* @param string $modelClassName model class name |
||||
|
* @return string generated class name |
||||
|
*/ |
||||
|
protected function generateQueryClassName($modelClassName) |
||||
|
{ |
||||
|
$queryClassName = $this->queryClass; |
||||
|
if (empty($queryClassName) || strpos($this->tableName, '*') !== false) { |
||||
|
$queryClassName = $modelClassName . 'Query'; |
||||
|
} |
||||
|
return $queryClassName; |
||||
|
} |
||||
|
|
||||
|
/** |
||||
|
* @return Connection the DB connection as specified by [[db]]. |
||||
|
*/ |
||||
|
protected function getDbConnection() |
||||
|
{ |
||||
|
return Yii::$app->get($this->db, false); |
||||
|
} |
||||
|
|
||||
|
/** |
||||
|
* @return string|null driver name of db connection. |
||||
|
* In case db is not instance of \yii\db\Connection null will be returned. |
||||
|
* @since 2.0.6 |
||||
|
*/ |
||||
|
protected function getDbDriverName() |
||||
|
{ |
||||
|
/** @var Connection $db */ |
||||
|
$db = $this->getDbConnection(); |
||||
|
return $db instanceof \yii\db\Connection ? $db->driverName : null; |
||||
|
} |
||||
|
|
||||
|
/** |
||||
|
* Checks if any of the specified columns is auto incremental. |
||||
|
* @param \yii\db\TableSchema $table the table schema |
||||
|
* @param array $columns columns to check for autoIncrement property |
||||
|
* @return bool whether any of the specified columns is auto incremental. |
||||
|
*/ |
||||
|
protected function isColumnAutoIncremental($table, $columns) |
||||
|
{ |
||||
|
foreach ($columns as $column) { |
||||
|
if (isset($table->columns[$column]) && $table->columns[$column]->autoIncrement) { |
||||
|
return true; |
||||
|
} |
||||
|
} |
||||
|
|
||||
|
return false; |
||||
|
} |
||||
|
} |
@ -0,0 +1,124 @@ |
|||||
|
<?php |
||||
|
/** |
||||
|
* This is the template for generating the model class of a specified table. |
||||
|
*/ |
||||
|
|
||||
|
/* @var $this yii\web\View */ |
||||
|
/* @var $generator yii\gii\generators\model\Generator */ |
||||
|
/* @var $tableName string full table name */ |
||||
|
/* @var $className string class name */ |
||||
|
/* @var $queryClassName string query class name */ |
||||
|
/* @var $tableSchema yii\db\TableSchema */ |
||||
|
/* @var $properties array list of properties (property => [type, name. comment]) */ |
||||
|
/* @var $labels string[] list of attribute labels (name => label) */ |
||||
|
/* @var $rules string[] list of validation rules */ |
||||
|
/* @var $relations array list of relations (name => relation declaration) */ |
||||
|
|
||||
|
echo "<?php\n"; |
||||
|
?>
|
||||
|
|
||||
|
namespace <?= $generator->ns ?>;
|
||||
|
|
||||
|
use Yii; |
||||
|
use yii\behaviors\TimestampBehavior; |
||||
|
|
||||
|
/** |
||||
|
* This is the model class for table "<?= $generator->generateTableName($tableName) ?>". |
||||
|
* |
||||
|
<?php foreach ($properties as $property => $data): ?>
|
||||
|
* @property <?= "{$data['type']} \${$property}" . ($data['comment'] ? ' ' . strtr($data['comment'], ["\n" => ' ']) : '') . "\n" ?>
|
||||
|
<?php endforeach; ?>
|
||||
|
<?php if (!empty($relations)): ?>
|
||||
|
* |
||||
|
<?php foreach ($relations as $name => $relation): ?>
|
||||
|
* @property <?= $relation[1] . ($relation[2] ? '[]' : '') . ' $' . lcfirst($name) . "\n" ?>
|
||||
|
<?php endforeach; ?>
|
||||
|
<?php endif; ?>
|
||||
|
*/ |
||||
|
class <?= $className ?> extends <?= '\\' . ltrim($generator->baseClass, '\\') . "\n" ?>
|
||||
|
{ |
||||
|
/** |
||||
|
* {@inheritdoc} |
||||
|
*/ |
||||
|
public static function tableName() |
||||
|
{ |
||||
|
return '<?= $generator->generateTableName($tableName) ?>'; |
||||
|
} |
||||
|
<?php if ($generator->db !== 'db'): ?>
|
||||
|
|
||||
|
/** |
||||
|
* @return \yii\db\Connection the database connection used by this AR class. |
||||
|
*/ |
||||
|
public static function getDb() |
||||
|
{ |
||||
|
return Yii::$app->get('<?= $generator->db ?>'); |
||||
|
} |
||||
|
<?php endif; ?>
|
||||
|
|
||||
|
/** |
||||
|
* {@inheritdoc} |
||||
|
*/ |
||||
|
public function rules() |
||||
|
{ |
||||
|
return [<?= empty($rules) ? '' : ("\n " . implode(",\n ", $rules) . ",\n ") ?>];
|
||||
|
} |
||||
|
|
||||
|
/** |
||||
|
* {@inheritdoc} |
||||
|
*/ |
||||
|
public function attributeLabels() |
||||
|
{ |
||||
|
return [ |
||||
|
<?php foreach ($labels as $name => $label): ?>
|
||||
|
<?= "'$name' => " . $generator->generateString($label) . ",\n" ?>
|
||||
|
<?php endforeach; ?>
|
||||
|
]; |
||||
|
} |
||||
|
<?php foreach ($relations as $name => $relation): ?>
|
||||
|
|
||||
|
/** |
||||
|
* @return \yii\db\ActiveQuery |
||||
|
*/ |
||||
|
public function get<?= $name ?>()
|
||||
|
{ |
||||
|
<?= $relation[0] . "\n" ?>
|
||||
|
} |
||||
|
<?php endforeach; ?>
|
||||
|
<?php if ($queryClassName): ?>
|
||||
|
<?php |
||||
|
$queryClassFullName = ($generator->ns === $generator->queryNs) ? $queryClassName : '\\' . $generator->queryNs . '\\' . $queryClassName; |
||||
|
echo "\n"; |
||||
|
?>
|
||||
|
/** |
||||
|
* {@inheritdoc} |
||||
|
* @return <?= $queryClassFullName ?> the active query used by this AR class.
|
||||
|
*/ |
||||
|
public static function find() |
||||
|
{ |
||||
|
return new <?= $queryClassFullName ?>(get_called_class());
|
||||
|
} |
||||
|
<?php endif; ?>
|
||||
|
|
||||
|
<?php if (array_key_exists('created_at', $labels) && array_key_exists('updated_at', $labels)): ?>
|
||||
|
/** |
||||
|
* @author linyao |
||||
|
* @email 602604991@qq.com |
||||
|
* @created Nov 8, 2019 |
||||
|
* |
||||
|
* 行为存储创建时间和更新时间 |
||||
|
*/ |
||||
|
public function behaviors() |
||||
|
{ |
||||
|
return [ |
||||
|
[ |
||||
|
'class' => TimestampBehavior::className(), |
||||
|
'createdAtAttribute' => 'created_at', |
||||
|
'updatedAtAttribute' => 'updated_at', |
||||
|
'value' => function() { |
||||
|
return time(); |
||||
|
}, |
||||
|
], |
||||
|
]; |
||||
|
} |
||||
|
<?php endif; ?>
|
||||
|
} |
@ -0,0 +1,56 @@ |
|||||
|
<?php |
||||
|
/** |
||||
|
* This is the template for generating the ActiveQuery class. |
||||
|
*/ |
||||
|
|
||||
|
/* @var $this yii\web\View */ |
||||
|
/* @var $generator yii\gii\generators\model\Generator */ |
||||
|
/* @var $tableName string full table name */ |
||||
|
/* @var $className string class name */ |
||||
|
/* @var $tableSchema yii\db\TableSchema */ |
||||
|
/* @var $labels string[] list of attribute labels (name => label) */ |
||||
|
/* @var $rules string[] list of validation rules */ |
||||
|
/* @var $relations array list of relations (name => relation declaration) */ |
||||
|
/* @var $className string class name */ |
||||
|
/* @var $modelClassName string related model class name */ |
||||
|
|
||||
|
$modelFullClassName = $modelClassName; |
||||
|
if ($generator->ns !== $generator->queryNs) { |
||||
|
$modelFullClassName = '\\' . $generator->ns . '\\' . $modelFullClassName; |
||||
|
} |
||||
|
|
||||
|
echo "<?php\n"; |
||||
|
?>
|
||||
|
|
||||
|
namespace <?= $generator->queryNs ?>;
|
||||
|
|
||||
|
/** |
||||
|
* This is the ActiveQuery class for [[<?= $modelFullClassName ?>]].
|
||||
|
* |
||||
|
* @see <?= $modelFullClassName . "\n" ?>
|
||||
|
*/ |
||||
|
class <?= $className ?> extends <?= '\\' . ltrim($generator->queryBaseClass, '\\') . "\n" ?>
|
||||
|
{ |
||||
|
/*public function active() |
||||
|
{ |
||||
|
return $this->andWhere('[[status]]=1'); |
||||
|
}*/ |
||||
|
|
||||
|
/** |
||||
|
* {@inheritdoc} |
||||
|
* @return <?= $modelFullClassName ?>[]|array
|
||||
|
*/ |
||||
|
public function all($db = null) |
||||
|
{ |
||||
|
return parent::all($db); |
||||
|
} |
||||
|
|
||||
|
/** |
||||
|
* {@inheritdoc} |
||||
|
* @return <?= $modelFullClassName ?>|array|null
|
||||
|
*/ |
||||
|
public function one($db = null) |
||||
|
{ |
||||
|
return parent::one($db); |
||||
|
} |
||||
|
} |
@ -0,0 +1,30 @@ |
|||||
|
<?php |
||||
|
|
||||
|
use yii\gii\generators\model\Generator; |
||||
|
|
||||
|
/* @var $this yii\web\View */ |
||||
|
/* @var $form yii\widgets\ActiveForm */ |
||||
|
/* @var $generator yii\gii\generators\model\Generator */ |
||||
|
|
||||
|
echo $form->field($generator, 'tableName')->textInput(['table_prefix' => $generator->getTablePrefix()]); |
||||
|
echo $form->field($generator, 'modelClass'); |
||||
|
echo $form->field($generator, 'standardizeCapitals')->checkbox(); |
||||
|
echo $form->field($generator, 'singularize')->checkbox(); |
||||
|
echo $form->field($generator, 'ns'); |
||||
|
echo $form->field($generator, 'baseClass'); |
||||
|
echo $form->field($generator, 'db'); |
||||
|
echo $form->field($generator, 'useTablePrefix')->checkbox(); |
||||
|
echo $form->field($generator, 'generateRelations')->dropDownList([ |
||||
|
Generator::RELATIONS_NONE => 'No relations', |
||||
|
Generator::RELATIONS_ALL => 'All relations', |
||||
|
Generator::RELATIONS_ALL_INVERSE => 'All relations with inverse', |
||||
|
]); |
||||
|
echo $form->field($generator, 'generateRelationsFromCurrentSchema')->checkbox(); |
||||
|
echo $form->field($generator, 'generateLabelsFromComments')->checkbox(); |
||||
|
echo $form->field($generator, 'generateQuery')->checkbox(); |
||||
|
echo $form->field($generator, 'queryNs'); |
||||
|
echo $form->field($generator, 'queryClass'); |
||||
|
echo $form->field($generator, 'queryBaseClass'); |
||||
|
echo $form->field($generator, 'enableI18N')->checkbox(); |
||||
|
echo $form->field($generator, 'messageCategory'); |
||||
|
echo $form->field($generator, 'useSchemaName')->checkbox(); |
Write
Preview
Loading…
Cancel
Save
Reference in new issue