mirror of
https://gitee.com/zoujingli/ThinkAdmin.git
synced 2025-04-05 05:52:43 +08:00
同步插件代码
This commit is contained in:
parent
a6afa20850
commit
15d33afdd2
3
.gitignore
vendored
3
.gitignore
vendored
@ -29,6 +29,7 @@
|
||||
|
||||
/database/migrations/20221013031925_install_admin.php
|
||||
/database/migrations/20221013031926_install_admin_data.php
|
||||
/database/migrations/20221013031927_install_admin20230325.php
|
||||
/database/migrations/20221013045829_install_wechat.php
|
||||
/database/migrations/20221013045830_install_wechat_data.php
|
||||
/database/migrations/20221013045838_install_user.php
|
||||
@ -37,4 +38,4 @@
|
||||
/database/migrations/20221215000001_install_center_data.php
|
||||
/database/migrations/20230209000001_install_wechat_service.php
|
||||
/database/migrations/20230224000001_install_account.php
|
||||
database/migrations/20230225000001_install_payment.php
|
||||
database/migrations/20230225000001_install_payment.php
|
||||
|
@ -20,7 +20,7 @@ use think\admin\Plugin;
|
||||
|
||||
/**
|
||||
* 插件服务注册
|
||||
* Class Service
|
||||
* @class Service
|
||||
* @package app\admin
|
||||
*/
|
||||
class Service extends Plugin
|
||||
|
@ -24,7 +24,7 @@ use think\admin\service\AdminService;
|
||||
|
||||
/**
|
||||
* 系统权限管理
|
||||
* Class Auth
|
||||
* @class Auth
|
||||
* @package app\admin\controller
|
||||
*/
|
||||
class Auth extends Controller
|
||||
|
@ -22,7 +22,7 @@ use think\admin\model\SystemBase;
|
||||
|
||||
/**
|
||||
* 数据字典管理
|
||||
* Class Base
|
||||
* @class Base
|
||||
* @package app\admin\controller
|
||||
*/
|
||||
class Base extends Controller
|
||||
|
@ -27,7 +27,7 @@ use think\admin\storage\TxcosStorage;
|
||||
|
||||
/**
|
||||
* 系统参数配置
|
||||
* Class Config
|
||||
* @class Config
|
||||
* @package app\admin\controller
|
||||
*/
|
||||
class Config extends Controller
|
||||
|
@ -24,7 +24,7 @@ use think\admin\Storage;
|
||||
|
||||
/**
|
||||
* 系统文件管理
|
||||
* Class File
|
||||
* @class File
|
||||
* @package app\admin\controller
|
||||
*/
|
||||
class File extends Controller
|
||||
@ -99,6 +99,7 @@ class File extends Controller
|
||||
* 清理重复文件
|
||||
* @auth true
|
||||
* @return void
|
||||
* @throws \think\db\exception\DbException
|
||||
*/
|
||||
public function distinct()
|
||||
{
|
||||
|
@ -23,7 +23,7 @@ use think\admin\service\MenuService;
|
||||
|
||||
/**
|
||||
* 后台界面入口
|
||||
* Class Index
|
||||
* @class Index
|
||||
* @package app\admin\controller
|
||||
*/
|
||||
class Index extends Controller
|
||||
@ -31,6 +31,7 @@ class Index extends Controller
|
||||
/**
|
||||
* 显示后台首页
|
||||
* @throws \ReflectionException
|
||||
* @throws \think\admin\Exception
|
||||
* @throws \think\db\exception\DataNotFoundException
|
||||
* @throws \think\db\exception\DbException
|
||||
* @throws \think\db\exception\ModelNotFoundException
|
||||
@ -58,9 +59,7 @@ class Index extends Controller
|
||||
* 后台主题切换
|
||||
* @login true
|
||||
* @return void
|
||||
* @throws \think\db\exception\DataNotFoundException
|
||||
* @throws \think\db\exception\DbException
|
||||
* @throws \think\db\exception\ModelNotFoundException
|
||||
* @throws \think\admin\Exception
|
||||
*/
|
||||
public function theme()
|
||||
{
|
||||
|
@ -26,7 +26,7 @@ use think\admin\service\SystemService;
|
||||
|
||||
/**
|
||||
* 用户登录管理
|
||||
* Class Login
|
||||
* @class Login
|
||||
* @package app\admin\controller
|
||||
*/
|
||||
class Login extends Controller
|
||||
@ -35,9 +35,7 @@ class Login extends Controller
|
||||
/**
|
||||
* 后台登录入口
|
||||
* @return void
|
||||
* @throws \think\db\exception\DataNotFoundException
|
||||
* @throws \think\db\exception\DbException
|
||||
* @throws \think\db\exception\ModelNotFoundException
|
||||
* @throws \think\admin\Exception
|
||||
*/
|
||||
public function index()
|
||||
{
|
||||
|
@ -25,7 +25,7 @@ use think\admin\service\NodeService;
|
||||
|
||||
/**
|
||||
* 系统菜单管理
|
||||
* Class Menu
|
||||
* @class Menu
|
||||
* @package app\admin\controller
|
||||
*/
|
||||
class Menu extends Controller
|
||||
|
@ -25,7 +25,7 @@ use think\exception\HttpResponseException;
|
||||
|
||||
/**
|
||||
* 系统日志管理
|
||||
* Class Oplog
|
||||
* @class Oplog
|
||||
* @package app\admin\controller
|
||||
*/
|
||||
class Oplog extends Controller
|
||||
|
@ -27,7 +27,7 @@ use think\exception\HttpResponseException;
|
||||
|
||||
/**
|
||||
* 系统任务管理
|
||||
* Class Queue
|
||||
* @class Queue
|
||||
* @package app\admin\controller
|
||||
*/
|
||||
class Queue extends Controller
|
||||
|
@ -25,7 +25,7 @@ use think\admin\service\AdminService;
|
||||
|
||||
/**
|
||||
* 系统用户管理
|
||||
* Class User
|
||||
* @class User
|
||||
* @package app\admin\controller
|
||||
*/
|
||||
class User extends Controller
|
||||
|
@ -21,8 +21,8 @@ use think\admin\service\AdminService;
|
||||
use think\Response;
|
||||
|
||||
/**
|
||||
* 通用插件管理
|
||||
* Class Plugs
|
||||
* 扩展插件管理
|
||||
* @class Plugs
|
||||
* @package app\admin\controller\api
|
||||
*/
|
||||
class Plugs extends Controller
|
||||
@ -42,17 +42,17 @@ class Plugs extends Controller
|
||||
/**
|
||||
* 前端脚本变量
|
||||
* @return \think\Response
|
||||
* @throws \think\db\exception\DataNotFoundException
|
||||
* @throws \think\db\exception\DbException
|
||||
* @throws \think\db\exception\ModelNotFoundException
|
||||
* @throws \think\admin\Exception
|
||||
*/
|
||||
public function script(): Response
|
||||
{
|
||||
$token = $this->request->get('uptoken', '');
|
||||
$domain = boolval(AdminService::withUploadUnid($token));
|
||||
return response(join("\r\n", [
|
||||
sprintf("window.taDebug = %s;", $this->app->isDebug() ? 'true' : 'false'),
|
||||
sprintf("window.taAdmin = '%s';", sysuri('admin/index/index', [], false)),
|
||||
sprintf("window.taAdmin = '%s';", sysuri('admin/index/index', [], false, $domain)),
|
||||
sprintf("window.taEditor = '%s';", sysconf('base.editor|raw') ?: 'ckeditor4'),
|
||||
]))->contentType('application/x-javascript');
|
||||
]))->contentType('application/javascript');
|
||||
}
|
||||
|
||||
/**
|
||||
@ -65,7 +65,7 @@ class Plugs extends Controller
|
||||
sysoplog('系统运维管理', '创建数据库优化任务');
|
||||
$this->_queue('优化数据库所有数据表', 'xadmin:database optimize');
|
||||
} else {
|
||||
$this->error('只有超级管理员才能操作!');
|
||||
$this->error('请使用超管账号操作!');
|
||||
}
|
||||
}
|
||||
}
|
@ -24,25 +24,25 @@ use think\admin\service\QueueService;
|
||||
use think\exception\HttpResponseException;
|
||||
|
||||
/**
|
||||
* 后台任务通用接口
|
||||
* Class Queue
|
||||
* 任务监听服务管理
|
||||
* @class Queue
|
||||
* @package app\admin\controller\api
|
||||
*/
|
||||
class Queue extends Controller
|
||||
{
|
||||
/**
|
||||
* WIN停止监听进程
|
||||
* 停止监听服务
|
||||
* @login true
|
||||
*/
|
||||
public function stop()
|
||||
{
|
||||
try {
|
||||
if (AdminService::isSuper()) try {
|
||||
$message = $this->app->console->call('xadmin:queue', ['stop'])->fetch();
|
||||
if (stripos($message, 'sent end signal to process')) {
|
||||
sysoplog('系统运维管理', '尝试停止后台服务主进程');
|
||||
$this->success('停止后台服务主进程成功!');
|
||||
sysoplog('系统运维管理', '尝试停止任务监听服务');
|
||||
$this->success('停止任务监听服务成功!');
|
||||
} elseif (stripos($message, 'processes to stop')) {
|
||||
$this->success('没有找到需要停止的进程!');
|
||||
$this->success('没有找到需要停止的服务!');
|
||||
} else {
|
||||
$this->error(nl2br($message));
|
||||
}
|
||||
@ -51,22 +51,24 @@ class Queue extends Controller
|
||||
} catch (Exception $exception) {
|
||||
trace_file($exception);
|
||||
$this->error($exception->getMessage());
|
||||
} else {
|
||||
$this->error('请使用超管账号操作!');
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* WIN创建监听进程
|
||||
* 启动监听服务
|
||||
* @login true
|
||||
*/
|
||||
public function start()
|
||||
{
|
||||
try {
|
||||
if (AdminService::isSuper()) try {
|
||||
$message = $this->app->console->call('xadmin:queue', ['start'])->fetch();
|
||||
if (stripos($message, 'daemons started successfully for pid')) {
|
||||
sysoplog('系统运维管理', '尝试启动后台服务主进程');
|
||||
$this->success('后台服务主进程启动成功!');
|
||||
sysoplog('系统运维管理', '尝试启动任务监听服务');
|
||||
$this->success('任务监听服务启动成功!');
|
||||
} elseif (stripos($message, 'daemons already exist for pid')) {
|
||||
$this->success('后台服务主进程已经存在!');
|
||||
$this->success('任务监听服务已经启动!');
|
||||
} else {
|
||||
$this->error(nl2br($message));
|
||||
}
|
||||
@ -75,11 +77,13 @@ class Queue extends Controller
|
||||
} catch (Exception $exception) {
|
||||
trace_file($exception);
|
||||
$this->error($exception->getMessage());
|
||||
} else {
|
||||
$this->error('请使用超管账号操作!');
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* 检查任务状态
|
||||
* 检查监听服务
|
||||
* @login true
|
||||
*/
|
||||
public function status()
|
||||
@ -99,7 +103,7 @@ class Queue extends Controller
|
||||
}
|
||||
|
||||
/**
|
||||
* 任务进度查询
|
||||
* 查询任务进度
|
||||
* @login true
|
||||
* @throws \think\admin\Exception
|
||||
*/
|
||||
|
@ -23,8 +23,8 @@ use think\admin\service\RuntimeService;
|
||||
use think\exception\HttpResponseException;
|
||||
|
||||
/**
|
||||
* 系统运行控制管理
|
||||
* Class System
|
||||
* 系统运行管理
|
||||
* @class System
|
||||
* @package app\admin\controller\api
|
||||
*/
|
||||
class System extends Controller
|
||||
@ -45,7 +45,7 @@ class System extends Controller
|
||||
trace_file($exception);
|
||||
$this->error($exception->getMessage());
|
||||
} else {
|
||||
$this->error('只有超级管理员才能操作!');
|
||||
$this->error('请使用超管账号操作!');
|
||||
}
|
||||
}
|
||||
|
||||
@ -64,7 +64,7 @@ class System extends Controller
|
||||
trace_file($exception);
|
||||
$this->error($exception->getMessage());
|
||||
} else {
|
||||
$this->error('只有超级管理员才能操作!');
|
||||
$this->error('请使用超管账号操作!');
|
||||
}
|
||||
}
|
||||
|
||||
@ -83,7 +83,7 @@ class System extends Controller
|
||||
sysoplog('系统运维管理', '生产模式切换为开发模式');
|
||||
$this->success('已切换为开发模式!', 'javascript:location.reload()');
|
||||
} else {
|
||||
$this->error('只有超级管理员才能操作!');
|
||||
$this->error('请使用超管账号操作!');
|
||||
}
|
||||
}
|
||||
|
||||
@ -102,7 +102,7 @@ class System extends Controller
|
||||
sysoplog('系统运维管理', "切换编辑器为{$editor}");
|
||||
$this->success('已切换后台编辑器!', 'javascript:location.reload()');
|
||||
} else {
|
||||
$this->error('只有超级管理员才能操作!');
|
||||
$this->error('请使用超管账号操作!');
|
||||
}
|
||||
}
|
||||
|
||||
@ -132,7 +132,7 @@ class System extends Controller
|
||||
trace_file($exception);
|
||||
$this->error($exception->getMessage());
|
||||
} else {
|
||||
$this->error('只有超级管理员才能操作!');
|
||||
$this->error('请使用超管账号操作!');
|
||||
}
|
||||
}
|
||||
}
|
@ -32,25 +32,23 @@ use think\Response;
|
||||
|
||||
/**
|
||||
* 文件上传接口
|
||||
* Class Upload
|
||||
* @class Upload
|
||||
* @package app\admin\controller\api
|
||||
*/
|
||||
class Upload extends Controller
|
||||
{
|
||||
|
||||
/**
|
||||
* 文件上传脚本
|
||||
* @return Response
|
||||
* @throws \think\db\exception\DataNotFoundException
|
||||
* @throws \think\db\exception\DbException
|
||||
* @throws \think\db\exception\ModelNotFoundException
|
||||
* @throws \think\admin\Exception
|
||||
*/
|
||||
public function index(): Response
|
||||
{
|
||||
$data = ['exts' => []];
|
||||
foreach (str2arr(sysconf('storage.allow_exts|raw')) as $ext) {
|
||||
$data['exts'][$ext] = Storage::mime($ext);
|
||||
}
|
||||
[$uuid, $unid, $exts] = $this->initUnid(false);
|
||||
$allows = str2arr(sysconf('storage.allow_exts|raw'));
|
||||
if (empty($uuid) && $unid > 0) $allows = array_intersect($exts, $allows);
|
||||
foreach ($allows as $ext) $data['exts'][$ext] = Storage::mime($ext);
|
||||
$template = realpath(__DIR__ . '/../../view/api/upload.js');
|
||||
$data['exts'] = json_encode($data['exts'], JSON_UNESCAPED_UNICODE);
|
||||
$data['nameType'] = sysconf('storage.name_type|raw') ?: 'xmd5';
|
||||
@ -58,21 +56,42 @@ class Upload extends Controller
|
||||
}
|
||||
|
||||
/**
|
||||
* 文件上传检查
|
||||
* @login true
|
||||
* @throws \think\admin\Exception
|
||||
* 文件选择器
|
||||
* @throws \think\db\exception\DataNotFoundException
|
||||
* @throws \think\db\exception\DbException
|
||||
* @throws \think\db\exception\ModelNotFoundException
|
||||
*/
|
||||
public function image()
|
||||
{
|
||||
[$uuid, $unid] = $this->initUnid();
|
||||
SystemFile::mQuery()->layTable(function () {
|
||||
$this->title = '文件选择器';
|
||||
}, function (QueryHelper $query) use ($unid, $uuid) {
|
||||
if ($unid && $uuid) $query->where(function ($query) use ($uuid, $unid) {
|
||||
/** @var \think\db\Query $query */
|
||||
$query->whereOr([['uuid', '=', $uuid], ['unid', '=', $unid]]);
|
||||
}); else {
|
||||
$query->where($unid ? ['unid' => $unid] : ['uuid' => $uuid]);
|
||||
}
|
||||
$query->where(['status' => 2, 'issafe' => 0])->in('xext#type');
|
||||
$query->like('name,hash')->dateBetween('create_at')->order('id desc');
|
||||
});
|
||||
}
|
||||
|
||||
/**
|
||||
* 文件上传检查
|
||||
* @throws \think\admin\Exception
|
||||
*/
|
||||
public function state()
|
||||
{
|
||||
[$uuid, $unid] = $this->initUnid();
|
||||
[$name, $safe] = [input('name'), $this->getSafe()];
|
||||
$data = ['uptype' => $this->getType(), 'safe' => intval($safe), 'key' => input('key')];
|
||||
$file = SystemFile::mk()->data($this->_vali([
|
||||
'xkey.value' => $data['key'],
|
||||
'type.value' => $this->getType(),
|
||||
'uuid.value' => AdminService::getUserId(),
|
||||
'uuid.value' => $uuid,
|
||||
'unid.value' => $unid,
|
||||
'name.require' => '名称不能为空!',
|
||||
'hash.require' => '哈希不能为空!',
|
||||
'xext.require' => '后缀不能为空!',
|
||||
@ -80,9 +99,10 @@ class Upload extends Controller
|
||||
'mime.default' => '',
|
||||
'status.value' => 1,
|
||||
]));
|
||||
if (empty($file['mime'])) $file['mime'] = Storage::mime($file['xext']);
|
||||
$mime = $file->getAttr('mime');
|
||||
if (empty($mime)) $file->setAttr('mime', Storage::mime($file->getAttr('xext')));
|
||||
$info = Storage::instance($data['uptype'])->info($data['key'], $safe, $name);
|
||||
if (is_array($info) && isset($info['url']) && isset($info['key'])) {
|
||||
if (isset($info['url']) && isset($info['key'])) {
|
||||
$file->save(['xurl' => $info['url'], 'isfast' => 1, 'issafe' => $data['safe']]);
|
||||
$extr = ['id' => $file->id ?? 0, 'url' => $info['url'], 'key' => $info['key']];
|
||||
$this->success('文件已经上传', array_merge($data, $extr), 200);
|
||||
@ -110,7 +130,7 @@ class Upload extends Controller
|
||||
$data['q-sign-algorithm'] = $token['q-sign-algorithm'];
|
||||
$data['server'] = TxcosStorage::instance()->upload();
|
||||
} elseif ('upyun' === $data['uptype']) {
|
||||
$token = UpyunStorage::instance()->buildUploadToken($data['key'], 3600, $name, input('size'), input('hash'));
|
||||
$token = UpyunStorage::instance()->buildUploadToken($data['key'], 3600, $name, input('hash', ''));
|
||||
$data['url'] = $token['siteurl'];
|
||||
$data['policy'] = $token['policy'];
|
||||
$data['authorization'] = $token['authorization'];
|
||||
@ -122,15 +142,16 @@ class Upload extends Controller
|
||||
|
||||
/**
|
||||
* 更新文件状态
|
||||
* @login true
|
||||
* @return void
|
||||
*/
|
||||
public function done()
|
||||
{
|
||||
[$uuid, $unid] = $this->initUnid();
|
||||
$data = $this->_vali([
|
||||
'id.require' => '编号不能为空!',
|
||||
'hash.require' => '哈希不能为空!',
|
||||
'uuid.value' => AdminService::getUserId(),
|
||||
'uuid.value' => $uuid,
|
||||
'unid.value' => $unid,
|
||||
]);
|
||||
$file = SystemFile::mk()->where($data)->findOrEmpty();
|
||||
if ($file->isEmpty()) $this->error('文件不存在!');
|
||||
@ -141,75 +162,63 @@ class Upload extends Controller
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* 文件选择器
|
||||
* @login true
|
||||
* @return void
|
||||
* @throws \think\db\exception\DataNotFoundException
|
||||
* @throws \think\db\exception\DbException
|
||||
* @throws \think\db\exception\ModelNotFoundException
|
||||
*/
|
||||
public function image()
|
||||
{
|
||||
SystemFile::mQuery()->layTable(function () {
|
||||
$this->title = '文件选择器';
|
||||
}, function (QueryHelper $query) {
|
||||
$query->where(['status' => 2, 'issafe' => 0, 'uuid' => AdminService::getUserId()]);
|
||||
$query->like('name,hash')->in('xext#type')->dateBetween('create_at')->order('id desc');
|
||||
});
|
||||
}
|
||||
|
||||
/**
|
||||
* 文件上传入口
|
||||
* @login true
|
||||
* @throws \think\db\exception\DataNotFoundException
|
||||
* @throws \think\db\exception\DbException
|
||||
* @throws \think\db\exception\ModelNotFoundException
|
||||
* @throws \think\admin\Exception
|
||||
*/
|
||||
public function file()
|
||||
{
|
||||
if (!($file = $this->getFile())->isValid()) {
|
||||
$this->error('文件上传异常,文件过大或未上传!');
|
||||
}
|
||||
$safeMode = $this->getSafe();
|
||||
[$uuid, $unid, $unexts] = $this->initUnid();
|
||||
// 开始处理文件上传
|
||||
$file = $this->getFile();
|
||||
$extension = strtolower($file->getOriginalExtension());
|
||||
$saveName = input('key') ?: Storage::name($file->getPathname(), $extension, '', 'md5_file');
|
||||
$saveFileName = input('key') ?: Storage::name($file->getPathname(), $extension, '', 'md5_file');
|
||||
// 检查文件名称是否合法
|
||||
if (strpos($saveName, '../') !== false) {
|
||||
if (strpos($saveFileName, '../') !== false) {
|
||||
$this->error('文件路径不能出现跳级操作!');
|
||||
}
|
||||
// 检查文件后缀是否被恶意修改
|
||||
if (strtolower(pathinfo(parse_url($saveName, PHP_URL_PATH), PATHINFO_EXTENSION)) !== $extension) {
|
||||
if (strtolower(pathinfo(parse_url($saveFileName, PHP_URL_PATH), PATHINFO_EXTENSION)) !== $extension) {
|
||||
$this->error('文件后缀异常,请重新上传文件!');
|
||||
}
|
||||
// 屏蔽禁止上传指定后缀的文件
|
||||
if (!in_array($extension, str2arr(sysconf('storage.allow_exts|raw')))) {
|
||||
$this->error('文件类型受限,请在后台配置规则!');
|
||||
}
|
||||
// 前端用户上传后缀检查处理
|
||||
if (empty($uuid) && $unid > 0 && !in_array($extension, $unexts)) {
|
||||
$this->error('文件类型受限,请上传允许的文件类型!');
|
||||
}
|
||||
if (in_array($extension, ['sh', 'asp', 'bat', 'cmd', 'exe', 'php'])) {
|
||||
$this->error('文件安全保护,禁止上传可执行文件!');
|
||||
}
|
||||
try {
|
||||
if ($this->getType() === 'local') {
|
||||
$safeMode = $this->getSafe();
|
||||
if (($type = $this->getType()) === 'local') {
|
||||
$local = LocalStorage::instance();
|
||||
$distName = $local->path($saveName, $safeMode);
|
||||
$file->move(dirname($distName), basename($distName));
|
||||
$info = $local->info($saveName, $safeMode, $file->getOriginalName());
|
||||
$distName = $local->path($saveFileName, $safeMode);
|
||||
if (PHP_SAPI === 'cli') {
|
||||
is_dir(dirname($distName)) || mkdir(dirname($distName), 0777, true);
|
||||
rename($file->getPathname(), $distName);
|
||||
} else {
|
||||
$file->move(dirname($distName), basename($distName));
|
||||
}
|
||||
$info = $local->info($saveFileName, $safeMode, $file->getOriginalName());
|
||||
if (in_array($extension, ['jpg', 'gif', 'png', 'bmp', 'jpeg', 'wbmp'])) {
|
||||
if ($this->imgNotSafe($distName) && $local->del($saveName)) {
|
||||
if ($this->imgNotSafe($distName) && $local->del($saveFileName)) {
|
||||
$this->error('图片未通过安全检查!');
|
||||
}
|
||||
[$width, $height] = getimagesize($distName);
|
||||
if (($width < 1 || $height < 1) && $local->del($saveName)) {
|
||||
if (($width < 1 || $height < 1) && $local->del($saveFileName)) {
|
||||
$this->error('读取图片的尺寸失败!');
|
||||
}
|
||||
}
|
||||
} else {
|
||||
$bina = file_get_contents($file->getPathname());
|
||||
$info = Storage::instance($this->getType())->set($saveName, $bina, $safeMode, $file->getOriginalName());
|
||||
$info = Storage::instance($type)->set($saveFileName, $bina, $safeMode, $file->getOriginalName());
|
||||
}
|
||||
if (isset($info['url'])) {
|
||||
$this->success('文件上传成功!', ['url' => $safeMode ? $saveName : $info['url']]);
|
||||
$this->success('文件上传成功!', ['url' => $safeMode ? $saveFileName : $info['url']]);
|
||||
} else {
|
||||
$this->error('文件处理失败,请稍候再试!');
|
||||
}
|
||||
@ -222,7 +231,7 @@ class Upload extends Controller
|
||||
}
|
||||
|
||||
/**
|
||||
* 获取文件上传类型
|
||||
* 获取上传类型
|
||||
* @return boolean
|
||||
*/
|
||||
private function getSafe(): bool
|
||||
@ -231,11 +240,9 @@ class Upload extends Controller
|
||||
}
|
||||
|
||||
/**
|
||||
* 获取文件上传方式
|
||||
* 获取上传方式
|
||||
* @return string
|
||||
* @throws \think\db\exception\DataNotFoundException
|
||||
* @throws \think\db\exception\DbException
|
||||
* @throws \think\db\exception\ModelNotFoundException
|
||||
* @throws \think\admin\Exception
|
||||
*/
|
||||
private function getType(): string
|
||||
{
|
||||
@ -248,7 +255,7 @@ class Upload extends Controller
|
||||
}
|
||||
|
||||
/**
|
||||
* 获取本地文件对象
|
||||
* 获取文件对象
|
||||
* @return UploadedFile|void
|
||||
*/
|
||||
private function getFile(): UploadedFile
|
||||
@ -268,6 +275,22 @@ class Upload extends Controller
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* 初始化用户状态
|
||||
* @param boolean $check
|
||||
* @return array
|
||||
*/
|
||||
private function initUnid(bool $check = true): array
|
||||
{
|
||||
$uuid = AdminService::getUserId();
|
||||
[$unid, $exts] = AdminService::withUploadUnid();
|
||||
if ($check && empty($uuid) && empty($unid)) {
|
||||
$this->error('未登录,禁止使用文件上传!');
|
||||
} else {
|
||||
return [$uuid, $unid, $exts];
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* 检查图片是否安全
|
||||
* @param string $filename
|
||||
|
@ -11,85 +11,86 @@
|
||||
<meta name="viewport" content="width=device-width,initial-scale=1,maximum-scale=1">
|
||||
<link rel="stylesheet" href="__ROOT__/static/theme/css/iconfont.css?at={:date('md')}">
|
||||
<link rel="stylesheet" href="__ROOT__/static/plugs/layui/css/layui.css?v={:date('ymd')}">
|
||||
<style> ::-webkit-input-placeholder {
|
||||
color: #aaa
|
||||
}
|
||||
<style>
|
||||
::-webkit-input-placeholder {
|
||||
color: #aaa
|
||||
}
|
||||
|
||||
::-webkit-scrollbar {
|
||||
width: 3px;
|
||||
height: 3px
|
||||
}
|
||||
::-webkit-scrollbar {
|
||||
width: 3px;
|
||||
height: 3px
|
||||
}
|
||||
|
||||
::-webkit-scrollbar-track {
|
||||
background: #ccc
|
||||
}
|
||||
::-webkit-scrollbar-track {
|
||||
background: #ccc
|
||||
}
|
||||
|
||||
::-webkit-scrollbar-thumb {
|
||||
background-color: #666
|
||||
}
|
||||
::-webkit-scrollbar-thumb {
|
||||
background-color: #666
|
||||
}
|
||||
|
||||
::selection {
|
||||
background-color: #ec494e;
|
||||
color: #fff
|
||||
}
|
||||
::selection {
|
||||
background-color: #ec494e;
|
||||
color: #fff
|
||||
}
|
||||
|
||||
::-moz-selection {
|
||||
background-color: #ec494e;
|
||||
color: #fff
|
||||
}
|
||||
::-moz-selection {
|
||||
background-color: #ec494e;
|
||||
color: #fff
|
||||
}
|
||||
|
||||
:-webkit-autofill {
|
||||
-webkit-box-shadow: 0 0 0 1000px white inset;
|
||||
-webkit-text-fill-color: #333
|
||||
}
|
||||
:-webkit-autofill {
|
||||
-webkit-box-shadow: 0 0 0 1000px white inset;
|
||||
-webkit-text-fill-color: #333
|
||||
}
|
||||
|
||||
ul li {
|
||||
width: 20%;
|
||||
height: 65px;
|
||||
display: block;
|
||||
float: left;
|
||||
margin-right: -1px;
|
||||
margin-left: -2px;
|
||||
margin-bottom: -2px;
|
||||
cursor: pointer;
|
||||
font-size: 14px;
|
||||
overflow: hidden;
|
||||
text-align: center;
|
||||
padding: 15px 0 10px 0;
|
||||
border: 1px solid #e2e2e2;
|
||||
background-color: #efefef;
|
||||
user-select: none;
|
||||
-ms-user-select: none;
|
||||
-moz-user-select: none;
|
||||
-webkit-user-select: none;
|
||||
transition: all .2s linear;
|
||||
-o-transition: all .2s linear;
|
||||
-moz-transition: all .2s linear;
|
||||
-webkit-transition: all .2s linear
|
||||
}
|
||||
ul li {
|
||||
width: 20%;
|
||||
height: 65px;
|
||||
display: block;
|
||||
float: left;
|
||||
margin-right: -1px;
|
||||
margin-left: -2px;
|
||||
margin-bottom: -2px;
|
||||
cursor: pointer;
|
||||
font-size: 14px;
|
||||
overflow: hidden;
|
||||
text-align: center;
|
||||
padding: 15px 0 10px 0;
|
||||
border: 1px solid #e2e2e2;
|
||||
background-color: #efefef;
|
||||
user-select: none;
|
||||
-ms-user-select: none;
|
||||
-moz-user-select: none;
|
||||
-webkit-user-select: none;
|
||||
transition: all .2s linear;
|
||||
-o-transition: all .2s linear;
|
||||
-moz-transition: all .2s linear;
|
||||
-webkit-transition: all .2s linear
|
||||
}
|
||||
|
||||
ul li:hover {
|
||||
color: #fff;
|
||||
background-color: #563d7c
|
||||
}
|
||||
ul li:hover {
|
||||
color: #fff;
|
||||
background-color: #563d7c
|
||||
}
|
||||
|
||||
ul li:hover i, ul li:hover div {
|
||||
color: #fff;
|
||||
}
|
||||
ul li:hover i, ul li:hover div {
|
||||
color: #fff;
|
||||
}
|
||||
|
||||
ul li i {
|
||||
color: #333;
|
||||
display: inline-block;
|
||||
font-size: 30px !important
|
||||
}
|
||||
ul li i {
|
||||
color: #333;
|
||||
display: inline-block;
|
||||
font-size: 30px !important
|
||||
}
|
||||
|
||||
ul li div {
|
||||
color: #333;
|
||||
height: 35px;
|
||||
font-size: 13px;
|
||||
line-height: 35px;
|
||||
white-space: nowrap
|
||||
} </style>
|
||||
ul li div {
|
||||
color: #333;
|
||||
height: 35px;
|
||||
font-size: 13px;
|
||||
line-height: 35px;
|
||||
white-space: nowrap
|
||||
} </style>
|
||||
</head>
|
||||
<body>
|
||||
|
||||
|
@ -32,7 +32,7 @@ define(['md5', 'notify'], function (SparkMD5, Notify, allowMime) {
|
||||
|
||||
/*! 初始化上传组件 */
|
||||
this.adapter = new Adapter(this.option, layui.upload.render({
|
||||
url: '{:url("admin/api.upload/file")}', auto: false, elem: elem, accept: 'file', multiple: this.option.mult, exts: this.option.exts.join('|'), acceptMime: this.option.mimes.join(','), choose: function (obj) {
|
||||
url: '{:url("admin/api.upload/file",[],false,true)}', auto: false, elem: elem, accept: 'file', multiple: this.option.mult, exts: this.option.exts.join('|'), acceptMime: this.option.mimes.join(','), choose: function (obj) {
|
||||
obj.items = [], obj.files = obj.pushFile();
|
||||
layui.each(obj.files, function (idx, file) {
|
||||
obj.items.push(file);
|
||||
@ -108,7 +108,7 @@ define(['md5', 'notify'], function (SparkMD5, Notify, allowMime) {
|
||||
Adapter.prototype.request = function (file, done) {
|
||||
var that = this, data = {key: file.xkey, safe: that.option.safe, uptype: that.option.type};
|
||||
data.size = file.size, data.name = file.name, data.hash = file.xmd5, data.mime = file.type, data.xext = file.xext;
|
||||
jQuery.ajax("{:url('admin/api.upload/state')}", {
|
||||
jQuery.ajax("{:url('admin/api.upload/state',[],false,true)}", {
|
||||
data: data, method: 'post', success: function (ret) {
|
||||
file.id = ret.data.id || 0, file.xurl = ret.data.url;
|
||||
file.xsafe = ret.data.safe, file.xpath = ret.data.key, file.xtype = ret.data.uptype;
|
||||
@ -192,7 +192,7 @@ define(['md5', 'notify'], function (SparkMD5, Notify, allowMime) {
|
||||
/*! 检查单个文件上传返回的结果 */
|
||||
if (ret.code < 1) return $.msg.tips(ret.info || '文件上传失败!');
|
||||
if (typeof file.xurl !== 'string') return $.msg.tips('无效的文件上传对象!');
|
||||
jQuery.post("{:url('admin/api.upload/done')}", {id: file.id, hash: file.xmd5});
|
||||
jQuery.post("{:url('admin/api.upload/done',[],false,true)}", {id: file.id, hash: file.xmd5});
|
||||
/*! 单个文件上传成功结果处理 */
|
||||
if (typeof done === 'function') {
|
||||
done.call(this.option.elem, file.xurl, this.files['id']);
|
||||
|
@ -109,7 +109,7 @@
|
||||
loadPage: function () {
|
||||
this.params = {page: this.page, limit: this.limit, output: 'layui.table', name: this.keys || ''};
|
||||
this.params.type = '{$get.type|default="gif,png,jpg,jpeg"}';
|
||||
$.form.load('{:url("image")}', this.params, 'get', function (ret) {
|
||||
$.form.load('{:url("image",[],false,true)}', this.params, 'get', function (ret) {
|
||||
return app.setList(ret.data, ret.count), false;
|
||||
});
|
||||
},
|
||||
|
@ -1,22 +1,26 @@
|
||||
{extend name='table'}
|
||||
|
||||
{block name="button"}
|
||||
<!--{if isset($super) and $super}-->
|
||||
<a data-table-id="QueueTable" class="layui-btn layui-btn-sm layui-btn-primary" data-queue="{:url('admin/api.plugs/optimize')}">优化数据库</a>
|
||||
<!--{/if}-->
|
||||
|
||||
<!--{if isset($super) and $super and $iswin}-->
|
||||
{if isset($super) and $super}
|
||||
|
||||
<a data-table-id="QueueTable" class="layui-btn layui-btn-sm layui-btn-primary" data-queue="{:url('admin/api.plugs/optimize')}">优化数据库</a>
|
||||
|
||||
{if isset($iswin) and ($iswin or php_sapi_name() eq 'cli')}
|
||||
<button data-load='{:url("admin/api.queue/start")}' class='layui-btn layui-btn-sm layui-btn-primary'>开启服务</button>
|
||||
<button data-load='{:url("admin/api.queue/stop")}' class='layui-btn layui-btn-sm layui-btn-primary'>关闭服务</button>
|
||||
<!--{/if}-->
|
||||
{/if}
|
||||
|
||||
<!--{if auth("clean")}-->
|
||||
{if auth("clean")}
|
||||
<button data-table-id="QueueTable" data-queue='{:url("clean")}' class='layui-btn layui-btn-sm layui-btn-primary'>定时清理</button>
|
||||
<!--{/if}-->
|
||||
{/if}
|
||||
|
||||
<!--{if auth("remove")}-->
|
||||
{/if}
|
||||
|
||||
{if auth("remove")}
|
||||
<button data-table-id="QueueTable" data-action='{:url("remove")}' data-rule="id#{id}" data-confirm="确定批量删除记录吗?" class='layui-btn layui-btn-sm layui-btn-primary'>批量删除</button>
|
||||
<!--{/if}-->
|
||||
{/if}
|
||||
|
||||
{/block}
|
||||
|
||||
{block name="content"}
|
||||
|
@ -78,7 +78,7 @@ class Menu extends Controller
|
||||
{
|
||||
try {
|
||||
WechatService::WeChatMenu()->delete();
|
||||
$this->success('菜单取消成功,重新订阅可立即生效!');
|
||||
$this->success('公众号菜单取消成功!');
|
||||
} catch (HttpResponseException $exception) {
|
||||
sysoplog('微信管理', '取消微信菜单成功');
|
||||
throw $exception;
|
||||
|
@ -16,45 +16,29 @@
|
||||
|
||||
namespace app\wechat\controller\api;
|
||||
|
||||
use app\wechat\service\MediaService;
|
||||
use app\wechat\service\WechatService;
|
||||
use app\wechat\service\LoginService;
|
||||
use think\admin\Controller;
|
||||
|
||||
/**
|
||||
* 微信扫码授权登录
|
||||
* 微信扫码登录
|
||||
* Class Login
|
||||
* @package app\wechat\controller\api
|
||||
*/
|
||||
class Login extends Controller
|
||||
{
|
||||
/**
|
||||
* 数据缓存时间
|
||||
* @var integer
|
||||
*/
|
||||
protected $expire = 3600;
|
||||
|
||||
/**
|
||||
* 授权码前缀
|
||||
* @var string
|
||||
*/
|
||||
protected $prefix = 'wxlogin';
|
||||
|
||||
/**
|
||||
* 扫描显示二维码
|
||||
* 显示二维码
|
||||
* @return void
|
||||
*/
|
||||
public function qrc()
|
||||
{
|
||||
$mode = intval(input('mode', '0'));
|
||||
$code = $this->prefix . md5(uniqid('t', true) . rand(10000, 99999));
|
||||
$text = url('wechat/api.login/oauth', [], false, true) . "?code={$code}&mode={$mode}";
|
||||
// 生成二维码并返回结果
|
||||
$qrcode = MediaService::getQrcode($text);
|
||||
$this->success('获取二维码成功', ['code' => $code, 'image' => $qrcode->getDataUri()]);
|
||||
$data = LoginService::qrcode(LoginService::gcode(), $mode);
|
||||
$this->success('登录二维码', $data);
|
||||
}
|
||||
|
||||
/**
|
||||
* 微信授权结果处理
|
||||
* 微信授权处理
|
||||
* @throws \WeChat\Exceptions\InvalidResponseException
|
||||
* @throws \WeChat\Exceptions\LocalCacheException
|
||||
* @throws \think\admin\Exception
|
||||
@ -64,41 +48,25 @@ class Login extends Controller
|
||||
*/
|
||||
public function oauth()
|
||||
{
|
||||
$this->code = input('code', '');
|
||||
$this->mode = input('mode', '0');
|
||||
if (stripos($this->code, $this->prefix) === 0) {
|
||||
$this->url = $this->request->url(true);
|
||||
$this->fans = WechatService::getWebOauthInfo($this->url, $this->mode);
|
||||
if (isset($this->fans['openid'])) {
|
||||
$this->fans['token'] = md5(uniqid('t', true) . rand(10000, 99999));
|
||||
$this->app->cache->set("wxlogin{$this->code}", $this->fans, $this->expire);
|
||||
$this->app->cache->set($this->fans['openid'], $this->fans['token'], $this->expire);
|
||||
$this->message = '授权成功';
|
||||
$this->fetch('success');
|
||||
} else {
|
||||
$this->message = '授权失败';
|
||||
$this->fetch('failed');
|
||||
}
|
||||
$data = $this->_vali(['auth.default' => '', 'mode.default' => '0']);
|
||||
if (LoginService::oauth($data['auth'], intval($data['mode']))) {
|
||||
$this->fetch('success', ['message' => '授权成功']);
|
||||
} else {
|
||||
$this->message = '授权失败';
|
||||
$this->fetch('failed');
|
||||
$this->fetch('failed', ['message' => '授权失败']);
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* 获取授权信息
|
||||
* 用定时器请求这个接口
|
||||
* @throws \think\exception\HttpResponseException
|
||||
*/
|
||||
public function query()
|
||||
{
|
||||
$this->code = input('code', '');
|
||||
if (stripos($this->code, $this->prefix) === 0) {
|
||||
$this->ckey = "wxlogin{$this->code}";
|
||||
$this->fans = $this->app->cache->get($this->ckey, new \stdClass());
|
||||
$this->success('获取授权信息', $this->fans);
|
||||
$data = $this->_vali(['code.require' => '编号不能为空!']);
|
||||
if ($fans = LoginService::query($data['code'])) {
|
||||
$this->success('获取授权信息', $fans);
|
||||
} else {
|
||||
$this->error("授权CODE不能为空!");
|
||||
$this->error('未获取到授权!');
|
||||
}
|
||||
}
|
||||
}
|
107
app/wechat/service/LoginService.php
Normal file
107
app/wechat/service/LoginService.php
Normal file
@ -0,0 +1,107 @@
|
||||
<?php
|
||||
|
||||
// +----------------------------------------------------------------------
|
||||
// | Wechat Plugin for ThinkAdmin
|
||||
// +----------------------------------------------------------------------
|
||||
// | 版权所有 2014~2023 Anyon <zoujingli@qq.com>
|
||||
// +----------------------------------------------------------------------
|
||||
// | 官方网站: https://thinkadmin.top
|
||||
// +----------------------------------------------------------------------
|
||||
// | 开源协议 ( https://mit-license.org )
|
||||
// | 免责声明 ( https://thinkadmin.top/disclaimer )
|
||||
// +----------------------------------------------------------------------
|
||||
// | gitee 代码仓库:https://gitee.com/zoujingli/think-plugs-wechat
|
||||
// | github 代码仓库:https://github.com/zoujingli/think-plugs-wechat
|
||||
// +----------------------------------------------------------------------
|
||||
|
||||
namespace app\wechat\service;
|
||||
|
||||
use think\admin\Library;
|
||||
|
||||
/**
|
||||
* 微信扫码登录服务
|
||||
* @class LoginService
|
||||
* @package app\wechat\service
|
||||
*/
|
||||
class LoginService
|
||||
{
|
||||
private const expire = 3600;
|
||||
private const prefix = 'wxlogin';
|
||||
|
||||
/**
|
||||
* 生成请求编号
|
||||
* @return string
|
||||
*/
|
||||
public static function gcode(): string
|
||||
{
|
||||
return md5(uniqid(strval(rand(0, 10000)), true));
|
||||
}
|
||||
|
||||
/**
|
||||
* 生成授权码
|
||||
* @param string $code 请求编号
|
||||
* @return string
|
||||
*/
|
||||
public static function gauth(string $code): string
|
||||
{
|
||||
return self::prefix . md5($code);
|
||||
}
|
||||
|
||||
/**
|
||||
* 生成授权二维码
|
||||
* @param string $code 请求编号
|
||||
* @param integer $mode 授权模式
|
||||
* @return array
|
||||
*/
|
||||
public static function qrcode(string $code, int $mode = 0): array
|
||||
{
|
||||
$data = ['auth' => self::gauth($code), 'mode' => $mode];
|
||||
$image = MediaService::getQrcode(sysuri('wechat/api.login/oauth', $data, false, true));
|
||||
return ['code' => $code, 'auth' => $data['auth'], 'image' => $image->getDataUri()];
|
||||
}
|
||||
|
||||
/**
|
||||
* 发起网页授权处理
|
||||
* @param string $auth 授权编号
|
||||
* @param integer $mode 授权模式
|
||||
* @return boolean
|
||||
* @throws \WeChat\Exceptions\InvalidResponseException
|
||||
* @throws \WeChat\Exceptions\LocalCacheException
|
||||
* @throws \think\admin\Exception
|
||||
* @throws \think\db\exception\DataNotFoundException
|
||||
* @throws \think\db\exception\DbException
|
||||
* @throws \think\db\exception\ModelNotFoundException
|
||||
*/
|
||||
public static function oauth(string $auth = '', int $mode = 0): bool
|
||||
{
|
||||
if (stripos($auth, self::prefix) === 0) {
|
||||
$url = Library::$sapp->request->url(true);
|
||||
$fans = WechatService::getWebOauthInfo($url, $mode);
|
||||
if (isset($fans['openid'])) {
|
||||
Library::$sapp->cache->set($auth, $fans, self::expire);
|
||||
return true;
|
||||
}
|
||||
}
|
||||
return false;
|
||||
}
|
||||
|
||||
/**
|
||||
* 检查是否授权
|
||||
* @param string $code 请求编号
|
||||
* @return ?array
|
||||
*/
|
||||
public static function query(string $code): ?array
|
||||
{
|
||||
return Library::$sapp->cache->get(self::gauth($code));
|
||||
}
|
||||
|
||||
/**
|
||||
* 删除授权缓存
|
||||
* @param string $code
|
||||
* @return bool
|
||||
*/
|
||||
public static function remove(string $code): bool
|
||||
{
|
||||
return Library::$sapp->cache->delete(self::gauth($code));
|
||||
}
|
||||
}
|
@ -34,60 +34,64 @@
|
||||
<div class="layui-card-body" ng-if="list.length<1">
|
||||
<blockquote class="layui-elem-quote border-0 text-center">请在左侧创建菜单...</blockquote>
|
||||
</div>
|
||||
<div class="layui-card-body" ng-if="list.length>0">
|
||||
<form class="layui-form menu-form padding-right-40" autocomplete="off">
|
||||
<div class="layui-form-item margin-top-20">
|
||||
<label class="layui-form-label">菜单名称</label>
|
||||
<div class="layui-input-block">
|
||||
<input required name="menu-name" ng-model="item.name" class="layui-input" placeholder="请输入菜单名称">
|
||||
<span class="help-block">字数不超过13个汉字或40个字母</span>
|
||||
</div>
|
||||
</div>
|
||||
<div class="layui-form-item margin-top-20" ng-if="!item.sub_button||item.sub_button.length<1">
|
||||
<label class="layui-form-label label-required">菜单类型</label>
|
||||
<div class="layui-input-block">
|
||||
{foreach $menuTypes as $key => $type}
|
||||
<label class="think-radio layui-elip"><input lay-ignore type="radio" ng-model="item.type" name="menu-type" value="{$key}"> {$type}</label>
|
||||
{/foreach}
|
||||
</div>
|
||||
</div>
|
||||
<div class="layui-form-item margin-top-20" ng-if="item.type==='customservice'">
|
||||
<label class="layui-form-label">提示文字</label>
|
||||
<div class="layui-input-block">
|
||||
<textarea required class="layui-textarea" ng-model="item.content"></textarea>
|
||||
</div>
|
||||
</div>
|
||||
<div class="layui-form-item margin-top-20" ng-if="(!item.sub_button||item.sub_button.length<1)&&item.type==='click'">
|
||||
<label class="layui-form-label">匹配规则</label>
|
||||
<div class="layui-input-block">
|
||||
<select required class="layui-select" lay-filter="key" lay-search>
|
||||
<option value="{{x.keys}}" ng-selected="x.keys===item.key" ng-repeat="x in keys" ng-bind="x.keys"></option>
|
||||
</select>
|
||||
</div>
|
||||
</div>
|
||||
<div class="layui-form-item margin-top-20" ng-if="item.type==='view'">
|
||||
<label class="layui-form-label">跳转链接</label>
|
||||
<div class="layui-input-block">
|
||||
<textarea required class="layui-textarea" ng-model="item.url" placeholder="请输入跳转链接"></textarea>
|
||||
</div>
|
||||
</div>
|
||||
<div ng-if="item.type==='miniprogram'">
|
||||
<div class="layui-card-body">
|
||||
<form class="layui-form padding-right-40" name="menu" role="form" onsubmit="return false">
|
||||
<div ng-if="list.length>0">
|
||||
<div class="layui-form-item margin-top-20">
|
||||
<label class="layui-form-label">小程序链接</label>
|
||||
<label class="layui-form-label">菜单名称</label>
|
||||
<div class="layui-input-block">
|
||||
<input type="text" required class="layui-input" ng-model="item.url" placeholder="请输入小程序链接">
|
||||
<input required vali-name="菜单名称" ng-model="item.name" class="layui-input" placeholder="请输入菜单名称">
|
||||
<span class="help-block">字数不超过13个汉字或40个字母</span>
|
||||
</div>
|
||||
</div>
|
||||
<div class="layui-form-item margin-top-20">
|
||||
<label class="layui-form-label">小程序APPID</label>
|
||||
<div class="layui-form-item margin-top-20" ng-if="!item.sub_button||item.sub_button.length<1">
|
||||
<label class="layui-form-label label-required">菜单类型</label>
|
||||
<div class="layui-input-block">
|
||||
<input type="text" required class="layui-input" ng-model="item.appid" placeholder="请输入小程序APPID">
|
||||
{foreach $menuTypes as $key => $type}
|
||||
<label class="think-radio layui-elip">
|
||||
<input lay-ignore type="radio" ng-model="item.type" name="menu-type" value="{$key}"> {$type}
|
||||
</label>
|
||||
{/foreach}
|
||||
</div>
|
||||
</div>
|
||||
<div class="layui-form-item margin-top-20">
|
||||
<label class="layui-form-label">小程序页面</label>
|
||||
<div class="layui-form-item margin-top-20" ng-if="item.type==='customservice'">
|
||||
<label class="layui-form-label">提示文字</label>
|
||||
<div class="layui-input-block">
|
||||
<input type="text" required class="layui-input" ng-model="item.pagepath" placeholder="请输入小程序页面">
|
||||
<textarea required vali-name="提示文字" class="layui-textarea" ng-model="item.content"></textarea>
|
||||
</div>
|
||||
</div>
|
||||
<div class="layui-form-item margin-top-20" ng-if="(!item.sub_button||item.sub_button.length<1)&&item.type==='click'">
|
||||
<label class="layui-form-label">匹配规则</label>
|
||||
<div class="layui-input-block">
|
||||
<select required class="layui-select" lay-filter="key" lay-search>
|
||||
<option value="{{x.keys}}" ng-selected="x.keys===item.key" ng-repeat="x in keys" ng-bind="x.keys"></option>
|
||||
</select>
|
||||
</div>
|
||||
</div>
|
||||
<div class="layui-form-item margin-top-20" ng-if="item.type==='view'">
|
||||
<label class="layui-form-label">跳转链接</label>
|
||||
<div class="layui-input-block">
|
||||
<textarea required pattern="url" class="layui-textarea" vali-name="跳转链接" ng-model="item.url" placeholder="请输入跳转链接"></textarea>
|
||||
</div>
|
||||
</div>
|
||||
<div ng-if="item.type==='miniprogram'">
|
||||
<div class="layui-form-item margin-top-20">
|
||||
<label class="layui-form-label">小程序链接</label>
|
||||
<div class="layui-input-block">
|
||||
<input type="text" required vali-name="小程序链接" class="layui-input" ng-model="item.url" placeholder="请输入小程序链接">
|
||||
</div>
|
||||
</div>
|
||||
<div class="layui-form-item margin-top-20">
|
||||
<label class="layui-form-label">小程序APPID</label>
|
||||
<div class="layui-input-block">
|
||||
<input type="text" required vali-name="小程序APPID" class="layui-input" ng-model="item.appid" placeholder="请输入小程序APPID">
|
||||
</div>
|
||||
</div>
|
||||
<div class="layui-form-item margin-top-20">
|
||||
<label class="layui-form-label">小程序页面</label>
|
||||
<div class="layui-input-block">
|
||||
<input type="text" required vali-name="小程序页面" class="layui-input" ng-model="item.pagepath" placeholder="请输入小程序页面">
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
@ -110,90 +114,86 @@
|
||||
<script>
|
||||
require(['angular'], function () {
|
||||
|
||||
var $form = $('.menu-form').vali();
|
||||
var $vali = $form.vali().data('validate');
|
||||
$('#MenuEditor.layui-hide').removeClass('layui-hide');
|
||||
$('#MenuEditor').removeClass('layui-hide');
|
||||
$('form[name=menu]').vali(null, function () {
|
||||
|
||||
var app = angular.module("MenuEditor", []).run(callback);
|
||||
angular.bootstrap(document.getElementById(app.name), [app.name]);
|
||||
var vali = this, app = angular.module('MenuEditor', []).run(callback);
|
||||
angular.bootstrap(document.getElementById(app.name), [app.name]);
|
||||
|
||||
function callback($rootScope) {
|
||||
$rootScope.item = {}, $rootScope.list = [], $rootScope.keys = [];
|
||||
$.form.load('{:url("index")}', {output: 'json'}, 'get', function (ret) {
|
||||
return $rootScope.$apply(function () {
|
||||
$rootScope.keys = ret.data.keysdata || [], $rootScope.list = ret.data.menudata || [];
|
||||
if ($rootScope.list.length < 1) $rootScope.list = [{name: '请输入名称', type: 'click', sub_button: []}];
|
||||
for (var i in $rootScope.list) $rootScope.list[i].sub_button = $rootScope.list[i].sub_button || [];
|
||||
$rootScope.list[0].show = true, $rootScope.list[0].active = true, $rootScope.item = $rootScope.list[0];
|
||||
}), false;
|
||||
});
|
||||
function callback($rootScope) {
|
||||
$rootScope.item = {}, $rootScope.list = [], $rootScope.keys = [];
|
||||
$.form.load('{:request()->url()}', {output: 'json'}, 'get', function (ret) {
|
||||
$rootScope.$apply(function () {
|
||||
$rootScope.keys = ret.data.keysdata || [], $rootScope.list = ret.data.menudata || [];
|
||||
if ($rootScope.list.length < 1) $rootScope.list = [{name: '请输入名称', type: 'click', sub_button: []}];
|
||||
for (var i in $rootScope.list) $rootScope.list[i].sub_button = $rootScope.list[i].sub_button || [];
|
||||
$rootScope.list[0].show = true, $rootScope.list[0].active = true, $rootScope.item = $rootScope.list[0];
|
||||
});
|
||||
return false;
|
||||
});
|
||||
|
||||
// 动态计算宽度
|
||||
$rootScope.getItemStyle = function (list) {
|
||||
return 'width:' + (100 / (list.length >= 3 ? 3 : (list.length + 1))) + '%';
|
||||
};
|
||||
// 动态计算宽度
|
||||
$rootScope.getItemStyle = function (list) {
|
||||
return 'width:' + (100 / (list.length >= 3 ? 3 : (list.length + 1))) + '%';
|
||||
};
|
||||
|
||||
// 增加菜单选项
|
||||
$rootScope.addItem = function (list) {
|
||||
$vali.checkAllInput();
|
||||
if ($form.find('.validate-error').size() > 0) {
|
||||
return $.msg.tips('表单验证不成功,请输入需要的内容!');
|
||||
}
|
||||
list.push({name: '请输入名称', type: 'click', sub_button: []});
|
||||
};
|
||||
// 增加菜单选项
|
||||
$rootScope.addItem = function (list) {
|
||||
if (!vali.checkAllInput()) return;
|
||||
list.push({name: '请输入名称', type: 'click', sub_button: []});
|
||||
};
|
||||
|
||||
// 移除菜单
|
||||
$rootScope.delItem = function (one, two) {
|
||||
var tmp = [], _two = null;
|
||||
if (two) {
|
||||
for (var i in one.sub_button) if (one.sub_button[i] !== two) {
|
||||
tmp.push(one.sub_button[i]);
|
||||
if (one.sub_button[i].active) _two = one.sub_button[i];
|
||||
// 移除菜单
|
||||
$rootScope.delItem = function (one, two) {
|
||||
var tmp = [], _two = null;
|
||||
if (two) {
|
||||
for (var i in one.sub_button) if (one.sub_button[i] !== two) {
|
||||
tmp.push(one.sub_button[i]);
|
||||
if (one.sub_button[i].active) _two = one.sub_button[i];
|
||||
}
|
||||
one.sub_button = tmp;
|
||||
return $rootScope.setActiveItem(one, _two);
|
||||
}
|
||||
one.sub_button = tmp;
|
||||
return $rootScope.setActiveItem(one, _two);
|
||||
for (var i in $rootScope.list) if (one !== $rootScope.list[i]) tmp.push($rootScope.list[i]);
|
||||
$rootScope.list = tmp;
|
||||
if ($rootScope.list.length > 1) $rootScope.setActiveItem($rootScope.list[0])
|
||||
};
|
||||
|
||||
// 切换选择菜单
|
||||
$rootScope.setActiveItem = function (one, two) {
|
||||
if (!vali.checkAllInput()) return;
|
||||
$rootScope.list.forEach(function (item) {
|
||||
item.show = item === one;
|
||||
item.active = two ? false : (item === one)
|
||||
item.sub_button.forEach(function (subitem) {
|
||||
subitem.active = subitem === two;
|
||||
});
|
||||
});
|
||||
$rootScope.item = two || one || {};
|
||||
$rootScope.item.type = $rootScope.item.type || 'click';
|
||||
if ($rootScope.item.type === 'click') setTimeout(function () {
|
||||
layui.form.render('select'), ($rootScope.item.key = $('[lay-filter="key"]').val());
|
||||
}, 50);
|
||||
};
|
||||
|
||||
// 下拉列表处理
|
||||
$rootScope.$watch('item', function () {
|
||||
if ($rootScope.item.type === 'click') setTimeout(function () {
|
||||
layui.form.render('select'), ($rootScope.item.key = $('[lay-filter="key"]').val());
|
||||
}, 50)
|
||||
}, true);
|
||||
|
||||
layui.form.on('select(key)', function (data) {
|
||||
$rootScope.item.key = data.value;
|
||||
});
|
||||
|
||||
// 提交数据
|
||||
$rootScope.submit = function () {
|
||||
if (!vali.checkAllInput()) return false;
|
||||
$.form.load('{:url("push")}', {data: angular.toJson($rootScope.list)}, 'post');
|
||||
}
|
||||
for (var i in $rootScope.list) if (one !== $rootScope.list[i]) tmp.push($rootScope.list[i]);
|
||||
$rootScope.list = tmp;
|
||||
if ($rootScope.list.length > 1) $rootScope.setActiveItem($rootScope.list[0])
|
||||
};
|
||||
|
||||
// 切换选择菜单
|
||||
$rootScope.setActiveItem = function (one, two) {
|
||||
for (var i in $rootScope.list) {
|
||||
$rootScope.list[i].show = ($rootScope.list[i] === one);
|
||||
$rootScope.list[i].active = two ? false : ($rootScope.list[i] === one);
|
||||
}
|
||||
for (var i in $rootScope.list) for (var j in $rootScope.list[i].sub_button) {
|
||||
$rootScope.list[i].sub_button[j].active = ($rootScope.list[i].sub_button[j] === two)
|
||||
}
|
||||
$rootScope.item = two || one || {};
|
||||
$rootScope.item.type = $rootScope.item.type || 'click';
|
||||
if ($rootScope.item.type === 'click') setTimeout(function () {
|
||||
layui.form.render('select');
|
||||
$rootScope.item.key = $('[lay-filter="key"]').val();
|
||||
}, 50);
|
||||
};
|
||||
|
||||
// 下拉列表处理
|
||||
$rootScope.$watch('item', function () {
|
||||
if ($rootScope.item.type === 'click') setTimeout(function () {
|
||||
layui.form.render('select');
|
||||
$rootScope.item.key = $('[lay-filter="key"]').val();
|
||||
}, 50)
|
||||
}, true);
|
||||
|
||||
layui.form.on('select(key)', function (data) {
|
||||
$rootScope.item.key = data.value;
|
||||
});
|
||||
|
||||
// 提交数据
|
||||
$rootScope.submit = function () {
|
||||
$vali.checkAllInput();
|
||||
if ($form.find('.validate-error').size() > 0) return false;
|
||||
$.form.load('{:url("push")}', {data: angular.toJson($rootScope.list)}, 'post');
|
||||
}
|
||||
}
|
||||
});
|
||||
});
|
||||
</script>
|
||||
|
||||
|
@ -81,135 +81,130 @@
|
||||
require(['angular', 'ckeditor'], function () {
|
||||
|
||||
var editor;
|
||||
var $form = $('form[name="news"]');
|
||||
var $vali = $form.vali().data('validate');
|
||||
$('form[name="news"]').vali(null, function () {
|
||||
var vali = this, app = angular.module("NewsEditor", []).run(callback);
|
||||
angular.bootstrap(document.getElementById(app.name), [app.name]);
|
||||
|
||||
var app = angular.module("NewsEditor", []).run(callback);
|
||||
angular.bootstrap(document.getElementById(app.name), [app.name]);
|
||||
function callback($rootScope) {
|
||||
$rootScope.list = [];
|
||||
$rootScope.item = {};
|
||||
|
||||
function callback($rootScope) {
|
||||
$rootScope.list = [];
|
||||
$rootScope.item = {};
|
||||
|
||||
$.form.load('{:request()->url()}', {output: 'json'}, 'get', function (ret) {
|
||||
return $rootScope.$apply(function () {
|
||||
apply((ret.data || {articles: []}).articles || []);
|
||||
}), false;
|
||||
});
|
||||
|
||||
function apply(list) {
|
||||
if (list.length < 1) list.push({
|
||||
title: '新建图文', author: '管理员', content: '文章内容',
|
||||
read_num: 0, local_url: '__FULL__/static/theme/img/image.png',
|
||||
});
|
||||
for (var i in list) {
|
||||
list[i].active = false;
|
||||
list[i].style = "background-image:url('" + list[i].local_url + "')";
|
||||
}
|
||||
$rootScope.list = list;
|
||||
$rootScope.item = $rootScope.list[0];
|
||||
$rootScope.setItemValue('active', true);
|
||||
$('.layui-card-body.layui-hide').removeClass('layui-hide');
|
||||
setTimeout(function () {
|
||||
if (editor) editor.destroy();
|
||||
editor = window.createEditor('[name="content"]');
|
||||
if (typeof editor !== 'undefined') {
|
||||
editor.setData($rootScope.item.content);
|
||||
$vali.checkAllInput();
|
||||
} else $('[name="content"]').on('editor.init', function (event, myEditor) {
|
||||
myEditor.setData($rootScope.item.content);
|
||||
editor = myEditor;
|
||||
$vali.checkAllInput();
|
||||
$.form.load('{:request()->url()}', {output: 'json'}, 'get', function (ret) {
|
||||
$rootScope.$apply(function () {
|
||||
apply((ret.data || {articles: []}).articles || []);
|
||||
});
|
||||
}, 100);
|
||||
}
|
||||
return false;
|
||||
});
|
||||
|
||||
$rootScope.upItem = function (index, $event) {
|
||||
$event.stopPropagation();
|
||||
var tmp = [], cur = $rootScope.list[index];
|
||||
if (index < 1) return false;
|
||||
for (var i in $rootScope.list) {
|
||||
(parseInt(i) === parseInt(index) - 1) && tmp.push(cur);
|
||||
(parseInt(i) !== parseInt(index)) && tmp.push($rootScope.list[i]);
|
||||
}
|
||||
apply(tmp);
|
||||
};
|
||||
$rootScope.dnItem = function (index, $event) {
|
||||
$event.stopPropagation();
|
||||
var tmp = [], cur = $rootScope.list[index];
|
||||
if (index > $rootScope.list.length - 2) return false;
|
||||
for (var i in $rootScope.list) {
|
||||
(parseInt(i) !== parseInt(index)) && tmp.push($rootScope.list[i]);
|
||||
(parseInt(i) === parseInt(index) + 1) && tmp.push(cur);
|
||||
}
|
||||
apply(tmp);
|
||||
};
|
||||
$rootScope.delItem = function (index, $event) {
|
||||
$event.stopPropagation();
|
||||
var list = $rootScope.list, temp = [];
|
||||
for (var i in list) (parseInt(i) !== parseInt(index)) && temp.push(list[i]);
|
||||
apply(temp);
|
||||
};
|
||||
$rootScope.setItem = function (index, $event) {
|
||||
$event.stopPropagation();
|
||||
$vali.checkAllInput();
|
||||
if ($form.find('.validate-error').size() > 0) return 0;
|
||||
if (editor.getData().length < 1) {
|
||||
return $.msg.tips('文章内容不能为空,请输入文章内容!');
|
||||
}
|
||||
for (var i in $rootScope.list) if (parseInt(i) !== parseInt(index)) {
|
||||
$rootScope.list[i].active = false;
|
||||
} else {
|
||||
$rootScope.item.content = editor.getData();
|
||||
$rootScope.item = $rootScope.list[i];
|
||||
editor.setData($rootScope.item.content);
|
||||
function apply(list) {
|
||||
if (list.length < 1) list.push({
|
||||
title: '新建图文', author: '管理员', content: '文章内容',
|
||||
read_num: 0, local_url: '__FULL__/static/theme/img/image.png',
|
||||
});
|
||||
for (var i in list) {
|
||||
list[i].active = false;
|
||||
list[i].style = "background-image:url('" + list[i].local_url + "')";
|
||||
}
|
||||
$rootScope.list = list;
|
||||
$rootScope.item = $rootScope.list[0];
|
||||
$rootScope.setItemValue('active', true);
|
||||
$('.layui-card-body.layui-hide').removeClass('layui-hide');
|
||||
setTimeout(function () {
|
||||
if (editor) editor.destroy();
|
||||
editor = window.createEditor('[name="content"]');
|
||||
if (typeof editor !== 'undefined') {
|
||||
editor.setData($rootScope.item.content);
|
||||
vali.checkAllInput();
|
||||
} else $('[name="content"]').on('editor.init', function (event, myEditor) {
|
||||
(editor = myEditor).setData($rootScope.item.content);
|
||||
vali.checkAllInput();
|
||||
});
|
||||
}, 100);
|
||||
}
|
||||
};
|
||||
$rootScope.setItemValue = function (name, value) {
|
||||
$rootScope.item[name] = value;
|
||||
$rootScope.item.style = "background-image:url('" + $rootScope.item.local_url + "')";
|
||||
};
|
||||
$rootScope.addItem = function () {
|
||||
if ($rootScope.list.length > 7) {
|
||||
return $.msg.tips('最多允许增加7篇文章哦!');
|
||||
}
|
||||
$rootScope.list.push({
|
||||
title: '新建图文',
|
||||
author: '管理员',
|
||||
content: '文章内容',
|
||||
read_num: 0,
|
||||
local_url: '__FULL__/static/theme/img/image.png',
|
||||
style: "background-image:url('__FULL__/static/theme/img/image.png')"
|
||||
|
||||
$rootScope.upItem = function (index, $event) {
|
||||
$event.stopPropagation();
|
||||
var tmp = [], cur = $rootScope.list[index];
|
||||
if (index < 1) return false;
|
||||
for (var i in $rootScope.list) {
|
||||
(parseInt(i) === parseInt(index) - 1) && tmp.push(cur);
|
||||
(parseInt(i) !== parseInt(index)) && tmp.push($rootScope.list[i]);
|
||||
}
|
||||
apply(tmp);
|
||||
};
|
||||
$rootScope.dnItem = function (index, $event) {
|
||||
$event.stopPropagation();
|
||||
var tmp = [], cur = $rootScope.list[index];
|
||||
if (index > $rootScope.list.length - 2) return false;
|
||||
for (var i in $rootScope.list) {
|
||||
(parseInt(i) !== parseInt(index)) && tmp.push($rootScope.list[i]);
|
||||
(parseInt(i) === parseInt(index) + 1) && tmp.push(cur);
|
||||
}
|
||||
apply(tmp);
|
||||
};
|
||||
$rootScope.delItem = function (index, $event) {
|
||||
$event.stopPropagation();
|
||||
var list = $rootScope.list, temp = [];
|
||||
for (var i in list) (parseInt(i) !== parseInt(index)) && temp.push(list[i]);
|
||||
apply(temp);
|
||||
};
|
||||
$rootScope.setItem = function (index, $event) {
|
||||
$event.stopPropagation();
|
||||
if (!vali.checkAllInput()) return;
|
||||
if (editor.getData().length < 1) {
|
||||
return $.msg.notify('操作提示', '文章内容不能为空,请输入内容!', 3000, {type: 'error', width: '400px'})
|
||||
}
|
||||
for (var i in $rootScope.list) if (parseInt(i) !== parseInt(index)) {
|
||||
$rootScope.list[i].active = false;
|
||||
} else {
|
||||
$rootScope.item.content = editor.getData();
|
||||
$rootScope.item = $rootScope.list[i];
|
||||
editor.setData($rootScope.item.content);
|
||||
$rootScope.setItemValue('active', true);
|
||||
}
|
||||
};
|
||||
$rootScope.setItemValue = function (name, value) {
|
||||
$rootScope.item[name] = value;
|
||||
$rootScope.item.style = "background-image:url('" + $rootScope.item.local_url + "')";
|
||||
};
|
||||
$rootScope.addItem = function () {
|
||||
if ($rootScope.list.length >= 7) {
|
||||
return $.msg.notify('操作提示', '最多允许增加 7 篇文章哦!', 3000, {type: 'error', width: '400px'})
|
||||
}
|
||||
$rootScope.list.push({
|
||||
title: '新建图文',
|
||||
author: '管理员',
|
||||
content: '文章内容',
|
||||
read_num: 0,
|
||||
local_url: '__FULL__/static/theme/img/image.png',
|
||||
style: "background-image:url('__FULL__/static/theme/img/image.png')"
|
||||
});
|
||||
};
|
||||
$rootScope.submit = function () {
|
||||
if (!vali.checkAllInput()) return false;
|
||||
$rootScope.item.content = editor.getData();
|
||||
var data = [];
|
||||
for (var i in $rootScope.list) data.push({
|
||||
id: $rootScope.list[i].id,
|
||||
title: $rootScope.list[i].title,
|
||||
author: $rootScope.list[i].author,
|
||||
digest: $rootScope.list[i].digest,
|
||||
content: $rootScope.list[i].content,
|
||||
read_num: $rootScope.list[i].read_num,
|
||||
local_url: $rootScope.list[i].local_url,
|
||||
show_cover_pic: $rootScope.list[i].show_cover_pic ? 1 : 0,
|
||||
content_source_url: $rootScope.list[i].content_source_url,
|
||||
});
|
||||
$.form.load('{:request()->url()}', {data: data}, "post");
|
||||
};
|
||||
$('[name="local_url"]').on('change', function () {
|
||||
var value = this.value;
|
||||
$rootScope.$apply(function () {
|
||||
$rootScope.setItemValue('local_url', value);
|
||||
});
|
||||
});
|
||||
};
|
||||
$rootScope.submit = function () {
|
||||
$vali.checkAllInput();
|
||||
if ($form.find('.validate-error').size() > 0) {
|
||||
return $.msg.tips('表单验证不成功,请输入需要的内容!');
|
||||
}
|
||||
$rootScope.item.content = editor.getData();
|
||||
var data = [];
|
||||
for (var i in $rootScope.list) data.push({
|
||||
id: $rootScope.list[i].id,
|
||||
title: $rootScope.list[i].title,
|
||||
author: $rootScope.list[i].author,
|
||||
digest: $rootScope.list[i].digest,
|
||||
content: $rootScope.list[i].content,
|
||||
read_num: $rootScope.list[i].read_num,
|
||||
local_url: $rootScope.list[i].local_url,
|
||||
show_cover_pic: $rootScope.list[i].show_cover_pic ? 1 : 0,
|
||||
content_source_url: $rootScope.list[i].content_source_url,
|
||||
});
|
||||
$.form.load('{:request()->url()}', {data: data}, "post");
|
||||
};
|
||||
$('[name="local_url"]').on('change', function (value) {
|
||||
value = this.value;
|
||||
$rootScope.$apply(function () {
|
||||
$rootScope.setItemValue('local_url', value);
|
||||
});
|
||||
});
|
||||
}
|
||||
}
|
||||
});
|
||||
});
|
||||
</script>
|
||||
{/block}
|
@ -23,6 +23,7 @@ layui.config({base: baseRoot + 'plugs/layui_exts/'});
|
||||
window.form = layui.form, window.layer = layui.layer;
|
||||
window.laytpl = layui.laytpl, window.laydate = layui.laydate;
|
||||
window.jQuery = window.$ = window.jQuery || window.$ || layui.$;
|
||||
window.jQuery.ajaxSetup({xhrFields: {withCredentials: true}});
|
||||
|
||||
/*! 配置 require 参数 */
|
||||
require.config({
|
||||
@ -61,7 +62,7 @@ require.config({
|
||||
}, shim: {
|
||||
'jszip': {deps: ['filesaver']},
|
||||
'excel': {deps: [baseRoot + 'plugs/layui_exts/excel.js']},
|
||||
'notify': {deps: ['css!' + baseRoot + 'plugs/notify/light.css']},
|
||||
'notify': {deps: ['css!' + baseRoot + 'plugs/notify/theme.css']},
|
||||
'cropper': {deps: ['css!' + baseRoot + 'plugs/cropper/cropper.min.css']},
|
||||
'websocket': {deps: [baseRoot + 'plugs/socket/swfobject.js']},
|
||||
'ckeditor5': {deps: ['jquery', 'upload', 'css!' + baseRoot + 'plugs/ckeditor5/ckeditor.css']},
|
||||
@ -203,7 +204,7 @@ $(function () {
|
||||
// https://www.jq22.com/demo/jquerygrowl-notification202104021049
|
||||
this.notify = function (title, message, time, option) {
|
||||
require(['notify'], function (Notify) {
|
||||
Notify.notify(Object.assign({title: title || '', description: message || '', position: 'top-right', closeTimeout: time || 3000}, (option || {})));
|
||||
Notify.notify(Object.assign({title: title || '', description: message || '', position: 'top-right', closeTimeout: time || 3000, width: '400px'}, option || {}));
|
||||
});
|
||||
};
|
||||
/*! 页面加载层 */
|
||||
@ -572,22 +573,21 @@ $(function () {
|
||||
var $this = $(this), tags = this.value ? this.value.split(',') : [];
|
||||
var $text = $('<textarea class="layui-input layui-input-inline layui-tag-input"></textarea>');
|
||||
var $tags = $('<div class="layui-tags"></div>').append($text);
|
||||
$this.parent().append($tags), $text.off('keydown blur'), (tags.length > 0 && showTags(tags));
|
||||
$this.parent().append($tags) && $text.off('keydown blur') && (tags.length > 0 && showTags(tags));
|
||||
$text.on('blur keydown', function (event, value) {
|
||||
if (event.keyCode === 13 || event.type === 'blur') {
|
||||
event.preventDefault(), (value = $text.val().replace(/^\s*|\s*$/g, ''));
|
||||
if (tags.indexOf($(this).val()) > -1) return layer.msg('该标签已经存在!');
|
||||
if (value.length > 0) tags.push(value), $this.val(tags.join(',')), showTags([value]), this.focus(), $text.val('');
|
||||
if (tags.indexOf($(this).val()) > -1) return $.msg.notify('温馨提示', '该标签已经存在!', 3000, {type: 'error', width: 280});
|
||||
else if (value.length > 0) tags.push(value), $this.val(tags.join(',')), showTags([value]), this.focus(), $text.val('');
|
||||
}
|
||||
});
|
||||
|
||||
function showTags(tagsArr) {
|
||||
$(tagsArr).each(function (idx, text, elem) {
|
||||
elem = $('<div class="layui-tag"></div>').html(text + '<i class="layui-icon">ဆ</i>');
|
||||
elem.on('click', 'i', function (tagText, tagIdx) {
|
||||
tagText = $(this).parent().text(), tagIdx = tags.indexOf(tagText);
|
||||
tags.splice(tagIdx, 1), $(this).parent().remove(), $this.val(tags.join(','));
|
||||
}), $tags.append(elem, $text);
|
||||
$(tagsArr).each(function (idx, text) {
|
||||
$('<div class="layui-tag"></div>').data('value', text).on('click', 'i', function () {
|
||||
tags.splice(tags.indexOf($(this).parent().data('value')), 1);
|
||||
$this.val(tags.join(',')) && $(this).parent().remove();
|
||||
}).insertBefore($text).html(text + '<i class="layui-icon">ဆ</i>');
|
||||
});
|
||||
}
|
||||
});
|
||||
@ -753,9 +753,9 @@ $(function () {
|
||||
/*! 创建表单验证 */
|
||||
$.vali = function (form, done, init) {
|
||||
require(['validate'], function (Validate) {
|
||||
/** @type {import("./plugs/admin/validate")|Validate} */
|
||||
/** @type {import("./plugs/admin/validate")|Validate}*/
|
||||
var vali = $(form).data('validate') || new Validate(form, onConfirm);
|
||||
typeof init === 'function' && init.call(vali, $(form).formToJson());
|
||||
typeof init === 'function' && init.call(vali, $(form).formToJson(), vali);
|
||||
typeof done === 'function' && vali.addDoneEvent(done);
|
||||
});
|
||||
};
|
||||
|
File diff suppressed because one or more lines are too long
File diff suppressed because one or more lines are too long
@ -1,27 +1,27 @@
|
||||
.growl-notification {
|
||||
/* min-height: 56px; */
|
||||
width: 320px;
|
||||
z-index: 99999999;
|
||||
position: fixed;
|
||||
word-break: break-all;
|
||||
background: #fff;
|
||||
box-shadow: 0 0 30px 0 rgba(0, 0, 0, .1);
|
||||
border-radius: 4px;
|
||||
box-shadow: 0 0 6px 1px rgba(0, 0, 0, .2)
|
||||
}
|
||||
|
||||
.growl-notification:before {
|
||||
border-radius: 4px 0 0 4px;
|
||||
top: 0;
|
||||
left: 0;
|
||||
width: 4px;
|
||||
bottom: 0;
|
||||
content: "";
|
||||
left: 0;
|
||||
position: absolute;
|
||||
top: 0;
|
||||
width: 4px
|
||||
border-radius: 4px 0 0 4px;
|
||||
}
|
||||
|
||||
.growl-notification__progress {
|
||||
display: none;
|
||||
width: 100%;
|
||||
height: 100%;
|
||||
display: none;
|
||||
position: absolute;
|
||||
border-radius: 4px 4px 0 0;
|
||||
}
|
||||
@ -40,7 +40,7 @@
|
||||
display: flex;
|
||||
padding: 10px 25px;
|
||||
position: relative;
|
||||
line-height: 1.6em;
|
||||
line-height: 2em;
|
||||
align-items: center;
|
||||
}
|
||||
|
||||
@ -81,12 +81,12 @@
|
||||
}
|
||||
|
||||
.growl-notification__close {
|
||||
top: 8px;
|
||||
right: 8px;
|
||||
cursor: pointer;
|
||||
position: absolute;
|
||||
font-size: 12px;
|
||||
line-height: 12px;
|
||||
position: absolute;
|
||||
right: 8px;
|
||||
top: 8px;
|
||||
transition: color .1s
|
||||
}
|
||||
|
||||
@ -110,17 +110,18 @@
|
||||
|
||||
.growl-notification__title {
|
||||
color: #333;
|
||||
font-size: 16px;
|
||||
font-size: 15px;
|
||||
font-weight: 600;
|
||||
}
|
||||
|
||||
.growl-notification__desc {
|
||||
color: #000
|
||||
color: #333;
|
||||
font-size: 13px;
|
||||
}
|
||||
|
||||
.growl-notification__title + .growl-notification__desc {
|
||||
color: rgba(0, 0, 0, .6);
|
||||
margin-top: 5px;
|
||||
color: rgba(0, 0, 0, .7);
|
||||
font-size: 14px;
|
||||
}
|
||||
|
||||
.growl-notification--close-on-click {
|
@ -954,6 +954,20 @@ input:not(.layui-hide,[type=hidden]) {
|
||||
position: relative;
|
||||
transform: scale(1);
|
||||
text-align: center;
|
||||
overflow: hidden;
|
||||
|
||||
&::before {
|
||||
top: 0;
|
||||
left: 0;
|
||||
right: 0;
|
||||
bottom: 0;
|
||||
width: 100%;
|
||||
height: 100%;
|
||||
content: '';
|
||||
display: block;
|
||||
position: absolute;
|
||||
background: rgba(0, 0, 0, .3);
|
||||
}
|
||||
|
||||
&::after {
|
||||
left: 50%;
|
||||
|
@ -177,6 +177,10 @@ form.layui-form fieldset {
|
||||
|
||||
.layui-nav {
|
||||
.layui-nav-item {
|
||||
.layui-elip {
|
||||
padding-right: 35px !important;
|
||||
}
|
||||
|
||||
.layui-nav-more {
|
||||
right: 15px;
|
||||
font-size: 14px !important;
|
||||
|
@ -100,7 +100,6 @@
|
||||
}
|
||||
|
||||
+ a.layui-elip {
|
||||
padding-right: 35px;
|
||||
|
||||
img {
|
||||
width: 20px;
|
||||
|
File diff suppressed because one or more lines are too long
File diff suppressed because one or more lines are too long
Loading…
x
Reference in New Issue
Block a user