同步插件代码

This commit is contained in:
邹景立 2023-03-25 12:57:39 +08:00
parent a6afa20850
commit 15d33afdd2
34 changed files with 631 additions and 512 deletions

1
.gitignore vendored
View File

@ -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

View File

@ -20,7 +20,7 @@ use think\admin\Plugin;
/**
* 插件服务注册
* Class Service
* @class Service
* @package app\admin
*/
class Service extends Plugin

View File

@ -24,7 +24,7 @@ use think\admin\service\AdminService;
/**
* 系统权限管理
* Class Auth
* @class Auth
* @package app\admin\controller
*/
class Auth extends Controller

View File

@ -22,7 +22,7 @@ use think\admin\model\SystemBase;
/**
* 数据字典管理
* Class Base
* @class Base
* @package app\admin\controller
*/
class Base extends Controller

View File

@ -27,7 +27,7 @@ use think\admin\storage\TxcosStorage;
/**
* 系统参数配置
* Class Config
* @class Config
* @package app\admin\controller
*/
class Config extends Controller

View File

@ -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()
{

View File

@ -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()
{

View File

@ -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()
{

View File

@ -25,7 +25,7 @@ use think\admin\service\NodeService;
/**
* 系统菜单管理
* Class Menu
* @class Menu
* @package app\admin\controller
*/
class Menu extends Controller

View File

@ -25,7 +25,7 @@ use think\exception\HttpResponseException;
/**
* 系统日志管理
* Class Oplog
* @class Oplog
* @package app\admin\controller
*/
class Oplog extends Controller

View File

@ -27,7 +27,7 @@ use think\exception\HttpResponseException;
/**
* 系统任务管理
* Class Queue
* @class Queue
* @package app\admin\controller
*/
class Queue extends Controller

View File

@ -25,7 +25,7 @@ use think\admin\service\AdminService;
/**
* 系统用户管理
* Class User
* @class User
* @package app\admin\controller
*/
class User extends Controller

View File

@ -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('请使用超管账号操作!');
}
}
}

View File

@ -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
*/

View File

@ -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('请使用超管账号操作!');
}
}
}

View File

@ -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

View File

@ -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>

View File

@ -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']);

View File

@ -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;
});
},

View File

@ -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"}

View File

@ -78,7 +78,7 @@ class Menu extends Controller
{
try {
WechatService::WeChatMenu()->delete();
$this->success('菜单取消成功,重新订阅可立即生效');
$this->success('公众号菜单取消成功!');
} catch (HttpResponseException $exception) {
sysoplog('微信管理', '取消微信菜单成功');
throw $exception;

View File

@ -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('未获取到授权!');
}
}
}

View 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));
}
}

View File

@ -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>

View File

@ -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}

View File

@ -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">&#x1006;</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">&#x1006;</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

View File

@ -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 {

View File

@ -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%;

View File

@ -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;

View File

@ -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