webman/plugin/admin/app/controller/TableController.php

1681 lines
60 KiB
PHP
Raw Permalink Normal View History

2025-02-15 12:13:10 +08:00
<?php
namespace plugin\admin\app\controller;
use Doctrine\Inflector\InflectorFactory;
use Illuminate\Database\Schema\Blueprint;
use plugin\admin\app\common\Layui;
use plugin\admin\app\common\Util;
use plugin\admin\app\model\Role;
use plugin\admin\app\model\Rule;
use plugin\admin\app\model\Option;
use support\exception\BusinessException;
use support\Request;
use support\Response;
use Throwable;
class TableController extends Base
{
/**
* 不需要鉴权的方法
* @var string[]
*/
protected $noNeedAuth = ['types'];
/**
* 浏览
* @return Response
* @throws Throwable
*/
public function index(): Response
{
return raw_view('table/index');
}
/**
* 查看表
* @param Request $request
* @return Response
* @throws BusinessException|Throwable
*/
public function view(Request $request): Response
{
$table = $request->get('table');
$table = Util::filterAlphaNum($table);
$form = Layui::buildForm($table, 'search');
$table_info = Util::getSchema($table, 'table');
$primary_key = $table_info['primary_key'][0] ?? null;
return raw_view("table/view", [
'form' => $form,
'table' => $table,
'primary_key' => $primary_key,
]);
}
/**
* 查询表
* @param Request $request
* @return Response
* @throws BusinessException
*/
public function show(Request $request): Response
{
$table_name = $request->get('table_name','');
$limit = (int)$request->get('limit', 10);
$page = (int)$request->get('page', 1);
$offset = ($page - 1) * $limit;
$database = config('database.connections')['plugin.admin.mysql']['database'];
$field = $request->get('field', 'TABLE_NAME');
$field = Util::filterAlphaNum($field);
$order = $request->get('order', 'asc');
$allow_column = ['TABLE_NAME', 'TABLE_COMMENT', 'ENGINE', 'TABLE_ROWS', 'CREATE_TIME', 'UPDATE_TIME', 'TABLE_COLLATION'];
if (!in_array($field, $allow_column)) {
$field = 'TABLE_NAME';
}
$order = $order === 'asc' ? 'asc' : 'desc';
$total = Util::db()->select("SELECT count(*)total FROM information_schema.`TABLES` WHERE TABLE_SCHEMA='$database' AND TABLE_NAME like '%{$table_name}%'")[0]->total ?? 0;
$tables = Util::db()->select("SELECT TABLE_NAME,TABLE_COMMENT,ENGINE,TABLE_ROWS,CREATE_TIME,UPDATE_TIME,TABLE_COLLATION FROM information_schema.`TABLES` WHERE TABLE_SCHEMA='$database' AND TABLE_NAME like '%{$table_name}%' order by $field $order limit $offset,$limit");
if ($tables) {
$table_names = array_column($tables, 'TABLE_NAME');
$table_rows_count = [];
foreach ($table_names as $table_name) {
$table_rows_count[$table_name] = Util::db()->table($table_name)->count();
}
foreach ($tables as $key => $table) {
$tables[$key]->TABLE_ROWS = $table_rows_count[$table->TABLE_NAME] ?? $table->TABLE_ROWS;
}
}
return json(['code' => 0, 'msg' => 'ok', 'count' => $total, 'data' => $tables]);
}
/**
* 创建表
* @param Request $request
* @return Response
* @throws BusinessException|Throwable
*/
public function create(Request $request): Response
{
if ($request->method() === 'GET') {
return raw_view("table/create", []);
}
$data = $request->post();
$table_name = Util::filterAlphaNum($data['table']);
$table_comment = Util::pdoQuote($data['table_comment']);
$columns = $data['columns'];
$forms = $data['forms'];
$keys = $data['keys'];
$primary_key_count = 0;
foreach ($columns as $index => $item) {
$columns[$index]['field'] = trim($item['field']);
if (!$item['field']) {
unset($columns[$index]);
continue;
}
$columns[$index]['primary_key'] = !empty($item['primary_key']);
if ($columns[$index]['primary_key']) {
$primary_key_count++;
}
$columns[$index]['auto_increment'] = !empty($item['auto_increment']);
$columns[$index]['nullable'] = !empty($item['nullable']);
if ($item['default'] === '') {
$columns[$index]['default'] = null;
} else if ($item['default'] === "''") {
$columns[$index]['default'] = '';
}
}
if ($primary_key_count > 1) {
throw new BusinessException('不支持复合主键');
}
foreach ($forms as $index => $item) {
if (!$item['field']) {
unset($forms[$index]);
continue;
}
$forms[$index]['form_show'] = !empty($item['form_show']);
$forms[$index]['list_show'] = !empty($item['list_show']);
$forms[$index]['enable_sort'] = !empty($item['enable_sort']);
$forms[$index]['searchable'] = !empty($item['searchable']);
}
foreach ($keys as $index => $item) {
if (!$item['name'] || !$item['columns']) {
unset($keys[$index]);
}
}
Util::schema()->create($table_name, function (Blueprint $table) use ($columns) {
$type_method_map = Util::methodControlMap();
foreach ($columns as $column) {
if (!isset($column['type'])) {
throw new BusinessException("请为{$column['field']}选择类型");
}
if (!isset($type_method_map[$column['type']])) {
throw new BusinessException("不支持的类型{$column['type']}");
}
$this->createColumn($column, $table);
}
$table->charset = 'utf8mb4';
$table->collation = 'utf8mb4_general_ci';
$table->engine = 'InnoDB';
});
Util::db()->statement("ALTER TABLE `$table_name` COMMENT $table_comment");
// 索引
Util::schema()->table($table_name, function (Blueprint $table) use ($keys) {
foreach ($keys as $key) {
$name = $key['name'];
$columns = is_array($key['columns']) ? $key['columns'] : explode(',', $key['columns']);
$type = $key['type'];
if ($type == 'unique') {
$table->unique($columns, $name);
continue;
}
$table->index($columns, $name);
}
});
$form_schema_map = [];
foreach ($forms as $item) {
$form_schema_map[$item['field']] = $item;
}
$form_schema_map = json_encode($form_schema_map, JSON_UNESCAPED_UNICODE);
$this->updateSchemaOption($table_name, $form_schema_map);
return $this->json(0, 'ok');
}
/**
* 修改表
* @param Request $request
* @return Response
* @throws BusinessException|Throwable
*/
public function modify(Request $request): Response
{
if ($request->method() === 'GET') {
return raw_view("table/modify", ['table' => $request->get('table')]);
}
$data = $request->post();
$old_table_name = Util::filterAlphaNum($data['old_table']);
$table_name = Util::filterAlphaNum($data['table']);
$table_comment = $data['table_comment'];
$columns = $data['columns'];
$forms = $data['forms'];
$keys = $data['keys'];
$primary_key = null;
$auto_increment_column = null;
$schema = Util::getSchema($old_table_name);
$old_columns = $schema['columns'];
$old_primary_key = $schema['table']['primary_key'][0] ?? null;
$primary_key_count = $auto_increment_count = 0;
foreach ($columns as $index => $item) {
$columns[$index]['field'] = trim($item['field']);
if (!$item['field']) {
unset($columns[$index]);
continue;
}
$field = $item['field'];
$columns[$index]['auto_increment'] = !empty($item['auto_increment']);
$columns[$index]['nullable'] = !empty($item['nullable']);
$columns[$index]['primary_key'] = !empty($item['primary_key']);
if ($columns[$index]['primary_key']) {
$primary_key = $item['field'];
$columns[$index]['nullable'] = false;
$primary_key_count++;
}
if ($item['default'] === '') {
$columns[$index]['default'] = null;
} else if ($item['default'] === "''") {
$columns[$index]['default'] = '';
}
if ($columns[$index]['auto_increment']) {
$auto_increment_count++;
if (!isset($old_columns[$field]) || !$old_columns[$field]['auto_increment']) {
$auto_increment_column = $columns[$index];
unset($auto_increment_column['old_field']);
$columns[$index]['auto_increment'] = false;
}
}
}
if ($primary_key_count > 1) {
throw new BusinessException('不支持复合主键');
}
if ($auto_increment_count > 1) {
throw new BusinessException('一个表只能有一个自增字段并且必须为key');
}
foreach ($forms as $index => $item) {
if (!$item['field']) {
unset($forms[$index]);
continue;
}
$forms[$index]['form_show'] = !empty($item['form_show']);
$forms[$index]['list_show'] = !empty($item['list_show']);
$forms[$index]['enable_sort'] = !empty($item['enable_sort']);
$forms[$index]['searchable'] = !empty($item['searchable']);
}
foreach ($keys as $index => $item) {
if (!$item['name'] || !$item['columns']) {
unset($keys[$index]);
}
}
// 改表名
if ($table_name != $old_table_name) {
Util::schema()->rename($old_table_name, $table_name);
}
$type_method_map = Util::methodControlMap();
foreach ($columns as $column) {
if (!isset($type_method_map[$column['type']])) {
throw new BusinessException("不支持的类型{$column['type']}");
}
$field = $column['old_field'] ?? $column['field'] ;
$old_column = $old_columns[$field] ?? [];
// 类型更改
foreach ($old_column as $key => $value) {
if (key_exists($key, $column) && ($column[$key] != $value || ($key === 'default' && $column[$key] !== $value))) {
$this->modifyColumn($column, $table_name);
break;
}
}
}
$table = Util::getSchema($table_name, 'table');
if ($table_comment !== $table['comment']) {
$table_comment = Util::pdoQuote($table_comment);
Util::db()->statement("ALTER TABLE `$table_name` COMMENT $table_comment");
}
$old_columns = Util::getSchema($table_name, 'columns');
Util::schema()->table($table_name, function (Blueprint $table) use ($columns, $old_columns, $keys, $table_name) {
foreach ($columns as $column) {
$field = $column['field'];
// 新字段
if (!isset($old_columns[$field])) {
$this->createColumn($column, $table);
}
}
// 更新索引名字
foreach ($keys as $key) {
if (!empty($key['old_name']) && $key['old_name'] !== $key['name']) {
$table->renameIndex($key['old_name'], $key['name']);
}
}
});
// 找到删除的字段
$old_columns = Util::getSchema($table_name, 'columns');
$exists_column_names = array_column($columns, 'field', 'field');
$old_columns_names = array_column($old_columns, 'field');
$drop_column_names = array_diff($old_columns_names, $exists_column_names);
$drop_column_names = Util::filterAlphaNum($drop_column_names);
foreach ($drop_column_names as $drop_column_name) {
Util::db()->statement("ALTER TABLE `$table_name` DROP COLUMN `$drop_column_name`");
}
$old_keys = Util::getSchema($table_name, 'keys');
Util::schema()->table($table_name, function (Blueprint $table) use ($keys, $old_keys, $table_name) {
foreach ($keys as $key) {
$key_name = $key['name'];
$old_key = $old_keys[$key_name] ?? [];
// 如果索引有变动,则删除索引,重新建立索引
if ($old_key && ($key['type'] != $old_key['type'] || $key['columns'] != implode(',', $old_key['columns']))) {
$old_key = [];
unset($old_keys[$key_name]);
echo "Drop Index $key_name\n";
$table->dropIndex($key_name);
}
// 重新建立索引
if (!$old_key) {
$name = $key['name'];
$columns = is_array($key['columns']) ? $key['columns'] : explode(',', $key['columns']);
$type = $key['type'];
if ($type == 'unique') {
$table->unique($columns, $name);
continue;
}
echo "Create Index $key_name\n";
$table->index($columns, $name);
}
}
// 找到删除的索引
$exists_key_names = array_column($keys, 'name', 'name');
$old_keys_names = array_column($old_keys, 'name');
$drop_keys_names = array_diff($old_keys_names, $exists_key_names);
foreach ($drop_keys_names as $name) {
echo "Drop Index $name\n";
$table->dropIndex($name);
}
});
// 变更主键
if ($old_primary_key != $primary_key) {
if ($old_primary_key) {
Util::db()->statement("ALTER TABLE `$table_name` DROP PRIMARY KEY");
}
if ($primary_key) {
$primary_key = Util::filterAlphaNum($primary_key);
Util::db()->statement("ALTER TABLE `$table_name` ADD PRIMARY KEY(`$primary_key`)");
}
}
// 一个表只能有一个 auto_increment 字段并且是key所以需要在最后设置
if ($auto_increment_column) {
$this->modifyColumn($auto_increment_column, $table_name);
}
$form_schema_map = [];
foreach ($forms as $item) {
$form_schema_map[$item['field']] = $item;
}
$form_schema_map = json_encode($form_schema_map, JSON_UNESCAPED_UNICODE);
$option_name = $this->updateSchemaOption($table_name, $form_schema_map);
return $this->json(0,$option_name);
}
/**
* 一键菜单
* @param Request $request
* @return Response
* @throws BusinessException|Throwable
*/
public function crud(Request $request): Response
{
$table_name = $request->input('table');
Util::checkTableName($table_name);
$prefix = 'wa_';
$table_basename = strpos($table_name, $prefix) === 0 ? substr($table_name, strlen($prefix)) : $table_name;
$inflector = InflectorFactory::create()->build();
$model_class = $inflector->classify($inflector->singularize($table_basename));
$base_path = '/plugin/admin/app';
if ($request->method() === 'GET') {
return raw_view("table/crud", [
'table' => $table_name,
'model' => "$base_path/model/$model_class.php",
'controller' => "$base_path/controller/{$model_class}Controller.php",
]);
}
$title = $request->post('title');
$pid = $request->post('pid', 0);
$icon = $request->post('icon', '');
$controller_file = '/' . trim($request->post('controller', ''), '/');
$model_file = '/' . trim($request->post('model', ''), '/');
$overwrite = $request->post('overwrite');
if ($controller_file === '/' || $model_file === '/') {
return $this->json(1, '控制器和model不能为空');
}
$controller_info = pathinfo($controller_file);
$model_info = pathinfo($model_file);
$controller_path = Util::filterPath($controller_info['dirname'] ?? '');
$model_path = Util::filterPath($model_info['dirname'] ?? '');
$controller_file_name = Util::filterAlphaNum($controller_info['filename'] ?? '');
$model_file_name = Util::filterAlphaNum($model_info['filename'] ?? '');
if ($controller_info['extension'] !== 'php' || $model_info['extension'] !== 'php' ) {
return $this->json(1, '控制器和model必须以.php为后缀');
}
$pid = (int)$pid;
if ($pid) {
$parent_menu = Rule::find($pid);
if (!$parent_menu) {
return $this->json(1, '父菜单不存在');
}
}
if (!$overwrite) {
if (is_file(base_path($controller_file))) {
return $this->json(1, "$controller_file 已经存在");
}
if (is_file(base_path($model_file))) {
return $this->json(1, "$model_file 已经存在");
}
}
$explode = explode('/', trim($controller_path, '/'));
$plugin = '';
if (strpos(strtolower($controller_file), '/controller/') === false) {
return $this->json(2, '控制器必须在controller目录下');
}
if ($explode[0] === 'plugin') {
if (count($explode) < 4) {
return $this->json(2, '控制器参数非法');
}
$plugin = $explode[1];
if (strtolower($explode[2]) !== 'app') {
return $this->json(2, '控制器必须在app目录');
}
$app = strtolower($explode[3]) !== 'controller' ? $explode[3] : '';
} else {
if (count($explode) < 2) {
return $this->json(3, '控制器参数非法');
}
if (strtolower($explode[0]) !== 'app') {
return $this->json(3, '控制器必须在app目录');
}
$app = strtolower($explode[1]) !== 'controller' ? $explode[1] : '';
}
Util::pauseFileMonitor();
try {
$model_class = $model_file_name;
$model_namespace = str_replace('/', '\\', trim($model_path, '/'));
// 创建model
$this->createModel($model_class, $model_namespace, base_path($model_file), $table_name);
$controller_suffix = $plugin ? config("plugin.$plugin.app.controller_suffix") : config('app.controller_suffix');
$controller_class = $controller_file_name;
$controller_namespace = str_replace('/', '\\', trim($controller_path, '/'));
// 创建controller
$controller_url_name = $controller_suffix && substr($controller_class, -strlen($controller_suffix)) === $controller_suffix ? substr($controller_class, 0, -strlen($controller_suffix)) : $controller_class;
$controller_url_name = str_replace('_', '-', $inflector->tableize($controller_url_name));
if ($plugin) {
array_splice($explode, 0, 2);
}
array_shift($explode);
if ($app) {
array_shift($explode);
}
foreach ($explode as $index => $item) {
if (strtolower($item) === 'controller') {
unset($explode[$index]);
}
}
$controller_base = implode('/', $explode);
$controller_class_with_namespace = "$controller_namespace\\$controller_class";
$template_path = $controller_base ? "$controller_base/$controller_url_name" : $controller_url_name;
$this->createController($controller_class, $controller_namespace, base_path($controller_file), $model_class, $model_namespace, $title, $template_path);
// 创建模版
$template_file_path = ($plugin ? "/plugin/$plugin" : '') . '/app/' . ($app ? "$app/" : '') . 'view/' . $template_path;
$model_class_with_namespace = "$model_namespace\\$model_class";
$primary_key = (new $model_class_with_namespace)->getKeyName();
$url_path_base = ($plugin ? "/app/$plugin/" : '/') . ($app ? "$app/" : '') . $template_path;
$this->createTemplate(base_path($template_file_path), $table_name, $url_path_base, $primary_key, "$controller_namespace\\$controller_class");
} finally {
Util::resumeFileMonitor();
}
$menu = Rule::where('key', $controller_class_with_namespace)->first();
if (!$menu) {
$menu = new Rule;
}
$menu->pid = $pid;
$menu->key = $controller_class_with_namespace;
$menu->title = $title;
$menu->icon = $icon;
$menu->href = "$url_path_base/index";
$menu->save();
$roles = admin('roles');
$rules = Role::whereIn('id', $roles)->pluck('rules');
$rule_ids = [];
foreach ($rules as $rule_string) {
if (!$rule_string) {
continue;
}
$rule_ids = array_merge($rule_ids, explode(',', $rule_string));
}
// 不是超级管理员,则需要给当前管理员这个菜单的权限
if (!in_array('*', $rule_ids) && $roles){
$role = Role::find(current($roles));
if ($role) {
$role->rules .= ",{$menu->id}";
}
$role->save();
}
return $this->json(0);
}
/**
* 创建model
* @param $class
* @param $namespace
* @param $file
* @param $table
* @return void
*/
protected function createModel($class, $namespace, $file, $table)
{
$this->mkdir($file);
$table_val = "'$table'";
$pk = 'id';
$properties = '';
$timestamps = '';
$incrementing = '';
$columns = [];
try {
$database = config('database.connections')['plugin.admin.mysql']['database'];
//plugin.admin.mysql
foreach (Util::db()->select("select COLUMN_NAME,DATA_TYPE,COLUMN_KEY,COLUMN_COMMENT from INFORMATION_SCHEMA.COLUMNS where table_name = '$table' and table_schema = '$database' order by ORDINAL_POSITION") as $item) {
if ($item->COLUMN_KEY === 'PRI') {
$pk = $item->COLUMN_NAME;
$item->COLUMN_COMMENT .= "(主键)";
if (strpos(strtolower($item->DATA_TYPE), 'int') === false) {
$incrementing = <<<EOF
/**
* Indicates if the model's ID is auto-incrementing.
*
* @var bool
*/
public \$incrementing = false;
EOF;
}
}
$type = $this->getType($item->DATA_TYPE);
$properties .= " * @property $type \${$item->COLUMN_NAME} {$item->COLUMN_COMMENT}\n";
$columns[$item->COLUMN_NAME] = $item->COLUMN_NAME;
}
} catch (Throwable $e) {echo $e;}
if (!isset($columns['created_at']) || !isset($columns['updated_at'])) {
$timestamps = <<<EOF
/**
* Indicates if the model should be timestamped.
*
* @var bool
*/
public \$timestamps = false;
EOF;
}
$properties = rtrim($properties) ?: ' *';
$model_content = <<<EOF
<?php
namespace $namespace;
use plugin\admin\app\model\Base;
/**
$properties
*/
class $class extends Base
{
/**
* The table associated with the model.
*
* @var string
*/
protected \$table = $table_val;
/**
* The primary key associated with the table.
*
* @var string
*/
protected \$primaryKey = '$pk';
$timestamps
$incrementing
}
EOF;
file_put_contents($file, $model_content);
}
/**
* 创建控制器
* @param $controller_class
* @param $namespace
* @param $file
* @param $model_class
* @param $model_namespace
* @param $name
* @param $template_path
* @return void
*/
protected function createController($controller_class, $namespace, $file, $model_class, $model_namespace, $name, $template_path)
{
$model_class_alias = $model_class;
if (strtolower($model_class) === strtolower($controller_class)) {
$model_class_alias = "$model_class as {$model_class}Model";
$model_class = "{$model_class}Model";
}
$this->mkdir($file);
$controller_content = <<<EOF
<?php
namespace $namespace;
use support\Request;
use support\Response;
use $model_namespace\\$model_class_alias;
use plugin\admin\app\controller\Crud;
use support\\exception\BusinessException;
/**
* $name
*/
class $controller_class extends Crud
{
/**
* @var $model_class
*/
protected \$model = null;
/**
* 构造函数
* @return void
*/
public function __construct()
{
\$this->model = new $model_class;
}
/**
* 浏览
* @return Response
*/
public function index(): Response
{
return view('$template_path/index');
}
/**
* 插入
* @param Request \$request
* @return Response
* @throws BusinessException
*/
public function insert(Request \$request): Response
{
if (\$request->method() === 'POST') {
return parent::insert(\$request);
}
return view('$template_path/insert');
}
/**
* 更新
* @param Request \$request
* @return Response
* @throws BusinessException
*/
public function update(Request \$request): Response
{
if (\$request->method() === 'POST') {
return parent::update(\$request);
}
return view('$template_path/update');
}
}
EOF;
file_put_contents($file, $controller_content);
}
/**
* 创建控制器
* @param $template_file_path
* @param $table
* @param $template_path
* @param $url_path_base
* @param $primary_key
* @param $controller_class_with_namespace
* @return void
*/
protected function createTemplate($template_file_path, $table, $url_path_base, $primary_key, $controller_class_with_namespace)
{
$this->mkdir($template_file_path . '/index.html');
$code_base = Util::controllerToUrlPath($controller_class_with_namespace);
$code_base = str_replace('/', '.', trim($code_base, '/'));
$form = Layui::buildForm($table, 'search');
$html = $form->html(3);
$html = $html ? <<<EOF
<div class="layui-card">
<div class="layui-card-body">
<form class="layui-form top-search-from">
$html
<div class="layui-form-item layui-inline">
<label class="layui-form-label"></label>
<button class="pear-btn pear-btn-md pear-btn-primary" lay-submit lay-filter="table-query">
<i class="layui-icon layui-icon-search"></i>查询
</button>
<button type="reset" class="pear-btn pear-btn-md" lay-submit lay-filter="table-reset">
<i class="layui-icon layui-icon-refresh"></i>重置
</button>
</div>
<div class="toggle-btn">
<a class="layui-hide">展开<i class="layui-icon layui-icon-down"></i></a>
<a class="layui-hide">收起<i class="layui-icon layui-icon-up"></i></a>
</div>
</form>
</div>
</div>
EOF
: '';
$html = str_replace("\n", "\n" . str_repeat(' ', 2), $html);
$js = $form->js(3);
$table_js = Layui::buildTable($table, 4);
$template_content = <<<EOF
<!DOCTYPE html>
<html lang="zh-cn">
<head>
<meta charset="utf-8">
<title>浏览页面</title>
<link rel="stylesheet" href="/app/admin/component/pear/css/pear.css" />
<link rel="stylesheet" href="/app/admin/admin/css/reset.css" />
</head>
<body class="pear-container">
<!-- 顶部查询表单 -->
$html
<!-- 数据表格 -->
<div class="layui-card">
<div class="layui-card-body">
<table id="data-table" lay-filter="data-table"></table>
</div>
</div>
<!-- 表格顶部工具栏 -->
<script type="text/html" id="table-toolbar">
<button class="pear-btn pear-btn-primary pear-btn-md" lay-event="add" permission="$code_base.insert">
<i class="layui-icon layui-icon-add-1"></i>新增
</button>
<button class="pear-btn pear-btn-danger pear-btn-md" lay-event="batchRemove" permission="$code_base.delete">
<i class="layui-icon layui-icon-delete"></i>删除
</button>
</script>
<!-- 表格行工具栏 -->
<script type="text/html" id="table-bar">
<button class="pear-btn pear-btn-xs tool-btn" lay-event="edit" permission="$code_base.update">编辑</button>
<button class="pear-btn pear-btn-xs tool-btn" lay-event="remove" permission="$code_base.delete">删除</button>
</script>
<script src="/app/admin/component/layui/layui.js?v=2.8.12"></script>
<script src="/app/admin/component/pear/pear.js"></script>
<script src="/app/admin/admin/js/permission.js"></script>
<script src="/app/admin/admin/js/common.js"></script>
<script>
// 相关常量
const PRIMARY_KEY = "$primary_key";
const SELECT_API = "$url_path_base/select";
const UPDATE_API = "$url_path_base/update";
const DELETE_API = "$url_path_base/delete";
const INSERT_URL = "$url_path_base/insert";
const UPDATE_URL = "$url_path_base/update";
$js
// 表格渲染
layui.use(["table", "form", "common", "popup", "util"], function() {
let table = layui.table;
let form = layui.form;
let $ = layui.$;
let common = layui.common;
let util = layui.util;
$table_js
// 编辑或删除行事件
table.on("tool(data-table)", function(obj) {
if (obj.event === "remove") {
remove(obj);
} else if (obj.event === "edit") {
edit(obj);
}
});
// 表格顶部工具栏事件
table.on("toolbar(data-table)", function(obj) {
if (obj.event === "add") {
add();
} else if (obj.event === "refresh") {
refreshTable();
} else if (obj.event === "batchRemove") {
batchRemove(obj);
}
});
// 表格顶部搜索事件
form.on("submit(table-query)", function(data) {
table.reload("data-table", {
page: {
curr: 1
},
where: data.field
})
return false;
});
// 表格顶部搜索重置事件
form.on("submit(table-reset)", function(data) {
table.reload("data-table", {
where: []
})
});
// 字段允许为空
form.verify({
phone: [/(^$)|^1\d{10}$/, "请输入正确的手机号"],
email: [/(^$)|^([a-zA-Z0-9_\.\-])+\@(([a-zA-Z0-9\-])+\.)+([a-zA-Z0-9]{2,4})+$/, "邮箱格式不正确"],
url: [/(^$)|(^#)|(^http(s*):\/\/[^\s]+\.[^\s]+)/, "链接格式不正确"],
number: [/(^$)|^\d+$/,'只能填写数字'],
date: [/(^$)|^(\d{4})[-\/](\d{1}|0\d{1}|1[0-2])([-\/](\d{1}|0\d{1}|[1-2][0-9]|3[0-1]))*$/, "日期格式不正确"],
identity: [/(^$)|(^\d{15}$)|(^\d{17}(x|X|\d)$)/, "请输入正确的身份证号"]
});
// 表格排序事件
table.on("sort(data-table)", function(obj){
table.reload("data-table", {
initSort: obj,
scrollPos: "fixed",
where: {
field: obj.field,
order: obj.type
}
});
});
// 表格新增数据
let add = function() {
layer.open({
type: 2,
title: "新增",
shade: 0.1,
maxmin: true,
area: [common.isModile()?"100%":"500px", common.isModile()?"100%":"450px"],
content: INSERT_URL
});
}
// 表格编辑数据
let edit = function(obj) {
let value = obj.data[PRIMARY_KEY];
layer.open({
type: 2,
title: "修改",
shade: 0.1,
maxmin: true,
area: [common.isModile()?"100%":"500px", common.isModile()?"100%":"450px"],
content: UPDATE_URL + "?" + PRIMARY_KEY + "=" + value
});
}
// 删除一行
let remove = function(obj) {
return doRemove(obj.data[PRIMARY_KEY]);
}
// 删除多行
let batchRemove = function(obj) {
let checkIds = common.checkField(obj, PRIMARY_KEY);
if (checkIds === "") {
layui.popup.warning("未选中数据");
return false;
}
doRemove(checkIds.split(","));
}
// 执行删除
let doRemove = function (ids) {
let data = {};
data[PRIMARY_KEY] = ids;
layer.confirm("确定删除?", {
icon: 3,
title: "提示"
}, function(index) {
layer.close(index);
let loading = layer.load();
$.ajax({
url: DELETE_API,
data: data,
dataType: "json",
type: "post",
success: function(res) {
layer.close(loading);
if (res.code) {
return layui.popup.failure(res.msg);
}
return layui.popup.success("操作成功", refreshTable);
}
})
});
}
// 刷新表格数据
window.refreshTable = function() {
table.reloadData("data-table", {
scrollPos: "fixed",
done: function (res, curr) {
if (curr > 1 && res.data && !res.data.length) {
curr = curr - 1;
table.reloadData("data-table", {
page: {
curr: curr
},
})
}
}
});
}
})
</script>
</body>
</html>
EOF;
file_put_contents("$template_file_path/index.html", $template_content);
$form = Layui::buildForm($table);
$html = $form->html(5);
$js = $form->js(3);
$template_content = <<<EOF
<!DOCTYPE html>
<html lang="zh-cn">
<head>
<meta charset="UTF-8">
<title>新增页面</title>
<link rel="stylesheet" href="/app/admin/component/pear/css/pear.css" />
<link rel="stylesheet" href="/app/admin/component/jsoneditor/css/jsoneditor.css" />
<link rel="stylesheet" href="/app/admin/admin/css/reset.css" />
</head>
<body>
<form class="layui-form" action="">
<div class="mainBox">
<div class="main-container mr-5">
$html
</div>
</div>
<div class="bottom">
<div class="button-container">
<button type="submit" class="pear-btn pear-btn-primary pear-btn-md" lay-submit=""
lay-filter="save">
提交
</button>
<button type="reset" class="pear-btn pear-btn-md">
重置
</button>
</div>
</div>
</form>
<script src="/app/admin/component/layui/layui.js?v=2.8.12"></script>
<script src="/app/admin/component/pear/pear.js"></script>
<script src="/app/admin/component/jsoneditor/jsoneditor.js"></script>
<script src="/app/admin/admin/js/permission.js"></script>
<script>
// 相关接口
const INSERT_API = "$url_path_base/insert";
$js
//提交事件
layui.use(["form", "popup"], function () {
// 字段验证允许为空
layui.form.verify({
phone: [/(^$)|^1\d{10}$/, "请输入正确的手机号"],
email: [/(^$)|^([a-zA-Z0-9_\.\-])+\@(([a-zA-Z0-9\-])+\.)+([a-zA-Z0-9]{2,4})+$/, "邮箱格式不正确"],
url: [/(^$)|(^#)|(^http(s*):\/\/[^\s]+\.[^\s]+)/, "链接格式不正确"],
number: [/(^$)|^\d+$/,'只能填写数字'],
date: [/(^$)|^(\d{4})[-\/](\d{1}|0\d{1}|1[0-2])([-\/](\d{1}|0\d{1}|[1-2][0-9]|3[0-1]))*$/, "日期格式不正确"],
identity: [/(^$)|(^\d{15}$)|(^\d{17}(x|X|\d)$)/, "请输入正确的身份证号"]
});
layui.form.on("submit(save)", function (data) {
layui.$.ajax({
url: INSERT_API,
type: "POST",
dateType: "json",
data: data.field,
success: function (res) {
if (res.code) {
return layui.popup.failure(res.msg);
}
return layui.popup.success("操作成功", function () {
parent.refreshTable();
parent.layer.close(parent.layer.getFrameIndex(window.name));
});
}
});
return false;
});
});
</script>
</body>
</html>
EOF;
file_put_contents("$template_file_path/insert.html", $template_content);
$form = Layui::buildForm($table, 'update');
$html = $form->html(5);
$js = $form->js(6);
$template_content = <<<EOF
<!DOCTYPE html>
<html lang="zh-cn">
<head>
<meta charset="UTF-8">
<title>更新页面</title>
<link rel="stylesheet" href="/app/admin/component/pear/css/pear.css" />
<link rel="stylesheet" href="/app/admin/component/jsoneditor/css/jsoneditor.css" />
<link rel="stylesheet" href="/app/admin/admin/css/reset.css" />
</head>
<body>
<form class="layui-form">
<div class="mainBox">
<div class="main-container mr-5">
$html
</div>
</div>
<div class="bottom">
<div class="button-container">
<button type="submit" class="pear-btn pear-btn-primary pear-btn-md" lay-submit="" lay-filter="save">
提交
</button>
<button type="reset" class="pear-btn pear-btn-md">
重置
</button>
</div>
</div>
</form>
<script src="/app/admin/component/layui/layui.js?v=2.8.12"></script>
<script src="/app/admin/component/pear/pear.js"></script>
<script src="/app/admin/component/jsoneditor/jsoneditor.js"></script>
<script src="/app/admin/admin/js/permission.js"></script>
<script>
// 相关接口
const PRIMARY_KEY = "$primary_key";
const SELECT_API = "$url_path_base/select" + location.search;
const UPDATE_API = "$url_path_base/update";
// 获取数据库记录
layui.use(["form", "util", "popup"], function () {
let $ = layui.$;
$.ajax({
url: SELECT_API,
dataType: "json",
success: function (res) {
// 给表单初始化数据
layui.each(res.data[0], function (key, value) {
let obj = $('*[name="'+key+'"]');
if (key === "password") {
obj.attr("placeholder", "不更新密码请留空");
return;
}
if (typeof obj[0] === "undefined" || !obj[0].nodeName) return;
if (obj[0].nodeName.toLowerCase() === "textarea") {
obj.val(value);
} else {
obj.attr("value", value);
obj[0].value = value;
}
});
$js
// ajax返回失败
if (res.code) {
layui.popup.failure(res.msg);
}
}
});
});
//提交事件
layui.use(["form", "popup"], function () {
// 字段验证允许为空
layui.form.verify({
phone: [/(^$)|^1\d{10}$/, "请输入正确的手机号"],
email: [/(^$)|^([a-zA-Z0-9_\.\-])+\@(([a-zA-Z0-9\-])+\.)+([a-zA-Z0-9]{2,4})+$/, "邮箱格式不正确"],
url: [/(^$)|(^#)|(^http(s*):\/\/[^\s]+\.[^\s]+)/, "链接格式不正确"],
number: [/(^$)|^\d+$/,'只能填写数字'],
date: [/(^$)|^(\d{4})[-\/](\d{1}|0\d{1}|1[0-2])([-\/](\d{1}|0\d{1}|[1-2][0-9]|3[0-1]))*$/, "日期格式不正确"],
identity: [/(^$)|(^\d{15}$)|(^\d{17}(x|X|\d)$)/, "请输入正确的身份证号"]
});
layui.form.on("submit(save)", function (data) {
data.field[PRIMARY_KEY] = layui.url().search[PRIMARY_KEY];
layui.$.ajax({
url: UPDATE_API,
type: "POST",
dateType: "json",
data: data.field,
success: function (res) {
if (res.code) {
return layui.popup.failure(res.msg);
}
return layui.popup.success("操作成功", function () {
parent.refreshTable();
parent.layer.close(parent.layer.getFrameIndex(window.name));
});
}
});
return false;
});
});
</script>
</body>
</html>
EOF;
file_put_contents("$template_file_path/update.html", $template_content);
}
/**
* 创建目录
* @param $file
* @return void
*/
protected function mkdir($file)
{
$path = pathinfo($file, PATHINFO_DIRNAME);
if (!is_dir($path)) {
mkdir($path, 0777, true);
}
}
/**
* 查询记录
* @param Request $request
* @return Response
* @throws BusinessException
*/
public function select(Request $request): Response
{
$page = $request->get('page', 1);
$field = $request->get('field');
$order = $request->get('order', 'asc');
$table = Util::filterAlphaNum($request->get('table', ''));
$format = $request->get('format', 'normal');
$limit = $request->get('limit', $format === 'tree' ? 5000 : 10);
$allow_column = Util::db()->select("desc `$table`");
if (!$allow_column) {
return $this->json(2, '表不存在');
}
$allow_column = array_column($allow_column, 'Field', 'Field');
if (!in_array($field, $allow_column)) {
$field = current($allow_column);
}
$order = $order === 'asc' ? 'asc' : 'desc';
$paginator = Util::db()->table($table);
foreach ($request->get() as $column => $value) {
if ($value === '') {
continue;
}
if (isset($allow_column[$column])) {
if (is_array($value)) {
if ($value[0] === 'like') {
$paginator = $paginator->where($column, 'like', "%$value[1]%");
} elseif (in_array($value[0], ['>', '=', '<', '<>', 'not like'])) {
$paginator = $paginator->where($column, $value[0], $value[1]);
} else {
if($value[0] !== '' || $value[1] !== '') {
$paginator = $paginator->whereBetween($column, $value);
}
}
} else {
$paginator = $paginator->where($column, $value);
}
}
}
$paginator = $paginator->orderBy($field, $order)->paginate($limit, '*', 'page', $page);
$items = $paginator->items();
if ($format == 'tree') {
$items_map = [];
foreach ($items as $item) {
$items_map[$item->id] = (array)$item;
}
$formatted_items = [];
foreach ($items_map as $index => $item) {
if ($item['pid'] && isset($items_map[$item['pid']])) {
$items_map[$item['pid']]['children'][] = &$items_map[$index];
}
}
foreach ($items_map as $item) {
if (!$item['pid']) {
$formatted_items[] = $item;
}
}
$items = $formatted_items;
}
return json(['code' => 0, 'msg' => 'ok', 'count' => $paginator->total(), 'data' => $items]);
}
/**
* 插入记录
* @param Request $request
* @return Response
* @throws BusinessException|Throwable
*/
public function insert(Request $request): Response
{
if ($request->method() === 'GET') {
$table = $request->get('table');
$form = Layui::buildForm($table);
return raw_view("table/insert", [
'form' => $form,
'table' => $table
]);
}
$table = Util::filterAlphaNum($request->input('table', ''));
$data = $request->post();
$allow_column = Util::db()->select("desc `$table`");
if (!$allow_column) {
throw new BusinessException('表不存在', 2);
}
$columns = array_column($allow_column, 'Type', 'Field');
foreach ($data as $col => $item) {
if (!isset($columns[$col])) {
unset($data[$col]);
continue;
}
// 非字符串类型传空则为null
if ($item === '' && strpos(strtolower($columns[$col]), 'varchar') === false && strpos(strtolower($columns[$col]), 'text') === false) {
$data[$col] = null;
}
if (is_array($item)) {
$data[$col] = implode(',', $item);
continue;
}
if ($col === 'password') {
$data[$col] = Util::passwordHash($item);
}
}
$datetime = date('Y-m-d H:i:s');
if (isset($columns['created_at']) && empty($data['created_at'])) {
$data['created_at'] = $datetime;
}
if (isset($columns['updated_at']) && empty($data['updated_at'])) {
$data['updated_at'] = $datetime;
}
$id = Util::db()->table($table)->insertGetId($data);
return $this->json(0, $id);
}
/**
* 更新记录
* @param Request $request
* @return Response
* @throws BusinessException|Throwable
*/
public function update(Request $request): Response
{
if ($request->method() === 'GET') {
$table = $request->get('table');
$table_info = Util::getSchema($table, 'table');
$primary_key = $table_info['primary_key'][0] ?? null;
$value = htmlspecialchars($request->get($primary_key, ''));
$form = Layui::buildForm($table,'update');
return raw_view("table/update", [
'primary_key' => $primary_key,
'value' => $value,
'form' => $form,
'table' => $table
]);
}
$table = Util::filterAlphaNum($request->post('table'));
$table_info = Util::getSchema($table, 'table');
$primary_keys = $table_info['primary_key'];
if (empty($primary_keys)) {
return $this->json(1, '该表没有主键,无法执行更新操作');
}
if (count($primary_keys) > 1) {
return $this->json(1, '不支持复合主键更新');
}
$primary_key = $primary_keys[0];
$value = $request->post($primary_key);
$data = $request->post();
$allow_column = Util::db()->select("desc `$table`");
if (!$allow_column) {
throw new BusinessException('表不存在', 2);
}
$columns = array_column($allow_column, 'Type', 'Field');
foreach ($data as $col => $item) {
if (!isset($columns[$col])) {
unset($data[$col]);
continue;
}
// 非字符串类型传空则为null
if ($item === '' && strpos(strtolower($columns[$col]), 'varchar') === false && strpos(strtolower($columns[$col]), 'text') === false) {
$data[$col] = null;
}
if (is_array($item)) {
$data[$col] = implode(',', $item);
}
if ($col === 'password') {
// 密码为空,则不更新密码
if ($item == '') {
unset($data[$col]);
continue;
}
$data[$col] = Util::passwordHash($item);
}
}
$datetime = date('Y-m-d H:i:s');
if (isset($columns['updated_at']) && empty($data['updated_at'])) {
$data['updated_at'] = $datetime;
}
Util::db()->table($table)->where($primary_key, $value)->update($data);
return $this->json(0);
}
/**
* 删除记录
* @param Request $request
* @return Response
* @throws BusinessException
*/
public function delete(Request $request): Response
{
$table = $request->post('table');
$table_info = Util::getSchema($table, 'table');
$primary_keys = $table_info['primary_key'];
if (empty($primary_keys)) {
return $this->json(1, '该表没有主键,无法执行删除操作');
}
if (count($primary_keys) > 1) {
return $this->json(1, '不支持复合主键删除');
}
$primary_key = $primary_keys[0];
$value = (array)$request->post($primary_key);
Util::db()->table($table)->whereIn($primary_key, $value)->delete();
return $this->json(0);
}
/**
* 删除表
* @param Request $request
* @return Response
*/
public function drop(Request $request): Response
{
$tables = $request->post('tables');
if (!$tables) {
return $this->json(0, 'not found');
}
$prefix = 'wa_';
$table_not_allow_drop = ["{$prefix}admins", "{$prefix}users", "{$prefix}options", "{$prefix}roles", "{$prefix}rules", "{$prefix}admin_roles", "{$prefix}uploads"];
if ($found = array_intersect($tables, $table_not_allow_drop)) {
return $this->json(400, implode(',', $found) . '不允许删除');
}
foreach ($tables as $table) {
Util::schema()->drop($table);
// 删除schema
Util::db()->table('wa_options')->where('name', "table_form_schema_$table")->delete();
}
return $this->json(0, 'ok');
}
/**
* 表摘要
* @param Request $request
* @return Response
*/
public function schema(Request $request): Response
{
$table = $request->get('table');
$data = Util::getSchema($table);
return $this->json(0, 'ok', [
'table' => $data['table'],
'columns' => array_values($data['columns']),
'forms' => array_values($data['forms']),
'keys' => array_values($data['keys']),
]);
}
/**
* 创建字段
* @param $column
* @param Blueprint $table
* @return mixed
*/
protected function createColumn($column, Blueprint $table)
{
$method = $column['type'];
$args = [$column['field']];
if (stripos($method, 'int') !== false) {
// auto_increment 会自动成为主键
if ($column['auto_increment']) {
$column['nullable'] = false;
$column['default'] = null;
$args[] = true;
}
} elseif (in_array($method, ['string', 'char']) || stripos($method, 'time') !== false) {
if ($column['length']) {
$args[] = $column['length'];
}
} elseif ($method === 'enum') {
$args[] = array_map('trim', explode(',', $column['length']));
} elseif (in_array($method, ['float', 'decimal', 'double'])) {
if ($column['length']) {
$args = array_merge($args, array_map('trim', explode(',', $column['length'])));
}
} else {
$column['auto_increment'] = false;
}
$column_def = call_user_func_array([$table, $method], $args);
if (!empty($column['comment'])) {
$column_def = $column_def->comment($column['comment']);
}
if (!$column['auto_increment'] && $column['primary_key']) {
$column_def = $column_def->primary(true);
}
if ($column['auto_increment'] && !$column['primary_key']) {
$column_def = $column_def->primary(false);
}
$column_def = $column_def->nullable($column['nullable']);
if ($column['primary_key']) {
$column_def = $column_def->nullable(false);
}
if ($method != 'text' && $column['default'] !== null) {
$column_def->default($column['default']);
}
return $column_def;
}
/**
* 更改字段
* @param $column
* @param $table
* @return mixed
* @throws BusinessException
*/
protected function modifyColumn($column, $table)
{
$table = Util::filterAlphaNum($table);
$method = Util::filterAlphaNum($column['type']);
$field = Util::filterAlphaNum($column['field']);
$old_field = Util::filterAlphaNum($column['old_field'] ?? null);
$nullable = $column['nullable'];
$default = $column['default'] !== null ? Util::pdoQuote($column['default']) : null;
$comment = Util::pdoQuote($column['comment']);
$auto_increment = $column['auto_increment'];
$length = (int)$column['length'];
if ($column['primary_key']) {
$default = null;
}
if ($old_field && $old_field !== $field) {
$sql = "ALTER TABLE `$table` CHANGE COLUMN `$old_field` `$field` ";
} else {
$sql = "ALTER TABLE `$table` MODIFY `$field` ";
}
if (stripos($method, 'integer') !== false) {
$type = str_ireplace('integer', 'int', $method);
if (stripos($method, 'unsigned') !== false) {
$type = str_ireplace('unsigned', '', $type);
$sql .= "$type ";
$sql .= 'unsigned ';
} else {
$sql .= "$type ";
}
if ($auto_increment) {
$column['nullable'] = false;
$column['default'] = null;
$sql .= 'AUTO_INCREMENT ';
}
} else {
switch ($method) {
case 'string':
$length = $length ?: 255;
$sql .= "varchar($length) ";
break;
case 'char':
case 'time':
$sql .= $length ? "$method($length) " : "$method ";
break;
case 'enum':
$args = array_map('trim', explode(',', (string)$column['length']));
foreach ($args as $key => $value) {
$args[$key] = Util::pdoQuote($value);
}
$sql .= 'enum(' . implode(',', $args) . ') ';
break;
case 'double':
case 'float':
case 'decimal':
if (trim($column['length'])) {
$args = array_map('intval', explode(',', $column['length']));
$args[1] = $args[1] ?? $args[0];
$sql .= "$method($args[0], $args[1]) ";
break;
}
$sql .= "$method ";
break;
default :
$sql .= "$method ";
}
}
if (!$nullable) {
$sql .= "NOT NULL ";
}
if ($method != 'text' && $default !== null) {
$sql .= "DEFAULT $default ";
}
if ($comment !== null) {
$sql .= "COMMENT $comment ";
}
echo "$sql\n";
Util::db()->statement($sql);
}
/**
* 字段类型列表
* @param Request $request
* @return Response
*/
public function types(Request $request): Response
{
$types = Util::methodControlMap();
return $this->json(0, 'ok', $types);
}
/**
* 更新表的form schema信息
* @param $table_name
* @param $data
* @return string
*/
protected function updateSchemaOption($table_name, $data): string
{
$option_name = "table_form_schema_$table_name";
$option = Option::where('name', $option_name)->first();
if ($option) {
Option::where('name', $option_name)->update(['value' => $data]);
} else {
Option::insert(['name' => $option_name, 'value' => $data]);
}
return $option_name;
}
/**
* 字段类型到php类型映射
* @param string $type
* @return string
*/
protected function getType(string $type): string
{
if (strpos($type, 'int') !== false) {
return 'integer';
}
switch ($type) {
case 'varchar':
case 'string':
case 'text':
case 'date':
case 'time':
case 'guid':
case 'datetimetz':
case 'datetime':
case 'decimal':
case 'enum':
return 'string';
case 'boolean':
return 'integer';
case 'float':
return 'float';
default:
return 'mixed';
}
}
}