first commit
This commit is contained in:
commit
ace33d202f
17
.env
Normal file
17
.env
Normal file
@ -0,0 +1,17 @@
|
|||||||
|
APP_DEBUG = false
|
||||||
|
|
||||||
|
[APP]
|
||||||
|
DEFAULT_TIMEZONE = Asia/Shanghai
|
||||||
|
|
||||||
|
[DATABASE]
|
||||||
|
TYPE = mysql
|
||||||
|
HOSTNAME = 127.0.0.1
|
||||||
|
DATABASE = ruankao
|
||||||
|
USERNAME = ruankao
|
||||||
|
PASSWORD = 123456
|
||||||
|
HOSTPORT = 3306
|
||||||
|
CHARSET = utf8
|
||||||
|
DEBUG = true
|
||||||
|
|
||||||
|
[LANG]
|
||||||
|
default_lang = zh-cn
|
||||||
1
.example.env
Normal file
1
.example.env
Normal file
@ -0,0 +1 @@
|
|||||||
|
APP_DEBUG = true
[APP]
DEFAULT_TIMEZONE = Asia/Shanghai
[DATABASE]
TYPE = mysql
HOSTNAME = 127.0.0.1
DATABASE = test
USERNAME = username
PASSWORD = password
HOSTPORT = 3306
CHARSET = utf8
DEBUG = true
[LANG]
default_lang = zh-cn
|
||||||
6
.gitignore
vendored
Normal file
6
.gitignore
vendored
Normal file
@ -0,0 +1,6 @@
|
|||||||
|
/.idea
|
||||||
|
/.vscode
|
||||||
|
/vendor
|
||||||
|
runtime
|
||||||
|
*.log
|
||||||
|
config/database.php
|
||||||
42
.travis.yml
Normal file
42
.travis.yml
Normal file
@ -0,0 +1,42 @@
|
|||||||
|
sudo: false
|
||||||
|
|
||||||
|
language: php
|
||||||
|
|
||||||
|
branches:
|
||||||
|
only:
|
||||||
|
- stable
|
||||||
|
|
||||||
|
cache:
|
||||||
|
directories:
|
||||||
|
- $HOME/.composer/cache
|
||||||
|
|
||||||
|
before_install:
|
||||||
|
- composer self-update
|
||||||
|
|
||||||
|
install:
|
||||||
|
- composer install --no-dev --no-interaction --ignore-platform-reqs
|
||||||
|
- zip -r --exclude='*.git*' --exclude='*.zip' --exclude='*.travis.yml' ThinkPHP_Core.zip .
|
||||||
|
- composer require --update-no-dev --no-interaction "topthink/think-image:^1.0"
|
||||||
|
- composer require --update-no-dev --no-interaction "topthink/think-migration:^1.0"
|
||||||
|
- composer require --update-no-dev --no-interaction "topthink/think-captcha:^1.0"
|
||||||
|
- composer require --update-no-dev --no-interaction "topthink/think-mongo:^1.0"
|
||||||
|
- composer require --update-no-dev --no-interaction "topthink/think-worker:^1.0"
|
||||||
|
- composer require --update-no-dev --no-interaction "topthink/think-helper:^1.0"
|
||||||
|
- composer require --update-no-dev --no-interaction "topthink/think-queue:^1.0"
|
||||||
|
- composer require --update-no-dev --no-interaction "topthink/think-angular:^1.0"
|
||||||
|
- composer require --dev --update-no-dev --no-interaction "topthink/think-testing:^1.0"
|
||||||
|
- zip -r --exclude='*.git*' --exclude='*.zip' --exclude='*.travis.yml' ThinkPHP_Full.zip .
|
||||||
|
|
||||||
|
script:
|
||||||
|
- php think unit
|
||||||
|
|
||||||
|
deploy:
|
||||||
|
provider: releases
|
||||||
|
api_key:
|
||||||
|
secure: TSF6bnl2JYN72UQOORAJYL+CqIryP2gHVKt6grfveQ7d9rleAEoxlq6PWxbvTI4jZ5nrPpUcBUpWIJHNgVcs+bzLFtyh5THaLqm39uCgBbrW7M8rI26L8sBh/6nsdtGgdeQrO/cLu31QoTzbwuz1WfAVoCdCkOSZeXyT/CclH99qV6RYyQYqaD2wpRjrhA5O4fSsEkiPVuk0GaOogFlrQHx+C+lHnf6pa1KxEoN1A0UxxVfGX6K4y5g4WQDO5zT4bLeubkWOXK0G51XSvACDOZVIyLdjApaOFTwamPcD3S1tfvuxRWWvsCD5ljFvb2kSmx5BIBNwN80MzuBmrGIC27XLGOxyMerwKxB6DskNUO9PflKHDPI61DRq0FTy1fv70SFMSiAtUv9aJRT41NQh9iJJ0vC8dl+xcxrWIjU1GG6+l/ZcRqVx9V1VuGQsLKndGhja7SQ+X1slHl76fRq223sMOql7MFCd0vvvxVQ2V39CcFKao/LB1aPH3VhODDEyxwx6aXoTznvC/QPepgWsHOWQzKj9ftsgDbsNiyFlXL4cu8DWUty6rQy8zT2b4O8b1xjcwSUCsy+auEjBamzQkMJFNlZAIUrukL/NbUhQU37TAbwsFyz7X0E/u/VMle/nBCNAzgkMwAUjiHM6FqrKKBRWFbPrSIixjfjkCnrMEPw=
|
||||||
|
file:
|
||||||
|
- ThinkPHP_Core.zip
|
||||||
|
- ThinkPHP_Full.zip
|
||||||
|
skip_cleanup: true
|
||||||
|
on:
|
||||||
|
tags: true
|
||||||
41
LICENSE.txt
Normal file
41
LICENSE.txt
Normal file
@ -0,0 +1,41 @@
|
|||||||
|
云泽网商业软件许可证协议
|
||||||
|
版本 1.0 - 2025年
|
||||||
|
|
||||||
|
版权所有 (c) 2025 云泽网。保留所有权利。
|
||||||
|
|
||||||
|
重要提示:在使用本软件前,请仔细阅读以下条款和条件。下载、安装或使用本软件即表示您同意受本协议的约束。
|
||||||
|
|
||||||
|
1. 授权范围
|
||||||
|
1.1 本软件仅供评估和测试目的使用,不得用于任何商业用途。
|
||||||
|
1.2 任何商业用途(包括但不限于生产环境使用、商业服务提供、商业产品集成等)必须获得云泽网的书面授权许可。
|
||||||
|
|
||||||
|
2. 限制条款
|
||||||
|
2.1 禁止移除、修改或隐藏任何版权声明、商标标识或专有权利通知。
|
||||||
|
2.2 禁止将本软件用于生产环境或任何可能产生商业收益的场景。
|
||||||
|
2.3 禁止对本软件进行转售、分发、出租、出借或再许可。
|
||||||
|
2.4 禁止对本软件进行反向工程、反编译或反汇编。
|
||||||
|
|
||||||
|
3. 商业授权
|
||||||
|
3.1 商业用户需通过官方渠道购买商业授权许可证。
|
||||||
|
3.2 商业授权将提供:
|
||||||
|
- 合法的商业使用权利
|
||||||
|
- 正式的技术支持服务
|
||||||
|
- 软件更新和维护
|
||||||
|
3.3 授权购买请联系:357099073@qq.com
|
||||||
|
3.4 官方网站:https://www.yunzer.cn
|
||||||
|
|
||||||
|
4. 免责声明
|
||||||
|
4.1 本软件按"原样"提供,不作任何明示或暗示的保证。
|
||||||
|
4.2 在任何情况下,云泽网不对因使用或不能使用本软件所发生的特殊、附带、间接或结果性损害承担任何责任。
|
||||||
|
|
||||||
|
5. 违约与终止
|
||||||
|
5.1 如违反本协议任何条款,您的使用权利将自动终止。
|
||||||
|
5.2 云泽网保留对任何未经授权商业使用行为追究法律责任的权利。
|
||||||
|
|
||||||
|
6. 其他条款
|
||||||
|
6.1 本协议受中华人民共和国法律管辖。
|
||||||
|
6.2 云泽网保留随时修改本协议条款的权利,修改后的协议将在官方网站公布后生效。
|
||||||
|
|
||||||
|
如需商业使用授权,请通过以下方式联系我们:
|
||||||
|
电子邮件:357099073@qq.com
|
||||||
|
官方网站:https://www.yunzer.cn
|
||||||
42
README.md
Normal file
42
README.md
Normal file
@ -0,0 +1,42 @@
|
|||||||
|
# phpAdmin 后台管理 (admin-thinkphp-Layui2)
|
||||||
|
|
||||||
|
- [官网:http://www.yunzer.cn](http://www.yunzer.cn)
|
||||||
|
- [演示:admin-thinkphp-layui](admin-thinkphp-layui)
|
||||||
|
|
||||||
|
## 一、介绍
|
||||||
|
|
||||||
|
- phpAdmin 后台管理
|
||||||
|
|
||||||
|
- [admin-thinkphp-layui]
|
||||||
|
- 后端框架:Thinkphp6.1
|
||||||
|
- Layui2.8
|
||||||
|
|
||||||
|
## 二、安装教程
|
||||||
|
|
||||||
|
- [根据 Thinkphp6 安装](https://www.kancloud.cn/manual/thinkphp6_0/1037479)
|
||||||
|
- 伪静态
|
||||||
|
```
|
||||||
|
location ~* (runtime|application)/{
|
||||||
|
return 403;
|
||||||
|
}
|
||||||
|
location / {
|
||||||
|
if (!-e $request_filename){
|
||||||
|
rewrite ^(.*)$ /index.php?s=$1 last; break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
```
|
||||||
|
|
||||||
|
## 三、使用说明
|
||||||
|
|
||||||
|
- 1、域名指向 public 目录下
|
||||||
|
- 2、后台管理系统访问网址:http://www.xxx.com/index.php/bews/index/index
|
||||||
|
|
||||||
|
## 四、鸣谢
|
||||||
|
|
||||||
|
- [PHP 中文网](https://www.php.cn)
|
||||||
|
- [Thinkphp](http://www.thinkphp.cn)
|
||||||
|
- [Layui](https://layui.dev)
|
||||||
|
|
||||||
|
## 五、技术支持
|
||||||
|
|
||||||
|
- QQ:357099073
|
||||||
1
app/.htaccess
Normal file
1
app/.htaccess
Normal file
@ -0,0 +1 @@
|
|||||||
|
deny from all
|
||||||
55
app/AppApi.php
Normal file
55
app/AppApi.php
Normal file
@ -0,0 +1,55 @@
|
|||||||
|
<?php
|
||||||
|
/**
|
||||||
|
* 商业使用授权协议
|
||||||
|
*
|
||||||
|
* Copyright (c) 2025 [云泽网]. 保留所有权利.
|
||||||
|
*
|
||||||
|
* 本软件仅供评估使用。任何商业用途必须获得书面授权许可。
|
||||||
|
* 未经授权商业使用本软件属于侵权行为,将承担法律责任。
|
||||||
|
*
|
||||||
|
* 授权购买请联系: 357099073@qq.com
|
||||||
|
* 官方网站: https://www.yunzer.cn
|
||||||
|
*
|
||||||
|
* 评估用户须知:
|
||||||
|
* 1. 禁止移除版权声明
|
||||||
|
* 2. 禁止用于生产环境
|
||||||
|
* 3. 禁止转售或分发
|
||||||
|
*/
|
||||||
|
|
||||||
|
namespace app;
|
||||||
|
/**
|
||||||
|
* app错误码
|
||||||
|
*/
|
||||||
|
class AppAPi{
|
||||||
|
public static function errorTip($code){
|
||||||
|
$array = array(
|
||||||
|
'10000001' => '系统错误,请稍后重试',
|
||||||
|
#############系统后台使用##############
|
||||||
|
'90000001' => '管理员账户不能为空',
|
||||||
|
'90000002' => '真实姓名不能为空',
|
||||||
|
'90000003' => '手机号不能为空',
|
||||||
|
'90000004' => '角色不能为空',
|
||||||
|
'90000005' => '密码不能为空',
|
||||||
|
'90000006' => '管理员账户要用邮箱',
|
||||||
|
'90000007' => '管理员账户已存在',
|
||||||
|
'90000008' => '角色名称不能为空',
|
||||||
|
'90000009' => '导航名称不能为空',
|
||||||
|
'90000010' => '导航下还有数据',
|
||||||
|
'90000011' => '请选择类型',
|
||||||
|
'90000012' => '内部代码不能为空',
|
||||||
|
'90000013' => '链接地址不能为空',
|
||||||
|
'90000015' => '按钮名称不能为空',
|
||||||
|
'90000029' => '管理员账户不存在',
|
||||||
|
'90000030' => '管理员已被禁用',
|
||||||
|
'90000031' => '密码不正确',
|
||||||
|
|
||||||
|
'91000001' => '添加失败',
|
||||||
|
'91000002' => '修改失败',
|
||||||
|
'91000003' => '删除失败',
|
||||||
|
'91000004' => '未改变数据',
|
||||||
|
'91000005' => '未查询到数据',
|
||||||
|
'91000006' => '失败',
|
||||||
|
);
|
||||||
|
return $array[$code];
|
||||||
|
}
|
||||||
|
}
|
||||||
39
app/AppService.php
Normal file
39
app/AppService.php
Normal file
@ -0,0 +1,39 @@
|
|||||||
|
<?php
|
||||||
|
/**
|
||||||
|
* 商业使用授权协议
|
||||||
|
*
|
||||||
|
* Copyright (c) 2025 [云泽网]. 保留所有权利.
|
||||||
|
*
|
||||||
|
* 本软件仅供评估使用。任何商业用途必须获得书面授权许可。
|
||||||
|
* 未经授权商业使用本软件属于侵权行为,将承担法律责任。
|
||||||
|
*
|
||||||
|
* 授权购买请联系: 357099073@qq.com
|
||||||
|
* 官方网站: https://www.yunzer.cn
|
||||||
|
*
|
||||||
|
* 评估用户须知:
|
||||||
|
* 1. 禁止移除版权声明
|
||||||
|
* 2. 禁止用于生产环境
|
||||||
|
* 3. 禁止转售或分发
|
||||||
|
*/
|
||||||
|
|
||||||
|
declare (strict_types = 1);
|
||||||
|
|
||||||
|
namespace app;
|
||||||
|
|
||||||
|
use think\Service;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 应用服务类
|
||||||
|
*/
|
||||||
|
class AppService extends Service
|
||||||
|
{
|
||||||
|
public function register()
|
||||||
|
{
|
||||||
|
// 服务注册
|
||||||
|
}
|
||||||
|
|
||||||
|
public function boot()
|
||||||
|
{
|
||||||
|
// 服务启动
|
||||||
|
}
|
||||||
|
}
|
||||||
75
app/ExceptionHandle.php
Normal file
75
app/ExceptionHandle.php
Normal file
@ -0,0 +1,75 @@
|
|||||||
|
<?php
|
||||||
|
/**
|
||||||
|
* 商业使用授权协议
|
||||||
|
*
|
||||||
|
* Copyright (c) 2025 [云泽网]. 保留所有权利.
|
||||||
|
*
|
||||||
|
* 本软件仅供评估使用。任何商业用途必须获得书面授权许可。
|
||||||
|
* 未经授权商业使用本软件属于侵权行为,将承担法律责任。
|
||||||
|
*
|
||||||
|
* 授权购买请联系: 357099073@qq.com
|
||||||
|
* 官方网站: https://www.yunzer.cn
|
||||||
|
*
|
||||||
|
* 评估用户须知:
|
||||||
|
* 1. 禁止移除版权声明
|
||||||
|
* 2. 禁止用于生产环境
|
||||||
|
* 3. 禁止转售或分发
|
||||||
|
*/
|
||||||
|
|
||||||
|
namespace app;
|
||||||
|
|
||||||
|
use think\db\exception\DataNotFoundException;
|
||||||
|
use think\db\exception\ModelNotFoundException;
|
||||||
|
use think\exception\Handle;
|
||||||
|
use think\exception\HttpException;
|
||||||
|
use think\exception\HttpResponseException;
|
||||||
|
use think\exception\ValidateException;
|
||||||
|
use think\Response;
|
||||||
|
use Throwable;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 应用异常处理类
|
||||||
|
*/
|
||||||
|
class ExceptionHandle extends Handle
|
||||||
|
{
|
||||||
|
/**
|
||||||
|
* 不需要记录信息(日志)的异常类列表
|
||||||
|
* @var array
|
||||||
|
*/
|
||||||
|
protected $ignoreReport = [
|
||||||
|
HttpException::class,
|
||||||
|
HttpResponseException::class,
|
||||||
|
ModelNotFoundException::class,
|
||||||
|
DataNotFoundException::class,
|
||||||
|
ValidateException::class,
|
||||||
|
];
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 记录异常信息(包括日志或者其它方式记录)
|
||||||
|
*
|
||||||
|
* @access public
|
||||||
|
* @param Throwable $exception
|
||||||
|
* @return void
|
||||||
|
*/
|
||||||
|
public function report(Throwable $exception): void
|
||||||
|
{
|
||||||
|
// 使用内置的方式记录异常日志
|
||||||
|
parent::report($exception);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Render an exception into an HTTP response.
|
||||||
|
*
|
||||||
|
* @access public
|
||||||
|
* @param \think\Request $request
|
||||||
|
* @param Throwable $e
|
||||||
|
* @return Response
|
||||||
|
*/
|
||||||
|
public function render($request, Throwable $e): Response
|
||||||
|
{
|
||||||
|
// 添加自定义异常处理机制
|
||||||
|
|
||||||
|
// 其他错误交给系统处理
|
||||||
|
return parent::render($request, $e);
|
||||||
|
}
|
||||||
|
}
|
||||||
25
app/Request.php
Normal file
25
app/Request.php
Normal file
@ -0,0 +1,25 @@
|
|||||||
|
<?php
|
||||||
|
/**
|
||||||
|
* 商业使用授权协议
|
||||||
|
*
|
||||||
|
* Copyright (c) 2025 [云泽网]. 保留所有权利.
|
||||||
|
*
|
||||||
|
* 本软件仅供评估使用。任何商业用途必须获得书面授权许可。
|
||||||
|
* 未经授权商业使用本软件属于侵权行为,将承担法律责任。
|
||||||
|
*
|
||||||
|
* 授权购买请联系: 357099073@qq.com
|
||||||
|
* 官方网站: https://www.yunzer.cn
|
||||||
|
*
|
||||||
|
* 评估用户须知:
|
||||||
|
* 1. 禁止移除版权声明
|
||||||
|
* 2. 禁止用于生产环境
|
||||||
|
* 3. 禁止转售或分发
|
||||||
|
*/
|
||||||
|
|
||||||
|
namespace app;
|
||||||
|
|
||||||
|
// 应用请求对象类
|
||||||
|
class Request extends \think\Request
|
||||||
|
{
|
||||||
|
|
||||||
|
}
|
||||||
508
app/admin/controller/ArticlesController.php
Normal file
508
app/admin/controller/ArticlesController.php
Normal file
@ -0,0 +1,508 @@
|
|||||||
|
<?php
|
||||||
|
/**
|
||||||
|
* 商业使用授权协议
|
||||||
|
*
|
||||||
|
* Copyright (c) 2025 [云泽网]. 保留所有权利.
|
||||||
|
*
|
||||||
|
* 本软件仅供评估使用。任何商业用途必须获得书面授权许可。
|
||||||
|
* 未经授权商业使用本软件属于侵权行为,将承担法律责任。
|
||||||
|
*
|
||||||
|
* 授权购买请联系: 357099073@qq.com
|
||||||
|
* 官方网站: https://www.yunzer.cn
|
||||||
|
*
|
||||||
|
* 评估用户须知:
|
||||||
|
* 1. 禁止移除版权声明
|
||||||
|
* 2. 禁止用于生产环境
|
||||||
|
* 3. 禁止转售或分发
|
||||||
|
*/
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 后台管理系统-文章管理
|
||||||
|
*/
|
||||||
|
namespace app\admin\controller;
|
||||||
|
use app\admin\model\Article\Articles;
|
||||||
|
use app\admin\model\Article\ArticlesCategory;
|
||||||
|
use think\facade\Db;
|
||||||
|
use think\facade\View;
|
||||||
|
use think\facade\Request;
|
||||||
|
use app\admin\controller\LogController as Log;
|
||||||
|
use app\admin\controller\BaseController;
|
||||||
|
|
||||||
|
class ArticlesController extends BaseController
|
||||||
|
{
|
||||||
|
/**
|
||||||
|
* 获取控制器名称
|
||||||
|
* @return string
|
||||||
|
*/
|
||||||
|
public function getControllerName()
|
||||||
|
{
|
||||||
|
return 'Article';
|
||||||
|
}
|
||||||
|
|
||||||
|
// 文章列表
|
||||||
|
public function articlelist()
|
||||||
|
{
|
||||||
|
if (Request::isPost()) {
|
||||||
|
$category = input('post.category');
|
||||||
|
$page = intval(input('post.page', 1));
|
||||||
|
$limit = intval(input('post.limit', 10));
|
||||||
|
$title = input('post.title');
|
||||||
|
$author = input('post.author');
|
||||||
|
|
||||||
|
$query = Articles::where('delete_time', null)
|
||||||
|
->where('status', '<>', 3);
|
||||||
|
|
||||||
|
// 分类筛选
|
||||||
|
if (!empty($category)) {
|
||||||
|
// 先获取分类ID
|
||||||
|
$cateInfo = ArticlesCategory::where('name', $category)
|
||||||
|
->where('delete_time', null)
|
||||||
|
->where('status', 1)
|
||||||
|
->find();
|
||||||
|
|
||||||
|
if ($cateInfo) {
|
||||||
|
$query = $query->where('cate', $cateInfo['id']);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// 标题搜索
|
||||||
|
if (!empty($title)) {
|
||||||
|
$query = $query->where('title', 'like', '%'.$title.'%');
|
||||||
|
}
|
||||||
|
|
||||||
|
// 作者搜索
|
||||||
|
if (!empty($author)) {
|
||||||
|
$query = $query->where('author', 'like', '%'.$author.'%');
|
||||||
|
}
|
||||||
|
|
||||||
|
// 获取总记录数
|
||||||
|
$count = $query->count();
|
||||||
|
|
||||||
|
// 获取分页数据
|
||||||
|
$lists = $query->order('id DESC')
|
||||||
|
->page($page, $limit)
|
||||||
|
->select()
|
||||||
|
->each(function ($item) {
|
||||||
|
// 获取分类信息
|
||||||
|
$cateInfo = ArticlesCategory::where('id', $item['cate'])
|
||||||
|
->field('name, image')
|
||||||
|
->find();
|
||||||
|
|
||||||
|
// 设置分类名称
|
||||||
|
$item['cate'] = $cateInfo ? $cateInfo['name'] : '';
|
||||||
|
|
||||||
|
// 如果文章没有图片,使用分类的图片
|
||||||
|
if (empty($item['image']) && $cateInfo && !empty($cateInfo['image'])) {
|
||||||
|
$item['image'] = $cateInfo['image'];
|
||||||
|
}
|
||||||
|
|
||||||
|
// 格式化时间
|
||||||
|
$item['create_time'] = is_numeric($item['create_time']) ? date('Y-m-d H:i:s', $item['create_time']) : $item['create_time'];
|
||||||
|
$item['publishdate'] = is_numeric($item['publishdate']) ? date('Y-m-d H:i:s', $item['publishdate']) : '';
|
||||||
|
|
||||||
|
return $item;
|
||||||
|
});
|
||||||
|
|
||||||
|
return json([
|
||||||
|
'code' => 0,
|
||||||
|
'msg' => '获取成功',
|
||||||
|
'count' => $count,
|
||||||
|
'data' => $lists
|
||||||
|
]);
|
||||||
|
} else {
|
||||||
|
// 获取所有分类并构建父子结构
|
||||||
|
$allCategories = ArticlesCategory::where('delete_time', null)
|
||||||
|
->where('status', 1)
|
||||||
|
->order('sort asc, id asc')
|
||||||
|
->select()
|
||||||
|
->toArray();
|
||||||
|
|
||||||
|
$categories = [];
|
||||||
|
foreach ($allCategories as $category) {
|
||||||
|
if ($category['cid'] == 0) {
|
||||||
|
$category['children'] = [];
|
||||||
|
foreach ($allCategories as $subCategory) {
|
||||||
|
if ($subCategory['cid'] == $category['id']) {
|
||||||
|
$category['children'][] = $subCategory;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
$categories[] = $category;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
View::assign([
|
||||||
|
'categories' => $categories
|
||||||
|
]);
|
||||||
|
return View::fetch();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// 添加文章
|
||||||
|
public function add()
|
||||||
|
{
|
||||||
|
if (Request::isPost()) {
|
||||||
|
$data = [
|
||||||
|
'title' => input('post.title'),
|
||||||
|
'cate' => input('post.cate'),
|
||||||
|
'image' => input('post.image'),
|
||||||
|
'content' => input('post.content'),
|
||||||
|
'author' => input('post.author'),
|
||||||
|
'desc' => input('post.desc'),
|
||||||
|
'status' => input('post.status', 2),
|
||||||
|
'publishdate' => time(),
|
||||||
|
'create_time' => time()
|
||||||
|
];
|
||||||
|
|
||||||
|
$insert = Articles::insert($data);
|
||||||
|
if (empty($insert)) {
|
||||||
|
Log::record('添加文章', 0, '添加文章失败', '文章管理');
|
||||||
|
return json(['code' => 1, 'msg' => '添加失败', 'data' => []]);
|
||||||
|
}
|
||||||
|
Log::record('添加文章', 1, '', '文章管理');
|
||||||
|
return json(['code' => 0, 'msg' => '添加成功', 'data' => []]);
|
||||||
|
} else {
|
||||||
|
$lists = Articles::order('id DESC')
|
||||||
|
->select()
|
||||||
|
->each(function ($item, $key) {
|
||||||
|
$item['create_time'] = time();
|
||||||
|
return $item;
|
||||||
|
});
|
||||||
|
View::assign([
|
||||||
|
'lists' => $lists
|
||||||
|
]);
|
||||||
|
return View::fetch();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// 编辑文章
|
||||||
|
public function edit()
|
||||||
|
{
|
||||||
|
if (Request::isPost()) {
|
||||||
|
$id = input('get.id');
|
||||||
|
$data = [
|
||||||
|
'title' => input('post.title'),
|
||||||
|
'cate' => input('post.cate'),
|
||||||
|
'image' => input('post.image'),
|
||||||
|
'content' => input('post.content'),
|
||||||
|
'author' => input('post.author'),
|
||||||
|
'desc' => input('post.desc'),
|
||||||
|
'status' => input('post.status', 2),
|
||||||
|
'update_time' => time()
|
||||||
|
];
|
||||||
|
|
||||||
|
$update = Articles::where('id', $id)->update($data);
|
||||||
|
if ($update === false) {
|
||||||
|
Log::record('编辑文章', 0, '编辑文章失败', '文章管理');
|
||||||
|
return json(['code' => 1, 'msg' => '更新失败', 'data' => []]);
|
||||||
|
}
|
||||||
|
Log::record('编辑文章', 1, '', '文章管理');
|
||||||
|
return json(['code' => 0, 'msg' => '更新成功', 'data' => []]);
|
||||||
|
} else {
|
||||||
|
$id = input('get.id');
|
||||||
|
$info = Articles::where('id', $id)->find();
|
||||||
|
if ($info === null) {
|
||||||
|
return json(['code' => 1, 'msg' => '文章不存在', 'data' => []]);
|
||||||
|
}
|
||||||
|
|
||||||
|
$cates = ArticlesCategory::where('delete_time', null)
|
||||||
|
->where('status', 1)
|
||||||
|
->order('sort asc, id asc')
|
||||||
|
->select()
|
||||||
|
->toArray();
|
||||||
|
|
||||||
|
$info['content'] = !empty($info['content']) ? htmlspecialchars_decode(str_replace(["\r\n", "\r", "\n"], '', addslashes($info['content']))) : '';
|
||||||
|
|
||||||
|
$currentCate = ArticlesCategory::where('id', $info['cate'])
|
||||||
|
->where('delete_time', null)
|
||||||
|
->where('status', 1)
|
||||||
|
->find();
|
||||||
|
$info['cate_name'] = $currentCate ? $currentCate['name'] : '';
|
||||||
|
|
||||||
|
View::assign([
|
||||||
|
'info' => $info,
|
||||||
|
'cates' => $cates
|
||||||
|
]);
|
||||||
|
return View::fetch();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// 删除文章
|
||||||
|
public function delete()
|
||||||
|
{
|
||||||
|
$id = input('post.id');
|
||||||
|
$data = [
|
||||||
|
'delete_time' => time(),
|
||||||
|
];
|
||||||
|
$delete = Articles::where('id', $id)->update($data);
|
||||||
|
if ($delete === false) {
|
||||||
|
Log::record('删除文章', 0, '删除文章失败', '文章管理');
|
||||||
|
return json(['code' => 1, 'msg' => '删除失败', 'data' => []]);
|
||||||
|
}
|
||||||
|
Log::record('删除文章', 1, '', '文章管理');
|
||||||
|
return json(['code' => 0, 'msg' => '删除成功', 'data' => []]);
|
||||||
|
}
|
||||||
|
|
||||||
|
// 文章分类
|
||||||
|
public function articlecate()
|
||||||
|
{
|
||||||
|
if (Request::isPost()) {
|
||||||
|
$lists = ArticlesCategory::where('delete_time', null)
|
||||||
|
->where('status', 1)
|
||||||
|
->order('sort asc, id asc')
|
||||||
|
->select()
|
||||||
|
->toArray();
|
||||||
|
|
||||||
|
// 构建树形结构
|
||||||
|
$tree = [];
|
||||||
|
foreach ($lists as $item) {
|
||||||
|
if ($item['cid'] == 0) {
|
||||||
|
$node = [
|
||||||
|
'id' => $item['id'],
|
||||||
|
'title' => $item['name'],
|
||||||
|
'children' => []
|
||||||
|
];
|
||||||
|
|
||||||
|
// 查找子分类
|
||||||
|
foreach ($lists as $subItem) {
|
||||||
|
if ($subItem['cid'] == $item['id']) {
|
||||||
|
$node['children'][] = [
|
||||||
|
'id' => $subItem['id'],
|
||||||
|
'title' => $subItem['name'],
|
||||||
|
'children' => []
|
||||||
|
];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
$tree[] = $node;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return json(['code' => 0, 'msg' => '获取成功', 'data' => $tree]);
|
||||||
|
}
|
||||||
|
|
||||||
|
// 非 POST 请求返回视图
|
||||||
|
return View::fetch();
|
||||||
|
}
|
||||||
|
|
||||||
|
//获取分类结构
|
||||||
|
public function getcate()
|
||||||
|
{
|
||||||
|
// 获取所有分类
|
||||||
|
$lists = ArticlesCategory::where('delete_time', null)
|
||||||
|
->where('status', 1)
|
||||||
|
->order('sort asc, id asc')
|
||||||
|
->select()
|
||||||
|
->toArray();
|
||||||
|
|
||||||
|
// 构建父子结构
|
||||||
|
$tree = [];
|
||||||
|
foreach ($lists as $item) {
|
||||||
|
if ($item['cid'] == 0) {
|
||||||
|
// 顶级分类
|
||||||
|
$tree[] = $item;
|
||||||
|
} else {
|
||||||
|
// 子分类
|
||||||
|
foreach ($tree as &$parent) {
|
||||||
|
if ($parent['id'] == $item['cid']) {
|
||||||
|
if (!isset($parent['children'])) {
|
||||||
|
$parent['children'] = [];
|
||||||
|
}
|
||||||
|
$parent['children'][] = $item;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return json(['code' => 0, 'msg' => '获取成功', 'data' => $tree]);
|
||||||
|
}
|
||||||
|
|
||||||
|
// 添加文章分类
|
||||||
|
public function cateadd()
|
||||||
|
{
|
||||||
|
if (Request::isPost()) {
|
||||||
|
$data = [
|
||||||
|
'name' => input('post.name'),
|
||||||
|
'image' => input('post.image'),
|
||||||
|
'cid' => input('post.cid'),
|
||||||
|
'sort' => input('post.sort', 0),
|
||||||
|
'status' => input('post.status', 1),
|
||||||
|
'create_time' => time()
|
||||||
|
];
|
||||||
|
|
||||||
|
$insert = ArticlesCategory::insert($data);
|
||||||
|
if (empty($insert)) {
|
||||||
|
Log::record('添加文章分类', 0, '添加文章分类失败', '文章分类');
|
||||||
|
return json(['code' => 1, 'msg' => '添加失败', 'data' => []]);
|
||||||
|
}
|
||||||
|
Log::record('添加文章分类', 1, '', '文章分类');
|
||||||
|
return json(['code' => 0, 'msg' => '添加成功', 'data' => []]);
|
||||||
|
} else {
|
||||||
|
// 获取所有可选的父级分类
|
||||||
|
$parentCategories = ArticlesCategory::where('delete_time', null)
|
||||||
|
->where('status', 1)
|
||||||
|
->where('cid', 0)
|
||||||
|
->field('id, name')
|
||||||
|
->select()
|
||||||
|
->toArray();
|
||||||
|
|
||||||
|
return json([
|
||||||
|
'code' => 0,
|
||||||
|
'msg' => '获取成功',
|
||||||
|
'data' => [
|
||||||
|
'parentOptions' => $parentCategories
|
||||||
|
]
|
||||||
|
]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
//编辑文章分类
|
||||||
|
public function cateedit()
|
||||||
|
{
|
||||||
|
if (Request::isPost()) {
|
||||||
|
$data = [
|
||||||
|
'id' => input('post.id'),
|
||||||
|
'name' => input('post.name'),
|
||||||
|
'image' => input('post.image'),
|
||||||
|
'cid' => input('post.cid'),
|
||||||
|
'sort' => input('post.sort', 0),
|
||||||
|
'status' => input('post.status', 1),
|
||||||
|
'update_time' => time()
|
||||||
|
];
|
||||||
|
|
||||||
|
$update = ArticlesCategory::where('id', $data['id'])
|
||||||
|
->update($data);
|
||||||
|
|
||||||
|
if ($update === false) {
|
||||||
|
Log::record('编辑文章分类', 0, '更新文章分类失败', '文章分类');
|
||||||
|
return json(['code' => 1, 'msg' => '更新失败', 'data' => []]);
|
||||||
|
}
|
||||||
|
Log::record('编辑文章分类', 1, '', '文章分类');
|
||||||
|
return json(['code' => 0, 'msg' => '更新成功', 'data' => []]);
|
||||||
|
} else {
|
||||||
|
$id = input('get.id');
|
||||||
|
$info = ArticlesCategory::where('id', $id)->find();
|
||||||
|
|
||||||
|
// 获取所有可选的父级分类
|
||||||
|
$parentCategories = ArticlesCategory::where('delete_time', null)
|
||||||
|
->where('status', 1)
|
||||||
|
->where('id', '<>', $id) // 排除自己
|
||||||
|
->where(function ($query) use ($id) {
|
||||||
|
// 排除自己的所有子分类
|
||||||
|
$query->where('cid', '<>', $id);
|
||||||
|
})
|
||||||
|
->field('id, name, cid')
|
||||||
|
->select()
|
||||||
|
->toArray();
|
||||||
|
|
||||||
|
// 构建父级分类选项
|
||||||
|
$parentOptions = [];
|
||||||
|
foreach ($parentCategories as $category) {
|
||||||
|
if ($category['cid'] == 0) {
|
||||||
|
$parentOptions[] = [
|
||||||
|
'id' => $category['id'],
|
||||||
|
'name' => $category['name']
|
||||||
|
];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return json([
|
||||||
|
'code' => 0,
|
||||||
|
'msg' => '获取成功',
|
||||||
|
'data' => [
|
||||||
|
'info' => $info,
|
||||||
|
'parentOptions' => $parentOptions
|
||||||
|
]
|
||||||
|
]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
//删除文章分类
|
||||||
|
public function catedel()
|
||||||
|
{
|
||||||
|
$id = input('post.id');
|
||||||
|
|
||||||
|
// 检查是否有子分类
|
||||||
|
$hasChildren = ArticlesCategory::where('cid', $id)
|
||||||
|
->where('delete_time', null)
|
||||||
|
->find();
|
||||||
|
|
||||||
|
if ($hasChildren) {
|
||||||
|
Log::record('删除文章分类', 0, '该分类下有子分类,无法删除', '文章分类');
|
||||||
|
return json(['code' => 1, 'msg' => '该分类下有子分类,无法删除', 'data' => []]);
|
||||||
|
}
|
||||||
|
|
||||||
|
$delete = ArticlesCategory::where('id', $id)
|
||||||
|
->update(['delete_time' => time()]);
|
||||||
|
|
||||||
|
if ($delete === false) {
|
||||||
|
Log::record('删除文章分类', 0, '删除文章分类失败', '文章分类');
|
||||||
|
return json(['code' => 1, 'msg' => '删除失败', 'data' => []]);
|
||||||
|
}
|
||||||
|
Log::record('删除文章分类', 1, '', '文章分类');
|
||||||
|
return json(['code' => 0, 'msg' => '删除成功', 'data' => []]);
|
||||||
|
}
|
||||||
|
|
||||||
|
//统计文章数量
|
||||||
|
public function counts() {
|
||||||
|
try {
|
||||||
|
// 获取文章总数
|
||||||
|
$total = Articles::where('delete_time', null)
|
||||||
|
->where('status', '<>', 3)
|
||||||
|
->count();
|
||||||
|
|
||||||
|
// 获取今日新增文章数
|
||||||
|
$today = strtotime(date('Y-m-d'));
|
||||||
|
$todayNew = Articles::where('delete_time', null)
|
||||||
|
->where('status', '<>', 3)
|
||||||
|
->where('create_time', '>=', $today)
|
||||||
|
->count();
|
||||||
|
|
||||||
|
// 获取最近7天的文章数据
|
||||||
|
$dates = [];
|
||||||
|
$counts = [];
|
||||||
|
$totalCounts = []; // 存储每天的总文章数
|
||||||
|
$totalSoFar = 0; // 用于累计总文章数
|
||||||
|
|
||||||
|
for ($i = 6; $i >= 0; $i--) {
|
||||||
|
$date = date('Y-m-d', strtotime("-$i days"));
|
||||||
|
$start = strtotime($date);
|
||||||
|
$end = $start + 86400;
|
||||||
|
|
||||||
|
// 获取当天新增文章数
|
||||||
|
$count = Articles::where('delete_time', null)
|
||||||
|
->where('status', '<>', 3)
|
||||||
|
->where('create_time', '>=', $start)
|
||||||
|
->where('create_time', '<', $end)
|
||||||
|
->count();
|
||||||
|
|
||||||
|
// 获取截至当天的总文章数
|
||||||
|
$totalCount = Articles::where('delete_time', null)
|
||||||
|
->where('status', '<>', 3)
|
||||||
|
->where('create_time', '<', $end)
|
||||||
|
->count();
|
||||||
|
|
||||||
|
$dates[] = $date;
|
||||||
|
$counts[] = $count;
|
||||||
|
$totalCounts[] = $totalCount;
|
||||||
|
}
|
||||||
|
|
||||||
|
return json([
|
||||||
|
'code' => 0,
|
||||||
|
'msg' => '获取成功',
|
||||||
|
'data' => [
|
||||||
|
'total' => $total,
|
||||||
|
'todayNew' => $todayNew,
|
||||||
|
'dates' => $dates,
|
||||||
|
'counts' => $counts,
|
||||||
|
'totalCounts' => $totalCounts
|
||||||
|
]
|
||||||
|
]);
|
||||||
|
} catch (\Exception $e) {
|
||||||
|
return json([
|
||||||
|
'code' => 1,
|
||||||
|
'msg' => '获取失败:' . $e->getMessage()
|
||||||
|
]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
181
app/admin/controller/Base.php
Normal file
181
app/admin/controller/Base.php
Normal file
@ -0,0 +1,181 @@
|
|||||||
|
<?php
|
||||||
|
/**
|
||||||
|
* 商业使用授权协议
|
||||||
|
*
|
||||||
|
* Copyright (c) 2025 [云泽网]. 保留所有权利.
|
||||||
|
*
|
||||||
|
* 本软件仅供评估使用。任何商业用途必须获得书面授权许可。
|
||||||
|
* 未经授权商业使用本软件属于侵权行为,将承担法律责任。
|
||||||
|
*
|
||||||
|
* 授权购买请联系: 357099073@qq.com
|
||||||
|
* 官方网站: https://www.yunzer.cn
|
||||||
|
*
|
||||||
|
* 评估用户须知:
|
||||||
|
* 1. 禁止移除版权声明
|
||||||
|
* 2. 禁止用于生产环境
|
||||||
|
* 3. 禁止转售或分发
|
||||||
|
*/
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 后台管理系统-管理员
|
||||||
|
*/
|
||||||
|
namespace app\admin\controller;
|
||||||
|
use app\AppApi;
|
||||||
|
use think\facade\Db;
|
||||||
|
use think\facade\View;
|
||||||
|
use think\facade\Cookie;
|
||||||
|
use think\facade\Config;
|
||||||
|
|
||||||
|
use app\admin\model\YzAdminConfig;
|
||||||
|
|
||||||
|
use think\exception\HttpResponseException;
|
||||||
|
use think\facade\Request;
|
||||||
|
use think\facade\Route;
|
||||||
|
use think\App;
|
||||||
|
|
||||||
|
class Base
|
||||||
|
{
|
||||||
|
protected $app;
|
||||||
|
protected $request;
|
||||||
|
public $adminId = null;
|
||||||
|
public $config = [];
|
||||||
|
public $aUser = [];
|
||||||
|
public function __construct()
|
||||||
|
{
|
||||||
|
date_default_timezone_set('PRC');
|
||||||
|
# 获取配置
|
||||||
|
$YzAdminConfig = new YzAdminConfig();
|
||||||
|
$this->config = $YzAdminConfig->getAll();
|
||||||
|
# 获取账户,账户判断
|
||||||
|
$this->adminId = Cookie::get('admin_id');
|
||||||
|
if (empty($this->adminId)) {
|
||||||
|
header('Location:' . $this->config['admin_route'] . 'Login/index');
|
||||||
|
exit;
|
||||||
|
}
|
||||||
|
$this->aUser = Db::table('yz_admin_user')->where('uid', $this->adminId)->find();
|
||||||
|
|
||||||
|
if (empty($this->aUser)) {
|
||||||
|
Cookie::delete('admin_id');
|
||||||
|
$this->error('管理员账户不存在');
|
||||||
|
}
|
||||||
|
if ($this->aUser['status'] != 1) {
|
||||||
|
Cookie::delete('admin_id');
|
||||||
|
$this->error('管理员已被禁用');
|
||||||
|
}
|
||||||
|
# 获取用户组权限
|
||||||
|
$group = Db::table('yz_admin_user_group')->where(['group_id' => $this->aUser['group_id']])->find();
|
||||||
|
if (empty($group)) {
|
||||||
|
$this->error('对不起,您没有权限');
|
||||||
|
}
|
||||||
|
# 获取当前链接,查询是否有权限
|
||||||
|
$controller = request()->controller();
|
||||||
|
$action = request()->action();
|
||||||
|
$key = $controller . '/' . $action;
|
||||||
|
View::assign([
|
||||||
|
'aUser' => $this->aUser,
|
||||||
|
'config' => $this->config
|
||||||
|
]);
|
||||||
|
}
|
||||||
|
/**
|
||||||
|
* 返回json对象
|
||||||
|
*/
|
||||||
|
protected function returnCode($code, $data = [], $count = 10)
|
||||||
|
{
|
||||||
|
header('Content-type:application/json');
|
||||||
|
if ($code == 0) {
|
||||||
|
$arr = array(
|
||||||
|
'code' => $code,
|
||||||
|
'msg' => '操作成功',
|
||||||
|
'count' => $count,
|
||||||
|
'data' => $data
|
||||||
|
);
|
||||||
|
} else if ($code >= 1 && $code <= 100) {
|
||||||
|
$arr = array(
|
||||||
|
'code' => $code,
|
||||||
|
'msg' => $data
|
||||||
|
);
|
||||||
|
} else {
|
||||||
|
$appapi = new AppApi();
|
||||||
|
$arr = array(
|
||||||
|
'code' => $code,
|
||||||
|
'msg' => $appapi::errorTip($code)
|
||||||
|
);
|
||||||
|
}
|
||||||
|
echo json_encode($arr);
|
||||||
|
if ($code != 0) {
|
||||||
|
exit;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 操作成功跳转的快捷方法
|
||||||
|
* @access protected
|
||||||
|
* @param mixed $msg 提示信息
|
||||||
|
* @return void
|
||||||
|
*/
|
||||||
|
protected function success($msg = '')
|
||||||
|
{
|
||||||
|
$result = [
|
||||||
|
'code' => 1,
|
||||||
|
'msg' => $msg
|
||||||
|
];
|
||||||
|
|
||||||
|
$type = $this->getResponseType();
|
||||||
|
if ($type == 'html') {
|
||||||
|
$response = view(Config::get('app.dispatch_success_tmpl'), $result);
|
||||||
|
} else if ($type == 'json') {
|
||||||
|
$response = json($result);
|
||||||
|
}
|
||||||
|
throw new HttpResponseException($response);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 操作错误跳转的快捷方法
|
||||||
|
* @access protected
|
||||||
|
* @param mixed $msg 提示信息
|
||||||
|
* @return void
|
||||||
|
*/
|
||||||
|
protected function error($msg = '')
|
||||||
|
{
|
||||||
|
$result = [
|
||||||
|
'code' => 0,
|
||||||
|
'msg' => $msg
|
||||||
|
];
|
||||||
|
$response = view(Config::get('app.dispatch_error_tmpl'), $result);
|
||||||
|
throw new HttpResponseException($response);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 获取当前的response 输出类型
|
||||||
|
* @access protected
|
||||||
|
* @return string
|
||||||
|
*/
|
||||||
|
protected function getResponseType()
|
||||||
|
{
|
||||||
|
return Request::isJson() || Request::isAjax() ? 'json' : 'html';
|
||||||
|
}
|
||||||
|
|
||||||
|
public function initialize(App $app)
|
||||||
|
{
|
||||||
|
$this->app = $app;
|
||||||
|
$this->request = $this->app->request;
|
||||||
|
|
||||||
|
// 检查是否是直接访问具体页面
|
||||||
|
$controller = $this->request->controller();
|
||||||
|
$action = $this->request->action();
|
||||||
|
|
||||||
|
// 如果不是访问index控制器,且不是通过iframe加载,且不是ajax请求
|
||||||
|
if (
|
||||||
|
$controller != 'Index' &&
|
||||||
|
!$this->request->isAjax() &&
|
||||||
|
!$this->request->header('X-Requested-With') &&
|
||||||
|
!$this->request->param('iframe')
|
||||||
|
) { // 添加iframe参数检查
|
||||||
|
|
||||||
|
// 重定向到index页面,并带上当前页面参数
|
||||||
|
$currentUrl = $controller . '/' . $action;
|
||||||
|
redirect(url('index/index', ['page' => $currentUrl]))->send();
|
||||||
|
exit;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
149
app/admin/controller/BaseController.php
Normal file
149
app/admin/controller/BaseController.php
Normal file
@ -0,0 +1,149 @@
|
|||||||
|
<?php
|
||||||
|
/**
|
||||||
|
* 商业使用授权协议
|
||||||
|
*
|
||||||
|
* Copyright (c) 2025 [云泽网]. 保留所有权利.
|
||||||
|
*
|
||||||
|
* 本软件仅供评估使用。任何商业用途必须获得书面授权许可。
|
||||||
|
* 未经授权商业使用本软件属于侵权行为,将承担法律责任。
|
||||||
|
*
|
||||||
|
* 授权购买请联系: 357099073@qq.com
|
||||||
|
* 官方网站: https://www.yunzer.cn
|
||||||
|
*
|
||||||
|
* 评估用户须知:
|
||||||
|
* 1. 禁止移除版权声明
|
||||||
|
* 2. 禁止用于生产环境
|
||||||
|
* 3. 禁止转售或分发
|
||||||
|
*/
|
||||||
|
|
||||||
|
declare (strict_types = 1);
|
||||||
|
|
||||||
|
namespace app\admin\controller;
|
||||||
|
|
||||||
|
use think\App;
|
||||||
|
use think\exception\ValidateException;
|
||||||
|
use think\Validate;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 后台控制器基础类
|
||||||
|
*/
|
||||||
|
abstract class BaseController extends Base
|
||||||
|
{
|
||||||
|
/**
|
||||||
|
* Request实例
|
||||||
|
* @var \think\Request
|
||||||
|
*/
|
||||||
|
protected $request;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 应用实例
|
||||||
|
* @var \think\App
|
||||||
|
*/
|
||||||
|
protected $app;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 是否批量验证
|
||||||
|
* @var bool
|
||||||
|
*/
|
||||||
|
protected $batchValidate = false;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 控制器中间件
|
||||||
|
* @var array
|
||||||
|
*/
|
||||||
|
protected $middleware = [];
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 构造方法
|
||||||
|
* @access public
|
||||||
|
* @param App $app 应用对象
|
||||||
|
*/
|
||||||
|
public function __construct(App $app)
|
||||||
|
{
|
||||||
|
parent::__construct();
|
||||||
|
$this->app = $app;
|
||||||
|
$this->request = $this->app->request;
|
||||||
|
|
||||||
|
// 控制器初始化
|
||||||
|
$this->initialize($app);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 初始化
|
||||||
|
* @access public
|
||||||
|
* @param App $app 应用对象
|
||||||
|
*/
|
||||||
|
public function initialize(App $app)
|
||||||
|
{
|
||||||
|
// 注册控制器映射
|
||||||
|
$this->registerControllerMap();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 注册控制器映射
|
||||||
|
*/
|
||||||
|
protected function registerControllerMap()
|
||||||
|
{
|
||||||
|
// 获取当前控制器类名
|
||||||
|
$className = get_class($this);
|
||||||
|
// 获取不带命名空间的类名
|
||||||
|
$shortName = substr($className, strrpos($className, '\\') + 1);
|
||||||
|
// 移除Controller后缀
|
||||||
|
$mapName = str_replace('Controller', '', $shortName);
|
||||||
|
|
||||||
|
// 调试信息
|
||||||
|
trace("Controller Mapping: {$mapName} => {$className}", 'debug');
|
||||||
|
|
||||||
|
// 注册控制器映射
|
||||||
|
$this->app->route->setControllerMap($mapName, $className);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 获取控制器名称(移除Controller后缀)
|
||||||
|
* @return string
|
||||||
|
*/
|
||||||
|
public function getControllerName()
|
||||||
|
{
|
||||||
|
$className = get_class($this);
|
||||||
|
$className = substr($className, strrpos($className, '\\') + 1);
|
||||||
|
return str_replace('Controller', '', $className);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 验证数据
|
||||||
|
* @access protected
|
||||||
|
* @param array $data 数据
|
||||||
|
* @param string|array $validate 验证器名或者验证规则数组
|
||||||
|
* @param array $message 提示信息
|
||||||
|
* @param bool $batch 是否批量验证
|
||||||
|
* @return array|string|true
|
||||||
|
* @throws ValidateException
|
||||||
|
*/
|
||||||
|
protected function validate(array $data, $validate, array $message = [], bool $batch = false)
|
||||||
|
{
|
||||||
|
if (is_array($validate)) {
|
||||||
|
$v = new Validate();
|
||||||
|
$v->rule($validate);
|
||||||
|
} else {
|
||||||
|
if (strpos($validate, '.')) {
|
||||||
|
// 支持场景
|
||||||
|
[$validate, $scene] = explode('.', $validate);
|
||||||
|
}
|
||||||
|
$class = false !== strpos($validate, '\\') ? $validate : $this->app->parseClass('validate', $validate);
|
||||||
|
$v = new $class();
|
||||||
|
if (!empty($scene)) {
|
||||||
|
$v->scene($scene);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
$v->message($message);
|
||||||
|
|
||||||
|
// 是否批量验证
|
||||||
|
if ($batch || $this->batchValidate) {
|
||||||
|
$v->batch(true);
|
||||||
|
}
|
||||||
|
|
||||||
|
return $v->failException(true)->check($data);
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
518
app/admin/controller/IndexController.php
Normal file
518
app/admin/controller/IndexController.php
Normal file
@ -0,0 +1,518 @@
|
|||||||
|
<?php
|
||||||
|
/**
|
||||||
|
* 商业使用授权协议
|
||||||
|
*
|
||||||
|
* Copyright (c) 2025 [云泽网]. 保留所有权利.
|
||||||
|
*
|
||||||
|
* 本软件仅供评估使用。任何商业用途必须获得书面授权许可。
|
||||||
|
* 未经授权商业使用本软件属于侵权行为,将承担法律责任。
|
||||||
|
*
|
||||||
|
* 授权购买请联系: 357099073@qq.com
|
||||||
|
* 官方网站: https://www.yunzer.cn
|
||||||
|
*
|
||||||
|
* 评估用户须知:
|
||||||
|
* 1. 禁止移除版权声明
|
||||||
|
* 2. 禁止用于生产环境
|
||||||
|
* 3. 禁止转售或分发
|
||||||
|
*/
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 后台管理系统-首页
|
||||||
|
*/
|
||||||
|
namespace app\admin\controller;
|
||||||
|
use app\admin\controller\Base;
|
||||||
|
use app\admin\model\DailyStats;
|
||||||
|
use app\admin\model\Log\LogsOperation;
|
||||||
|
use app\index\model\Attachments;
|
||||||
|
use think\facade\Db;
|
||||||
|
use think\facade\View;
|
||||||
|
use think\facade\Env;
|
||||||
|
use think\facade\Config;
|
||||||
|
use app\admin\controller\Log;
|
||||||
|
use \think\facade\Filesystem;
|
||||||
|
|
||||||
|
use app\admin\model\AdminUserGroup;
|
||||||
|
use app\admin\model\AdminSysMenu;
|
||||||
|
|
||||||
|
class IndexController extends Base{
|
||||||
|
# 首页
|
||||||
|
public function index(){
|
||||||
|
$menus = [];
|
||||||
|
$menu = [];
|
||||||
|
$where = ['group_id'=>$this->aUser['group_id']];
|
||||||
|
$role = AdminUserGroup::where($where)->find();
|
||||||
|
if($role){
|
||||||
|
$role['rights'] = (isset($role['rights']) && $role['rights']) ? json_decode($role['rights'],true) : [];
|
||||||
|
}
|
||||||
|
if($role['rights']){
|
||||||
|
$where = [
|
||||||
|
['smid','in',implode(',',$role['rights']) ],
|
||||||
|
['status','=',1]
|
||||||
|
];
|
||||||
|
// 获取所有菜单
|
||||||
|
$menus = AdminSysMenu::order('type,sort desc')->where($where)->select()->toArray();
|
||||||
|
|
||||||
|
// 构建树形结构菜单
|
||||||
|
$menuTree = [];
|
||||||
|
$menuMap = [];
|
||||||
|
|
||||||
|
// 先将所有菜单项映射到一个关联数组中
|
||||||
|
foreach($menus as $item){
|
||||||
|
$item['children'] = [];
|
||||||
|
$menuMap[$item['smid']] = $item;
|
||||||
|
}
|
||||||
|
|
||||||
|
// 构建树形结构
|
||||||
|
foreach($menus as $item){
|
||||||
|
if($item['parent_id'] == 0){
|
||||||
|
// 顶级菜单
|
||||||
|
$menuTree[$item['smid']] = &$menuMap[$item['smid']];
|
||||||
|
}else{
|
||||||
|
// 子菜单,添加到父菜单的children数组中
|
||||||
|
if(isset($menuMap[$item['parent_id']])){
|
||||||
|
$menuMap[$item['parent_id']]['children'][] = &$menuMap[$item['smid']];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
$menu = $menuTree;
|
||||||
|
}
|
||||||
|
|
||||||
|
View::assign([
|
||||||
|
'role' => $role,
|
||||||
|
'menu' => $menu
|
||||||
|
]);
|
||||||
|
return View::fetch();
|
||||||
|
}
|
||||||
|
# 欢迎页面
|
||||||
|
public function welcome(){
|
||||||
|
try {
|
||||||
|
// 获取最近7天的日期
|
||||||
|
$dates = [];
|
||||||
|
for ($i = 6; $i >= 0; $i--) {
|
||||||
|
$dates[] = date('Y-m-d', strtotime("-$i day"));
|
||||||
|
}
|
||||||
|
|
||||||
|
// 初始化数据数组
|
||||||
|
$visitData = [];
|
||||||
|
$userData = [];
|
||||||
|
$resourceData = [];
|
||||||
|
$articleData = [];
|
||||||
|
|
||||||
|
// 直接查询每天的数据
|
||||||
|
foreach ($dates as $date) {
|
||||||
|
$dayStats = Db::name('daily_stats')
|
||||||
|
->where('date', $date)
|
||||||
|
->find();
|
||||||
|
|
||||||
|
// 访问数据
|
||||||
|
$visitData[] = [
|
||||||
|
'date' => $date,
|
||||||
|
'visits' => $dayStats ? intval($dayStats['daily_visits']) : 0,
|
||||||
|
'uv' => $dayStats ? intval($dayStats['unique_visitors']) : 0
|
||||||
|
];
|
||||||
|
|
||||||
|
// 用户数据
|
||||||
|
$userData[] = [
|
||||||
|
'date' => $date,
|
||||||
|
'total' => $dayStats ? intval($dayStats['total_users']) : 0,
|
||||||
|
'new' => $dayStats ? intval($dayStats['new_users']) : 0
|
||||||
|
];
|
||||||
|
|
||||||
|
// 资源数据
|
||||||
|
$resourceData[] = [
|
||||||
|
'date' => $date,
|
||||||
|
'total' => $dayStats ? intval($dayStats['total_resources']) : 0,
|
||||||
|
'new' => $dayStats ? intval($dayStats['daily_resources']) : 0,
|
||||||
|
'downloads' => $dayStats ? intval($dayStats['resource_downloads']) : 0
|
||||||
|
];
|
||||||
|
|
||||||
|
// 文章数据
|
||||||
|
$articleData[] = [
|
||||||
|
'date' => $date,
|
||||||
|
'total' => $dayStats ? intval($dayStats['total_articles']) : 0,
|
||||||
|
'new' => $dayStats ? intval($dayStats['daily_articles']) : 0,
|
||||||
|
'views' => $dayStats ? intval($dayStats['article_views']) : 0
|
||||||
|
];
|
||||||
|
}
|
||||||
|
|
||||||
|
// 获取今日统计数据
|
||||||
|
$today = date('Y-m-d');
|
||||||
|
$todayStats = Db::name('daily_stats')
|
||||||
|
->where('date', $today)
|
||||||
|
->find();
|
||||||
|
|
||||||
|
// 获取最近的操作日志
|
||||||
|
$recentActivities = Db::name('logs_operation')
|
||||||
|
->field('operation_time, module, operation')
|
||||||
|
->order('operation_time DESC')
|
||||||
|
->limit(5)
|
||||||
|
->select()
|
||||||
|
->each(function($item) {
|
||||||
|
$item['content'] = date('Y年m月d日 H:i:s', strtotime($item['operation_time'])) . ' 在 ' .
|
||||||
|
($item['module'] ?: '未知模块') . ' ' .
|
||||||
|
($item['operation'] ?: '未知操作');
|
||||||
|
$item['icon'] = $this->getActivityIcon($item['module'] ?: '其他');
|
||||||
|
return $item;
|
||||||
|
});
|
||||||
|
|
||||||
|
// 处理图表数据
|
||||||
|
$chartData = [
|
||||||
|
'visitTrend' => [
|
||||||
|
'dates' => array_map(function($item) { return date('m-d', strtotime($item['date'])); }, $visitData),
|
||||||
|
'visits' => array_column($visitData, 'visits'),
|
||||||
|
'uvs' => array_column($visitData, 'uv')
|
||||||
|
],
|
||||||
|
'userGrowth' => [
|
||||||
|
'dates' => array_map(function($item) { return date('m-d', strtotime($item['date'])); }, $userData),
|
||||||
|
'newUsers' => array_column($userData, 'new'),
|
||||||
|
'totalUsers' => array_column($userData, 'total')
|
||||||
|
],
|
||||||
|
'resourceStats' => [
|
||||||
|
'dates' => array_map(function($item) { return date('m-d', strtotime($item['date'])); }, $resourceData),
|
||||||
|
'newResources' => array_column($resourceData, 'new'),
|
||||||
|
'totalResources' => array_column($resourceData, 'total'),
|
||||||
|
'downloads' => array_column($resourceData, 'downloads')
|
||||||
|
],
|
||||||
|
'articleStats' => [
|
||||||
|
'dates' => array_map(function($item) { return date('m-d', strtotime($item['date'])); }, $articleData),
|
||||||
|
'newArticles' => array_column($articleData, 'new'),
|
||||||
|
'totalArticles' => array_column($articleData, 'total'),
|
||||||
|
'views' => array_column($articleData, 'views')
|
||||||
|
]
|
||||||
|
];
|
||||||
|
|
||||||
|
// 传递给视图
|
||||||
|
View::assign([
|
||||||
|
'todayStats' => $todayStats ?: [
|
||||||
|
'total_users' => 0,
|
||||||
|
'new_users' => 0,
|
||||||
|
'total_visits' => 0,
|
||||||
|
'daily_visits' => 0,
|
||||||
|
'unique_visitors' => 0,
|
||||||
|
'total_articles' => 0,
|
||||||
|
'daily_articles' => 0,
|
||||||
|
'article_views' => 0,
|
||||||
|
'total_resources' => 0,
|
||||||
|
'daily_resources' => 0,
|
||||||
|
'resource_downloads' => 0
|
||||||
|
],
|
||||||
|
'recentActivities' => $recentActivities,
|
||||||
|
'chartData' => $chartData
|
||||||
|
]);
|
||||||
|
|
||||||
|
return View::fetch();
|
||||||
|
} catch (\Exception $e) {
|
||||||
|
// 记录错误日志
|
||||||
|
\think\facade\Log::error('获取统计数据失败:' . $e->getMessage());
|
||||||
|
// 返回空数据
|
||||||
|
View::assign([
|
||||||
|
'todayStats' => [
|
||||||
|
'total_users' => 0,
|
||||||
|
'new_users' => 0,
|
||||||
|
'total_visits' => 0,
|
||||||
|
'daily_visits' => 0,
|
||||||
|
'unique_visitors' => 0,
|
||||||
|
'total_articles' => 0,
|
||||||
|
'daily_articles' => 0,
|
||||||
|
'article_views' => 0,
|
||||||
|
'total_resources' => 0,
|
||||||
|
'daily_resources' => 0,
|
||||||
|
'resource_downloads' => 0
|
||||||
|
],
|
||||||
|
'recentActivities' => [],
|
||||||
|
'chartData' => [
|
||||||
|
'visitTrend' => ['dates' => [], 'visits' => [], 'uvs' => []],
|
||||||
|
'userGrowth' => ['dates' => [], 'newUsers' => [], 'totalUsers' => []],
|
||||||
|
'resourceStats' => ['dates' => [], 'newResources' => [], 'totalResources' => [], 'downloads' => []],
|
||||||
|
'articleStats' => ['dates' => [], 'newArticles' => [], 'totalArticles' => [], 'views' => []]
|
||||||
|
]
|
||||||
|
]);
|
||||||
|
return View::fetch();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 根据操作类型获取对应的图标
|
||||||
|
*/
|
||||||
|
private function getActivityIcon($type)
|
||||||
|
{
|
||||||
|
$icons = [
|
||||||
|
'用户管理' => '👥',
|
||||||
|
'文章管理' => '📝',
|
||||||
|
'资源管理' => '📦',
|
||||||
|
'系统设置' => '⚙️',
|
||||||
|
'登录' => '🔑',
|
||||||
|
'退出' => '🚪',
|
||||||
|
'其他' => '📌'
|
||||||
|
];
|
||||||
|
|
||||||
|
return $icons[$type] ?? '📌';
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 格式化访问趋势数据
|
||||||
|
*/
|
||||||
|
private function formatVisitTrendData($data)
|
||||||
|
{
|
||||||
|
$dates = [];
|
||||||
|
$visits = [];
|
||||||
|
$uvs = [];
|
||||||
|
|
||||||
|
foreach ($data as $item) {
|
||||||
|
$dates[] = date('m-d', strtotime($item['date']));
|
||||||
|
$visits[] = $item['daily_visits'];
|
||||||
|
$uvs[] = $item['unique_visitors'];
|
||||||
|
}
|
||||||
|
|
||||||
|
return [
|
||||||
|
'dates' => $dates,
|
||||||
|
'visits' => $visits,
|
||||||
|
'uvs' => $uvs
|
||||||
|
];
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 格式化用户增长数据
|
||||||
|
*/
|
||||||
|
private function formatUserGrowthData($data)
|
||||||
|
{
|
||||||
|
$dates = [];
|
||||||
|
$newUsers = [];
|
||||||
|
$totalUsers = [];
|
||||||
|
|
||||||
|
foreach ($data as $item) {
|
||||||
|
$dates[] = date('m-d', strtotime($item['date']));
|
||||||
|
$newUsers[] = $item['new_users'];
|
||||||
|
$totalUsers[] = $item['total_users'];
|
||||||
|
}
|
||||||
|
|
||||||
|
return [
|
||||||
|
'dates' => $dates,
|
||||||
|
'newUsers' => $newUsers,
|
||||||
|
'totalUsers' => $totalUsers
|
||||||
|
];
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 格式化资源统计数据
|
||||||
|
*/
|
||||||
|
private function formatResourceStatsData($data)
|
||||||
|
{
|
||||||
|
$dates = [];
|
||||||
|
$resources = [];
|
||||||
|
$downloads = [];
|
||||||
|
|
||||||
|
foreach ($data as $item) {
|
||||||
|
$dates[] = date('m-d', strtotime($item['date']));
|
||||||
|
$resources[] = $item['daily_resources'];
|
||||||
|
$downloads[] = $item['resource_downloads'];
|
||||||
|
}
|
||||||
|
|
||||||
|
return [
|
||||||
|
'dates' => $dates,
|
||||||
|
'resources' => $resources,
|
||||||
|
'downloads' => $downloads
|
||||||
|
];
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 格式化文章统计数据
|
||||||
|
*/
|
||||||
|
private function formatArticleStatsData($data)
|
||||||
|
{
|
||||||
|
$dates = [];
|
||||||
|
$articles = [];
|
||||||
|
$views = [];
|
||||||
|
|
||||||
|
foreach ($data as $item) {
|
||||||
|
$dates[] = date('m-d', strtotime($item['date']));
|
||||||
|
$articles[] = $item['daily_articles'];
|
||||||
|
$views[] = $item['article_views'];
|
||||||
|
}
|
||||||
|
|
||||||
|
return [
|
||||||
|
'dates' => $dates,
|
||||||
|
'articles' => $articles,
|
||||||
|
'views' => $views
|
||||||
|
];
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 保存附件信息到数据库
|
||||||
|
* @param string $name 文件名
|
||||||
|
* @param int $type 附件类型
|
||||||
|
* @param int $size 文件大小
|
||||||
|
* @param string $src 文件路径
|
||||||
|
* @return int 附件ID
|
||||||
|
*/
|
||||||
|
private function saveAttachment($name, $type, $size, $src) {
|
||||||
|
$data = [
|
||||||
|
'name' => $name,
|
||||||
|
'type' => $type,
|
||||||
|
'size' => $size,
|
||||||
|
'src' => $src,
|
||||||
|
'create_time' => time(),
|
||||||
|
'update_time' => time()
|
||||||
|
];
|
||||||
|
return Attachments::insertGetId($data);
|
||||||
|
}
|
||||||
|
|
||||||
|
# 图片上传
|
||||||
|
public function upload_img(){
|
||||||
|
// 获取上传的文件
|
||||||
|
$file = request()->file();
|
||||||
|
$files = request()->file('file');
|
||||||
|
|
||||||
|
// 检查是否有文件上传
|
||||||
|
if(empty($file)){
|
||||||
|
return json(['code'=>1, 'msg'=>'没有文件上传'])->send();
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
// 验证上传的文件
|
||||||
|
validate([
|
||||||
|
'image'=>'filesize:51200|fileExt:jpg,png,gif,jpeg'
|
||||||
|
])->check($file);
|
||||||
|
|
||||||
|
// 存储文件到public磁盘的uploads目录
|
||||||
|
$info = Filesystem::disk('public')->putFile('uploads', $files);
|
||||||
|
|
||||||
|
// 处理文件路径,统一使用正斜杠
|
||||||
|
$info = str_replace("\\", "/", $info);
|
||||||
|
$img = '/storage/'.$info;
|
||||||
|
|
||||||
|
// 保存附件信息
|
||||||
|
$fileName = $files->getOriginalName();
|
||||||
|
$fileSize = $files->getSize();
|
||||||
|
$attachmentId = $this->saveAttachment($fileName, 1, $fileSize, $img); // 1: 图片
|
||||||
|
|
||||||
|
// 返回成功信息
|
||||||
|
return json([
|
||||||
|
'code' => 0,
|
||||||
|
'data' => $img,
|
||||||
|
'url' => $this->config['admin_domain'].$img,
|
||||||
|
'attachment_id' => $attachmentId
|
||||||
|
])->send();
|
||||||
|
|
||||||
|
} catch (\think\exception\ValidateException $e) {
|
||||||
|
// 捕获验证异常并返回错误信息
|
||||||
|
return json(['code'=>1, 'msg'=>$e->getMessage()])->send();
|
||||||
|
} catch (\Exception $e) {
|
||||||
|
// 捕获其他异常
|
||||||
|
return json(['code'=>1, 'msg'=>'上传失败:'.$e->getMessage()])->send();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
# 清除缓存
|
||||||
|
public function clear(){
|
||||||
|
$a = delete_dir_file(Env::get('runtime_path').'cache/');
|
||||||
|
$b = delete_dir_file(Env::get('runtime_path').'temp/');
|
||||||
|
if ($a || $b) {
|
||||||
|
$this->returnCode(0, '清除缓存成功');
|
||||||
|
} else {
|
||||||
|
$this->returnCode(1, '清除缓存失败');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
# 文件上传
|
||||||
|
public function upload_file(){
|
||||||
|
$file = request()->file();
|
||||||
|
$files = request()->file('file');
|
||||||
|
if(empty($file)){
|
||||||
|
return json(['code'=>1, 'msg'=>'没有文件上传'])->send();
|
||||||
|
}
|
||||||
|
try {
|
||||||
|
// 只验证文件扩展名,不验证大小
|
||||||
|
validate([
|
||||||
|
'file'=>'fileExt:doc,docx,xls,xlsx,ppt,pptx,pdf,txt,zip,rar,7z'
|
||||||
|
])->check($file);
|
||||||
|
|
||||||
|
$info = Filesystem::disk('public')->putFile('uploads/files', $files);
|
||||||
|
|
||||||
|
// 处理文件路径
|
||||||
|
$info = str_replace("\\", "/", $info);
|
||||||
|
$filePath = '/storage/'.$info;
|
||||||
|
|
||||||
|
// 保存附件信息
|
||||||
|
$fileName = $files->getOriginalName();
|
||||||
|
$fileSize = $files->getSize();
|
||||||
|
$attachmentId = $this->saveAttachment($fileName, 2, $fileSize, $filePath); // 2: 文件
|
||||||
|
|
||||||
|
return json([
|
||||||
|
'code' => 0,
|
||||||
|
'data' => [
|
||||||
|
'src' => $filePath,
|
||||||
|
'attachment_id' => $attachmentId
|
||||||
|
]
|
||||||
|
])->send();
|
||||||
|
} catch (\think\exception\ValidateException $e) {
|
||||||
|
return json(['code'=>1, 'msg'=>$e->getMessage()])->send();
|
||||||
|
} catch (\Exception $e) {
|
||||||
|
return json(['code'=>1, 'msg'=>'上传失败:'.$e->getMessage()])->send();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
# 视频上传
|
||||||
|
public function upload_video(){
|
||||||
|
$file = request()->file();
|
||||||
|
$files = request()->file('file');
|
||||||
|
if(empty($file)){
|
||||||
|
return json(['code'=>1, 'msg'=>'没有文件上传'])->send();
|
||||||
|
}
|
||||||
|
try {
|
||||||
|
validate(['video'=>'filesize:102400|fileExt:mp4,avi,mov,wmv,flv'])->check($file);
|
||||||
|
$info = Filesystem::disk('public')->putFile('uploads/videos', $files);
|
||||||
|
|
||||||
|
// 处理文件路径
|
||||||
|
$info = str_replace("\\", "/", $info);
|
||||||
|
$videoPath = '/storage/'.$info;
|
||||||
|
|
||||||
|
// 保存附件信息
|
||||||
|
$fileName = $files->getOriginalName();
|
||||||
|
$fileSize = $files->getSize();
|
||||||
|
$attachmentId = $this->saveAttachment($fileName, 3, $fileSize, $videoPath); // 3: 视频
|
||||||
|
|
||||||
|
return json([
|
||||||
|
'code' => 0,
|
||||||
|
'data' => [
|
||||||
|
'src' => $videoPath,
|
||||||
|
'attachment_id' => $attachmentId
|
||||||
|
]
|
||||||
|
])->send();
|
||||||
|
} catch (\think\exception\ValidateException $e) {
|
||||||
|
return json(['code'=>1, 'msg'=>$e->getMessage()])->send();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
# 音频上传
|
||||||
|
public function upload_audio(){
|
||||||
|
$file = request()->file();
|
||||||
|
$files = request()->file('file');
|
||||||
|
if(empty($file)){
|
||||||
|
return json(['code'=>1, 'msg'=>'没有文件上传'])->send();
|
||||||
|
}
|
||||||
|
try {
|
||||||
|
validate(['audio'=>'filesize:51200|fileExt:mp3,wav,ogg,m4a'])->check($file);
|
||||||
|
$info = Filesystem::disk('public')->putFile('uploads/audios', $files);
|
||||||
|
|
||||||
|
// 处理文件路径
|
||||||
|
$info = str_replace("\\", "/", $info);
|
||||||
|
$audioPath = '/storage/'.$info;
|
||||||
|
|
||||||
|
// 保存附件信息
|
||||||
|
$fileName = $files->getOriginalName();
|
||||||
|
$fileSize = $files->getSize();
|
||||||
|
$attachmentId = $this->saveAttachment($fileName, 4, $fileSize, $audioPath); // 4: 音频
|
||||||
|
|
||||||
|
return json([
|
||||||
|
'code' => 0,
|
||||||
|
'data' => [
|
||||||
|
'src' => $audioPath,
|
||||||
|
'attachment_id' => $attachmentId
|
||||||
|
]
|
||||||
|
])->send();
|
||||||
|
} catch (\think\exception\ValidateException $e) {
|
||||||
|
return json(['code'=>1, 'msg'=>$e->getMessage()])->send();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
177
app/admin/controller/LogController.php
Normal file
177
app/admin/controller/LogController.php
Normal file
@ -0,0 +1,177 @@
|
|||||||
|
<?php
|
||||||
|
/**
|
||||||
|
* 商业使用授权协议
|
||||||
|
*
|
||||||
|
* Copyright (c) 2025 [云泽网]. 保留所有权利.
|
||||||
|
*
|
||||||
|
* 本软件仅供评估使用。任何商业用途必须获得书面授权许可。
|
||||||
|
* 未经授权商业使用本软件属于侵权行为,将承担法律责任。
|
||||||
|
*
|
||||||
|
* 授权购买请联系: 357099073@qq.com
|
||||||
|
* 官方网站: https://www.yunzer.cn
|
||||||
|
*
|
||||||
|
* 评估用户须知:
|
||||||
|
* 1. 禁止移除版权声明
|
||||||
|
* 2. 禁止用于生产环境
|
||||||
|
* 3. 禁止转售或分发
|
||||||
|
*/
|
||||||
|
|
||||||
|
namespace app\admin\controller;
|
||||||
|
|
||||||
|
use app\admin\controller\Base;
|
||||||
|
use think\facade\Db;
|
||||||
|
use think\facade\Request;
|
||||||
|
use think\facade\View;
|
||||||
|
use think\facade\Cookie;
|
||||||
|
use app\admin\model\Log\LogsLogin;
|
||||||
|
use app\admin\model\Log\LogsOperation;
|
||||||
|
|
||||||
|
class LogController extends Base
|
||||||
|
{
|
||||||
|
/**
|
||||||
|
* 登录日志列表
|
||||||
|
*/
|
||||||
|
public function login()
|
||||||
|
{
|
||||||
|
if (Request::isPost()) {
|
||||||
|
$page = input('post.page', 1);
|
||||||
|
$limit = input('post.limit', 10);
|
||||||
|
$username = input('post.username');
|
||||||
|
$ip = input('post.ip');
|
||||||
|
$status = input('post.status');
|
||||||
|
$startTime = input('post.start_time');
|
||||||
|
$endTime = input('post.end_time');
|
||||||
|
|
||||||
|
// 搜索条件
|
||||||
|
if ($username) {
|
||||||
|
$query = LogsLogin::where('username', 'like', "%{$username}%");
|
||||||
|
}
|
||||||
|
if ($ip) {
|
||||||
|
$query = LogsLogin::where('ip_address', 'like', "%{$ip}%");
|
||||||
|
}
|
||||||
|
if ($status !== '') {
|
||||||
|
$query = LogsLogin::where('login_status', $status);
|
||||||
|
}
|
||||||
|
if ($startTime) {
|
||||||
|
$query = LogsLogin::where('login_time', '>=', $startTime);
|
||||||
|
}
|
||||||
|
if ($endTime) {
|
||||||
|
$query = LogsLogin::where('login_time', '<=', $endTime);
|
||||||
|
}
|
||||||
|
|
||||||
|
$count = LogsLogin::count();
|
||||||
|
$list = LogsLogin::order('id desc')
|
||||||
|
->page($page, $limit)
|
||||||
|
->select();
|
||||||
|
|
||||||
|
return json([
|
||||||
|
'code' => 0,
|
||||||
|
'msg' => '获取成功',
|
||||||
|
'count' => $count,
|
||||||
|
'data' => $list
|
||||||
|
]);
|
||||||
|
}
|
||||||
|
|
||||||
|
return View::fetch();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 操作日志列表
|
||||||
|
*/
|
||||||
|
public function operation()
|
||||||
|
{
|
||||||
|
$params = input();
|
||||||
|
$page = $params['page'] ?? 1;
|
||||||
|
$limit = $params['limit'] ?? 10;
|
||||||
|
|
||||||
|
// 构建搜索条件
|
||||||
|
$searchFields = [
|
||||||
|
'username' => ['field' => 'username', 'type' => 'like'],
|
||||||
|
'module' => ['field' => 'module', 'type' => 'like'],
|
||||||
|
'operation' => ['field' => 'operation', 'type' => 'like'],
|
||||||
|
'status' => ['field' => 'status', 'type' => '='],
|
||||||
|
'start_time' => ['field' => 'operation_time', 'type' => '>='],
|
||||||
|
'end_time' => ['field' => 'operation_time', 'type' => '<=']
|
||||||
|
];
|
||||||
|
|
||||||
|
$query = LogsOperation::where('1=1');
|
||||||
|
|
||||||
|
foreach ($searchFields as $param => $config) {
|
||||||
|
if (!empty($params[$param])) {
|
||||||
|
$value = $params[$param];
|
||||||
|
if ($config['type'] === 'like') {
|
||||||
|
$value = "%{$value}%";
|
||||||
|
}
|
||||||
|
$query = $query->where($config['field'], $config['type'], $value);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
$count = LogsOperation::where('1=1')->count();
|
||||||
|
$list = LogsOperation::where('1=1')
|
||||||
|
->order('id desc')
|
||||||
|
->page($page, $limit)
|
||||||
|
->select();
|
||||||
|
|
||||||
|
if (Request::isAjax()) {
|
||||||
|
return json([
|
||||||
|
'code' => 0,
|
||||||
|
'msg' => '获取成功',
|
||||||
|
'count' => $count,
|
||||||
|
'data' => $list
|
||||||
|
]);
|
||||||
|
}
|
||||||
|
|
||||||
|
return View::fetch();
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 记录操作日志
|
||||||
|
* @param string $operation 操作名称
|
||||||
|
* @param int $status 状态 1成功 0失败
|
||||||
|
* @param string $error_message 错误信息
|
||||||
|
* @param string $module 模块名称
|
||||||
|
*/
|
||||||
|
public static function record($operation, $status = 1, $error_message = '', $module = '')
|
||||||
|
{
|
||||||
|
$data = [
|
||||||
|
'username' => Cookie::get('admin_name') ?: '未知用户',
|
||||||
|
'module' => $module ?: '系统管理',
|
||||||
|
'operation' => $operation,
|
||||||
|
'request_method' => Request::method(),
|
||||||
|
'request_url' => Request::url(true),
|
||||||
|
'request_params' => json_encode(Request::param(), JSON_UNESCAPED_UNICODE),
|
||||||
|
'ip_address' => Request::ip(),
|
||||||
|
'status' => $status,
|
||||||
|
'error_message' => $error_message,
|
||||||
|
'operation_time' => date('Y-m-d H:i:s'),
|
||||||
|
'execution_time' => 0
|
||||||
|
];
|
||||||
|
|
||||||
|
LogsOperation::insert($data);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 获取操作日志详情
|
||||||
|
*/
|
||||||
|
public function getOperationDetail()
|
||||||
|
{
|
||||||
|
$id = input('id/d', 0);
|
||||||
|
if (!$id) {
|
||||||
|
return json(['code' => 1, 'msg' => '参数错误']);
|
||||||
|
}
|
||||||
|
|
||||||
|
$info = LogsOperation::where('id', $id)
|
||||||
|
->find();
|
||||||
|
|
||||||
|
if (!$info) {
|
||||||
|
return json(['code' => 1, 'msg' => '日志不存在']);
|
||||||
|
}
|
||||||
|
|
||||||
|
// 格式化请求参数
|
||||||
|
if (!empty($info['request_params'])) {
|
||||||
|
$info['request_params'] = json_decode($info['request_params'], true);
|
||||||
|
}
|
||||||
|
|
||||||
|
return json(['code' => 0, 'msg' => '获取成功', 'data' => $info]);
|
||||||
|
}
|
||||||
|
}
|
||||||
195
app/admin/controller/LoginController.php
Normal file
195
app/admin/controller/LoginController.php
Normal file
@ -0,0 +1,195 @@
|
|||||||
|
<?php
|
||||||
|
/**
|
||||||
|
* 商业使用授权协议
|
||||||
|
*
|
||||||
|
* Copyright (c) 2025 [云泽网]. 保留所有权利.
|
||||||
|
*
|
||||||
|
* 本软件仅供评估使用。任何商业用途必须获得书面授权许可。
|
||||||
|
* 未经授权商业使用本软件属于侵权行为,将承担法律责任。
|
||||||
|
*
|
||||||
|
* 授权购买请联系: 357099073@qq.com
|
||||||
|
* 官方网站: https://www.yunzer.cn
|
||||||
|
*
|
||||||
|
* 评估用户须知:
|
||||||
|
* 1. 禁止移除版权声明
|
||||||
|
* 2. 禁止用于生产环境
|
||||||
|
* 3. 禁止转售或分发
|
||||||
|
*/
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 后台管理系统-登录
|
||||||
|
*/
|
||||||
|
namespace app\admin\controller;
|
||||||
|
use think\App;
|
||||||
|
use app\AppApi;
|
||||||
|
use think\facade\Db;
|
||||||
|
use think\facade\View;
|
||||||
|
use think\facade\Cookie;
|
||||||
|
use think\facade\Request;
|
||||||
|
use app\admin\model\YzAdminConfig;
|
||||||
|
use app\admin\model\AdminUser;
|
||||||
|
use app\admin\model\Log\LogsLogin;
|
||||||
|
|
||||||
|
class LoginController extends Base
|
||||||
|
{
|
||||||
|
public $app;
|
||||||
|
public $config;
|
||||||
|
|
||||||
|
public function __construct(App $app)
|
||||||
|
{
|
||||||
|
$this->app = $app;
|
||||||
|
$this->config = new YzAdminConfig();
|
||||||
|
}
|
||||||
|
|
||||||
|
// 登录页面
|
||||||
|
public function index()
|
||||||
|
{
|
||||||
|
# 获取配置
|
||||||
|
$config = $this->config->getAll();
|
||||||
|
View::assign([
|
||||||
|
'config' => $config
|
||||||
|
]);
|
||||||
|
return View::fetch();
|
||||||
|
}
|
||||||
|
|
||||||
|
// 记录登录日志
|
||||||
|
public function recordLoginLog($username, $status, $reason = '')
|
||||||
|
{
|
||||||
|
$data = [
|
||||||
|
'username' => $username,
|
||||||
|
'ip_address' => Request::ip(),
|
||||||
|
'location' => $this->getLocation(Request::ip()),
|
||||||
|
'device_type' => $this->getDeviceType(),
|
||||||
|
'user_agent' => Request::header('user-agent'),
|
||||||
|
'login_status' => $status,
|
||||||
|
'failure_reason' => $reason,
|
||||||
|
'login_time' => date('Y-m-d H:i:s')
|
||||||
|
];
|
||||||
|
LogsLogin::create($data);
|
||||||
|
}
|
||||||
|
|
||||||
|
// 获取IP地址位置
|
||||||
|
public function getLocation($ip)
|
||||||
|
{
|
||||||
|
// 这里可以接入IP地址库或第三方API
|
||||||
|
return '未知';
|
||||||
|
}
|
||||||
|
|
||||||
|
// 获取设备类型
|
||||||
|
public function getDeviceType()
|
||||||
|
{
|
||||||
|
$agent = Request::header('user-agent');
|
||||||
|
if (preg_match('/(iPhone|iPod|Android|ios|iPad|Mobile)/i', $agent)) {
|
||||||
|
return '移动端';
|
||||||
|
}
|
||||||
|
return 'PC端';
|
||||||
|
}
|
||||||
|
|
||||||
|
// 登录
|
||||||
|
public function login()
|
||||||
|
{
|
||||||
|
if (Request::isPost()) {
|
||||||
|
$account = trim(input('post.account'));
|
||||||
|
if (empty($account)) {
|
||||||
|
$this->recordLoginLog($account, 0, '账号不能为空');
|
||||||
|
return json(['code' => 1, 'msg' => '账号不能为空']);
|
||||||
|
}
|
||||||
|
$pattern = "/^([0-9A-Za-z-_.]+)@([0-9a-z]+.[a-z]{2,3}(.[a-z]{2})?)$/i";
|
||||||
|
if (!preg_match($pattern, $account)) {
|
||||||
|
$this->recordLoginLog($account, 0, '邮箱格式不正确');
|
||||||
|
return json(['code' => 1, 'msg' => '邮箱格式不正确']);
|
||||||
|
}
|
||||||
|
$password = trim(input('post.password'));
|
||||||
|
if (empty($password)) {
|
||||||
|
$this->recordLoginLog($account, 0, '密码不能为空');
|
||||||
|
return json(['code' => 1, 'msg' => '密码不能为空']);
|
||||||
|
}
|
||||||
|
$code = trim(input('post.code'));
|
||||||
|
if ($code == '') {
|
||||||
|
$this->recordLoginLog($account, 0, '验证码不能为空');
|
||||||
|
return json(['code' => 1, 'msg' => '验证码不能为空']);
|
||||||
|
}
|
||||||
|
if (!captcha_check($code)) {
|
||||||
|
$this->recordLoginLog($account, 0, '验证码错误');
|
||||||
|
return json(['code' => 1, 'msg' => '验证码错误']);
|
||||||
|
}
|
||||||
|
$aUser = AdminUser::where('account', $account)->find();
|
||||||
|
if (empty($aUser)) {
|
||||||
|
$this->recordLoginLog($account, 0, '账号不存在');
|
||||||
|
return json(['code' => 1, 'msg' => '账号不存在']);
|
||||||
|
}
|
||||||
|
if ($aUser['status'] != 1) {
|
||||||
|
$this->recordLoginLog($account, 0, '账号已被禁用');
|
||||||
|
return json(['code' => 1, 'msg' => '账号已被禁用']);
|
||||||
|
}
|
||||||
|
if ($aUser['password'] != md5($password)) {
|
||||||
|
$this->recordLoginLog($account, 0, '密码错误');
|
||||||
|
return json(['code' => 1, 'msg' => '密码错误']);
|
||||||
|
}
|
||||||
|
$remember = input('post.remember');
|
||||||
|
if (!empty($remember)) {
|
||||||
|
Cookie::set('admin_id', $aUser['uid'], 60 * 60 * 24 * 7);
|
||||||
|
Cookie::set('admin_name', $aUser['name'], 60 * 60 * 24 * 7);
|
||||||
|
} else {
|
||||||
|
Cookie::set('admin_id', $aUser['uid']);
|
||||||
|
Cookie::set('admin_name', $aUser['name']);
|
||||||
|
}
|
||||||
|
AdminUser::where('uid', $aUser['uid'])->update(
|
||||||
|
['login_count' => $aUser['login_count'] + 1, 'update_time' => time()]
|
||||||
|
);
|
||||||
|
// 记录登录成功日志
|
||||||
|
$this->recordLoginLog($account, 1);
|
||||||
|
return json(['code' => 0, 'msg' => '登录成功', 'data' => []]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// 退出
|
||||||
|
public function logout()
|
||||||
|
{
|
||||||
|
Cookie::delete('admin_id');
|
||||||
|
Cookie::delete('admin_name');
|
||||||
|
return json(['code' => 0, 'msg' => '退出成功', 'data' => []]);
|
||||||
|
}
|
||||||
|
|
||||||
|
// 密码重置页面
|
||||||
|
public function resetpwdindex()
|
||||||
|
{
|
||||||
|
return View::fetch('resetpwd');
|
||||||
|
}
|
||||||
|
|
||||||
|
//管理员密码重置
|
||||||
|
public function resetpwd()
|
||||||
|
{
|
||||||
|
$account = trim(input('post.account'));
|
||||||
|
if (empty($account)) {
|
||||||
|
return json(['code' => 1, 'msg' => '账号不能为空']);
|
||||||
|
}
|
||||||
|
|
||||||
|
$user = AdminUser::where('account', $account)->find();
|
||||||
|
|
||||||
|
if (!$user) {
|
||||||
|
return json(['code' => 1, 'msg' => '未找到该用户名']);
|
||||||
|
}
|
||||||
|
|
||||||
|
// 使用md5进行密码加密处理
|
||||||
|
$password = md5('123456');
|
||||||
|
|
||||||
|
try {
|
||||||
|
$res = AdminUser::where('account', $account)
|
||||||
|
->update(['password' => $password]);
|
||||||
|
|
||||||
|
if ($res === false) {
|
||||||
|
return json(['code' => 1, 'msg' => '数据库更新失败']);
|
||||||
|
}
|
||||||
|
|
||||||
|
if ($res === 0) {
|
||||||
|
return json(['code' => 1, 'msg' => '密码未发生变化']);
|
||||||
|
}
|
||||||
|
|
||||||
|
return json(['code' => 0, 'msg' => '密码重置成功', 'data' => []]);
|
||||||
|
} catch (\Exception $e) {
|
||||||
|
return json(['code' => 1, 'msg' => '系统错误:' . $e->getMessage()]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
554
app/admin/controller/ResourcesController.php
Normal file
554
app/admin/controller/ResourcesController.php
Normal file
@ -0,0 +1,554 @@
|
|||||||
|
<?php
|
||||||
|
/**
|
||||||
|
* 商业使用授权协议
|
||||||
|
*
|
||||||
|
* Copyright (c) 2025 [云泽网]. 保留所有权利.
|
||||||
|
*
|
||||||
|
* 本软件仅供评估使用。任何商业用途必须获得书面授权许可。
|
||||||
|
* 未经授权商业使用本软件属于侵权行为,将承担法律责任。
|
||||||
|
*
|
||||||
|
* 授权购买请联系: 357099073@qq.com
|
||||||
|
* 官方网站: https://www.yunzer.cn
|
||||||
|
*
|
||||||
|
* 评估用户须知:
|
||||||
|
* 1. 禁止移除版权声明
|
||||||
|
* 2. 禁止用于生产环境
|
||||||
|
* 3. 禁止转售或分发
|
||||||
|
*/
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 后台管理系统-资源管理
|
||||||
|
*/
|
||||||
|
namespace app\admin\controller;
|
||||||
|
use app\admin\controller\BaseController;
|
||||||
|
use app\admin\model\Resource\Resource;
|
||||||
|
use app\admin\model\Resource\ResourceCategory;
|
||||||
|
use think\facade\View;
|
||||||
|
use think\facade\Request;
|
||||||
|
use think\facade\Db;
|
||||||
|
use app\admin\controller\LogController as Log;
|
||||||
|
use think\App;
|
||||||
|
|
||||||
|
class ResourcesController extends BaseController
|
||||||
|
{
|
||||||
|
// 资源列表
|
||||||
|
public function lists()
|
||||||
|
{
|
||||||
|
if (Request::isPost()) {
|
||||||
|
$params = [
|
||||||
|
'category' => input('post.category'),
|
||||||
|
'name' => input('post.name'),
|
||||||
|
'uploader' => input('post.uploader')
|
||||||
|
];
|
||||||
|
$page = (int) input('post.page', 1);
|
||||||
|
$limit = (int) input('post.limit', 10);
|
||||||
|
|
||||||
|
$query = Resource::where('delete_time', null)
|
||||||
|
->where('status', 1);
|
||||||
|
|
||||||
|
// 分类筛选
|
||||||
|
if (!empty($params['category'])) {
|
||||||
|
$cateInfo = ResourceCategory::where('name', $params['category'])
|
||||||
|
->where('delete_time', null)
|
||||||
|
->where('status', 1)
|
||||||
|
->field('id')
|
||||||
|
->find();
|
||||||
|
|
||||||
|
if ($cateInfo) {
|
||||||
|
$query = $query->where('cate', (int) $cateInfo['id']);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// 名称搜索
|
||||||
|
if (!empty($params['name'])) {
|
||||||
|
$query = $query->where('name', 'like', '%' . $params['name'] . '%');
|
||||||
|
}
|
||||||
|
|
||||||
|
// 上传者搜索
|
||||||
|
if (!empty($params['uploader'])) {
|
||||||
|
$query = $query->where('uploader', 'like', '%' . $params['uploader'] . '%');
|
||||||
|
}
|
||||||
|
|
||||||
|
$count = $query->count();
|
||||||
|
|
||||||
|
$lists = $query->order('id DESC')
|
||||||
|
->page($page, $limit)
|
||||||
|
->select()
|
||||||
|
->each(function ($item) {
|
||||||
|
// 获取分类信息
|
||||||
|
$cateInfo = ResourceCategory::where('id', (int) $item['cate'])
|
||||||
|
->field('name, icon')
|
||||||
|
->find();
|
||||||
|
if ($cateInfo) {
|
||||||
|
$item['cate'] = $cateInfo['name'];
|
||||||
|
if (empty($item['icon']) && !empty($cateInfo['icon'])) {
|
||||||
|
$item['icon'] = $cateInfo['icon'];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
$item['create_time'] = date('Y-m-d H:i:s', (int) $item['create_time']);
|
||||||
|
return $item;
|
||||||
|
});
|
||||||
|
|
||||||
|
return json([
|
||||||
|
'code' => 0,
|
||||||
|
'msg' => '获取成功',
|
||||||
|
'count' => $count,
|
||||||
|
'data' => $lists
|
||||||
|
]);
|
||||||
|
} else {
|
||||||
|
$allCategories = ResourceCategory::where('delete_time', null)
|
||||||
|
->where('status', 1)
|
||||||
|
->field('id, name, cid, icon')
|
||||||
|
->order('sort asc, id asc')
|
||||||
|
->select()
|
||||||
|
->toArray();
|
||||||
|
|
||||||
|
$categories = $this->buildParentChild($allCategories);
|
||||||
|
|
||||||
|
View::assign([
|
||||||
|
'categories' => $categories
|
||||||
|
]);
|
||||||
|
return View::fetch();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// 添加资源
|
||||||
|
public function add()
|
||||||
|
{
|
||||||
|
if (Request::isPost()) {
|
||||||
|
$data = [
|
||||||
|
'title' => input('post.title'),
|
||||||
|
'cate' => input('post.cate'),
|
||||||
|
'desc' => input('post.desc'),
|
||||||
|
'icon' => input('post.icon'),
|
||||||
|
'images' => input('post.images'),
|
||||||
|
'url' => input('post.url'),
|
||||||
|
'fileurl' => input('post.fileurl'),
|
||||||
|
'code' => input('post.code'),
|
||||||
|
'zipcode' => input('post.zipcode'),
|
||||||
|
'uploader' => input('post.uploader'),
|
||||||
|
'content' => input('post.content'),
|
||||||
|
'number' => input('post.number'),
|
||||||
|
'status' => input('post.status', 1),
|
||||||
|
'create_time' => time()
|
||||||
|
];
|
||||||
|
|
||||||
|
$insert = Resource::insert($data);
|
||||||
|
if (empty($insert)) {
|
||||||
|
Log::record('添加资源', 0, '添加资源失败', '资源管理');
|
||||||
|
$this->error('添加失败');
|
||||||
|
}
|
||||||
|
Log::record('添加资源', 1, '', '资源管理');
|
||||||
|
return json(['code' => 0, 'msg' => '添加成功', 'data' => []]);
|
||||||
|
} else {
|
||||||
|
// 获取所有分类
|
||||||
|
$allCategories = ResourceCategory::where('delete_time', null)
|
||||||
|
->where('status', 1)
|
||||||
|
->field('id, name, cid, icon, number')
|
||||||
|
->order('sort asc, id asc')
|
||||||
|
->select()
|
||||||
|
->toArray();
|
||||||
|
|
||||||
|
$categories = $this->buildParentChild($allCategories);
|
||||||
|
|
||||||
|
View::assign([
|
||||||
|
'categories' => $categories
|
||||||
|
]);
|
||||||
|
|
||||||
|
return View::fetch();
|
||||||
|
} catch (\Exception $e) {
|
||||||
|
Log::record('添加资源页面加载', 0, $e->getMessage(), '资源管理');
|
||||||
|
$this->error('页面加载失败:' . $e->getMessage());
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// 编辑资源
|
||||||
|
public function edit()
|
||||||
|
{
|
||||||
|
if (Request::isPost()) {
|
||||||
|
$data = input('post.');
|
||||||
|
$id = input('id/d', 0);
|
||||||
|
|
||||||
|
if (!$id) {
|
||||||
|
Log::record('编辑资源', 0, '参数错误', '资源管理');
|
||||||
|
return json(['code' => 1, 'msg' => '参数错误']);
|
||||||
|
}
|
||||||
|
|
||||||
|
$updateData = [
|
||||||
|
'title' => $data['title'],
|
||||||
|
'cate' => $data['cate'],
|
||||||
|
'desc' => $data['desc'],
|
||||||
|
'uploader' => $data['uploader'],
|
||||||
|
'icon' => $data['icon'],
|
||||||
|
'images' => $data['images'],
|
||||||
|
'fileurl' => $data['fileurl'],
|
||||||
|
'url' => $data['url'],
|
||||||
|
'code' => $data['code'],
|
||||||
|
'zipcode' => $data['zipcode'],
|
||||||
|
'sort' => $data['sort'],
|
||||||
|
'number' => $data['number'],
|
||||||
|
'content' => $data['content'],
|
||||||
|
'update_time' => time()
|
||||||
|
];
|
||||||
|
|
||||||
|
$result = Resource::where('id', $id)
|
||||||
|
->update($updateData);
|
||||||
|
|
||||||
|
if ($result !== false) {
|
||||||
|
Log::record('编辑资源', 1, '', '资源管理');
|
||||||
|
return json(['code' => 0, 'msg' => '编辑成功']);
|
||||||
|
} else {
|
||||||
|
Log::record('编辑资源', 0, '编辑资源失败', '资源管理');
|
||||||
|
return json(['code' => 1, 'msg' => '编辑失败']);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
$id = input('id/d', 0);
|
||||||
|
if (!$id) {
|
||||||
|
Log::record('编辑资源', 0, '参数错误', '资源管理');
|
||||||
|
$this->error('参数错误');
|
||||||
|
}
|
||||||
|
|
||||||
|
$resource = Resource::where('id', $id)
|
||||||
|
->where('delete_time', null)
|
||||||
|
->find();
|
||||||
|
|
||||||
|
if (!$resource) {
|
||||||
|
Log::record('编辑资源', 0, '资源不存在', '资源管理');
|
||||||
|
$this->error('资源不存在');
|
||||||
|
}
|
||||||
|
|
||||||
|
// 处理图片路径
|
||||||
|
if (!empty($resource['images'])) {
|
||||||
|
$domain = request()->domain();
|
||||||
|
$images = explode(',', $resource['images']);
|
||||||
|
// $images = array_map(function ($image) use ($domain) {
|
||||||
|
// return $domain . $image;
|
||||||
|
// }, $images);
|
||||||
|
$resource['images'] = implode(',', $images);
|
||||||
|
}
|
||||||
|
|
||||||
|
View::assign('resource', $resource);
|
||||||
|
return View::fetch();
|
||||||
|
}
|
||||||
|
|
||||||
|
// 删除资源
|
||||||
|
public function delete()
|
||||||
|
{
|
||||||
|
$id = input('post.id');
|
||||||
|
$delete = Resource::where('id', $id)
|
||||||
|
->update(['delete_time' => time()]);
|
||||||
|
if ($delete === false) {
|
||||||
|
Log::record('删除资源', 0, '删除资源失败', '资源管理');
|
||||||
|
return json(['code' => 1, 'msg' => '删除失败', 'data' => []]);
|
||||||
|
}
|
||||||
|
Log::record('删除资源', 1, '', '资源管理');
|
||||||
|
return json(['code' => 0, 'msg' => '删除成功', 'data' => []]);
|
||||||
|
}
|
||||||
|
|
||||||
|
// 资源分类
|
||||||
|
public function cate()
|
||||||
|
{
|
||||||
|
if (Request::isPost()) {
|
||||||
|
$lists = ResourceCategory::where('delete_time', null)
|
||||||
|
->where('status', 1)
|
||||||
|
->order('sort asc, id asc')
|
||||||
|
->select()
|
||||||
|
->toArray();
|
||||||
|
$tree = $this->buildTree($lists);
|
||||||
|
return json(['code' => 0, 'msg' => '获取成功', 'data' => $tree]);
|
||||||
|
}
|
||||||
|
return View::fetch();
|
||||||
|
}
|
||||||
|
|
||||||
|
//获取分类结构
|
||||||
|
public function getcate()
|
||||||
|
{
|
||||||
|
$lists = ResourceCategory::where('delete_time', null)
|
||||||
|
->where('status', 1)
|
||||||
|
->order('sort asc, id asc')
|
||||||
|
->select()
|
||||||
|
->toArray();
|
||||||
|
|
||||||
|
// 获取每个分类下的资源总数
|
||||||
|
foreach ($lists as &$item) {
|
||||||
|
if ($item['cid'] == 0) {
|
||||||
|
// 父级分类 - 统计所有子分类的资源总数
|
||||||
|
$childIds = ResourceCategory::where('cid', $item['id'])
|
||||||
|
->where('delete_time', null)
|
||||||
|
->where('status', 1)
|
||||||
|
->column('id');
|
||||||
|
|
||||||
|
$item['total'] = Resource::where('cate', 'in', array_merge([$item['id']], $childIds))
|
||||||
|
->where('delete_time', null)
|
||||||
|
->where('status', '<>', 3)
|
||||||
|
->count();
|
||||||
|
} else {
|
||||||
|
// 子分类 - 只统计当前分类的资源
|
||||||
|
$item['total'] = Resource::where('cate', $item['id'])
|
||||||
|
->where('delete_time', null)
|
||||||
|
->where('status', '<>', 3)
|
||||||
|
->count();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
$tree = $this->buildParentChild($lists);
|
||||||
|
return json(['code' => 0, 'msg' => '获取成功', 'data' => $tree]);
|
||||||
|
}
|
||||||
|
|
||||||
|
// 添加资源分类
|
||||||
|
public function cateadd()
|
||||||
|
{
|
||||||
|
if (Request::isPost()) {
|
||||||
|
$data = [
|
||||||
|
'name' => input('post.name'),
|
||||||
|
'icon' => input('post.icon'),
|
||||||
|
'cid' => input('post.cid'),
|
||||||
|
'number' => input('post.number'),
|
||||||
|
'sort' => input('post.sort', 0),
|
||||||
|
'status' => input('post.status', 1),
|
||||||
|
'create_time' => time()
|
||||||
|
];
|
||||||
|
|
||||||
|
$insert = ResourceCategory::insert($data);
|
||||||
|
if (empty($insert)) {
|
||||||
|
Log::record('添加资源分类', 0, '添加资源分类失败', '资源分类');
|
||||||
|
return json(['code' => 1, 'msg' => '添加失败', 'data' => []]);
|
||||||
|
}
|
||||||
|
Log::record('添加资源分类', 1, '', '资源分类');
|
||||||
|
return json(['code' => 0, 'msg' => '添加成功', 'data' => []]);
|
||||||
|
} else {
|
||||||
|
$parentCategories = ResourceCategory::where('delete_time', null)
|
||||||
|
->where('status', 1)
|
||||||
|
->where('cid', 0)
|
||||||
|
->field('id, name')
|
||||||
|
->select()
|
||||||
|
->toArray();
|
||||||
|
return json([
|
||||||
|
'code' => 0,
|
||||||
|
'msg' => '获取成功',
|
||||||
|
'data' => [
|
||||||
|
'parentOptions' => $parentCategories
|
||||||
|
]
|
||||||
|
]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
//编辑资源分类
|
||||||
|
public function cateedit()
|
||||||
|
{
|
||||||
|
if (Request::isPost()) {
|
||||||
|
$data = [
|
||||||
|
'id' => input('post.id'),
|
||||||
|
'name' => input('post.name'),
|
||||||
|
'icon' => input('post.icon'),
|
||||||
|
'cid' => input('post.cid'),
|
||||||
|
'number' => input('post.number'),
|
||||||
|
'sort' => input('post.sort', 0),
|
||||||
|
'status' => input('post.status', 1),
|
||||||
|
'update_time' => time()
|
||||||
|
];
|
||||||
|
|
||||||
|
$update = ResourceCategory::where('id', $data['id'])
|
||||||
|
->update($data);
|
||||||
|
if ($update === false) {
|
||||||
|
Log::record('编辑资源分类', 0, '更新资源分类失败', '资源分类');
|
||||||
|
return json(['code' => 1, 'msg' => '更新失败', 'data' => []]);
|
||||||
|
}
|
||||||
|
Log::record('编辑资源分类', 1, '', '资源分类');
|
||||||
|
return json(['code' => 0, 'msg' => '更新成功', 'data' => []]);
|
||||||
|
} else {
|
||||||
|
$id = input('get.id');
|
||||||
|
$info = ResourceCategory::where('id', $id)->find();
|
||||||
|
$parentCategories = ResourceCategory::where('delete_time', null)
|
||||||
|
->where('status', 1)
|
||||||
|
->where('cid', 0)
|
||||||
|
->where('id', '<>', $id)
|
||||||
|
->field('id, name')
|
||||||
|
->select()
|
||||||
|
->toArray();
|
||||||
|
|
||||||
|
return json([
|
||||||
|
'code' => 0,
|
||||||
|
'msg' => '获取成功',
|
||||||
|
'data' => [
|
||||||
|
'info' => $info,
|
||||||
|
'parentOptions' => $parentCategories
|
||||||
|
]
|
||||||
|
]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
//删除资源分类
|
||||||
|
public function catedel()
|
||||||
|
{
|
||||||
|
$id = input('post.id');
|
||||||
|
|
||||||
|
// 检查是否有子分类
|
||||||
|
$hasChildren = ResourceCategory::where('cid', $id)
|
||||||
|
->where('delete_time', null)
|
||||||
|
->find();
|
||||||
|
if ($hasChildren) {
|
||||||
|
Log::record('删除资源分类', 0, '该分类下有子分类,无法删除', '资源分类');
|
||||||
|
return json(['code' => 1, 'msg' => '该分类下有子分类,无法删除', 'data' => []]);
|
||||||
|
}
|
||||||
|
|
||||||
|
$delete = ResourceCategory::where('id', $id)
|
||||||
|
->update(['delete_time' => time()]);
|
||||||
|
if ($delete === false) {
|
||||||
|
Log::record('删除资源分类', 0, '删除资源分类失败', '资源分类');
|
||||||
|
return json(['code' => 1, 'msg' => '删除失败', 'data' => []]);
|
||||||
|
}
|
||||||
|
Log::record('删除资源分类', 1, '', '资源分类');
|
||||||
|
return json(['code' => 0, 'msg' => '删除成功', 'data' => []]);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 获取资源详情
|
||||||
|
*/
|
||||||
|
public function get()
|
||||||
|
{
|
||||||
|
$id = input('id/d', 0);
|
||||||
|
if (!$id) {
|
||||||
|
Log::record('获取资源详情', 0, '参数错误', '资源管理');
|
||||||
|
return json(['code' => 1, 'msg' => '参数错误']);
|
||||||
|
}
|
||||||
|
|
||||||
|
$resource = Resource::where('id', $id)
|
||||||
|
->where('delete_time', null)
|
||||||
|
->find();
|
||||||
|
|
||||||
|
if (!$resource) {
|
||||||
|
Log::record('获取资源详情', 0, '资源不存在', '资源管理');
|
||||||
|
return json(['code' => 1, 'msg' => '资源不存在']);
|
||||||
|
}
|
||||||
|
|
||||||
|
// 获取分类信息
|
||||||
|
$cateInfo = ResourceCategory::where('id', $resource['cate'])
|
||||||
|
->field('name')
|
||||||
|
->find();
|
||||||
|
|
||||||
|
if ($cateInfo) {
|
||||||
|
$resource['cate_name'] = $cateInfo['name'];
|
||||||
|
}
|
||||||
|
|
||||||
|
Log::record('获取资源详情', 1, '', '资源管理');
|
||||||
|
return json(['code' => 0, 'msg' => '获取成功', 'data' => $resource]);
|
||||||
|
}
|
||||||
|
|
||||||
|
//统计资源数量
|
||||||
|
public function counts()
|
||||||
|
{
|
||||||
|
try {
|
||||||
|
// 获取资源总数
|
||||||
|
$total = Resource::where('delete_time', null)
|
||||||
|
->where('status', '<>', 3)
|
||||||
|
->count();
|
||||||
|
|
||||||
|
// 获取今日新增资源数
|
||||||
|
$today = strtotime(date('Y-m-d'));
|
||||||
|
$todayNew = Resource::where('delete_time', null)
|
||||||
|
->where('status', '<>', 3)
|
||||||
|
->where('create_time', '>=', $today)
|
||||||
|
->count();
|
||||||
|
|
||||||
|
// 获取最近7天的资源数据
|
||||||
|
$dates = [];
|
||||||
|
$counts = [];
|
||||||
|
$totalCounts = []; // 存储每天的总资源数
|
||||||
|
|
||||||
|
for ($i = 6; $i >= 0; $i--) {
|
||||||
|
$date = date('Y-m-d', strtotime("-$i days"));
|
||||||
|
$start = strtotime($date);
|
||||||
|
$end = $start + 86400;
|
||||||
|
|
||||||
|
// 获取当天新增资源数
|
||||||
|
$count = Resource::where('delete_time', null)
|
||||||
|
->where('status', '<>', 3)
|
||||||
|
->where('create_time', '>=', $start)
|
||||||
|
->where('create_time', '<', $end)
|
||||||
|
->count();
|
||||||
|
|
||||||
|
// 获取截至当天的总资源数
|
||||||
|
$totalCount = Resource::where('delete_time', null)
|
||||||
|
->where('status', '<>', 3)
|
||||||
|
->where('create_time', '<', $end)
|
||||||
|
->count();
|
||||||
|
|
||||||
|
$dates[] = $date;
|
||||||
|
$counts[] = $count;
|
||||||
|
$totalCounts[] = $totalCount;
|
||||||
|
}
|
||||||
|
|
||||||
|
return json([
|
||||||
|
'code' => 0,
|
||||||
|
'msg' => '获取成功',
|
||||||
|
'data' => [
|
||||||
|
'total' => $total,
|
||||||
|
'todayNew' => $todayNew,
|
||||||
|
'dates' => $dates,
|
||||||
|
'counts' => $counts,
|
||||||
|
'totalCounts' => $totalCounts
|
||||||
|
]
|
||||||
|
]);
|
||||||
|
} catch (\Exception $e) {
|
||||||
|
return json([
|
||||||
|
'code' => 1,
|
||||||
|
'msg' => '获取失败:' . $e->getMessage()
|
||||||
|
]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// 构建树形结构
|
||||||
|
private function buildTree($lists)
|
||||||
|
{
|
||||||
|
$tree = [];
|
||||||
|
foreach ($lists as $item) {
|
||||||
|
if ($item['cid'] == 0) {
|
||||||
|
$node = [
|
||||||
|
'id' => $item['id'],
|
||||||
|
'title' => $item['name'],
|
||||||
|
'children' => []
|
||||||
|
];
|
||||||
|
|
||||||
|
// 查找子分类
|
||||||
|
foreach ($lists as $subItem) {
|
||||||
|
if ($subItem['cid'] == $item['id']) {
|
||||||
|
$node['children'][] = [
|
||||||
|
'id' => $subItem['id'],
|
||||||
|
'title' => $subItem['name'],
|
||||||
|
'children' => []
|
||||||
|
];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
$tree[] = $node;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return $tree;
|
||||||
|
}
|
||||||
|
|
||||||
|
// 构建父子结构
|
||||||
|
private function buildParentChild($lists)
|
||||||
|
{
|
||||||
|
$tree = [];
|
||||||
|
foreach ($lists as $item) {
|
||||||
|
if ($item['cid'] == 0) {
|
||||||
|
// 顶级分类
|
||||||
|
$tree[] = $item;
|
||||||
|
} else {
|
||||||
|
// 子分类
|
||||||
|
foreach ($tree as &$parent) {
|
||||||
|
if ($parent['id'] == $item['cid']) {
|
||||||
|
if (!isset($parent['children'])) {
|
||||||
|
$parent['children'] = [];
|
||||||
|
}
|
||||||
|
$parent['children'][] = $item;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return $tree;
|
||||||
|
}
|
||||||
|
}
|
||||||
98
app/admin/controller/UsersController.php
Normal file
98
app/admin/controller/UsersController.php
Normal file
@ -0,0 +1,98 @@
|
|||||||
|
<?php
|
||||||
|
/**
|
||||||
|
* 商业使用授权协议
|
||||||
|
*
|
||||||
|
* Copyright (c) 2025 [云泽网]. 保留所有权利.
|
||||||
|
*
|
||||||
|
* 本软件仅供评估使用。任何商业用途必须获得书面授权许可。
|
||||||
|
* 未经授权商业使用本软件属于侵权行为,将承担法律责任。
|
||||||
|
*
|
||||||
|
* 授权购买请联系: 357099073@qq.com
|
||||||
|
* 官方网站: https://www.yunzer.cn
|
||||||
|
*
|
||||||
|
* 评估用户须知:
|
||||||
|
* 1. 禁止移除版权声明
|
||||||
|
* 2. 禁止用于生产环境
|
||||||
|
* 3. 禁止转售或分发
|
||||||
|
*/
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 后台管理系统-用户管理
|
||||||
|
*/
|
||||||
|
namespace app\admin\controller;
|
||||||
|
use app\admin\controller\BaseController;
|
||||||
|
use think\facade\View;
|
||||||
|
use think\facade\Request;
|
||||||
|
use think\facade\Db;
|
||||||
|
use app\admin\controller\Log;
|
||||||
|
use think\App;
|
||||||
|
use app\admin\model\User\Users;
|
||||||
|
|
||||||
|
|
||||||
|
class UsersController extends BaseController
|
||||||
|
{
|
||||||
|
/**
|
||||||
|
* 统计用户数量
|
||||||
|
*/
|
||||||
|
public function counts()
|
||||||
|
{
|
||||||
|
try {
|
||||||
|
// 统计用户总数
|
||||||
|
$total = Users::where('delete_time', 0)
|
||||||
|
->where('status', 1)
|
||||||
|
->count();
|
||||||
|
|
||||||
|
// 获取今日新增用户数
|
||||||
|
$today = strtotime(date('Y-m-d'));
|
||||||
|
$todayNew = Users::where('delete_time', 0)
|
||||||
|
->where('status', 1)
|
||||||
|
->where('create_time', '>=', $today)
|
||||||
|
->count();
|
||||||
|
|
||||||
|
// 获取最近7天的用户数据
|
||||||
|
$dates = [];
|
||||||
|
$counts = [];
|
||||||
|
$totalCounts = []; // 存储每天的总用户数
|
||||||
|
|
||||||
|
for ($i = 6; $i >= 0; $i--) {
|
||||||
|
$date = date('Y-m-d', strtotime("-$i days"));
|
||||||
|
$start = strtotime($date);
|
||||||
|
$end = $start + 86400;
|
||||||
|
|
||||||
|
// 获取当天新增用户数
|
||||||
|
$count = Users::where('delete_time', 0)
|
||||||
|
->where('status', 1)
|
||||||
|
->where('create_time', '>=', $start)
|
||||||
|
->where('create_time', '<', $end)
|
||||||
|
->count();
|
||||||
|
|
||||||
|
// 获取截至当天的总用户数
|
||||||
|
$totalCount = Users::where('delete_time', 0)
|
||||||
|
->where('status', 1)
|
||||||
|
->where('create_time', '<', $end)
|
||||||
|
->count();
|
||||||
|
|
||||||
|
$dates[] = $date;
|
||||||
|
$counts[] = $count;
|
||||||
|
$totalCounts[] = $totalCount;
|
||||||
|
}
|
||||||
|
|
||||||
|
return json([
|
||||||
|
'code' => 0,
|
||||||
|
'msg' => '获取成功',
|
||||||
|
'data' => [
|
||||||
|
'total' => $total,
|
||||||
|
'todayNew' => $todayNew,
|
||||||
|
'dates' => $dates,
|
||||||
|
'counts' => $counts,
|
||||||
|
'totalCounts' => $totalCounts
|
||||||
|
]
|
||||||
|
]);
|
||||||
|
} catch (\Exception $e) {
|
||||||
|
return json([
|
||||||
|
'code' => 1,
|
||||||
|
'msg' => '获取失败:' . $e->getMessage()
|
||||||
|
]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
556
app/admin/controller/YunzerController.php
Normal file
556
app/admin/controller/YunzerController.php
Normal file
@ -0,0 +1,556 @@
|
|||||||
|
<?php
|
||||||
|
/**
|
||||||
|
* 商业使用授权协议
|
||||||
|
*
|
||||||
|
* Copyright (c) 2025 [云泽网]. 保留所有权利.
|
||||||
|
*
|
||||||
|
* 本软件仅供评估使用。任何商业用途必须获得书面授权许可。
|
||||||
|
* 未经授权商业使用本软件属于侵权行为,将承担法律责任。
|
||||||
|
*
|
||||||
|
* 授权购买请联系: 357099073@qq.com
|
||||||
|
* 官方网站: https://www.yunzer.cn
|
||||||
|
*
|
||||||
|
* 评估用户须知:
|
||||||
|
* 1. 禁止移除版权声明
|
||||||
|
* 2. 禁止用于生产环境
|
||||||
|
* 3. 禁止转售或分发
|
||||||
|
*/
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 后台管理系统-核心功能
|
||||||
|
*/
|
||||||
|
namespace app\admin\controller;
|
||||||
|
use app\admin\controller\Base;
|
||||||
|
use think\facade\Db;
|
||||||
|
use think\facade\View;
|
||||||
|
use think\facade\Request;
|
||||||
|
use app\admin\model\YzAdminConfig;
|
||||||
|
use app\admin\controller\LogController as Log;
|
||||||
|
use app\admin\model\AdminSysMenu;
|
||||||
|
use app\admin\model\AdminUserGroup;
|
||||||
|
use app\admin\model\AdminConfig;
|
||||||
|
use app\admin\model\ZIconfont;
|
||||||
|
use app\admin\model\MailConfig;
|
||||||
|
use PHPMailer\PHPMailer\PHPMailer;
|
||||||
|
use PHPMailer\PHPMailer\Exception;
|
||||||
|
|
||||||
|
class YunzerController extends Base
|
||||||
|
{
|
||||||
|
# 菜单列表
|
||||||
|
public function menuinfo()
|
||||||
|
{
|
||||||
|
$lists = AdminSysMenu::where('parent_id', 0)->order('sort DESC,smid DESC')->select();
|
||||||
|
View::assign([
|
||||||
|
'lists' => $lists
|
||||||
|
]);
|
||||||
|
return View::fetch();
|
||||||
|
}
|
||||||
|
# 菜单添加
|
||||||
|
public function menuadd()
|
||||||
|
{
|
||||||
|
$req = request();
|
||||||
|
if ($req->isPost()) {
|
||||||
|
$data['label'] = trim(input('post.label'));
|
||||||
|
if (empty($data['label'])) {
|
||||||
|
Log::record('添加菜单', 0, '请输入菜单名称', '菜单管理');
|
||||||
|
$this->returnCode(1, '请输入菜单名称');
|
||||||
|
}
|
||||||
|
$data['icon_class'] = trim(input('post.icon_class'));
|
||||||
|
$data['sort'] = (int) trim(input('post.sort'));
|
||||||
|
$data['status'] = (int) trim(input('post.status'));
|
||||||
|
$data['type'] = (int) trim(input('post.type', 0));
|
||||||
|
if ($data['type'] == 1) {
|
||||||
|
$data['src'] = trim(input('post.src1'));
|
||||||
|
if (empty($data['src'])) {
|
||||||
|
Log::record('添加菜单', 0, '请输入内部跳转地址', '菜单管理');
|
||||||
|
$this->returnCode(1, '请输入内部跳转地址');
|
||||||
|
}
|
||||||
|
} else if ($data['type'] == 2) {
|
||||||
|
$data['src'] = trim(input('post.src2'));
|
||||||
|
if (empty($data['src'])) {
|
||||||
|
Log::record('添加菜单', 0, '请输入超链接地址', '菜单管理');
|
||||||
|
$this->returnCode(1, '请输入超链接地址');
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
$data['src'] = '';
|
||||||
|
}
|
||||||
|
// 保存菜单
|
||||||
|
$res = AdminSysMenu::insert($data);
|
||||||
|
if (!$res) {
|
||||||
|
Log::record('添加菜单', 0, '添加菜单失败', '菜单管理');
|
||||||
|
$this->returnCode(1, '添加菜单失败');
|
||||||
|
}
|
||||||
|
|
||||||
|
// 获取新插入的菜单ID
|
||||||
|
$newMenuId = AdminSysMenu::getLastInsID();
|
||||||
|
|
||||||
|
// 获取所有管理员用户组
|
||||||
|
$adminGroups = AdminUserGroup::select();
|
||||||
|
foreach ($adminGroups as $group) {
|
||||||
|
// 获取现有权限
|
||||||
|
$rights = [];
|
||||||
|
if (!empty($group['rights'])) {
|
||||||
|
$rights = json_decode($group['rights'], true);
|
||||||
|
}
|
||||||
|
|
||||||
|
// 添加新菜单ID到权限列表
|
||||||
|
if (!in_array($newMenuId, $rights)) {
|
||||||
|
$rights[] = $newMenuId;
|
||||||
|
|
||||||
|
// 更新用户组权限
|
||||||
|
AdminUserGroup::where('group_id', $group['group_id'])
|
||||||
|
->update(['rights' => json_encode($rights)]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
Log::record('添加菜单', 1, '', '菜单管理');
|
||||||
|
$this->returnCode(0);
|
||||||
|
} else {
|
||||||
|
$iconfont = ZIconfont::where('status', 1)->select();
|
||||||
|
View::assign([
|
||||||
|
'iconfont' => $iconfont
|
||||||
|
]);
|
||||||
|
return View::fetch();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
# 菜单修改
|
||||||
|
public function menuedit()
|
||||||
|
{
|
||||||
|
$req = request();
|
||||||
|
if ($req->isPost()) {
|
||||||
|
$smid = (int) input('post.smid');
|
||||||
|
$data['label'] = trim(input('post.label'));
|
||||||
|
if (!$data['label']) {
|
||||||
|
Log::record('编辑菜单', 0, '请输入菜单名称', '菜单管理');
|
||||||
|
$this->returnCode(1, '请输入菜单名称');
|
||||||
|
}
|
||||||
|
$data['icon_class'] = trim(input('post.icon_class'));
|
||||||
|
$data['sort'] = (int) trim(input('post.sort'));
|
||||||
|
$data['status'] = (int) trim(input('post.status'));
|
||||||
|
$data['type'] = (int) trim(input('post.type', 0));
|
||||||
|
if ($data['type'] == 1) {
|
||||||
|
$data['src'] = trim(input('post.src1'));
|
||||||
|
if (empty($data['src'])) {
|
||||||
|
Log::record('编辑菜单', 0, '请输入内部跳转地址', '菜单管理');
|
||||||
|
$this->returnCode(1, '请输入内部跳转地址');
|
||||||
|
}
|
||||||
|
} else if ($data['type'] == 2) {
|
||||||
|
$data['src'] = trim(input('post.src2'));
|
||||||
|
if (empty($data['src'])) {
|
||||||
|
Log::record('编辑菜单', 0, '请输入超链接地址', '菜单管理');
|
||||||
|
$this->returnCode(1, '请输入超链接地址');
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
$data['src'] = '';
|
||||||
|
}
|
||||||
|
// 保存用户
|
||||||
|
$res = AdminSysMenu::where('smid', $smid)->update($data);
|
||||||
|
if (!$res) {
|
||||||
|
Log::record('编辑菜单', 0, '修改菜单失败', '菜单管理');
|
||||||
|
$this->returnCode(1, '修改菜单失败');
|
||||||
|
}
|
||||||
|
Log::record('编辑菜单', 1, '', '菜单管理');
|
||||||
|
$this->returnCode(0);
|
||||||
|
} else {
|
||||||
|
$smid = (int) input('get.smid');
|
||||||
|
$lists = AdminSysMenu::where('smid', $smid)->find();
|
||||||
|
$iconfont = ZIconfont::where('status', 1)->select();
|
||||||
|
View::assign([
|
||||||
|
'lists' => $lists,
|
||||||
|
'iconfont' => $iconfont
|
||||||
|
]);
|
||||||
|
return View::fetch();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
# 菜单删除
|
||||||
|
public function menudel()
|
||||||
|
{
|
||||||
|
$smid = (int) input('post.smid');
|
||||||
|
$count = AdminSysMenu::where('parent_id', $smid)->count();
|
||||||
|
if ($count > 0) {
|
||||||
|
Log::record('删除菜单', 0, '该菜单下还有子菜单,不能删除', '菜单管理');
|
||||||
|
$this->returnCode(1, '该菜单下还有子菜单,不能删除');
|
||||||
|
}
|
||||||
|
$res = AdminSysMenu::where('smid', $smid)->delete();
|
||||||
|
if (empty($res)) {
|
||||||
|
Log::record('删除菜单', 0, '删除菜单失败', '菜单管理');
|
||||||
|
$this->returnCode(1, '删除菜单失败');
|
||||||
|
}
|
||||||
|
Log::record('删除菜单', 1, '', '菜单管理');
|
||||||
|
$this->returnCode(0);
|
||||||
|
}
|
||||||
|
# 按钮管理
|
||||||
|
public function buttoninfo()
|
||||||
|
{
|
||||||
|
$smid = (int) input('get.smid');
|
||||||
|
$lists = AdminSysMenu::where('parent_id', $smid)->order('sort DESC')->select()->toArray();
|
||||||
|
if (!empty($lists)) {
|
||||||
|
foreach ($lists as &$v) {
|
||||||
|
switch ($v['type']) {
|
||||||
|
case 0:
|
||||||
|
$v['type_name'] = '顶级菜单';
|
||||||
|
break;
|
||||||
|
case 1:
|
||||||
|
$v['type_name'] = '内部跳转';
|
||||||
|
break;
|
||||||
|
case 2:
|
||||||
|
$v['type_name'] = '超链接';
|
||||||
|
break;
|
||||||
|
default:
|
||||||
|
$v['type_name'] = '未规划类型';
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
View::assign([
|
||||||
|
'lists' => $lists,
|
||||||
|
'smid' => $smid
|
||||||
|
]);
|
||||||
|
return View::fetch();
|
||||||
|
}
|
||||||
|
# 按钮添加
|
||||||
|
public function buttonadd()
|
||||||
|
{
|
||||||
|
$req = request();
|
||||||
|
if ($req->isPost()) {
|
||||||
|
$smid = (int) input('post.smid');
|
||||||
|
$data['label'] = trim(input('post.label'));
|
||||||
|
if (!$data['label']) {
|
||||||
|
Log::record('添加按钮', 0, '请输入按钮名称', '按钮管理');
|
||||||
|
$this->returnCode(1, '请输入按钮名称');
|
||||||
|
}
|
||||||
|
$data['icon_class'] = trim(input('post.icon_class'));
|
||||||
|
$data['sort'] = (int) trim(input('post.sort'));
|
||||||
|
$data['status'] = (int) trim(input('post.status'));
|
||||||
|
$data['type'] = (int) trim(input('post.type'));
|
||||||
|
if (empty($data['type'])) {
|
||||||
|
Log::record('添加按钮', 0, '请选择按钮类型', '按钮管理');
|
||||||
|
$this->returnCode(1, '请选择按钮类型');
|
||||||
|
}
|
||||||
|
if ($data['type'] == 1) {
|
||||||
|
$data['src'] = trim(input('post.src1'));
|
||||||
|
if (empty($data['src'])) {
|
||||||
|
Log::record('添加按钮', 0, '请输入内部跳转地址', '按钮管理');
|
||||||
|
$this->returnCode(1, '请输入内部跳转地址');
|
||||||
|
}
|
||||||
|
} else if ($data['type'] == 2) {
|
||||||
|
$data['src'] = trim(input('post.src2'));
|
||||||
|
if (empty($data['src'])) {
|
||||||
|
Log::record('添加按钮', 0, '请输入超链接地址', '按钮管理');
|
||||||
|
$this->returnCode(1, '请输入超链接地址');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
$data['parent_id'] = $smid;
|
||||||
|
$res = AdminSysMenu::insert($data);
|
||||||
|
if (!$res) {
|
||||||
|
Log::record('添加按钮', 0, '添加按钮失败', '按钮管理');
|
||||||
|
$this->returnCode(1, '添加按钮失败');
|
||||||
|
}
|
||||||
|
Log::record('添加按钮', 1, '', '按钮管理');
|
||||||
|
$this->returnCode(0);
|
||||||
|
} else {
|
||||||
|
$smid = (int) input('get.smid');
|
||||||
|
$iconfont = ZIconfont::where('status', 1)->select();
|
||||||
|
View::assign([
|
||||||
|
'smid' => $smid,
|
||||||
|
'iconfont' => $iconfont
|
||||||
|
]);
|
||||||
|
return View::fetch();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
# 按钮修改
|
||||||
|
public function buttonedit()
|
||||||
|
{
|
||||||
|
$req = request();
|
||||||
|
if ($req->isPost()) {
|
||||||
|
$smid = (int) input('post.smid');
|
||||||
|
$data['label'] = trim(input('post.label'));
|
||||||
|
if (!$data['label']) {
|
||||||
|
Log::record('编辑按钮', 0, '请输入按钮名称', '按钮管理');
|
||||||
|
$this->returnCode(1, '请输入按钮名称');
|
||||||
|
}
|
||||||
|
$data['icon_class'] = trim(input('post.icon_class'));
|
||||||
|
$data['sort'] = (int) trim(input('post.sort'));
|
||||||
|
$data['status'] = (int) trim(input('post.status'));
|
||||||
|
$data['type'] = (int) trim(input('post.type'));
|
||||||
|
if (empty($data['type'])) {
|
||||||
|
Log::record('编辑按钮', 0, '请选择按钮类型', '按钮管理');
|
||||||
|
$this->returnCode(1, '请选择按钮类型');
|
||||||
|
}
|
||||||
|
if ($data['type'] == 1) {
|
||||||
|
$data['src'] = trim(input('post.src1'));
|
||||||
|
if (empty($data['src'])) {
|
||||||
|
Log::record('编辑按钮', 0, '请输入内部跳转地址', '按钮管理');
|
||||||
|
$this->returnCode(1, '请输入内部跳转地址');
|
||||||
|
}
|
||||||
|
} else if ($data['type'] == 2) {
|
||||||
|
$data['src'] = trim(input('post.src2'));
|
||||||
|
if (empty($data['src'])) {
|
||||||
|
Log::record('编辑按钮', 0, '请输入超链接地址', '按钮管理');
|
||||||
|
$this->returnCode(1, '请输入超链接地址');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
$res = AdminSysMenu::where('smid', $smid)->update($data);
|
||||||
|
if (!$res) {
|
||||||
|
Log::record('编辑按钮', 0, '修改按钮失败', '按钮管理');
|
||||||
|
$this->returnCode(1, '修改按钮失败');
|
||||||
|
}
|
||||||
|
Log::record('编辑按钮', 1, '', '按钮管理');
|
||||||
|
$this->returnCode(0);
|
||||||
|
} else {
|
||||||
|
$smid = (int) input('get.smid');
|
||||||
|
$lists = AdminSysMenu::where('smid', $smid)->find();
|
||||||
|
$iconfont = ZIconfont::where('status', 1)->select();
|
||||||
|
View::assign([
|
||||||
|
'lists' => $lists,
|
||||||
|
'iconfont' => $iconfont
|
||||||
|
]);
|
||||||
|
return View::fetch();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
# 按钮删除
|
||||||
|
public function buttondel()
|
||||||
|
{
|
||||||
|
$smid = (int) input('post.smid');
|
||||||
|
$res = AdminSysMenu::where('smid', $smid)->delete();
|
||||||
|
if (empty($res)) {
|
||||||
|
Log::record('删除按钮', 0, '删除按钮失败', '按钮管理');
|
||||||
|
$this->returnCode(1, '删除按钮失败');
|
||||||
|
}
|
||||||
|
Log::record('删除按钮', 1, '', '按钮管理');
|
||||||
|
$this->returnCode(0);
|
||||||
|
}
|
||||||
|
# 配置列表
|
||||||
|
public function configlist()
|
||||||
|
{
|
||||||
|
$req = request();
|
||||||
|
if ($req->isPost()) {
|
||||||
|
$page = (int) input('post.page', 1);
|
||||||
|
$limit = (int) input('post.limit', $this->config['admin_page']);
|
||||||
|
$count = AdminConfig::count();
|
||||||
|
$lists = AdminConfig::page($page, $limit)->order('config_sort DESC,config_id DESC')->select();
|
||||||
|
$this->returnCode(0, $lists, $count);
|
||||||
|
} else {
|
||||||
|
return View::fetch();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
# 配置添加
|
||||||
|
public function configadd()
|
||||||
|
{
|
||||||
|
$req = request();
|
||||||
|
if ($req->isPost()) {
|
||||||
|
$data['config_name'] = trim(input('post.config_name'));
|
||||||
|
if (empty($data['config_name'])) {
|
||||||
|
Log::record('添加配置', 0, '请输入关键词', '系统配置');
|
||||||
|
$this->returnCode(1, '请输入关键词');
|
||||||
|
}
|
||||||
|
$data['config_info'] = trim(input('post.config_info'));
|
||||||
|
if (empty($data['config_info'])) {
|
||||||
|
Log::record('添加配置', 0, '请输入作用', '系统配置');
|
||||||
|
$this->returnCode(1, '请输入作用');
|
||||||
|
}
|
||||||
|
$data['config_type'] = trim(input('post.config_type'));
|
||||||
|
$data['config_desc'] = trim(input('post.config_desc'));
|
||||||
|
$data['config_status'] = trim(input('post.config_status'));
|
||||||
|
$data['config_sort'] = trim(input('post.config_sort'));
|
||||||
|
$res = AdminConfig::insert($data);
|
||||||
|
if (empty($res)) {
|
||||||
|
Log::record('添加配置', 0, '添加配置失败', '系统配置');
|
||||||
|
$this->returnCode(1, '添加配置失败');
|
||||||
|
}
|
||||||
|
Log::record('添加配置', 1, '', '系统配置');
|
||||||
|
$this->returnCode(0);
|
||||||
|
} else {
|
||||||
|
return View::fetch();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
# 配置修改
|
||||||
|
public function configedit()
|
||||||
|
{
|
||||||
|
$req = request();
|
||||||
|
if ($req->isPost()) {
|
||||||
|
$config_id = (int) input('post.config_id');
|
||||||
|
if (empty($config_id)) {
|
||||||
|
Log::record('编辑配置', 0, '请选择一条数据', '系统配置');
|
||||||
|
$this->returnCode(1, '请选择一条数据');
|
||||||
|
}
|
||||||
|
$data['config_name'] = trim(input('post.config_name'));
|
||||||
|
if (empty($data['config_name'])) {
|
||||||
|
Log::record('编辑配置', 0, '请输入关键词', '系统配置');
|
||||||
|
$this->returnCode(1, '请输入关键词');
|
||||||
|
}
|
||||||
|
$data['config_info'] = trim(input('post.config_info'));
|
||||||
|
if (empty($data['config_info'])) {
|
||||||
|
Log::record('编辑配置', 0, '请输入作用', '系统配置');
|
||||||
|
$this->returnCode(1, '请输入作用');
|
||||||
|
}
|
||||||
|
$data['config_type'] = trim(input('post.config_type'));
|
||||||
|
$data['config_desc'] = trim(input('post.config_desc'));
|
||||||
|
$data['config_status'] = trim(input('post.config_status'));
|
||||||
|
$data['config_sort'] = trim(input('post.config_sort'));
|
||||||
|
$res = AdminConfig::where('config_id', $config_id)->update($data);
|
||||||
|
if (empty($res)) {
|
||||||
|
Log::record('编辑配置', 0, '修改配置失败', '系统配置');
|
||||||
|
$this->returnCode(1, '修改配置失败');
|
||||||
|
}
|
||||||
|
Log::record('编辑配置', 1, '', '系统配置');
|
||||||
|
$this->returnCode(0);
|
||||||
|
} else {
|
||||||
|
$config_id = (int) input('get.config_id');
|
||||||
|
$find = AdminConfig::where('config_id', $config_id)->find();
|
||||||
|
View::assign([
|
||||||
|
'find' => $find
|
||||||
|
]);
|
||||||
|
return View::fetch();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
# 配置删除
|
||||||
|
public function configdel()
|
||||||
|
{
|
||||||
|
$config_id = (int) input('post.config_id');
|
||||||
|
if (empty($config_id)) {
|
||||||
|
Log::record('删除配置', 0, '请选择一条数据', '系统配置');
|
||||||
|
$this->returnCode(1, '请选择一条数据');
|
||||||
|
}
|
||||||
|
$res = AdminConfig::where('config_id', $config_id)->delete();
|
||||||
|
if (empty($res)) {
|
||||||
|
Log::record('删除配置', 0, '删除配置失败', '系统配置');
|
||||||
|
$this->returnCode(1, '删除配置失败');
|
||||||
|
}
|
||||||
|
Log::record('删除配置', 1, '', '系统配置');
|
||||||
|
$this->returnCode(0);
|
||||||
|
}
|
||||||
|
# 配置值
|
||||||
|
public function configvalue()
|
||||||
|
{
|
||||||
|
$req = request();
|
||||||
|
if ($req->isPost()) {
|
||||||
|
$post = input('post.');
|
||||||
|
if (empty($post)) {
|
||||||
|
Log::record('更新配置值', 0, '数据不能为空', '系统配置');
|
||||||
|
$this->returnCode(1, '数据不能为空');
|
||||||
|
}
|
||||||
|
|
||||||
|
// 获取所有配置项,检查 config_type 为 4 的项
|
||||||
|
$allConfigs = AdminConfig::order('config_sort DESC,config_id')->select();
|
||||||
|
foreach ($allConfigs as $config) {
|
||||||
|
if ($config['config_type'] == 4) {
|
||||||
|
$checkboxName = $config['config_name'] . '_checkbox';
|
||||||
|
$configName = $config['config_name'];
|
||||||
|
if (!isset($post[$checkboxName])) {
|
||||||
|
// 复选框未选中,手动添加对应的值为 0
|
||||||
|
$post[$configName] = 0;
|
||||||
|
} else {
|
||||||
|
// 复选框选中,确保值为 1
|
||||||
|
$post[$configName] = 1;
|
||||||
|
}
|
||||||
|
// 移除 checkbox 字段
|
||||||
|
unset($post[$checkboxName]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
$oConfig = new YzAdminConfig();
|
||||||
|
$updateAll = $oConfig->updateAll($post);
|
||||||
|
if (empty($updateAll)) {
|
||||||
|
Log::record('更新配置值', 0, '更新配置值失败', '系统配置');
|
||||||
|
$this->returnCode(1, '更新配置值失败');
|
||||||
|
}
|
||||||
|
Log::record('更新配置值', 1, '', '系统配置');
|
||||||
|
$this->returnCode(0);
|
||||||
|
} else {
|
||||||
|
$lists = AdminConfig::order('config_sort DESC,config_id')->select();
|
||||||
|
View::assign([
|
||||||
|
'lists' => $lists
|
||||||
|
]);
|
||||||
|
return View::fetch();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
// 邮件配置
|
||||||
|
public function mailconfig()
|
||||||
|
{
|
||||||
|
if (Request::isPost()) {
|
||||||
|
$data = [
|
||||||
|
'smtp_host' => input('post.smtp_host'),
|
||||||
|
'smtp_port' => input('post.smtp_port'),
|
||||||
|
'smtp_email' => input('post.smtp_email'),
|
||||||
|
'smtp_password' => input('post.smtp_password'),
|
||||||
|
'smtp_name' => input('post.smtp_name')
|
||||||
|
];
|
||||||
|
|
||||||
|
// 验证必填字段
|
||||||
|
if (empty($data['smtp_host']) || empty($data['smtp_port']) || empty($data['smtp_email']) || empty($data['smtp_password'])) {
|
||||||
|
// Log::record('修改邮件配置', 0, '必填字段不能为空', '邮件配置');
|
||||||
|
return json(['code' => 1, 'msg' => '必填字段不能为空']);
|
||||||
|
}
|
||||||
|
|
||||||
|
$res = MailConfig::where('id', 1)->update($data);
|
||||||
|
if ($res === false) {
|
||||||
|
// Log::record('修改邮件配置', 0, '更新邮件配置失败', '邮件配置');
|
||||||
|
return json(['code' => 1, 'msg' => '更新邮件配置失败']);
|
||||||
|
}
|
||||||
|
// Log::record('修改邮件配置', 1, '', '邮件配置');
|
||||||
|
return json(['code' => 0, 'msg' => '更新成功']);
|
||||||
|
}
|
||||||
|
|
||||||
|
$config = MailConfig::where('id', 1)->find();
|
||||||
|
View::assign([
|
||||||
|
'config' => $config
|
||||||
|
]);
|
||||||
|
return View::fetch();
|
||||||
|
}
|
||||||
|
|
||||||
|
// 获取邮件配置
|
||||||
|
public function getMailConfig()
|
||||||
|
{
|
||||||
|
$config = MailConfig::where('id', 1)->find();
|
||||||
|
if ($config) {
|
||||||
|
return json(['code' => 0, 'msg' => '获取成功', 'data' => $config]);
|
||||||
|
}
|
||||||
|
return json(['code' => 1, 'msg' => '获取配置失败']);
|
||||||
|
}
|
||||||
|
|
||||||
|
// 测试邮件配置
|
||||||
|
public function testMailConfig()
|
||||||
|
{
|
||||||
|
if (!Request::isPost()) {
|
||||||
|
return json(['code' => 1, 'msg' => '请求方法无效']);
|
||||||
|
}
|
||||||
|
|
||||||
|
$email = input('post.email');
|
||||||
|
$config = input('post.config');
|
||||||
|
|
||||||
|
if (empty($email) || empty($config)) {
|
||||||
|
return json(['code' => 1, 'msg' => '参数错误']);
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
// 配置邮件服务器
|
||||||
|
$mail = new PHPMailer(true);
|
||||||
|
$mail->isSMTP();
|
||||||
|
$mail->Host = $config['smtp_host'];
|
||||||
|
$mail->Port = $config['smtp_port'];
|
||||||
|
$mail->SMTPAuth = true;
|
||||||
|
$mail->Username = $config['smtp_email'];
|
||||||
|
$mail->Password = $config['smtp_password'];
|
||||||
|
$mail->SMTPSecure = 'ssl';
|
||||||
|
$mail->CharSet = 'UTF-8';
|
||||||
|
|
||||||
|
// 设置发件人
|
||||||
|
$mail->setFrom($config['smtp_email'], $config['smtp_name']);
|
||||||
|
$mail->addAddress($email);
|
||||||
|
|
||||||
|
// 设置邮件内容
|
||||||
|
$mail->isHTML(true);
|
||||||
|
$mail->Subject = '邮件配置测试';
|
||||||
|
$mail->Body = '这是一封测试邮件,如果您收到这封邮件,说明邮件配置正确。';
|
||||||
|
|
||||||
|
$mail->send();
|
||||||
|
return json(['code' => 0, 'msg' => '发送成功']);
|
||||||
|
} catch (\Exception $e) {
|
||||||
|
//Log::record('测试邮件配置', 0, $e->getMessage(), '邮件配置');
|
||||||
|
return json(['code' => 1, 'msg' => '发送失败:' . $e->getMessage()]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
1209
app/admin/controller/YunzeradminController.php
Normal file
1209
app/admin/controller/YunzeradminController.php
Normal file
File diff suppressed because it is too large
Load Diff
184
app/admin/controller/Yunzertest.php
Normal file
184
app/admin/controller/Yunzertest.php
Normal file
@ -0,0 +1,184 @@
|
|||||||
|
<?php
|
||||||
|
/**
|
||||||
|
* 商业使用授权协议
|
||||||
|
*
|
||||||
|
* Copyright (c) 2025 [云泽网]. 保留所有权利.
|
||||||
|
*
|
||||||
|
* 本软件仅供评估使用。任何商业用途必须获得书面授权许可。
|
||||||
|
* 未经授权商业使用本软件属于侵权行为,将承担法律责任。
|
||||||
|
*
|
||||||
|
* 授权购买请联系: 357099073@qq.com
|
||||||
|
* 官方网站: https://www.yunzer.cn
|
||||||
|
*
|
||||||
|
* 评估用户须知:
|
||||||
|
* 1. 禁止移除版权声明
|
||||||
|
* 2. 禁止用于生产环境
|
||||||
|
* 3. 禁止转售或分发
|
||||||
|
*/
|
||||||
|
|
||||||
|
namespace app\admin\controller;
|
||||||
|
use app\admin\controller\Base;
|
||||||
|
use think\facade\Db;
|
||||||
|
use think\facade\View;
|
||||||
|
use think\facade\Request;
|
||||||
|
|
||||||
|
class Yunzertest extends Base{
|
||||||
|
public function icon_list(){
|
||||||
|
$lists = Db::table('yz_z_iconfont')->where('status','=',1)->select();
|
||||||
|
View::assign([
|
||||||
|
'lists' => $lists
|
||||||
|
]);
|
||||||
|
return View::fetch();
|
||||||
|
}
|
||||||
|
public function test_list(){
|
||||||
|
if(Request::isPost()){
|
||||||
|
$count = Db::table('yz_z_test')->count();
|
||||||
|
$page = (int)input('post.page',1);
|
||||||
|
$limit = (int)input('post.limit',10);
|
||||||
|
$lists = Db::table('yz_z_test')->order('test_id DESC')->page($page,$limit)->select()->each(function($item, $key){
|
||||||
|
if($item['test_reference'] == 1){
|
||||||
|
$item['test_reference'] = '开启';
|
||||||
|
}else{
|
||||||
|
$item['test_reference'] = '关闭';
|
||||||
|
}
|
||||||
|
$item['test_time'] = date('Y-m-d H:i:s',$item['test_time']);
|
||||||
|
return $item;
|
||||||
|
});
|
||||||
|
$this->returnCode(0,$lists,$count);
|
||||||
|
}else{
|
||||||
|
return View::fetch();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
public function test_add(){
|
||||||
|
if(Request::isPost()){
|
||||||
|
$data['test_input'] = input('post.test_input');
|
||||||
|
$data['test_reference'] = input('post.test_reference');
|
||||||
|
$data['test_time'] = input('post.test_time');
|
||||||
|
if(!empty($data['test_time'])){
|
||||||
|
$data['test_time'] = strtotime($data['test_time']);
|
||||||
|
}
|
||||||
|
$data['test_data'] = input('post.test_data');
|
||||||
|
$data['test_datatime'] = input('post.test_datatime');
|
||||||
|
$data['test_img'] = input('post.test_img');
|
||||||
|
$data['test_rich_baidu'] = input('post.test_rich_baidu');
|
||||||
|
$data['test_url'] = input('post.test_url');
|
||||||
|
|
||||||
|
$insert = Db::table('yz_z_test')->insert($data);
|
||||||
|
if(empty($insert)){
|
||||||
|
$this->returnCode('91000001');
|
||||||
|
}
|
||||||
|
$this->returnCode(0);
|
||||||
|
}else{
|
||||||
|
return View::fetch();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
public function test_edit(){
|
||||||
|
if(Request::isPost()){
|
||||||
|
$test_id = input('post.test_id');
|
||||||
|
$data['test_input'] = input('post.test_input');
|
||||||
|
$data['test_reference'] = input('post.test_reference');
|
||||||
|
$data['test_time'] = input('post.test_time');
|
||||||
|
if(!empty($data['test_time'])){
|
||||||
|
$data['test_time'] = strtotime($data['test_time']);
|
||||||
|
}
|
||||||
|
$data['test_data'] = input('post.test_data');
|
||||||
|
$data['test_datatime'] = input('post.test_datatime');
|
||||||
|
$data['test_img'] = input('post.test_img');
|
||||||
|
$data['test_rich'] = input('post.test_rich');
|
||||||
|
$data['test_rich_baidu'] = input('post.test_rich_baidu');
|
||||||
|
$data['test_url'] = input('post.test_url');
|
||||||
|
|
||||||
|
$update = Db::table('yz_z_test')->where('test_id',$test_id)->update($data);
|
||||||
|
if(empty($update)){
|
||||||
|
$this->returnCode('91000002');
|
||||||
|
}
|
||||||
|
$this->returnCode(0);
|
||||||
|
}else{
|
||||||
|
$test_id = input('get.test_id');
|
||||||
|
$find = Db::table('yz_z_test')->where('test_id',$test_id)->find();
|
||||||
|
if(!empty($find)){
|
||||||
|
$find['test_time'] = date('Y-m-d H:i:s',$find['test_time']);
|
||||||
|
}
|
||||||
|
View::assign([
|
||||||
|
'find' => $find
|
||||||
|
]);
|
||||||
|
return View::fetch();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
public function test_del(){
|
||||||
|
$test_id = (int)input('post.test_id');
|
||||||
|
$res = Db::table('yz_z_test')->where('test_id',$test_id)->delete();
|
||||||
|
if(empty($res)){
|
||||||
|
$this->returnCode('91000003');
|
||||||
|
}
|
||||||
|
$this->returnCode(0);
|
||||||
|
}
|
||||||
|
public function test_static_list(){
|
||||||
|
$lists = Db::table('yz_z_test')
|
||||||
|
->order('test_id DESC')
|
||||||
|
->paginate();
|
||||||
|
View::assign([
|
||||||
|
'lists' => $lists
|
||||||
|
]);
|
||||||
|
return View::fetch();
|
||||||
|
}
|
||||||
|
public function test_static_add(){
|
||||||
|
if(Request::isPost()){
|
||||||
|
$data['test_input'] = input('post.test_input');
|
||||||
|
$data['test_reference'] = input('post.test_reference');
|
||||||
|
$data['test_time'] = input('post.test_time');
|
||||||
|
if(!empty($data['test_time'])){
|
||||||
|
$data['test_time'] = strtotime($data['test_time']);
|
||||||
|
}
|
||||||
|
$data['test_data'] = input('post.test_data');
|
||||||
|
$data['test_datatime'] = input('post.test_datatime');
|
||||||
|
$data['test_img'] = input('post.test_img');
|
||||||
|
$data['test_rich_baidu'] = input('post.test_rich_baidu');
|
||||||
|
$data['test_url'] = input('post.test_url');
|
||||||
|
|
||||||
|
$insert = Db::table('yz_z_test')->insert($data);
|
||||||
|
if(empty($insert)){
|
||||||
|
$this->returnCode('91000001');
|
||||||
|
}
|
||||||
|
$this->returnCode(0);
|
||||||
|
}else{
|
||||||
|
return View::fetch();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
public function test_static_edit(){
|
||||||
|
if(Request::isPost()){
|
||||||
|
$test_id = input('post.test_id');
|
||||||
|
$data['test_input'] = input('post.test_input');
|
||||||
|
$data['test_reference'] = input('post.test_reference');
|
||||||
|
$data['test_time'] = input('post.test_time');
|
||||||
|
if(!empty($data['test_time'])){
|
||||||
|
$data['test_time'] = strtotime($data['test_time']);
|
||||||
|
}
|
||||||
|
$data['test_data'] = input('post.test_data');
|
||||||
|
$data['test_datatime'] = input('post.test_datatime');
|
||||||
|
$data['test_img'] = input('post.test_img');
|
||||||
|
$data['test_rich'] = input('post.test_rich');
|
||||||
|
$data['test_rich_baidu'] = input('post.test_rich_baidu');
|
||||||
|
$data['test_url'] = input('post.test_url');
|
||||||
|
|
||||||
|
$update = Db::table('yz_z_test')->where('test_id',$test_id)->update($data);
|
||||||
|
if(empty($update)){
|
||||||
|
$this->returnCode('91000002');
|
||||||
|
}
|
||||||
|
$this->returnCode(0);
|
||||||
|
}else{
|
||||||
|
$test_id = input('get.test_id');
|
||||||
|
$find = Db::table('yz_z_test')->where('test_id',$test_id)->find();
|
||||||
|
if(!empty($find)){
|
||||||
|
$find['test_time'] = date('Y-m-d H:i:s',$find['test_time']);
|
||||||
|
if(!empty($find['test_img'])){
|
||||||
|
$find['test_img_s'] = explode(';',$find['test_img']);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
View::assign([
|
||||||
|
'find' => $find
|
||||||
|
]);
|
||||||
|
return View::fetch();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
25
app/admin/model/AdminConfig.php
Normal file
25
app/admin/model/AdminConfig.php
Normal file
@ -0,0 +1,25 @@
|
|||||||
|
<?php
|
||||||
|
/**
|
||||||
|
* 商业使用授权协议
|
||||||
|
*
|
||||||
|
* Copyright (c) 2025 [云泽网]. 保留所有权利.
|
||||||
|
*
|
||||||
|
* 本软件仅供评估使用。任何商业用途必须获得书面授权许可。
|
||||||
|
* 未经授权商业使用本软件属于侵权行为,将承担法律责任。
|
||||||
|
*
|
||||||
|
* 授权购买请联系: 357099073@qq.com
|
||||||
|
* 官方网站: https://www.yunzer.cn
|
||||||
|
*
|
||||||
|
* 评估用户须知:
|
||||||
|
* 1. 禁止移除版权声明
|
||||||
|
* 2. 禁止用于生产环境
|
||||||
|
* 3. 禁止转售或分发
|
||||||
|
*/
|
||||||
|
|
||||||
|
namespace app\admin\model;
|
||||||
|
|
||||||
|
use think\Model;
|
||||||
|
|
||||||
|
class AdminConfig extends Model
|
||||||
|
{
|
||||||
|
}
|
||||||
57
app/admin/model/AdminSysMenu.php
Normal file
57
app/admin/model/AdminSysMenu.php
Normal file
@ -0,0 +1,57 @@
|
|||||||
|
<?php
|
||||||
|
/**
|
||||||
|
* 商业使用授权协议
|
||||||
|
*
|
||||||
|
* Copyright (c) 2025 [云泽网]. 保留所有权利.
|
||||||
|
*
|
||||||
|
* 本软件仅供评估使用。任何商业用途必须获得书面授权许可。
|
||||||
|
* 未经授权商业使用本软件属于侵权行为,将承担法律责任。
|
||||||
|
*
|
||||||
|
* 授权购买请联系: 357099073@qq.com
|
||||||
|
* 官方网站: https://www.yunzer.cn
|
||||||
|
*
|
||||||
|
* 评估用户须知:
|
||||||
|
* 1. 禁止移除版权声明
|
||||||
|
* 2. 禁止用于生产环境
|
||||||
|
* 3. 禁止转售或分发
|
||||||
|
*/
|
||||||
|
|
||||||
|
namespace app\admin\model;
|
||||||
|
|
||||||
|
use think\Model;
|
||||||
|
|
||||||
|
class AdminSysMenu extends Model
|
||||||
|
{
|
||||||
|
/**
|
||||||
|
* 获取菜单树形结构
|
||||||
|
* @return array
|
||||||
|
*/
|
||||||
|
public static function getMenuTree()
|
||||||
|
{
|
||||||
|
// 获取所有启用的菜单
|
||||||
|
$menus = self::where('status', 1)
|
||||||
|
->order('type', 'asc')
|
||||||
|
->order('sort', 'desc')
|
||||||
|
->select()
|
||||||
|
->toArray();
|
||||||
|
|
||||||
|
$menuTree = [];
|
||||||
|
|
||||||
|
// 先处理所有父菜单
|
||||||
|
foreach ($menus as $menu) {
|
||||||
|
if ($menu['parent_id'] == 0) {
|
||||||
|
$menuTree[$menu['smid']] = $menu;
|
||||||
|
$menuTree[$menu['smid']]['children'] = [];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// 再处理子菜单
|
||||||
|
foreach ($menus as $menu) {
|
||||||
|
if ($menu['parent_id'] != 0 && isset($menuTree[$menu['parent_id']])) {
|
||||||
|
$menuTree[$menu['parent_id']]['children'][] = $menu;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return array_values($menuTree);
|
||||||
|
}
|
||||||
|
}
|
||||||
25
app/admin/model/AdminUser.php
Normal file
25
app/admin/model/AdminUser.php
Normal file
@ -0,0 +1,25 @@
|
|||||||
|
<?php
|
||||||
|
/**
|
||||||
|
* 商业使用授权协议
|
||||||
|
*
|
||||||
|
* Copyright (c) 2025 [云泽网]. 保留所有权利.
|
||||||
|
*
|
||||||
|
* 本软件仅供评估使用。任何商业用途必须获得书面授权许可。
|
||||||
|
* 未经授权商业使用本软件属于侵权行为,将承担法律责任。
|
||||||
|
*
|
||||||
|
* 授权购买请联系: 357099073@qq.com
|
||||||
|
* 官方网站: https://www.yunzer.cn
|
||||||
|
*
|
||||||
|
* 评估用户须知:
|
||||||
|
* 1. 禁止移除版权声明
|
||||||
|
* 2. 禁止用于生产环境
|
||||||
|
* 3. 禁止转售或分发
|
||||||
|
*/
|
||||||
|
|
||||||
|
namespace app\admin\model;
|
||||||
|
|
||||||
|
use think\Model;
|
||||||
|
|
||||||
|
class AdminUser extends Model
|
||||||
|
{
|
||||||
|
}
|
||||||
25
app/admin/model/AdminUserGroup.php
Normal file
25
app/admin/model/AdminUserGroup.php
Normal file
@ -0,0 +1,25 @@
|
|||||||
|
<?php
|
||||||
|
/**
|
||||||
|
* 商业使用授权协议
|
||||||
|
*
|
||||||
|
* Copyright (c) 2025 [云泽网]. 保留所有权利.
|
||||||
|
*
|
||||||
|
* 本软件仅供评估使用。任何商业用途必须获得书面授权许可。
|
||||||
|
* 未经授权商业使用本软件属于侵权行为,将承担法律责任。
|
||||||
|
*
|
||||||
|
* 授权购买请联系: 357099073@qq.com
|
||||||
|
* 官方网站: https://www.yunzer.cn
|
||||||
|
*
|
||||||
|
* 评估用户须知:
|
||||||
|
* 1. 禁止移除版权声明
|
||||||
|
* 2. 禁止用于生产环境
|
||||||
|
* 3. 禁止转售或分发
|
||||||
|
*/
|
||||||
|
|
||||||
|
namespace app\admin\model;
|
||||||
|
|
||||||
|
use think\Model;
|
||||||
|
|
||||||
|
class AdminUserGroup extends Model
|
||||||
|
{
|
||||||
|
}
|
||||||
25
app/admin/model/ApiKey.php
Normal file
25
app/admin/model/ApiKey.php
Normal file
@ -0,0 +1,25 @@
|
|||||||
|
<?php
|
||||||
|
/**
|
||||||
|
* 商业使用授权协议
|
||||||
|
*
|
||||||
|
* Copyright (c) 2025 [云泽网]. 保留所有权利.
|
||||||
|
*
|
||||||
|
* 本软件仅供评估使用。任何商业用途必须获得书面授权许可。
|
||||||
|
* 未经授权商业使用本软件属于侵权行为,将承担法律责任。
|
||||||
|
*
|
||||||
|
* 授权购买请联系: 357099073@qq.com
|
||||||
|
* 官方网站: https://www.yunzer.cn
|
||||||
|
*
|
||||||
|
* 评估用户须知:
|
||||||
|
* 1. 禁止移除版权声明
|
||||||
|
* 2. 禁止用于生产环境
|
||||||
|
* 3. 禁止转售或分发
|
||||||
|
*/
|
||||||
|
|
||||||
|
namespace app\admin\model;
|
||||||
|
|
||||||
|
use think\Model;
|
||||||
|
|
||||||
|
class ApiKey extends Model
|
||||||
|
{
|
||||||
|
}
|
||||||
25
app/admin/model/Article/Articles.php
Normal file
25
app/admin/model/Article/Articles.php
Normal file
@ -0,0 +1,25 @@
|
|||||||
|
<?php
|
||||||
|
/**
|
||||||
|
* 商业使用授权协议
|
||||||
|
*
|
||||||
|
* Copyright (c) 2025 [云泽网]. 保留所有权利.
|
||||||
|
*
|
||||||
|
* 本软件仅供评估使用。任何商业用途必须获得书面授权许可。
|
||||||
|
* 未经授权商业使用本软件属于侵权行为,将承担法律责任。
|
||||||
|
*
|
||||||
|
* 授权购买请联系: 357099073@qq.com
|
||||||
|
* 官方网站: https://www.yunzer.cn
|
||||||
|
*
|
||||||
|
* 评估用户须知:
|
||||||
|
* 1. 禁止移除版权声明
|
||||||
|
* 2. 禁止用于生产环境
|
||||||
|
* 3. 禁止转售或分发
|
||||||
|
*/
|
||||||
|
|
||||||
|
namespace app\admin\model\Article;
|
||||||
|
|
||||||
|
use think\Model;
|
||||||
|
|
||||||
|
class Articles extends Model
|
||||||
|
{
|
||||||
|
}
|
||||||
25
app/admin/model/Article/ArticlesCategory.php
Normal file
25
app/admin/model/Article/ArticlesCategory.php
Normal file
@ -0,0 +1,25 @@
|
|||||||
|
<?php
|
||||||
|
/**
|
||||||
|
* 商业使用授权协议
|
||||||
|
*
|
||||||
|
* Copyright (c) 2025 [云泽网]. 保留所有权利.
|
||||||
|
*
|
||||||
|
* 本软件仅供评估使用。任何商业用途必须获得书面授权许可。
|
||||||
|
* 未经授权商业使用本软件属于侵权行为,将承担法律责任。
|
||||||
|
*
|
||||||
|
* 授权购买请联系: 357099073@qq.com
|
||||||
|
* 官方网站: https://www.yunzer.cn
|
||||||
|
*
|
||||||
|
* 评估用户须知:
|
||||||
|
* 1. 禁止移除版权声明
|
||||||
|
* 2. 禁止用于生产环境
|
||||||
|
* 3. 禁止转售或分发
|
||||||
|
*/
|
||||||
|
|
||||||
|
namespace app\admin\model\Article;
|
||||||
|
|
||||||
|
use think\Model;
|
||||||
|
|
||||||
|
class ArticlesCategory extends Model
|
||||||
|
{
|
||||||
|
}
|
||||||
25
app/admin/model/Banner.php
Normal file
25
app/admin/model/Banner.php
Normal file
@ -0,0 +1,25 @@
|
|||||||
|
<?php
|
||||||
|
/**
|
||||||
|
* 商业使用授权协议
|
||||||
|
*
|
||||||
|
* Copyright (c) 2025 [云泽网]. 保留所有权利.
|
||||||
|
*
|
||||||
|
* 本软件仅供评估使用。任何商业用途必须获得书面授权许可。
|
||||||
|
* 未经授权商业使用本软件属于侵权行为,将承担法律责任。
|
||||||
|
*
|
||||||
|
* 授权购买请联系: 357099073@qq.com
|
||||||
|
* 官方网站: https://www.yunzer.cn
|
||||||
|
*
|
||||||
|
* 评估用户须知:
|
||||||
|
* 1. 禁止移除版权声明
|
||||||
|
* 2. 禁止用于生产环境
|
||||||
|
* 3. 禁止转售或分发
|
||||||
|
*/
|
||||||
|
|
||||||
|
namespace app\admin\model;
|
||||||
|
|
||||||
|
use think\Model;
|
||||||
|
|
||||||
|
class Banner extends Model
|
||||||
|
{
|
||||||
|
}
|
||||||
54
app/admin/model/Base.php
Normal file
54
app/admin/model/Base.php
Normal file
@ -0,0 +1,54 @@
|
|||||||
|
<?php
|
||||||
|
/**
|
||||||
|
* 商业使用授权协议
|
||||||
|
*
|
||||||
|
* Copyright (c) 2025 [云泽网]. 保留所有权利.
|
||||||
|
*
|
||||||
|
* 本软件仅供评估使用。任何商业用途必须获得书面授权许可。
|
||||||
|
* 未经授权商业使用本软件属于侵权行为,将承担法律责任。
|
||||||
|
*
|
||||||
|
* 授权购买请联系: 357099073@qq.com
|
||||||
|
* 官方网站: https://www.yunzer.cn
|
||||||
|
*
|
||||||
|
* 评估用户须知:
|
||||||
|
* 1. 禁止移除版权声明
|
||||||
|
* 2. 禁止用于生产环境
|
||||||
|
* 3. 禁止转售或分发
|
||||||
|
*/
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 后台管理系统
|
||||||
|
*/
|
||||||
|
namespace app\admin\model;
|
||||||
|
use think\Model;
|
||||||
|
use think\facade\App;
|
||||||
|
|
||||||
|
class Base extends Model{
|
||||||
|
public function logs($data=null,$fileName=''){
|
||||||
|
if(is_null($data) || is_null($fileName)){
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
//获取Runtime路径
|
||||||
|
$path = App::getRuntimePath() . 'logs' . DIRECTORY_SEPARATOR;
|
||||||
|
if(!is_dir($path)){
|
||||||
|
$mkdir_re = mkdir($path,0777,TRUE);
|
||||||
|
if(!$mkdir_re){
|
||||||
|
$this -> logs($data,$fileName);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
$info = ['data'=>$data];
|
||||||
|
$content = json_encode($info, JSON_UNESCAPED_UNICODE | JSON_UNESCAPED_SLASHES) . PHP_EOL;
|
||||||
|
if(empty($fileName)){
|
||||||
|
$filePath = $path . "/" . date("Ymd",time()).'.info.log';
|
||||||
|
}else{
|
||||||
|
$filePath = $path . "/" . $fileName . '.info.log';
|
||||||
|
}
|
||||||
|
$time = "[".date("Y-m-d H:i:s",time())."]";
|
||||||
|
$re = file_put_contents($filePath, $time." ".$content , FILE_APPEND);
|
||||||
|
if(!$re){
|
||||||
|
$this -> logs($data,$fileName);
|
||||||
|
}else{
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
34
app/admin/model/ContentPush/ContentPush.php
Normal file
34
app/admin/model/ContentPush/ContentPush.php
Normal file
@ -0,0 +1,34 @@
|
|||||||
|
<?php
|
||||||
|
/**
|
||||||
|
* 商业使用授权协议
|
||||||
|
*
|
||||||
|
* Copyright (c) 2025 [云泽网]. 保留所有权利.
|
||||||
|
*
|
||||||
|
* 本软件仅供评估使用。任何商业用途必须获得书面授权许可。
|
||||||
|
* 未经授权商业使用本软件属于侵权行为,将承担法律责任。
|
||||||
|
*
|
||||||
|
* 授权购买请联系: 357099073@qq.com
|
||||||
|
* 官方网站: https://www.yunzer.cn
|
||||||
|
*
|
||||||
|
* 评估用户须知:
|
||||||
|
* 1. 禁止移除版权声明
|
||||||
|
* 2. 禁止用于生产环境
|
||||||
|
* 3. 禁止转售或分发
|
||||||
|
*/
|
||||||
|
|
||||||
|
namespace app\admin\model\ContentPush;
|
||||||
|
|
||||||
|
use think\Model;
|
||||||
|
|
||||||
|
class ContentPush extends Model
|
||||||
|
{
|
||||||
|
protected $name = 'content_push';
|
||||||
|
|
||||||
|
// 自动写入时间戳
|
||||||
|
protected $autoWriteTimestamp = true;
|
||||||
|
|
||||||
|
// 定义时间戳字段名
|
||||||
|
protected $createTime = 'create_time';
|
||||||
|
protected $updateTime = 'update_time';
|
||||||
|
protected $deleteTime = 'delete_time';
|
||||||
|
}
|
||||||
34
app/admin/model/ContentPush/ContentPushSetting.php
Normal file
34
app/admin/model/ContentPush/ContentPushSetting.php
Normal file
@ -0,0 +1,34 @@
|
|||||||
|
<?php
|
||||||
|
/**
|
||||||
|
* 商业使用授权协议
|
||||||
|
*
|
||||||
|
* Copyright (c) 2025 [云泽网]. 保留所有权利.
|
||||||
|
*
|
||||||
|
* 本软件仅供评估使用。任何商业用途必须获得书面授权许可。
|
||||||
|
* 未经授权商业使用本软件属于侵权行为,将承担法律责任。
|
||||||
|
*
|
||||||
|
* 授权购买请联系: 357099073@qq.com
|
||||||
|
* 官方网站: https://www.yunzer.cn
|
||||||
|
*
|
||||||
|
* 评估用户须知:
|
||||||
|
* 1. 禁止移除版权声明
|
||||||
|
* 2. 禁止用于生产环境
|
||||||
|
* 3. 禁止转售或分发
|
||||||
|
*/
|
||||||
|
|
||||||
|
namespace app\admin\model\ContentPush;
|
||||||
|
|
||||||
|
use think\Model;
|
||||||
|
|
||||||
|
class ContentPushSetting extends Model
|
||||||
|
{
|
||||||
|
protected $name = 'content_push_setting';
|
||||||
|
|
||||||
|
// 自动写入时间戳
|
||||||
|
protected $autoWriteTimestamp = true;
|
||||||
|
|
||||||
|
// 定义时间戳字段名
|
||||||
|
protected $createTime = 'create_time';
|
||||||
|
protected $updateTime = 'update_time';
|
||||||
|
protected $deleteTime = 'delete_time';
|
||||||
|
}
|
||||||
25
app/admin/model/DailyStats.php
Normal file
25
app/admin/model/DailyStats.php
Normal file
@ -0,0 +1,25 @@
|
|||||||
|
<?php
|
||||||
|
/**
|
||||||
|
* 商业使用授权协议
|
||||||
|
*
|
||||||
|
* Copyright (c) 2025 [云泽网]. 保留所有权利.
|
||||||
|
*
|
||||||
|
* 本软件仅供评估使用。任何商业用途必须获得书面授权许可。
|
||||||
|
* 未经授权商业使用本软件属于侵权行为,将承担法律责任。
|
||||||
|
*
|
||||||
|
* 授权购买请联系: 357099073@qq.com
|
||||||
|
* 官方网站: https://www.yunzer.cn
|
||||||
|
*
|
||||||
|
* 评估用户须知:
|
||||||
|
* 1. 禁止移除版权声明
|
||||||
|
* 2. 禁止用于生产环境
|
||||||
|
* 3. 禁止转售或分发
|
||||||
|
*/
|
||||||
|
|
||||||
|
namespace app\admin\model;
|
||||||
|
|
||||||
|
use think\Model;
|
||||||
|
|
||||||
|
class DailyStats extends Model
|
||||||
|
{
|
||||||
|
}
|
||||||
25
app/admin/model/Log/LogsLogin.php
Normal file
25
app/admin/model/Log/LogsLogin.php
Normal file
@ -0,0 +1,25 @@
|
|||||||
|
<?php
|
||||||
|
/**
|
||||||
|
* 商业使用授权协议
|
||||||
|
*
|
||||||
|
* Copyright (c) 2025 [云泽网]. 保留所有权利.
|
||||||
|
*
|
||||||
|
* 本软件仅供评估使用。任何商业用途必须获得书面授权许可。
|
||||||
|
* 未经授权商业使用本软件属于侵权行为,将承担法律责任。
|
||||||
|
*
|
||||||
|
* 授权购买请联系: 357099073@qq.com
|
||||||
|
* 官方网站: https://www.yunzer.cn
|
||||||
|
*
|
||||||
|
* 评估用户须知:
|
||||||
|
* 1. 禁止移除版权声明
|
||||||
|
* 2. 禁止用于生产环境
|
||||||
|
* 3. 禁止转售或分发
|
||||||
|
*/
|
||||||
|
|
||||||
|
namespace app\admin\model\Log;
|
||||||
|
|
||||||
|
use think\Model;
|
||||||
|
|
||||||
|
class LogsLogin extends Model
|
||||||
|
{
|
||||||
|
}
|
||||||
25
app/admin/model/Log/LogsOperation.php
Normal file
25
app/admin/model/Log/LogsOperation.php
Normal file
@ -0,0 +1,25 @@
|
|||||||
|
<?php
|
||||||
|
/**
|
||||||
|
* 商业使用授权协议
|
||||||
|
*
|
||||||
|
* Copyright (c) 2025 [云泽网]. 保留所有权利.
|
||||||
|
*
|
||||||
|
* 本软件仅供评估使用。任何商业用途必须获得书面授权许可。
|
||||||
|
* 未经授权商业使用本软件属于侵权行为,将承担法律责任。
|
||||||
|
*
|
||||||
|
* 授权购买请联系: 357099073@qq.com
|
||||||
|
* 官方网站: https://www.yunzer.cn
|
||||||
|
*
|
||||||
|
* 评估用户须知:
|
||||||
|
* 1. 禁止移除版权声明
|
||||||
|
* 2. 禁止用于生产环境
|
||||||
|
* 3. 禁止转售或分发
|
||||||
|
*/
|
||||||
|
|
||||||
|
namespace app\admin\model\Log;
|
||||||
|
|
||||||
|
use think\Model;
|
||||||
|
|
||||||
|
class LogsOperation extends Model
|
||||||
|
{
|
||||||
|
}
|
||||||
25
app/admin/model/MailConfig.php
Normal file
25
app/admin/model/MailConfig.php
Normal file
@ -0,0 +1,25 @@
|
|||||||
|
<?php
|
||||||
|
/**
|
||||||
|
* 商业使用授权协议
|
||||||
|
*
|
||||||
|
* Copyright (c) 2025 [云泽网]. 保留所有权利.
|
||||||
|
*
|
||||||
|
* 本软件仅供评估使用。任何商业用途必须获得书面授权许可。
|
||||||
|
* 未经授权商业使用本软件属于侵权行为,将承担法律责任。
|
||||||
|
*
|
||||||
|
* 授权购买请联系: 357099073@qq.com
|
||||||
|
* 官方网站: https://www.yunzer.cn
|
||||||
|
*
|
||||||
|
* 评估用户须知:
|
||||||
|
* 1. 禁止移除版权声明
|
||||||
|
* 2. 禁止用于生产环境
|
||||||
|
* 3. 禁止转售或分发
|
||||||
|
*/
|
||||||
|
|
||||||
|
namespace app\admin\model;
|
||||||
|
|
||||||
|
use think\Model;
|
||||||
|
|
||||||
|
class MailConfig extends Model
|
||||||
|
{
|
||||||
|
}
|
||||||
30
app/admin/model/Resource/Resource.php
Normal file
30
app/admin/model/Resource/Resource.php
Normal file
@ -0,0 +1,30 @@
|
|||||||
|
<?php
|
||||||
|
/**
|
||||||
|
* 商业使用授权协议
|
||||||
|
*
|
||||||
|
* Copyright (c) 2025 [云泽网]. 保留所有权利.
|
||||||
|
*
|
||||||
|
* 本软件仅供评估使用。任何商业用途必须获得书面授权许可。
|
||||||
|
* 未经授权商业使用本软件属于侵权行为,将承担法律责任。
|
||||||
|
*
|
||||||
|
* 授权购买请联系: 357099073@qq.com
|
||||||
|
* 官方网站: https://www.yunzer.cn
|
||||||
|
*
|
||||||
|
* 评估用户须知:
|
||||||
|
* 1. 禁止移除版权声明
|
||||||
|
* 2. 禁止用于生产环境
|
||||||
|
* 3. 禁止转售或分发
|
||||||
|
*/
|
||||||
|
|
||||||
|
namespace app\admin\model\Resource;
|
||||||
|
|
||||||
|
use think\Model;
|
||||||
|
|
||||||
|
class Resource extends Model
|
||||||
|
{
|
||||||
|
// 设置当前模型对应的数据表名称(不含前缀)
|
||||||
|
protected $name = 'resources';
|
||||||
|
|
||||||
|
// 设置主键
|
||||||
|
protected $pk = 'id';
|
||||||
|
}
|
||||||
30
app/admin/model/Resource/ResourceCategory.php
Normal file
30
app/admin/model/Resource/ResourceCategory.php
Normal file
@ -0,0 +1,30 @@
|
|||||||
|
<?php
|
||||||
|
/**
|
||||||
|
* 商业使用授权协议
|
||||||
|
*
|
||||||
|
* Copyright (c) 2025 [云泽网]. 保留所有权利.
|
||||||
|
*
|
||||||
|
* 本软件仅供评估使用。任何商业用途必须获得书面授权许可。
|
||||||
|
* 未经授权商业使用本软件属于侵权行为,将承担法律责任。
|
||||||
|
*
|
||||||
|
* 授权购买请联系: 357099073@qq.com
|
||||||
|
* 官方网站: https://www.yunzer.cn
|
||||||
|
*
|
||||||
|
* 评估用户须知:
|
||||||
|
* 1. 禁止移除版权声明
|
||||||
|
* 2. 禁止用于生产环境
|
||||||
|
* 3. 禁止转售或分发
|
||||||
|
*/
|
||||||
|
|
||||||
|
namespace app\admin\model\Resource;
|
||||||
|
|
||||||
|
use think\Model;
|
||||||
|
|
||||||
|
class ResourceCategory extends Model
|
||||||
|
{
|
||||||
|
// 设置当前模型对应的数据表名称(不含前缀)
|
||||||
|
protected $name = 'resources_category';
|
||||||
|
|
||||||
|
// 设置主键
|
||||||
|
protected $pk = 'id';
|
||||||
|
}
|
||||||
25
app/admin/model/User/Users.php
Normal file
25
app/admin/model/User/Users.php
Normal file
@ -0,0 +1,25 @@
|
|||||||
|
<?php
|
||||||
|
/**
|
||||||
|
* 商业使用授权协议
|
||||||
|
*
|
||||||
|
* Copyright (c) 2025 [云泽网]. 保留所有权利.
|
||||||
|
*
|
||||||
|
* 本软件仅供评估使用。任何商业用途必须获得书面授权许可。
|
||||||
|
* 未经授权商业使用本软件属于侵权行为,将承担法律责任。
|
||||||
|
*
|
||||||
|
* 授权购买请联系: 357099073@qq.com
|
||||||
|
* 官方网站: https://www.yunzer.cn
|
||||||
|
*
|
||||||
|
* 评估用户须知:
|
||||||
|
* 1. 禁止移除版权声明
|
||||||
|
* 2. 禁止用于生产环境
|
||||||
|
* 3. 禁止转售或分发
|
||||||
|
*/
|
||||||
|
|
||||||
|
namespace app\admin\model\User;
|
||||||
|
|
||||||
|
use think\Model;
|
||||||
|
|
||||||
|
class Users extends Model
|
||||||
|
{
|
||||||
|
}
|
||||||
25
app/admin/model/User/UsersGroup.php
Normal file
25
app/admin/model/User/UsersGroup.php
Normal file
@ -0,0 +1,25 @@
|
|||||||
|
<?php
|
||||||
|
/**
|
||||||
|
* 商业使用授权协议
|
||||||
|
*
|
||||||
|
* Copyright (c) 2025 [云泽网]. 保留所有权利.
|
||||||
|
*
|
||||||
|
* 本软件仅供评估使用。任何商业用途必须获得书面授权许可。
|
||||||
|
* 未经授权商业使用本软件属于侵权行为,将承担法律责任。
|
||||||
|
*
|
||||||
|
* 授权购买请联系: 357099073@qq.com
|
||||||
|
* 官方网站: https://www.yunzer.cn
|
||||||
|
*
|
||||||
|
* 评估用户须知:
|
||||||
|
* 1. 禁止移除版权声明
|
||||||
|
* 2. 禁止用于生产环境
|
||||||
|
* 3. 禁止转售或分发
|
||||||
|
*/
|
||||||
|
|
||||||
|
namespace app\admin\model\User;
|
||||||
|
|
||||||
|
use think\Model;
|
||||||
|
|
||||||
|
class UsersGroup extends Model
|
||||||
|
{
|
||||||
|
}
|
||||||
68
app/admin/model/YzAdminConfig.php
Normal file
68
app/admin/model/YzAdminConfig.php
Normal file
@ -0,0 +1,68 @@
|
|||||||
|
<?php
|
||||||
|
/**
|
||||||
|
* 商业使用授权协议
|
||||||
|
*
|
||||||
|
* Copyright (c) 2025 [云泽网]. 保留所有权利.
|
||||||
|
*
|
||||||
|
* 本软件仅供评估使用。任何商业用途必须获得书面授权许可。
|
||||||
|
* 未经授权商业使用本软件属于侵权行为,将承担法律责任。
|
||||||
|
*
|
||||||
|
* 授权购买请联系: 357099073@qq.com
|
||||||
|
* 官方网站: https://www.yunzer.cn
|
||||||
|
*
|
||||||
|
* 评估用户须知:
|
||||||
|
* 1. 禁止移除版权声明
|
||||||
|
* 2. 禁止用于生产环境
|
||||||
|
* 3. 禁止转售或分发
|
||||||
|
*/
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 配置表
|
||||||
|
*/
|
||||||
|
namespace app\admin\model;
|
||||||
|
|
||||||
|
use think\Model;
|
||||||
|
|
||||||
|
class YzAdminConfig extends Model
|
||||||
|
{
|
||||||
|
// 设置当前模型对应的数据表名称(不含前缀)
|
||||||
|
protected $name = 'admin_config';
|
||||||
|
|
||||||
|
// 设置主键
|
||||||
|
protected $pk = 'config_id';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 列出全部配置,key对应value
|
||||||
|
*/
|
||||||
|
public function getAll(){
|
||||||
|
$aList = static::where('config_status',1)->order('config_sort DESC')->select()->toArray();
|
||||||
|
if(empty($aList)){
|
||||||
|
return [];
|
||||||
|
}else{
|
||||||
|
$return = [];
|
||||||
|
foreach($aList as $k=>$v){
|
||||||
|
$return[$v['config_name']] = $v['config_value'];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return $return;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* 多条数据更新
|
||||||
|
*/
|
||||||
|
public function updateAll($data){
|
||||||
|
$lists = static::order('config_sort DESC,config_id')->select()->toArray();
|
||||||
|
if(empty($lists)){
|
||||||
|
return false;
|
||||||
|
}else{
|
||||||
|
foreach($lists as &$lists_v){
|
||||||
|
$lists_v['config_value'] = $data[$lists_v['config_name']];
|
||||||
|
}
|
||||||
|
$save = static::saveAll($lists);
|
||||||
|
if(empty($save)){
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
25
app/admin/model/ZIconfont.php
Normal file
25
app/admin/model/ZIconfont.php
Normal file
@ -0,0 +1,25 @@
|
|||||||
|
<?php
|
||||||
|
/**
|
||||||
|
* 商业使用授权协议
|
||||||
|
*
|
||||||
|
* Copyright (c) 2025 [云泽网]. 保留所有权利.
|
||||||
|
*
|
||||||
|
* 本软件仅供评估使用。任何商业用途必须获得书面授权许可。
|
||||||
|
* 未经授权商业使用本软件属于侵权行为,将承担法律责任。
|
||||||
|
*
|
||||||
|
* 授权购买请联系: 357099073@qq.com
|
||||||
|
* 官方网站: https://www.yunzer.cn
|
||||||
|
*
|
||||||
|
* 评估用户须知:
|
||||||
|
* 1. 禁止移除版权声明
|
||||||
|
* 2. 禁止用于生产环境
|
||||||
|
* 3. 禁止转售或分发
|
||||||
|
*/
|
||||||
|
|
||||||
|
namespace app\admin\model;
|
||||||
|
|
||||||
|
use think\Model;
|
||||||
|
|
||||||
|
class ZIconfont extends Model
|
||||||
|
{
|
||||||
|
}
|
||||||
288
app/admin/view/articles/add.php
Normal file
288
app/admin/view/articles/add.php
Normal file
@ -0,0 +1,288 @@
|
|||||||
|
{include file="public/header" /}
|
||||||
|
<div class="config-container">
|
||||||
|
<div class="config-header" style="display:flex;justify-content: space-between;">
|
||||||
|
<div>
|
||||||
|
<span>添加文章</span>
|
||||||
|
</div>
|
||||||
|
<div>
|
||||||
|
<button type="button" class="layui-btn layui-btn-primary layui-btn-sm" onclick="goBack()">
|
||||||
|
<i class="layui-icon layui-icon-return"></i>返回
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<form class="layui-form" action="" method="post">
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">标题</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<input type="text" name="title" required lay-verify="required" placeholder="请输入文章标题" autocomplete="off"
|
||||||
|
class="layui-input">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">分类</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<select name="cate" lay-verify="required">
|
||||||
|
<option value="">请选择分类</option>
|
||||||
|
</select>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">描述</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<textarea name="desc" placeholder="请输入描述内容" class="layui-textarea"></textarea>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">作者</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<input type="text" name="author" required lay-verify="required" placeholder="请输入作者" autocomplete="off"
|
||||||
|
class="layui-input" value="{$aUser['name']}">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">是否转载</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<input type="checkbox" name="is_trans" lay-skin="switch" lay-text="是|否" lay-filter="isTrans">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="layui-form-item" id="transUrlItem" style="display: none;">
|
||||||
|
<label class="layui-form-label">转载地址</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<input type="text" name="transurl" placeholder="请输入转载地址" autocomplete="off" class="layui-input">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">封面</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<button type="button" class="layui-btn" id="upload-btn">
|
||||||
|
<i class="layui-icon layui-icon-upload"></i> 图片上传
|
||||||
|
</button>
|
||||||
|
<div style="width: 120px;">
|
||||||
|
<div class="layui-upload-list">
|
||||||
|
<img class="layui-upload-img" id="upload-img"
|
||||||
|
style="width: 118px; height: 118px;object-fit: cover;">
|
||||||
|
<div id="upload-text"></div>
|
||||||
|
</div>
|
||||||
|
<div class="layui-progress layui-progress-big" lay-showPercent="yes" lay-filter="filter-demo">
|
||||||
|
<div class="layui-progress-bar" lay-percent=""></div>
|
||||||
|
</div>
|
||||||
|
<input type="hidden" name="image" id="image" value="">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="layui-form-item layui-form-text">
|
||||||
|
<label class="layui-form-label">内容</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<div id="editor—wrapper" id="content" name="content" style="border: 1px solid #ccc;">
|
||||||
|
<div id="toolbar-container" style="border-bottom: 1px solid #ccc;"><!-- 工具栏 --></div>
|
||||||
|
<div id="editor-container" style="height: 800px;"><!-- 编辑器 --></div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<button class="layui-btn" lay-submit lay-filter="formSubmit">立即提交</button>
|
||||||
|
<button class="layui-btn layui-btn-primary" lay-submit lay-filter="formDraft">存草稿</button>
|
||||||
|
<button type="reset" class="layui-btn layui-btn-primary">重置</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</form>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<script src="/static/js/wangeditor.js"></script>
|
||||||
|
<script>
|
||||||
|
layui.use(['form', 'layer'], function () {
|
||||||
|
var form = layui.form;
|
||||||
|
var layer = layui.layer;
|
||||||
|
var $ = layui.jquery;
|
||||||
|
var upload = layui.upload;
|
||||||
|
var element = layui.element;
|
||||||
|
|
||||||
|
// 监听转载开关
|
||||||
|
form.on('switch(isTrans)', function(data){
|
||||||
|
$('#transUrlItem')[data.elem.checked ? 'show' : 'hide']();
|
||||||
|
});
|
||||||
|
|
||||||
|
// 图片上传
|
||||||
|
var uploadInst = upload.render({
|
||||||
|
elem: '#upload-btn',
|
||||||
|
url: '{:url("index/upload_img")}', // 上传图片接口
|
||||||
|
before: function (obj) {
|
||||||
|
// 预读本地文件示例,不支持ie8
|
||||||
|
obj.preview(function (index, file, result) {
|
||||||
|
$('#upload-img').attr('src', result); // 图片链接(base64)
|
||||||
|
});
|
||||||
|
element.progress('filter-demo', '0%'); // 进度条复位
|
||||||
|
layer.msg('上传中', { icon: 16, time: 0 });
|
||||||
|
},
|
||||||
|
done: function (res) {
|
||||||
|
// 若上传失败
|
||||||
|
if (res.code > 0) {
|
||||||
|
return layer.msg('上传失败');
|
||||||
|
}
|
||||||
|
// 上传成功
|
||||||
|
$('#image').val(res.data); // 设置图片路径到隐藏输入框
|
||||||
|
$('#upload-text').html(''); // 置空上传失败的状态
|
||||||
|
layer.msg('上传成功', { icon: 1 });
|
||||||
|
},
|
||||||
|
uploadError: function () { // 这里改为 uploadError
|
||||||
|
// 演示失败状态,并实现重传
|
||||||
|
var demoText = $('#upload-text');
|
||||||
|
demoText.html('<span style="color: #FF5722;">上传失败</span> <a class="layui-btn layui-btn-xs demo-reload">重试</a>');
|
||||||
|
demoText.find('.demo-reload').on('click', function () {
|
||||||
|
uploadInst.upload();
|
||||||
|
});
|
||||||
|
},
|
||||||
|
// 进度条
|
||||||
|
progress: function (n, elem, e) {
|
||||||
|
element.progress('filter-demo', n + '%'); // 可配合 layui 进度条元素使用
|
||||||
|
if (n == 100) {
|
||||||
|
layer.msg('上传完毕', { icon: 1 });
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
// 获取分类列表
|
||||||
|
$.get('{:url("articles/getcate")}', function (res) {
|
||||||
|
if (res.code == 0) {
|
||||||
|
var html = '<option value="">请选择分类</option>';
|
||||||
|
res.data.forEach(function (item) {
|
||||||
|
// 如果cid为0,则设置为禁用
|
||||||
|
var disabled = item.cid == 0 ? 'disabled' : '';
|
||||||
|
html += '<option value="' + item.id + '" ' + disabled + '>' + item.name + '</option>';
|
||||||
|
// 如果有子分类,添加子分类选项
|
||||||
|
if (item.children && item.children.length > 0) {
|
||||||
|
item.children.forEach(function (child) {
|
||||||
|
html += '<option value="' + child.id + '">├─ ' + child.name + '</option>';
|
||||||
|
});
|
||||||
|
}
|
||||||
|
});
|
||||||
|
$('select[name="cate"]').html(html);
|
||||||
|
form.render('select');
|
||||||
|
} else {
|
||||||
|
layer.msg(res.msg, { icon: 2 });
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
// 表单提交
|
||||||
|
form.on('submit(formSubmit)', function (data) {
|
||||||
|
// 获取编辑器内容
|
||||||
|
var content = editor.getHtml();
|
||||||
|
if (!content || content === '<p><br></p>') {
|
||||||
|
layer.msg('请输入文章内容', { icon: 2 });
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
var loadIndex = layer.load(2);
|
||||||
|
data.field.content = content;
|
||||||
|
|
||||||
|
$.ajax({
|
||||||
|
url: '{:url("articles/add")}',
|
||||||
|
type: 'POST',
|
||||||
|
data: data.field,
|
||||||
|
success: function (res) {
|
||||||
|
layer.close(loadIndex);
|
||||||
|
if (res.code == 0) {
|
||||||
|
layer.msg(res.msg, { icon: 1 });
|
||||||
|
setTimeout(function () {
|
||||||
|
window.location.href = '{:url("articles/articlelist")}';
|
||||||
|
}, 1000);
|
||||||
|
} else {
|
||||||
|
layer.msg(res.msg, { icon: 2 });
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
return false;
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<!-- wangeditor编辑器脚本 -->
|
||||||
|
<script>
|
||||||
|
const { createEditor, createToolbar } = window.wangEditor
|
||||||
|
|
||||||
|
const editorConfig = {
|
||||||
|
MENU_CONF: {},
|
||||||
|
placeholder: '请输入内容...',
|
||||||
|
onChange(editor) {
|
||||||
|
const html = editor.getHtml()
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
// 配置图片上传
|
||||||
|
editorConfig.MENU_CONF['uploadImage'] = {
|
||||||
|
server: '{:url("index/upload_img")}',
|
||||||
|
fieldName: 'file',
|
||||||
|
maxFileSize: 10 * 1024 * 1024, // 10M
|
||||||
|
maxNumberOfFiles: 10,
|
||||||
|
allowedFileTypes: ['image/*'],
|
||||||
|
meta: {
|
||||||
|
token: 'xxx'
|
||||||
|
},
|
||||||
|
metaWithUrl: true,
|
||||||
|
headers: {
|
||||||
|
Accept: 'text/x-json'
|
||||||
|
},
|
||||||
|
timeout: 5 * 1000, // 5s
|
||||||
|
|
||||||
|
onBeforeUpload(file) {
|
||||||
|
console.log('准备上传图片', file)
|
||||||
|
return file
|
||||||
|
},
|
||||||
|
onProgress(progress) {
|
||||||
|
console.log('上传进度', progress)
|
||||||
|
},
|
||||||
|
onSuccess(file, res) {
|
||||||
|
console.log('上传成功', file, res)
|
||||||
|
},
|
||||||
|
onFailed(file, res) {
|
||||||
|
layer.msg('上传失败:' + res.msg, { icon: 2 })
|
||||||
|
console.log('上传失败', file, res)
|
||||||
|
},
|
||||||
|
onError(file, err, res) {
|
||||||
|
layer.msg('上传出错:' + err.message, { icon: 2 })
|
||||||
|
console.error('上传出错', file, err, res)
|
||||||
|
},
|
||||||
|
customInsert(res, insertFn) {
|
||||||
|
// res 即服务端的返回结果
|
||||||
|
if (res.code === 0 && res.data) {
|
||||||
|
// 直接使用res.data作为图片地址
|
||||||
|
insertFn(res.data);
|
||||||
|
} else {
|
||||||
|
layer.msg('图片上传失败', { icon: 2 });
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const editor = createEditor({
|
||||||
|
selector: '#editor-container',
|
||||||
|
html: '<p><br></p>',
|
||||||
|
config: editorConfig,
|
||||||
|
mode: 'default', // or 'simple'
|
||||||
|
})
|
||||||
|
|
||||||
|
const toolbarConfig = {}
|
||||||
|
|
||||||
|
const toolbar = createToolbar({
|
||||||
|
editor,
|
||||||
|
selector: '#toolbar-container',
|
||||||
|
config: toolbarConfig,
|
||||||
|
mode: 'default', // or 'simple'
|
||||||
|
})
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
//返回文章列表
|
||||||
|
function goBack() {
|
||||||
|
window.location.href = '{:url("articles/articlelist")}';
|
||||||
|
}
|
||||||
|
</script>
|
||||||
525
app/admin/view/articles/articlecate.php
Normal file
525
app/admin/view/articles/articlecate.php
Normal file
@ -0,0 +1,525 @@
|
|||||||
|
{include file="public/header" /}
|
||||||
|
<div class="config-container">
|
||||||
|
<!-- 页面头部样式 -->
|
||||||
|
<div class="config-header" style="display: flex;flex-direction: column;flex-wrap: wrap;align-items: flex-start;">
|
||||||
|
<div class="maintitle">
|
||||||
|
<i class="layui-icon layui-icon-app"></i>
|
||||||
|
<span>文章分类管理</span>
|
||||||
|
</div>
|
||||||
|
<div style="display: flex;align-items: flex-start;flex-direction: column;gap: 15px;margin-bottom: 10px;">
|
||||||
|
<div>
|
||||||
|
<button type="button" class="layui-btn layui-btn-normal" onclick="add()">
|
||||||
|
<i class="layui-icon layui-icon-add-1"></i>添加分类
|
||||||
|
</button>
|
||||||
|
<button type="button" class="layui-btn layui-btn-primary layui-border-blue" onclick="refresh()">
|
||||||
|
<i class="layui-icon layui-icon-refresh"></i>刷新
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<!-- 主要内容区 -->
|
||||||
|
<div class="main-content">
|
||||||
|
<div class="layui-row layui-col-space20">
|
||||||
|
<!-- 左侧分类列表 -->
|
||||||
|
<div class="layui-col-md7">
|
||||||
|
<div class="layui-card">
|
||||||
|
<div class="layui-card-header">
|
||||||
|
<span>分类列表</span>
|
||||||
|
<small class="text-muted">支持两级分类结构</small>
|
||||||
|
</div>
|
||||||
|
<div class="layui-card-body">
|
||||||
|
<div id="categoryList" class="category-tree"></div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<!-- 右侧分类信息 -->
|
||||||
|
<div class="layui-col-md5">
|
||||||
|
<div class="layui-card">
|
||||||
|
<div class="layui-card-header">
|
||||||
|
<span>分类信息</span>
|
||||||
|
</div>
|
||||||
|
<div class="layui-card-body">
|
||||||
|
<!-- 默认提示 -->
|
||||||
|
<div id="defaultTip" class="empty-tip">
|
||||||
|
<i class="layui-icon layui-icon-face-surprised"></i>
|
||||||
|
<p>请选择左侧分类或点击新增按钮</p>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<!-- 分类表单 -->
|
||||||
|
<form class="layui-form category-form" lay-filter="categoryForm" style="display: none;">
|
||||||
|
<input type="hidden" name="id" id="categoryId">
|
||||||
|
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">分类名称</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<input type="text" name="name" required lay-verify="required" placeholder="请输入分类名称"
|
||||||
|
autocomplete="off" class="layui-input">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">父级分类</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<select name="cid" lay-verify="required">
|
||||||
|
<option value="0">顶级分类</option>
|
||||||
|
</select>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">封面图片</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<div class="layui-upload-drag" id="uploadImage">
|
||||||
|
<i class="layui-icon layui-icon-upload"></i>
|
||||||
|
<p>点击上传或拖拽图片至此处</p>
|
||||||
|
<div class="layui-hide" id="uploadPreview">
|
||||||
|
<hr>
|
||||||
|
<img src="" alt="封面图片" style="max-width: 100%">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<input type="hidden" name="image" id="imageInput">
|
||||||
|
<div class="layui-form-mid layui-word-aux">建议尺寸:250px * 140px</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">排序</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<input type="number" name="sort" value="0" class="layui-input"
|
||||||
|
placeholder="数字越大越靠前">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">状态</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<input type="radio" name="status" value="1" title="正常" checked>
|
||||||
|
<input type="radio" name="status" value="0" title="禁用">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="layui-form-item form-actions">
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<button class="layui-btn" lay-submit lay-filter="saveCategory">保存</button>
|
||||||
|
<button type="reset" class="layui-btn layui-btn-primary">重置</button>
|
||||||
|
<button type="button" class="layui-btn layui-btn-danger" id="deleteBtn"
|
||||||
|
style="display: none;">删除</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</form>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<style>
|
||||||
|
/* 页面整体样式 */
|
||||||
|
.config-container {
|
||||||
|
padding: 15px;
|
||||||
|
/* background: #f2f2f2; */
|
||||||
|
}
|
||||||
|
|
||||||
|
.layui-col-md7 .layui-btn-primary {
|
||||||
|
border-color: #d2d2d2;
|
||||||
|
background: 0 0;
|
||||||
|
color: #5f5f5f
|
||||||
|
}
|
||||||
|
|
||||||
|
.layui-col-md7 .layui-btn-primary:hover {
|
||||||
|
background-color: #1e9fff;
|
||||||
|
color: #efefef
|
||||||
|
}
|
||||||
|
|
||||||
|
/* 页面头部样式 */
|
||||||
|
.page-header {
|
||||||
|
display: flex;
|
||||||
|
justify-content: space-between;
|
||||||
|
align-items: center;
|
||||||
|
margin-bottom: 15px;
|
||||||
|
padding: 10px 15px;
|
||||||
|
background: #fff;
|
||||||
|
border-radius: 2px;
|
||||||
|
box-shadow: 0 1px 2px rgba(0, 0, 0, 0.05);
|
||||||
|
}
|
||||||
|
|
||||||
|
.header-title {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
font-size: 16px;
|
||||||
|
font-weight: 500;
|
||||||
|
}
|
||||||
|
|
||||||
|
.header-title .layui-icon {
|
||||||
|
margin-right: 8px;
|
||||||
|
font-size: 18px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.header-actions .layui-btn {
|
||||||
|
margin-left: 8px;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* 主要内容区样式 */
|
||||||
|
.main-content {
|
||||||
|
min-height: calc(100vh - 170px);
|
||||||
|
}
|
||||||
|
|
||||||
|
.layui-card {
|
||||||
|
margin-bottom: 0;
|
||||||
|
border-radius: 8px;
|
||||||
|
box-shadow: 0 3px 6px rgba(0, 0, 0, 0.1);
|
||||||
|
}
|
||||||
|
|
||||||
|
.layui-card-header {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
height: auto;
|
||||||
|
padding: 12px 15px;
|
||||||
|
font-size: 15px;
|
||||||
|
font-weight: 500;
|
||||||
|
}
|
||||||
|
|
||||||
|
.text-muted {
|
||||||
|
margin-left: 10px;
|
||||||
|
font-size: 12px;
|
||||||
|
font-weight: normal;
|
||||||
|
color: #999;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* 分类树样式 */
|
||||||
|
.category-tree {
|
||||||
|
padding: 10px 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.category-item {
|
||||||
|
margin: 5px 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.category-header {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: space-between;
|
||||||
|
padding: 10px 15px;
|
||||||
|
background-color: #f8f8f8;
|
||||||
|
border-radius: 2px;
|
||||||
|
transition: all 0.3s;
|
||||||
|
cursor: pointer;
|
||||||
|
}
|
||||||
|
|
||||||
|
.category-header:hover {
|
||||||
|
background-color: #f2f2f2;
|
||||||
|
}
|
||||||
|
|
||||||
|
.category-header.active {
|
||||||
|
background-color: #e6f7ff;
|
||||||
|
border-right: 3px solid #1890ff;
|
||||||
|
}
|
||||||
|
|
||||||
|
.category-name {
|
||||||
|
flex: 1;
|
||||||
|
font-size: 14px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.category-actions {
|
||||||
|
/* opacity: 0; */
|
||||||
|
transition: opacity 0.3s;
|
||||||
|
}
|
||||||
|
|
||||||
|
.category-header:hover .category-actions {
|
||||||
|
opacity: 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
.category-children {
|
||||||
|
margin-left: 20px;
|
||||||
|
padding-left: 15px;
|
||||||
|
border-left: 1px dashed #e6e6e6;
|
||||||
|
}
|
||||||
|
|
||||||
|
.add-child {
|
||||||
|
padding: 3px 8px;
|
||||||
|
font-size: 12px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.add-child .layui-icon {
|
||||||
|
font-size: 12px;
|
||||||
|
margin-right: 3px;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* 空状态提示 */
|
||||||
|
.empty-tip {
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
padding: 40px 0;
|
||||||
|
color: #999;
|
||||||
|
}
|
||||||
|
|
||||||
|
.empty-tip .layui-icon {
|
||||||
|
font-size: 32px;
|
||||||
|
margin-bottom: 10px;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* 表单样式优化 */
|
||||||
|
.category-form {
|
||||||
|
padding: 10px 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.layui-form-label {
|
||||||
|
width: 100px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.layui-input-block {
|
||||||
|
margin-left: 130px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.form-actions {
|
||||||
|
margin-top: 30px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.layui-upload-drag {
|
||||||
|
padding: 20px;
|
||||||
|
border: 1px dashed #e2e2e2;
|
||||||
|
background-color: #fff;
|
||||||
|
text-align: center;
|
||||||
|
cursor: pointer;
|
||||||
|
}
|
||||||
|
|
||||||
|
.layui-upload-drag:hover {
|
||||||
|
border-color: #009688;
|
||||||
|
}
|
||||||
|
|
||||||
|
.layui-upload-drag img {
|
||||||
|
max-width: 100%;
|
||||||
|
margin: 10px 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.layui-btn-xs {
|
||||||
|
height: 30px;
|
||||||
|
line-height: 25px;
|
||||||
|
font-size: 12px;
|
||||||
|
}
|
||||||
|
</style>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
// 定义全局变量和函数
|
||||||
|
var categoryManager = {
|
||||||
|
init: function () {
|
||||||
|
this.initLayui();
|
||||||
|
},
|
||||||
|
|
||||||
|
initLayui: function () {
|
||||||
|
var that = this;
|
||||||
|
layui.use(['layer', 'form', 'upload'], function () {
|
||||||
|
var layer = layui.layer;
|
||||||
|
var form = layui.form;
|
||||||
|
var upload = layui.upload;
|
||||||
|
var $ = layui.jquery;
|
||||||
|
|
||||||
|
// 初始化分类列表
|
||||||
|
that.initCategoryList = function () {
|
||||||
|
$.ajax({
|
||||||
|
url: '/admin/articles/articlecate',
|
||||||
|
type: 'POST',
|
||||||
|
success: function (res) {
|
||||||
|
if (res.code === 0) {
|
||||||
|
var html = '';
|
||||||
|
res.data.forEach(function (item) {
|
||||||
|
html += that.renderCategory(item);
|
||||||
|
});
|
||||||
|
$('#categoryList').html(html);
|
||||||
|
that.bindEvents();
|
||||||
|
} else {
|
||||||
|
layer.msg('获取分类数据失败', { icon: 2 });
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
// 渲染分类项
|
||||||
|
that.renderCategory = function (category, level = 0) {
|
||||||
|
var html = '<div class="category-item" data-id="' + category.id + '">';
|
||||||
|
html += '<div class="category-header">';
|
||||||
|
html += '<div class="category-name">' + category.title + '</div>';
|
||||||
|
if (level === 0) {
|
||||||
|
html += '<div class="category-actions">';
|
||||||
|
html += '<button type="button" class="layui-btn layui-btn-primary layui-btn-xs add-child">';
|
||||||
|
html += '<i class="layui-icon layui-icon-add-1"></i>添加子分类</button>';
|
||||||
|
html += '</div>';
|
||||||
|
}
|
||||||
|
html += '</div>';
|
||||||
|
|
||||||
|
if (category.children && category.children.length > 0) {
|
||||||
|
html += '<div class="category-children">';
|
||||||
|
category.children.forEach(function (child) {
|
||||||
|
html += that.renderCategory(child, level + 1);
|
||||||
|
});
|
||||||
|
html += '</div>';
|
||||||
|
}
|
||||||
|
html += '</div>';
|
||||||
|
return html;
|
||||||
|
};
|
||||||
|
|
||||||
|
// 绑定事件
|
||||||
|
that.bindEvents = function () {
|
||||||
|
// 点击分类项加载编辑信息
|
||||||
|
$('.category-header').off('click').on('click', function (e) {
|
||||||
|
if (!$(e.target).closest('.add-child').length) {
|
||||||
|
var id = $(this).closest('.category-item').data('id');
|
||||||
|
that.loadCategoryInfo(id);
|
||||||
|
|
||||||
|
// 添加选中效果
|
||||||
|
$('.category-header').removeClass('active');
|
||||||
|
$(this).addClass('active');
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
// 添加子分类
|
||||||
|
$('.add-child').off('click').on('click', function (e) {
|
||||||
|
e.stopPropagation();
|
||||||
|
var parentId = $(this).closest('.category-item').data('id');
|
||||||
|
that.showCategoryForm(parentId);
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
// 加载分类信息
|
||||||
|
that.loadCategoryInfo = function (id) {
|
||||||
|
$.get('/admin/articles/cateedit?id=' + id, function (res) {
|
||||||
|
if (res.code === 0) {
|
||||||
|
that.showCategoryForm(0, res.data);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
// 显示分类表单
|
||||||
|
that.showCategoryForm = function (parentId = 0, data = null) {
|
||||||
|
$('#defaultTip').hide();
|
||||||
|
$('.category-form').show();
|
||||||
|
|
||||||
|
// 重置表单
|
||||||
|
form.val('categoryForm', {
|
||||||
|
id: data ? data.info.id : '',
|
||||||
|
name: data ? data.info.name : '',
|
||||||
|
cid: data ? data.info.cid : parentId,
|
||||||
|
sort: data ? data.info.sort : 0,
|
||||||
|
status: data ? data.info.status : 1
|
||||||
|
});
|
||||||
|
|
||||||
|
// 更新父级分类选项
|
||||||
|
var $select = $('select[name="cid"]');
|
||||||
|
$select.empty().append('<option value="0">顶级分类</option>');
|
||||||
|
// 获取所有分类作为父级选项
|
||||||
|
$.ajax({
|
||||||
|
url: '/admin/articles/articlecate',
|
||||||
|
type: 'POST',
|
||||||
|
async: false,
|
||||||
|
success: function (res) {
|
||||||
|
if (res.code === 0) {
|
||||||
|
// 当前编辑的分类ID
|
||||||
|
var currentId = data ? data.info.id : 0;
|
||||||
|
// 递归构建分类选项
|
||||||
|
function buildOptions(categories, level) {
|
||||||
|
categories.forEach(function (category) {
|
||||||
|
// 不能选择自己或自己的子分类作为父级
|
||||||
|
if (category.id != currentId) {
|
||||||
|
var prefix = new Array(level + 1).join('├─ ');
|
||||||
|
$select.append('<option value="' + category.id + '">' + prefix + category.title + '</option>'); if (category.children && category.children.length > 0) { buildOptions(category.children, level + 1); }
|
||||||
|
}
|
||||||
|
});
|
||||||
|
} buildOptions(res.data, 0);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
// 设置选中的父级分类
|
||||||
|
if (data && data.info.cid) {
|
||||||
|
$select.val(data.info.cid);
|
||||||
|
}
|
||||||
|
|
||||||
|
// 更新图片预览
|
||||||
|
if (data && data.info.image) {
|
||||||
|
$('#uploadPreview').removeClass('layui-hide').find('img').attr('src', data.info.image);
|
||||||
|
$('#imageInput').val(data.info.image);
|
||||||
|
} else {
|
||||||
|
$('#uploadPreview').addClass('layui-hide').find('img').attr('src', '');
|
||||||
|
$('#imageInput').val('');
|
||||||
|
}
|
||||||
|
|
||||||
|
// 显示/隐藏删除按钮
|
||||||
|
$('#deleteBtn')[data ? 'show' : 'hide']();
|
||||||
|
|
||||||
|
form.render();
|
||||||
|
};
|
||||||
|
|
||||||
|
// 初始化上传组件
|
||||||
|
upload.render({
|
||||||
|
elem: '#uploadImage',
|
||||||
|
url: '/admin/index/upload_img',
|
||||||
|
accept: 'images',
|
||||||
|
acceptMime: 'image/*',
|
||||||
|
done: function (res) {
|
||||||
|
if (res.code === 0) {
|
||||||
|
$('#uploadPreview').removeClass('layui-hide').find('img').attr('src', typeof res.data === 'object' ? res.data.url : res.data);
|
||||||
|
$('#imageInput').val(typeof res.data === 'object' ? res.data : res.data);
|
||||||
|
layer.msg('上传成功');
|
||||||
|
} else {
|
||||||
|
layer.msg('上传失败');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
// 监听表单提交
|
||||||
|
form.on('submit(saveCategory)', function (data) {
|
||||||
|
var url = data.field.id ? '/admin/articles/cateedit' : '/admin/articles/cateadd';
|
||||||
|
$.post(url, data.field, function (res) {
|
||||||
|
if (res.code === 0) {
|
||||||
|
layer.msg(res.msg, { icon: 1 });
|
||||||
|
that.initCategoryList();
|
||||||
|
} else {
|
||||||
|
layer.msg(res.msg, { icon: 2 });
|
||||||
|
}
|
||||||
|
});
|
||||||
|
return false;
|
||||||
|
});
|
||||||
|
|
||||||
|
// 监听删除按钮
|
||||||
|
$('#deleteBtn').on('click', function () {
|
||||||
|
var id = $('#categoryId').val();
|
||||||
|
if (!id) return;
|
||||||
|
|
||||||
|
layer.confirm('确定要删除该分类吗?', function (index) {
|
||||||
|
$.post('/admin/articles/catedel', { id: id }, function (res) {
|
||||||
|
if (res.code === 0) {
|
||||||
|
layer.msg(res.msg, { icon: 1 });
|
||||||
|
that.initCategoryList();
|
||||||
|
$('#defaultTip').show();
|
||||||
|
$('.category-form').hide();
|
||||||
|
} else {
|
||||||
|
layer.msg(res.msg, { icon: 2 });
|
||||||
|
}
|
||||||
|
});
|
||||||
|
layer.close(index);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
// 初始化页面
|
||||||
|
that.initCategoryList();
|
||||||
|
});
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
// 初始化
|
||||||
|
categoryManager.init();
|
||||||
|
|
||||||
|
// 新增分类
|
||||||
|
function add() {
|
||||||
|
categoryManager.showCategoryForm();
|
||||||
|
}
|
||||||
|
|
||||||
|
// 刷新列表
|
||||||
|
function refresh() {
|
||||||
|
categoryManager.initCategoryList();
|
||||||
|
}
|
||||||
|
</script>
|
||||||
204
app/admin/view/articles/articlelist.php
Normal file
204
app/admin/view/articles/articlelist.php
Normal file
@ -0,0 +1,204 @@
|
|||||||
|
{include file="public/header" /}
|
||||||
|
<div class="config-container">
|
||||||
|
<!-- 页面头部样式 -->
|
||||||
|
<div class="config-header" style="display: flex;flex-direction: column;flex-wrap: wrap;align-items: flex-start;">
|
||||||
|
<div class="maintitle">
|
||||||
|
<i class="layui-icon layui-icon-list"></i>
|
||||||
|
<span>文章列表</span>
|
||||||
|
</div>
|
||||||
|
<div style="display: flex;align-items: flex-start;flex-direction: column;gap: 15px;margin-bottom: 10px;">
|
||||||
|
<div class="shaixuan">
|
||||||
|
<label>筛选:</label>
|
||||||
|
<div class="layui-form" style="display: flex; gap: 10px;">
|
||||||
|
<div class="layui-input-inline">
|
||||||
|
<select id="categoryFilter" lay-filter="categoryFilter" lay-verify="">
|
||||||
|
<option value="">全部分类</option>
|
||||||
|
{volist name="categories" id="category"}
|
||||||
|
<optgroup label="{$category.name}">
|
||||||
|
{volist name="category.children" id="subCategory"}
|
||||||
|
<option value="{$subCategory.id}">{$subCategory.name}</option>
|
||||||
|
{/volist}
|
||||||
|
</optgroup>
|
||||||
|
{/volist}
|
||||||
|
</select>
|
||||||
|
</div>
|
||||||
|
<div class="layui-input-inline">
|
||||||
|
<input type="text" id="titleSearch" placeholder="搜索标题" class="layui-input">
|
||||||
|
</div>
|
||||||
|
<div class="layui-input-inline">
|
||||||
|
<input type="text" id="authorSearch" placeholder="搜索作者" class="layui-input">
|
||||||
|
</div>
|
||||||
|
<button type="button" class="layui-btn layui-btn-normal" onclick="doSearch()">
|
||||||
|
<i class="layui-icon layui-icon-search"></i>搜索
|
||||||
|
</button>
|
||||||
|
<button type="button" class="layui-btn layui-btn-primary layui-border-blue" onclick="doRefresh()">
|
||||||
|
<i class="layui-icon layui-icon-refresh"></i>重置
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div>
|
||||||
|
<button type="button" class="layui-btn layui-btn-normal" onclick="add()">
|
||||||
|
<i class="layui-icon layui-icon-add-1"></i>添加文章
|
||||||
|
</button>
|
||||||
|
<button type="button" class="layui-btn layui-btn-primary layui-border-blue" onclick="refresh()">
|
||||||
|
<i class="layui-icon layui-icon-refresh"></i>刷新
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<table id="articleTable" lay-filter="articleTable"></table>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<script type="text/html" id="imageTemplate">
|
||||||
|
{{# if(d.image){ }}
|
||||||
|
<img src="{{ d.image }}" style="max-width: 50px; max-height: 50px;">
|
||||||
|
{{# } }}
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<script type="text/html" id="statusTemplate">
|
||||||
|
{{# if(d.status === 0){ }}
|
||||||
|
<span style="color:red;">草稿</span>
|
||||||
|
{{# } else if(d.status === 1){ }}
|
||||||
|
<span style="color:orange;">待审核</span>
|
||||||
|
{{# } else if(d.status === 2){ }}
|
||||||
|
<span style="color:green;">已发布</span>
|
||||||
|
{{# } else if(d.status === 3){ }}
|
||||||
|
<span style="color:gray;">隐藏</span>
|
||||||
|
{{# } }}
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<script type="text/html" id="operationBar">
|
||||||
|
<button type="button" class="layui-btn layui-btn-normal layui-btn-xs" lay-event="edit">
|
||||||
|
<i class="layui-icon layui-icon-edit"></i>编辑
|
||||||
|
</button>
|
||||||
|
<button type="button" class="layui-btn layui-btn-primary layui-btn-xs" lay-event="del">
|
||||||
|
<i class="layui-icon layui-icon-delete"></i>删除
|
||||||
|
</button>
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<script type="text/javascript">
|
||||||
|
layui.use(['layer', 'form', 'table'], function () {
|
||||||
|
var layer = layui.layer;
|
||||||
|
var $ = layui.jquery;
|
||||||
|
var form = layui.form;
|
||||||
|
var table = layui.table;
|
||||||
|
|
||||||
|
// 初始化表格
|
||||||
|
table.render({
|
||||||
|
elem: '#articleTable',
|
||||||
|
url: '/admin/articles/articlelist',
|
||||||
|
method: 'post',
|
||||||
|
cols: [[
|
||||||
|
{ field: 'id', title: 'ID', align: 'center', width: 80 },
|
||||||
|
{ field: 'title', title: '标题' },
|
||||||
|
{ field: 'cate', title: '分类', align: 'center', width: 180 },
|
||||||
|
{ field: 'image', title: '封面', templet: '#imageTemplate', align: 'center', width: 180 },
|
||||||
|
{ field: 'author', title: '作者', align: 'center', width: 120 },
|
||||||
|
{ field: 'status', title: '状态', templet: '#statusTemplate', align: 'center', width: 80 },
|
||||||
|
{ field: 'publishdate', title: '发布时间', align: 'center', width: 180 },
|
||||||
|
{ title: '操作', toolbar: '#operationBar', align: 'center', width: 150 }
|
||||||
|
]],
|
||||||
|
page: true,
|
||||||
|
limit: 10,
|
||||||
|
limits: [10, 50, 100],
|
||||||
|
//height: 'full-220'
|
||||||
|
});
|
||||||
|
|
||||||
|
// 监听工具条事件
|
||||||
|
table.on('tool(articleTable)', function (obj) {
|
||||||
|
var data = obj.data;
|
||||||
|
if (obj.event === 'edit') {
|
||||||
|
edit(data.id);
|
||||||
|
} else if (obj.event === 'del') {
|
||||||
|
del(data.id);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
// 监听分类筛选变化
|
||||||
|
form.on('select(categoryFilter)', function (data) {
|
||||||
|
filterByCategory(data.value);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
function filterByCategory(categoryId) {
|
||||||
|
reloadTable();
|
||||||
|
}
|
||||||
|
|
||||||
|
function doSearch() {
|
||||||
|
var titleKeyword = $('#titleSearch').val().trim();
|
||||||
|
var authorKeyword = $('#authorSearch').val().trim();
|
||||||
|
|
||||||
|
if (!titleKeyword && !authorKeyword && !$('#categoryFilter').val()) {
|
||||||
|
layer.msg('请输入搜索条件', { icon: 0 });
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
reloadTable();
|
||||||
|
}
|
||||||
|
|
||||||
|
function doRefresh() {
|
||||||
|
// 清空搜索条件
|
||||||
|
$('#titleSearch').val('');
|
||||||
|
$('#authorSearch').val('');
|
||||||
|
$('#categoryFilter').val('');
|
||||||
|
layui.form.render('select'); // 重新渲染select
|
||||||
|
|
||||||
|
// 重新加载表格,不带任何筛选条件
|
||||||
|
layui.table.reload('articleTable', {
|
||||||
|
where: {},
|
||||||
|
page: {
|
||||||
|
curr: 1
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
layer.msg('已重置', { icon: 1 });
|
||||||
|
}
|
||||||
|
|
||||||
|
function reloadTable() {
|
||||||
|
var categoryId = $('#categoryFilter').val();
|
||||||
|
var categoryName = categoryId ? $('#categoryFilter option[value="' + categoryId + '"]').text() : '';
|
||||||
|
var titleKeyword = $('#titleSearch').val().trim();
|
||||||
|
var authorKeyword = $('#authorSearch').val().trim();
|
||||||
|
|
||||||
|
layui.table.reload('articleTable', {
|
||||||
|
where: {
|
||||||
|
category: categoryName,
|
||||||
|
title: titleKeyword,
|
||||||
|
author: authorKeyword
|
||||||
|
},
|
||||||
|
page: {
|
||||||
|
curr: 1
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
function add() {
|
||||||
|
window.location.href = '/admin/articles/add';
|
||||||
|
}
|
||||||
|
|
||||||
|
function edit(id) {
|
||||||
|
window.location.href = '/admin/articles/edit?id=' + id;
|
||||||
|
}
|
||||||
|
|
||||||
|
function del(id) {
|
||||||
|
layer.confirm('确定要删除该文章吗?', {
|
||||||
|
btn: ['确定', '取消']
|
||||||
|
}, function () {
|
||||||
|
$.post('/admin/articles/delete', { id: id }, function (res) {
|
||||||
|
if (res.code == 0) {
|
||||||
|
layer.msg(res.msg, { icon: 1 });
|
||||||
|
setTimeout(function () {
|
||||||
|
layui.table.reload('articleTable');
|
||||||
|
}, 1000);
|
||||||
|
} else {
|
||||||
|
layer.msg(res.msg, { icon: 2 });
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
function refresh() {
|
||||||
|
layui.table.reload('articleTable');
|
||||||
|
}
|
||||||
|
</script>
|
||||||
77
app/admin/view/articles/cateadd.php
Normal file
77
app/admin/view/articles/cateadd.php
Normal file
@ -0,0 +1,77 @@
|
|||||||
|
{include file="public/header" /}
|
||||||
|
<div style="padding: 50px;padding-bottom: 0px;">
|
||||||
|
<form class="layui-form" action="" method="post">
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">分类名称</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<input type="text" id="name" name="name" required lay-verify="required" placeholder="请输入分类名称"
|
||||||
|
class="layui-input">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">父级分类</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<select id="cid" name="cid" lay-verify="required">
|
||||||
|
<option value="0">顶级分类</option>
|
||||||
|
</select>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">排序</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<input type="number" id="sort" name="sort" value="0" placeholder="请输入排序值" class="layui-input">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<button class="layui-btn" lay-submit lay-filter="formSubmit">添加分类</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</form>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
layui.use(['form', 'layer'], function () {
|
||||||
|
var form = layui.form;
|
||||||
|
var layer = layui.layer;
|
||||||
|
var $ = layui.jquery;
|
||||||
|
|
||||||
|
// 获取URL中的cid参数
|
||||||
|
var urlParams = new URLSearchParams(window.location.search);
|
||||||
|
var cid = urlParams.get('cid');
|
||||||
|
|
||||||
|
// 获取分类列表
|
||||||
|
$.get('{:url("articles/getcate")}', function (res) {
|
||||||
|
if (res.code == 0) {
|
||||||
|
var html = '<option value="0">顶级分类</option>';
|
||||||
|
res.data.forEach(function (item) {
|
||||||
|
html += '<option value="' + item.id + '"' + (cid == item.id ? ' selected' : '') + '>' + item.name + '</option>';
|
||||||
|
});
|
||||||
|
$('#cid').html(html);
|
||||||
|
form.render('select');
|
||||||
|
} else {
|
||||||
|
layer.msg(res.msg, { icon: 2 });
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
// 表单提交
|
||||||
|
form.on('submit(formSubmit)', function (data) {
|
||||||
|
$.post('{:url("articles/cateadd")}', data.field, function (res) {
|
||||||
|
if (res.code == 0) {
|
||||||
|
layer.msg(res.msg, { icon: 1 });
|
||||||
|
setTimeout(function () {
|
||||||
|
var index = parent.layer.getFrameIndex(window.name);
|
||||||
|
parent.layer.close(index);
|
||||||
|
parent.location.reload();
|
||||||
|
}, 1000);
|
||||||
|
} else {
|
||||||
|
layer.msg(res.msg, { icon: 2 });
|
||||||
|
}
|
||||||
|
});
|
||||||
|
return false;
|
||||||
|
});
|
||||||
|
});
|
||||||
|
</script>
|
||||||
154
app/admin/view/articles/cateedit.php
Normal file
154
app/admin/view/articles/cateedit.php
Normal file
@ -0,0 +1,154 @@
|
|||||||
|
{include file="public/header" /}
|
||||||
|
<div style="padding: 50px;padding-bottom: 0px;">
|
||||||
|
<form class="layui-form" action="" method="post">
|
||||||
|
<input type="hidden" name="id" value="{$info.id}">
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">分类图片</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<button type="button" class="layui-btn" id="upload-btn">
|
||||||
|
<i class="layui-icon layui-icon-upload"></i> 图片上传
|
||||||
|
</button>
|
||||||
|
<div style="width: 120px;">
|
||||||
|
<div class="layui-upload-list">
|
||||||
|
<img class="layui-upload-img" id="upload-img"
|
||||||
|
style="width: 118px; height: 118px;object-fit: cover;" src="{$info.image}">
|
||||||
|
<div id="upload-text"></div>
|
||||||
|
</div>
|
||||||
|
<div class="layui-progress layui-progress-big" lay-showPercent="yes" lay-filter="filter-demo">
|
||||||
|
<div class="layui-progress-bar" lay-percent=""></div>
|
||||||
|
</div>
|
||||||
|
<input type="hidden" name="image" id="image" value="{$info.image}">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">分类名称</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<input type="text" id="name" name="name" required lay-verify="required" placeholder="请输入分类名称"
|
||||||
|
value="{$info.name}" class="layui-input">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">父级分类</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<select id="cid" name="cid" lay-verify="required">
|
||||||
|
</select>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">状态</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<select id="status" name="status" lay-verify="required">
|
||||||
|
<option value="1">启用</option>
|
||||||
|
<option value="0">关闭</option>
|
||||||
|
</select>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">排序</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<input type="number" id="sort" name="sort" value="{$info.sort}" placeholder="请输入排序值"
|
||||||
|
class="layui-input">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<button class="layui-btn" lay-submit lay-filter="formSubmit">保存修改</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</form>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
// 首先定义PHP变量,确保类型正确
|
||||||
|
var currentCid = parseInt('{$info.cid|default=0}'); // 转换为数字
|
||||||
|
var currentId = parseInt('{$info.id|default=0}'); // 转换为数字
|
||||||
|
|
||||||
|
layui.use(['form', 'layer', 'upload', 'element'], function () {
|
||||||
|
var form = layui.form;
|
||||||
|
var layer = layui.layer;
|
||||||
|
var $ = layui.jquery;
|
||||||
|
var upload = layui.upload;
|
||||||
|
var element = layui.element;
|
||||||
|
|
||||||
|
// 获取URL中的id参数
|
||||||
|
var urlParams = new URLSearchParams(window.location.search);
|
||||||
|
var id = parseInt(urlParams.get('id')) || 0; // 转换为数字
|
||||||
|
|
||||||
|
// 图片上传
|
||||||
|
var uploadInst = upload.render({
|
||||||
|
elem: '#upload-btn',
|
||||||
|
url: '{:url("index/upload_img")}',
|
||||||
|
before: function (obj) {
|
||||||
|
obj.preview(function (index, file, result) {
|
||||||
|
$('#upload-img').attr('src', result);
|
||||||
|
});
|
||||||
|
element.progress('filter-demo', '0%');
|
||||||
|
layer.msg('上传中', { icon: 16, time: 0 });
|
||||||
|
},
|
||||||
|
done: function (res) {
|
||||||
|
if (res.code > 0) {
|
||||||
|
return layer.msg('上传失败');
|
||||||
|
}
|
||||||
|
$('#image').val(res.data);
|
||||||
|
$('#upload-text').html('');
|
||||||
|
layer.msg('上传成功', { icon: 1 });
|
||||||
|
},
|
||||||
|
uploadError: function () {
|
||||||
|
var demoText = $('#upload-text');
|
||||||
|
demoText.html('<span style="color: #FF5722;">上传失败</span> <a class="layui-btn layui-btn-xs demo-reload">重试</a>');
|
||||||
|
demoText.find('.demo-reload').on('click', function () {
|
||||||
|
uploadInst.upload();
|
||||||
|
});
|
||||||
|
},
|
||||||
|
progress: function (n, elem, e) {
|
||||||
|
element.progress('filter-demo', n + '%');
|
||||||
|
if (n == 100) {
|
||||||
|
layer.msg('上传完毕', { icon: 1 });
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
// 获取分类列表
|
||||||
|
$.get('{:url("articles/getcate")}', function (res) {
|
||||||
|
if (res.code == 0) {
|
||||||
|
var html = '<option value="0">顶级分类</option>';
|
||||||
|
res.data.forEach(function (item) {
|
||||||
|
// 确保item.id是数字类型
|
||||||
|
var itemId = parseInt(item.id);
|
||||||
|
if (itemId !== id) { // 使用严格比较
|
||||||
|
html += '<option value="' + itemId + '"' +
|
||||||
|
(itemId === currentCid ? ' selected' : '') +
|
||||||
|
'>' + item.name + '</option>';
|
||||||
|
}
|
||||||
|
});
|
||||||
|
$('#cid').html(html);
|
||||||
|
form.render('select');
|
||||||
|
} else {
|
||||||
|
layer.msg(res.msg, { icon: 2 });
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
// 表单提交
|
||||||
|
form.on('submit(formSubmit)', function (data) {
|
||||||
|
$.post('{:url("articles/cateedit")}', data.field, function (res) {
|
||||||
|
if (res.code == 0) {
|
||||||
|
layer.msg(res.msg, { icon: 1 });
|
||||||
|
setTimeout(function () {
|
||||||
|
var index = parent.layer.getFrameIndex(window.name);
|
||||||
|
parent.layer.close(index);
|
||||||
|
parent.location.reload();
|
||||||
|
}, 1000);
|
||||||
|
} else {
|
||||||
|
layer.msg(res.msg, { icon: 2 });
|
||||||
|
}
|
||||||
|
});
|
||||||
|
return false;
|
||||||
|
});
|
||||||
|
});
|
||||||
|
</script>
|
||||||
313
app/admin/view/articles/edit.php
Normal file
313
app/admin/view/articles/edit.php
Normal file
@ -0,0 +1,313 @@
|
|||||||
|
{include file="public/header" /}
|
||||||
|
<div class="config-container">
|
||||||
|
<div class="config-header" style="display:flex;justify-content: space-between;">
|
||||||
|
<div>
|
||||||
|
<span>编辑文章</span>
|
||||||
|
</div>
|
||||||
|
<div>
|
||||||
|
<button type="button" class="layui-btn layui-btn-primary layui-btn-sm" onclick="goBack()">
|
||||||
|
<i class="layui-icon layui-icon-return"></i>返回
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<form class="layui-form" action="" method="post">
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">标题</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<input type="text" name="title" required lay-verify="required" placeholder="请输入文章标题" autocomplete="off"
|
||||||
|
class="layui-input" value="{$info.title}">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">分类</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<select name="cate" lay-verify="required">
|
||||||
|
<option value="">请选择分类</option>
|
||||||
|
{foreach $cates as $item}
|
||||||
|
<option value="{$item.id}" {if $info.cate==$item.id}selected{/if}>{$item.name}</option>
|
||||||
|
{/foreach}
|
||||||
|
</select>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">描述</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<textarea name="desc" placeholder="请输入描述内容" class="layui-textarea">{$info.desc}</textarea>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">作者</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<input type="text" name="author" required lay-verify="required" placeholder="请输入作者" autocomplete="off"
|
||||||
|
class="layui-input" value="{$info.author}">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">是否转载</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<input type="checkbox" name="is_trans" lay-skin="switch" lay-text="是|否" lay-filter="isTrans" {if $info.is_trans eq '是'}checked{/if}>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="layui-form-item" id="transUrlItem" style="display: none;">
|
||||||
|
<label class="layui-form-label">转载地址</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<input type="text" name="transurl" placeholder="请输入转载地址" autocomplete="off" class="layui-input"
|
||||||
|
value="{$info.transurl}">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">封面</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<button type="button" class="layui-btn" id="upload-btn">
|
||||||
|
<i class="layui-icon layui-icon-upload"></i> 图片上传
|
||||||
|
</button>
|
||||||
|
<div style="width: 120px;">
|
||||||
|
<div class="layui-upload-list">
|
||||||
|
<img class="layui-upload-img" id="upload-img"
|
||||||
|
style="width: 118px; height: 118px;object-fit: cover;" src="{$info.image}">
|
||||||
|
<div id="upload-text"></div>
|
||||||
|
</div>
|
||||||
|
<div class="layui-progress layui-progress-big" lay-showPercent="yes" lay-filter="filter-demo">
|
||||||
|
<div class="layui-progress-bar" lay-percent=""></div>
|
||||||
|
</div>
|
||||||
|
<input type="hidden" name="image" id="image" value="{$info.image}">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="layui-form-item layui-form-text">
|
||||||
|
<label class="layui-form-label">内容</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<div id="editor—wrapper" id="content" name="content" style="border: 1px solid #ccc;">
|
||||||
|
<div id="toolbar-container" style="border-bottom: 1px solid #ccc;"><!-- 工具栏 --></div>
|
||||||
|
<div id="editor-container" style="height: 800px;"><!-- 编辑器 --></div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<button class="layui-btn" lay-submit lay-filter="formSubmit">立即提交</button>
|
||||||
|
<button class="layui-btn layui-btn-primary" lay-submit lay-filter="formDraft">存草稿</button>
|
||||||
|
<button type="reset" class="layui-btn layui-btn-primary">重置</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</form>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<script src="/static/js/wangeditor.js"></script>
|
||||||
|
<script>
|
||||||
|
layui.use(['form', 'layer'], function () {
|
||||||
|
var form = layui.form;
|
||||||
|
var layer = layui.layer;
|
||||||
|
var $ = layui.jquery;
|
||||||
|
var upload = layui.upload;
|
||||||
|
var element = layui.element;
|
||||||
|
|
||||||
|
// 页面加载时检查转载状态
|
||||||
|
var isTrans = '{$info.is_trans|default="否"}';
|
||||||
|
if(isTrans === '是') {
|
||||||
|
$('#transUrlItem').show();
|
||||||
|
} else {
|
||||||
|
$('#transUrlItem').hide();
|
||||||
|
}
|
||||||
|
|
||||||
|
// 监听转载开关
|
||||||
|
form.on('switch(isTrans)', function (data) {
|
||||||
|
$('#transUrlItem')[data.elem.checked ? 'show' : 'hide']();
|
||||||
|
});
|
||||||
|
|
||||||
|
// 图片上传
|
||||||
|
var uploadInst = upload.render({
|
||||||
|
elem: '#upload-btn',
|
||||||
|
url: '{:url("index/upload_img")}', // 上传图片接口
|
||||||
|
before: function (obj) {
|
||||||
|
// 预读本地文件示例,不支持ie8
|
||||||
|
obj.preview(function (index, file, result) {
|
||||||
|
$('#upload-img').attr('src', result); // 图片链接(base64)
|
||||||
|
});
|
||||||
|
element.progress('filter-demo', '0%'); // 进度条复位
|
||||||
|
layer.msg('上传中', { icon: 16, time: 0 });
|
||||||
|
},
|
||||||
|
done: function (res) {
|
||||||
|
// 若上传失败
|
||||||
|
if (res.code > 0) {
|
||||||
|
return layer.msg('上传失败');
|
||||||
|
}
|
||||||
|
// 上传成功
|
||||||
|
$('#image').val(res.data); // 设置图片路径到隐藏输入框
|
||||||
|
$('#upload-text').html(''); // 置空上传失败的状态
|
||||||
|
layer.msg('上传成功', { icon: 1 });
|
||||||
|
},
|
||||||
|
uploadError: function () { // 这里改为 uploadError
|
||||||
|
// 演示失败状态,并实现重传
|
||||||
|
var demoText = $('#upload-text');
|
||||||
|
demoText.html('<span style="color: #FF5722;">上传失败</span> <a class="layui-btn layui-btn-xs demo-reload">重试</a>');
|
||||||
|
demoText.find('.demo-reload').on('click', function () {
|
||||||
|
uploadInst.upload();
|
||||||
|
});
|
||||||
|
},
|
||||||
|
// 进度条
|
||||||
|
progress: function (n, elem, e) {
|
||||||
|
element.progress('filter-demo', n + '%'); // 可配合 layui 进度条元素使用
|
||||||
|
if (n == 100) {
|
||||||
|
layer.msg('上传完毕', { icon: 1 });
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
// 获取分类列表
|
||||||
|
$.get('{:url("articles/getcate")}', function (res) {
|
||||||
|
if (res.code == 0) {
|
||||||
|
var html = '<option value="">请选择分类</option>';
|
||||||
|
res.data.forEach(function (item) {
|
||||||
|
// 如果cid为0,则设置为禁用
|
||||||
|
var disabled = item.cid == 0 ? 'disabled' : '';
|
||||||
|
var selected = item.id == '{$info.cate}' ? 'selected' : '';
|
||||||
|
html += '<option value="' + item.id + '" ' + disabled + ' ' + selected + '>' + item.name + '</option>';
|
||||||
|
// 如果有子分类,添加子分类选项
|
||||||
|
if (item.children && item.children.length > 0) {
|
||||||
|
item.children.forEach(function (child) {
|
||||||
|
var childSelected = child.id == '{$info.cate}' ? 'selected' : '';
|
||||||
|
html += '<option value="' + child.id + '" ' + childSelected + '>├─ ' + child.name + '</option>';
|
||||||
|
});
|
||||||
|
}
|
||||||
|
});
|
||||||
|
$('select[name="cate"]').html(html);
|
||||||
|
form.render('select');
|
||||||
|
} else {
|
||||||
|
layer.msg(res.msg, { icon: 2 });
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
// 表单提交
|
||||||
|
form.on('submit(formSubmit)', function (data) {
|
||||||
|
// 获取编辑器内容
|
||||||
|
var content = editor.getHtml();
|
||||||
|
if (!content || content === '<p><br></p>') {
|
||||||
|
layer.msg('请输入文章内容', { icon: 2 });
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
var loadIndex = layer.load(2);
|
||||||
|
data.field.content = content;
|
||||||
|
|
||||||
|
$.ajax({
|
||||||
|
url: '{:url("articles/edit")}?id={$info.id}',
|
||||||
|
type: 'POST',
|
||||||
|
data: data.field,
|
||||||
|
success: function (res) {
|
||||||
|
layer.close(loadIndex);
|
||||||
|
if (res.code == 0) {
|
||||||
|
layer.msg(res.msg, { icon: 1 });
|
||||||
|
setTimeout(function () {
|
||||||
|
window.location.href = '{:url("articles/articlelist")}';
|
||||||
|
}, 1000);
|
||||||
|
} else {
|
||||||
|
layer.msg(res.msg, { icon: 2 });
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
return false;
|
||||||
|
});
|
||||||
|
|
||||||
|
// 返回上一页
|
||||||
|
function goBack() {
|
||||||
|
var index = parent.layer.getFrameIndex(window.name);
|
||||||
|
parent.layer.close(index);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<!-- wangeditor编辑器脚本 -->
|
||||||
|
<script>
|
||||||
|
const { createEditor, createToolbar } = window.wangEditor
|
||||||
|
|
||||||
|
const editorConfig = {
|
||||||
|
MENU_CONF: {},
|
||||||
|
placeholder: '请输入内容...',
|
||||||
|
onChange(editor) {
|
||||||
|
const html = editor.getHtml()
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
// 配置图片上传
|
||||||
|
editorConfig.MENU_CONF['uploadImage'] = {
|
||||||
|
server: '{:url("index/upload_img")}',
|
||||||
|
fieldName: 'file',
|
||||||
|
maxFileSize: 10 * 1024 * 1024, // 10M
|
||||||
|
maxNumberOfFiles: 10,
|
||||||
|
allowedFileTypes: ['image/*'],
|
||||||
|
meta: {
|
||||||
|
token: 'xxx'
|
||||||
|
},
|
||||||
|
metaWithUrl: true,
|
||||||
|
headers: {
|
||||||
|
Accept: 'text/x-json'
|
||||||
|
},
|
||||||
|
timeout: 5 * 1000, // 5s
|
||||||
|
|
||||||
|
onBeforeUpload(file) {
|
||||||
|
console.log('准备上传图片', file)
|
||||||
|
return file
|
||||||
|
},
|
||||||
|
onProgress(progress) {
|
||||||
|
console.log('上传进度', progress)
|
||||||
|
},
|
||||||
|
onSuccess(file, res) {
|
||||||
|
console.log('上传成功', file, res)
|
||||||
|
},
|
||||||
|
onFailed(file, res) {
|
||||||
|
layer.msg('上传失败:' + res.msg, { icon: 2 })
|
||||||
|
console.log('上传失败', file, res)
|
||||||
|
},
|
||||||
|
onError(file, err, res) {
|
||||||
|
layer.msg('上传出错:' + err.message, { icon: 2 })
|
||||||
|
console.error('上传出错', file, err, res)
|
||||||
|
},
|
||||||
|
customInsert(res, insertFn) {
|
||||||
|
// res 即服务端的返回结果
|
||||||
|
if (res.code === 0 && res.data) {
|
||||||
|
// 从res.data中获取src字段
|
||||||
|
const url = String(res.data.src || '');
|
||||||
|
if (url) {
|
||||||
|
insertFn(url);
|
||||||
|
} else {
|
||||||
|
layer.msg('图片地址无效', { icon: 2 });
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
layer.msg('图片上传失败', { icon: 2 });
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const editor = createEditor({
|
||||||
|
selector: '#editor-container',
|
||||||
|
html: `{$info.content|raw|default=''}`,
|
||||||
|
config: editorConfig,
|
||||||
|
mode: 'default', // or 'simple'
|
||||||
|
})
|
||||||
|
|
||||||
|
const toolbarConfig = {}
|
||||||
|
|
||||||
|
const toolbar = createToolbar({
|
||||||
|
editor,
|
||||||
|
selector: '#toolbar-container',
|
||||||
|
config: toolbarConfig,
|
||||||
|
mode: 'default', // or 'simple'
|
||||||
|
})
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
//返回文章列表
|
||||||
|
function goBack() {
|
||||||
|
window.location.href = '{:url("articles/articlelist")}';
|
||||||
|
}
|
||||||
|
</script>
|
||||||
424
app/admin/view/index/index.php
Normal file
424
app/admin/view/index/index.php
Normal file
@ -0,0 +1,424 @@
|
|||||||
|
<!DOCTYPE html>
|
||||||
|
<html>
|
||||||
|
|
||||||
|
<head>
|
||||||
|
<meta charset="utf-8">
|
||||||
|
<title>{$config['web_title']}</title>
|
||||||
|
<meta name="renderer" content="webkit">
|
||||||
|
<meta http-equiv="X-UA-Compatible" content="IE=edge,chrome=1">
|
||||||
|
<meta name="viewport"
|
||||||
|
content="width=device-width, initial-scale=1.0, minimum-scale=1.0, maximum-scale=1.0, user-scalable=0">
|
||||||
|
<link rel="stylesheet" type="text/css" href="/static/third/layui/css/layui.css" media="all">
|
||||||
|
<link rel="stylesheet" type="text/css" href="/static/css/index.css" media="all">
|
||||||
|
<script type="text/javascript" src="/static/third/layui/layui.js"></script>
|
||||||
|
<script type="text/javascript" src="/static/js/admin.js"></script>
|
||||||
|
<style>
|
||||||
|
.layadmin-side-shrink .layui-layout-admin .layui-logo {
|
||||||
|
width: 60px;
|
||||||
|
background-image: url("/static/images/logob32.jpg");
|
||||||
|
}
|
||||||
|
.main-content {
|
||||||
|
height: 100%;
|
||||||
|
background: #fff;
|
||||||
|
}
|
||||||
|
#mainWorkspace {
|
||||||
|
height: 100%;
|
||||||
|
}
|
||||||
|
#mainTabs.layui-tab {
|
||||||
|
margin: 0;
|
||||||
|
height: 100%;
|
||||||
|
}
|
||||||
|
.layui-tab-content {
|
||||||
|
padding: 0;
|
||||||
|
height: calc(100% - 41px);
|
||||||
|
}
|
||||||
|
.layui-tab-item {
|
||||||
|
height: 100%;
|
||||||
|
}
|
||||||
|
.main-iframe {
|
||||||
|
width: 100%;
|
||||||
|
height: 100%;
|
||||||
|
border: none;
|
||||||
|
/* background-color: #f2f2f2; */
|
||||||
|
}
|
||||||
|
.layui-tab-title {
|
||||||
|
background: #fff;
|
||||||
|
box-shadow: 0 1px 2px 0 rgba(0,0,0,.1);
|
||||||
|
}
|
||||||
|
.layui-tab-title .layui-this:after {
|
||||||
|
border-bottom-color: #009688;
|
||||||
|
}
|
||||||
|
#LAY_app_body {
|
||||||
|
overflow: hidden;
|
||||||
|
}
|
||||||
|
.layui-tab-content .layui-tab-item {
|
||||||
|
position: relative;
|
||||||
|
}
|
||||||
|
</style>
|
||||||
|
</head>
|
||||||
|
|
||||||
|
<body layadmin-themealias="default" class="layui-layout-body">
|
||||||
|
<div id="LAY_app" class="layadmin-tabspage-none">
|
||||||
|
<div class="layui-layout layui-layout-admin">
|
||||||
|
<div class="layui-header">
|
||||||
|
<!-- 头部区域 -->
|
||||||
|
<div style="display: flex;align-items: center;height:70px;">
|
||||||
|
<ul class="layui-nav layui-layout-left">
|
||||||
|
<li class="layui-nav-item layadmin-flexible" lay-unselect onclick="shrink()">
|
||||||
|
<a href="javascript:;" layadmin-event="flexible" title="侧边伸缩">
|
||||||
|
<i class="layui-icon layui-icon-shrink-right" id="LAY_app_flexible"></i>
|
||||||
|
</a>
|
||||||
|
</li>
|
||||||
|
</ul>
|
||||||
|
<ul class="layui-nav layui-layout-right" lay-filter="layadmin-layout-right">
|
||||||
|
<li class="layui-nav-item layui-hide-xs" lay-unselect title="前端站点" onclick="gotoFront()">
|
||||||
|
<a href="javascript:;" layadmin-event="gotoFront">
|
||||||
|
<i class="layui-icon layui-icon-website"></i>
|
||||||
|
</a>
|
||||||
|
</li>
|
||||||
|
<li class="layui-nav-item layui-hide-xs" lay-unselect title="全屏" onclick="fullScreen()">
|
||||||
|
<a href="javascript:;" layadmin-event="fullscreen">
|
||||||
|
<i class="layui-icon layui-icon-screen-full"></i>
|
||||||
|
</a>
|
||||||
|
</li>
|
||||||
|
<li class="layui-nav-item" lay-unselect>
|
||||||
|
<a href="javascript:;">
|
||||||
|
<cite>{$aUser['name']}</cite>
|
||||||
|
</a>
|
||||||
|
<dl class="layui-nav-child">
|
||||||
|
<dd><a lay-href="" onclick="menuFire('yunzeradmin/admininfo',1)">个人中心</a></dd>
|
||||||
|
<hr>
|
||||||
|
<dd layadmin-event="logout" style="text-align:center;" onclick="logout()">
|
||||||
|
<a>退出</a>
|
||||||
|
</dd>
|
||||||
|
</dl>
|
||||||
|
</li>
|
||||||
|
</ul>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<!-- 侧边菜单 -->
|
||||||
|
<div class="layui-side layui-side-menu">
|
||||||
|
<div class="layui-side-scroll">
|
||||||
|
<!-- <div class="layui-logo" lay-href="" style="display: flex;align-items: center;">
|
||||||
|
<img src="/static/images/logo-l-w.png" alt="{$config['web_title']}"
|
||||||
|
style="max-width: 100%; max-height: 100%;">
|
||||||
|
</div> -->
|
||||||
|
<ul class="layui-nav layui-nav-tree" lay-shrink="all" id="LAY-system-side-menu"
|
||||||
|
lay-filter="layadmin-system-side-menu">
|
||||||
|
<!-- 固定的工作台菜单项 -->
|
||||||
|
<li class="layui-nav-item" data-name="index/welcome">
|
||||||
|
<a href="javascript:;" lay-tips="工作台" lay-direction="2"
|
||||||
|
onclick="menuFire('index/welcome',1)">
|
||||||
|
<i class="layui-icon layui-icon-home" style="margin-top: -20px;"></i>
|
||||||
|
<cite>工作台</cite>
|
||||||
|
</a>
|
||||||
|
</li>
|
||||||
|
<!-- 下面是原有的菜单循环 -->
|
||||||
|
{volist name="menu" id="vo"}
|
||||||
|
<li data-name="{$vo.src}" data-jump="" class="layui-nav-item">
|
||||||
|
<!-- 修改一级菜单的点击事件,只有当有src时才跳转 -->
|
||||||
|
<a href="javascript:;" lay-tips="{$vo['label']}" lay-direction="2" {if isset($vo['src']) &&
|
||||||
|
$vo['src']}onclick="menuFire('{$vo.src}',1)" {/if}>
|
||||||
|
<i class="layui-icon layui-icons {$vo['icon_class']}"></i>
|
||||||
|
<cite>{$vo['label']}</cite>
|
||||||
|
</a>
|
||||||
|
{if (isset($vo['children']) && $vo['children'])}
|
||||||
|
<dl class="layui-nav-child">
|
||||||
|
{volist name="vo.children" id="cvo"}
|
||||||
|
<dd data-name="" data-jump="/">
|
||||||
|
{if $cvo['type'] == 1}
|
||||||
|
<a href="javascript:;" onclick="menuFire('{$cvo.src}',1)">
|
||||||
|
<i class="layui-icon layui-icons {$cvo.icon_class}"></i>{$cvo.label}
|
||||||
|
</a>
|
||||||
|
{elseif $cvo['type'] == 2 /}
|
||||||
|
<a href="{$cvo.src}" target="_blank">
|
||||||
|
<i class="layui-icon layui-icons {$cvo.icon_class}"></i>{$cvo.label}
|
||||||
|
</a>
|
||||||
|
{/if}
|
||||||
|
</dd>
|
||||||
|
{/volist}
|
||||||
|
</dl>
|
||||||
|
{/if}
|
||||||
|
</li>
|
||||||
|
{/volist}
|
||||||
|
</ul>
|
||||||
|
<div style="position: absolute;bottom:20px;width:200px;display:flex;justify-content: center;">
|
||||||
|
<a style="color:#848484" href="https://www.yunzer.cn/">POWER BY 云泽网</a>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<!-- 主体内容 -->
|
||||||
|
<div class="layui-body" id="LAY_app_body">
|
||||||
|
<div class="main-content">
|
||||||
|
<!-- 默认工作台界面 -->
|
||||||
|
<div id="mainWorkspace">
|
||||||
|
<iframe src="{$config['admin_route']}index/welcome" class="main-iframe" frameborder="0" scrolling="0"></iframe>
|
||||||
|
</div>
|
||||||
|
<!-- 动态标签页 -->
|
||||||
|
<div id="mainTabs" class="layui-tab" lay-allowClose="true" lay-filter="mainTabs" style="margin-top: 10px;">
|
||||||
|
<ul class="layui-tab-title"></ul>
|
||||||
|
<div class="layui-tab-content"></div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<!-- 辅助元素,一般用于移动设备下遮罩 -->
|
||||||
|
<div class="layadmin-body-shade" layadmin-event="shade" onclick="shrink()"></div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<!-- 辅助元素,一般用于移动设备下遮罩 -->
|
||||||
|
<div class="layadmin-body-shade" layadmin-event="shade"></div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<script type="text/javascript">
|
||||||
|
layui.use(['element', 'layer', 'jquery'], function () {
|
||||||
|
var element = layui.element;
|
||||||
|
$ = layui.jquery;
|
||||||
|
layer = layui.layer;
|
||||||
|
setter = layui.setter;
|
||||||
|
|
||||||
|
// 保存标签状态
|
||||||
|
function saveTabState(layid, title, url) {
|
||||||
|
var tabState = {
|
||||||
|
layid: layid,
|
||||||
|
title: title,
|
||||||
|
url: url
|
||||||
|
};
|
||||||
|
sessionStorage.setItem('currentTab', JSON.stringify(tabState));
|
||||||
|
}
|
||||||
|
|
||||||
|
// 恢复标签状态
|
||||||
|
function restoreTabState() {
|
||||||
|
var tabState = sessionStorage.getItem('currentTab');
|
||||||
|
if (tabState) {
|
||||||
|
try {
|
||||||
|
var tab = JSON.parse(tabState);
|
||||||
|
if (tab.url && tab.title) {
|
||||||
|
addTab(tab.title, tab.url, tab.layid);
|
||||||
|
}
|
||||||
|
} catch (e) {
|
||||||
|
console.error('Failed to restore tab state:', e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// 在页面加载完成后执行
|
||||||
|
$(document).ready(function () {
|
||||||
|
// 获取URL参数
|
||||||
|
var urlParams = new URLSearchParams(window.location.search);
|
||||||
|
var page = urlParams.get('page');
|
||||||
|
|
||||||
|
if (page) {
|
||||||
|
// 如果有page参数,加载对应页面到iframe
|
||||||
|
$('iframe').attr('src', "{$config['admin_route']}" + page);
|
||||||
|
} else {
|
||||||
|
// 否则加载默认页面
|
||||||
|
$('iframe').attr('src', "{$config['admin_route']}index/welcome");
|
||||||
|
}
|
||||||
|
|
||||||
|
// 恢复标签状态
|
||||||
|
restoreTabState();
|
||||||
|
});
|
||||||
|
|
||||||
|
// 添加或切换到标签页
|
||||||
|
window.addTab = function(title, url, id) {
|
||||||
|
var element = layui.element;
|
||||||
|
var $ = layui.jquery;
|
||||||
|
var layid = id || url.replace(/\//g, '_');
|
||||||
|
|
||||||
|
// 如果是首页/工作台,直接显示主工作区
|
||||||
|
if(url.indexOf('index/welcome') > -1 || url.indexOf('welcome') > -1) {
|
||||||
|
$('#mainTabs').hide();
|
||||||
|
$('#mainWorkspace').show();
|
||||||
|
$('#mainWorkspace iframe').attr('src', url);
|
||||||
|
sessionStorage.removeItem('currentTab');
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// 显示标签区域
|
||||||
|
$('#mainWorkspace').hide();
|
||||||
|
$('#mainTabs').show();
|
||||||
|
|
||||||
|
// 如果标签已存在,直接切换
|
||||||
|
if($('.layui-tab-title li[lay-id="'+ layid +'"]').length > 0) {
|
||||||
|
element.tabChange('mainTabs', layid);
|
||||||
|
saveTabState(layid, title, url);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// 添加新标签
|
||||||
|
element.tabAdd('mainTabs', {
|
||||||
|
title: title,
|
||||||
|
content: '<iframe src="' + url + '" class="main-iframe" frameborder="0"></iframe>',
|
||||||
|
id: layid
|
||||||
|
});
|
||||||
|
|
||||||
|
// 切换到新标签页
|
||||||
|
element.tabChange('mainTabs', layid);
|
||||||
|
|
||||||
|
// 保存当前标签状态
|
||||||
|
saveTabState(layid, title, url);
|
||||||
|
};
|
||||||
|
|
||||||
|
// 监听标签切换事件
|
||||||
|
element.on('tab(mainTabs)', function(data){
|
||||||
|
var layid = $(this).attr('lay-id');
|
||||||
|
// 确保当前标签页内容可见
|
||||||
|
$('.layui-tab-content .layui-tab-item').eq(data.index).addClass('layui-show')
|
||||||
|
.siblings().removeClass('layui-show');
|
||||||
|
|
||||||
|
// 更新保存的标签状态
|
||||||
|
var title = $(this).text();
|
||||||
|
var url = $('.layui-tab-content .layui-tab-item').eq(data.index).find('iframe').attr('src');
|
||||||
|
saveTabState(layid, title, url);
|
||||||
|
});
|
||||||
|
|
||||||
|
// 监听标签删除事件
|
||||||
|
element.on('tabDelete(mainTabs)', function(data){
|
||||||
|
// 如果没有标签了,显示工作台
|
||||||
|
if($('.layui-tab-title li').length === 0) {
|
||||||
|
$('#mainTabs').hide();
|
||||||
|
$('#mainWorkspace').show();
|
||||||
|
sessionStorage.removeItem('currentTab');
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
// 左侧菜单点击事件
|
||||||
|
$('.left-nav #nav li').click(function (event) {
|
||||||
|
if ($(this).children('.sub-menu').length) {
|
||||||
|
if ($(this).hasClass('open')) {
|
||||||
|
$(this).removeClass('open');
|
||||||
|
$(this).find('.nav_right').html('');
|
||||||
|
$(this).children('.sub-menu').stop().slideUp();
|
||||||
|
$(this).siblings().children('.sub-menu').slideUp();
|
||||||
|
} else {
|
||||||
|
$(this).addClass('open');
|
||||||
|
$(this).children('a').find('.nav_right').html('');
|
||||||
|
$(this).children('.sub-menu').stop().slideDown();
|
||||||
|
$(this).siblings().children('.sub-menu').stop().slideUp();
|
||||||
|
$(this).siblings().find('.nav_right').html('');
|
||||||
|
$(this).siblings().removeClass('open');
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
var url = $(this).children('a').attr('_href');
|
||||||
|
var title = $(this).children('a').html();
|
||||||
|
title = title.replace(/<[^>]+>/g, "").trim(); // 移除HTML标签
|
||||||
|
|
||||||
|
if (url) {
|
||||||
|
window.addTab(title, url);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
event.stopPropagation();
|
||||||
|
})
|
||||||
|
|
||||||
|
// 修改resetMainHeight函数
|
||||||
|
function resetMainHeight(iframe) {
|
||||||
|
if (!iframe) return;
|
||||||
|
try {
|
||||||
|
// 获取视口高度
|
||||||
|
var clientHeight = document.documentElement.clientHeight;
|
||||||
|
// 计算iframe应该的高度(减去头部和tab标签的高度)
|
||||||
|
var iframeHeight = clientHeight - 60 - 40; // 60px是头部高度,40px是tab标签高度
|
||||||
|
$(iframe).css({
|
||||||
|
'height': iframeHeight + 'px',
|
||||||
|
'width': '100%',
|
||||||
|
'display': 'block'
|
||||||
|
});
|
||||||
|
} catch(e) {
|
||||||
|
console.error('Reset iframe height failed:', e);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// 在窗口大小改变时重置所有iframe高度
|
||||||
|
$(window).on('resize', function() {
|
||||||
|
$('.main-iframe').each(function() {
|
||||||
|
resetMainHeight(this);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
// 菜单点击
|
||||||
|
function menuFire(obj, num) {
|
||||||
|
if (num == 1) {
|
||||||
|
var title = '';
|
||||||
|
// 获取菜单标题
|
||||||
|
$('.layui-nav-item a').each(function() {
|
||||||
|
if($(this).attr('onclick') && $(this).attr('onclick').indexOf(obj) > -1) {
|
||||||
|
title = $(this).text().trim();
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
// 如果没找到标题,使用子菜单查找
|
||||||
|
if(!title) {
|
||||||
|
$('.layui-nav-child a').each(function() {
|
||||||
|
if($(this).attr('onclick') && $(this).attr('onclick').indexOf(obj) > -1) {
|
||||||
|
title = $(this).text().trim();
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
// 添加或切换到标签页
|
||||||
|
window.addTab(title || '新页面', "{$config['admin_route']}" + obj, obj.replace(/\//g, '_'));
|
||||||
|
|
||||||
|
// 更新浏览器URL,但保持在index页面
|
||||||
|
window.history.pushState({}, '', "{$config['admin_route']}index/index?page=" + obj);
|
||||||
|
|
||||||
|
// 如果是子菜单,确保父菜单展开
|
||||||
|
if (obj.indexOf('/') > -1) {
|
||||||
|
var parentMenu = obj.split('/')[0];
|
||||||
|
$('.layui-nav-item').each(function () {
|
||||||
|
var menuSrc = $(this).find('a').attr('onclick');
|
||||||
|
if (menuSrc && menuSrc.indexOf(parentMenu) > -1) {
|
||||||
|
$(this).addClass('layui-nav-itemed');
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
var width = screen();
|
||||||
|
if (width < 2) {
|
||||||
|
shrink();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// 监听浏览器前进后退按钮
|
||||||
|
window.addEventListener('popstate', function (event) {
|
||||||
|
// 获取URL参数
|
||||||
|
var urlParams = new URLSearchParams(window.location.search);
|
||||||
|
var page = urlParams.get('page');
|
||||||
|
|
||||||
|
if (page) {
|
||||||
|
// 如果有page参数,通过标签系统加载页面
|
||||||
|
var title = '新页面';
|
||||||
|
window.addTab(title, "{$config['admin_route']}" + page, page.replace(/\//g, '_'));
|
||||||
|
} else {
|
||||||
|
// 否则加载默认页面
|
||||||
|
element.tabChange('mainTabs', 'welcome');
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
// 退出
|
||||||
|
function logout() {
|
||||||
|
layer.confirm('确定要退出吗?', {
|
||||||
|
icon: 3,
|
||||||
|
btn: ['确定', '取消']
|
||||||
|
}, function () {
|
||||||
|
$.get("{$config['admin_route']}login/logout", function (res) {
|
||||||
|
if (res.code > 0) {
|
||||||
|
layer.msg(res.msg, { 'icon': 2 });
|
||||||
|
} else {
|
||||||
|
layer.msg(res.msg, { 'icon': 1 });
|
||||||
|
setTimeout(function () { window.location.href = "{$config['admin_route']}login/index"; }, 1000);
|
||||||
|
}
|
||||||
|
}, 'json');
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
//跳转前端站点
|
||||||
|
function gotoFront() {
|
||||||
|
window.open("//{$config['admin_domain']}", "_blank");
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
</body>
|
||||||
|
|
||||||
|
</html>
|
||||||
725
app/admin/view/index/welcome.php
Normal file
725
app/admin/view/index/welcome.php
Normal file
@ -0,0 +1,725 @@
|
|||||||
|
{include file="public/header" /}
|
||||||
|
<div class="dashboard-container">
|
||||||
|
<div class="welcome-header">
|
||||||
|
<h1>欢迎使用{$config['web_title']}</h1>
|
||||||
|
<p>今天是 <span id="current-time"></span>,祝您工作愉快</p>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="stats-container">
|
||||||
|
<div class="stat-card">
|
||||||
|
<div class="stat-title">用户总数</div>
|
||||||
|
<div class="stat-value">{$todayStats.total_users|number_format}</div>
|
||||||
|
<div class="stat-icon">👥</div>
|
||||||
|
</div>
|
||||||
|
<div class="stat-card">
|
||||||
|
<div class="stat-title">今日访问</div>
|
||||||
|
<div class="stat-value">{$todayStats.daily_visits|number_format}</div>
|
||||||
|
<div class="stat-icon">📊</div>
|
||||||
|
</div>
|
||||||
|
<div class="stat-card">
|
||||||
|
<div class="stat-title">文章总数</div>
|
||||||
|
<div class="stat-value">{$todayStats.total_articles|number_format}</div>
|
||||||
|
<div class="stat-icon">📝</div>
|
||||||
|
</div>
|
||||||
|
<!-- <div class="stat-card">
|
||||||
|
<div class="stat-title">资源总数</div>
|
||||||
|
<div class="stat-value">{$todayStats.total_resources|number_format}</div>
|
||||||
|
<div class="stat-icon">📦</div>
|
||||||
|
</div> -->
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="quick-actions">
|
||||||
|
<h2>快捷操作</h2>
|
||||||
|
<div class="action-buttons">
|
||||||
|
<a href="{:url('user/index')}" class="action-button">
|
||||||
|
<i class="fas fa-users"></i>用户管理
|
||||||
|
</a>
|
||||||
|
<a href="{:url('content/publish')}" class="action-button">
|
||||||
|
<i class="fas fa-edit"></i>内容发布
|
||||||
|
</a>
|
||||||
|
<a href="{:url('statistics/index')}" class="action-button">
|
||||||
|
<i class="fas fa-chart-bar"></i>数据统计
|
||||||
|
</a>
|
||||||
|
<a href="{:url('system/settings')}" class="action-button">
|
||||||
|
<i class="fas fa-cog"></i>系统设置
|
||||||
|
</a>
|
||||||
|
<a href="{:url('system/clear_cache')}" class="action-button">
|
||||||
|
<i class="fas fa-broom"></i>清除缓存
|
||||||
|
</a>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="recent-activity">
|
||||||
|
<h2>最近动态</h2>
|
||||||
|
<div class="activity-list">
|
||||||
|
{volist name="recentActivities" id="activity"}
|
||||||
|
<div class="activity-item">
|
||||||
|
<div class="activity-icon">{$activity.icon|default='📌'}</div>
|
||||||
|
<div class="activity-content">
|
||||||
|
<div class="activity-title">{$activity.content}</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
{/volist}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="charts-container">
|
||||||
|
<div class="chart-card">
|
||||||
|
<h2>访问趋势</h2>
|
||||||
|
<div id="visitTrend" class="chart-container"></div>
|
||||||
|
</div>
|
||||||
|
<div class="chart-card">
|
||||||
|
<h2>用户增长</h2>
|
||||||
|
<div id="userGrowth" class="chart-container"></div>
|
||||||
|
</div>
|
||||||
|
<!-- <div class="chart-card">
|
||||||
|
<h2>资源统计</h2>
|
||||||
|
<div id="resourceStats" class="chart-container"></div>
|
||||||
|
</div> -->
|
||||||
|
<div class="chart-card">
|
||||||
|
<h2>文章统计</h2>
|
||||||
|
<div id="articleStats" class="chart-container"></div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
<script src="__JS__/echarts.min.js"></script>
|
||||||
|
<script>
|
||||||
|
function updateTime() {
|
||||||
|
var now = new Date();
|
||||||
|
var year = now.getFullYear();
|
||||||
|
var month = now.getMonth() + 1;
|
||||||
|
var date = now.getDate();
|
||||||
|
var hours = now.getHours();
|
||||||
|
var minutes = now.getMinutes();
|
||||||
|
var seconds = now.getSeconds();
|
||||||
|
|
||||||
|
var padZero = function(num) {
|
||||||
|
return num < 10 ? '0' + num : num;
|
||||||
|
};
|
||||||
|
|
||||||
|
var timeString = year + '年' +
|
||||||
|
padZero(month) + '月' +
|
||||||
|
padZero(date) + '日 ' +
|
||||||
|
padZero(hours) + ':' +
|
||||||
|
padZero(minutes) + ':' +
|
||||||
|
padZero(seconds);
|
||||||
|
|
||||||
|
document.getElementById('current-time').innerHTML = timeString;
|
||||||
|
}
|
||||||
|
|
||||||
|
// 获取用户统计数据
|
||||||
|
function getUserCounts() {
|
||||||
|
fetch('{:url("users/counts")}')
|
||||||
|
.then(response => response.json())
|
||||||
|
.then(res => {
|
||||||
|
// console.log('用户统计接口返回数据:', res);
|
||||||
|
if (res.code === 0 && res.data) {
|
||||||
|
// 更新用户总数
|
||||||
|
document.querySelector('.stat-card:nth-child(1) .stat-value').textContent = res.data.total.toLocaleString();
|
||||||
|
|
||||||
|
// 更新用户增长图表
|
||||||
|
if (window.userChart) {
|
||||||
|
window.userChart.setOption({
|
||||||
|
xAxis: {
|
||||||
|
data: res.data.dates
|
||||||
|
},
|
||||||
|
series: [{
|
||||||
|
name: '新增用户',
|
||||||
|
data: res.data.counts
|
||||||
|
}, {
|
||||||
|
name: '总用户数',
|
||||||
|
data: res.data.totalCounts
|
||||||
|
}]
|
||||||
|
});
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
console.warn('用户统计接口返回异常:', res);
|
||||||
|
document.querySelector('.stat-card:nth-child(1) .stat-value').textContent = '0';
|
||||||
|
}
|
||||||
|
})
|
||||||
|
.catch(error => {
|
||||||
|
console.error('获取用户统计失败:', error);
|
||||||
|
document.querySelector('.stat-card:nth-child(1) .stat-value').textContent = '0';
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
// 获取文章统计数据
|
||||||
|
function getArticleCounts() {
|
||||||
|
fetch('{:url("articles/counts")}')
|
||||||
|
.then(response => response.json())
|
||||||
|
.then(res => {
|
||||||
|
// console.log('文章统计接口返回数据:', res);
|
||||||
|
if (res.code === 0 && res.data) {
|
||||||
|
// 更新文章总数
|
||||||
|
document.querySelector('.stat-card:nth-child(3) .stat-value').textContent = res.data.total.toLocaleString();
|
||||||
|
|
||||||
|
// 更新文章统计图表
|
||||||
|
if (window.articleChart) {
|
||||||
|
window.articleChart.setOption({
|
||||||
|
xAxis: {
|
||||||
|
data: res.data.dates
|
||||||
|
},
|
||||||
|
series: [{
|
||||||
|
name: '新增文章',
|
||||||
|
data: res.data.counts
|
||||||
|
}, {
|
||||||
|
name: '总文章数',
|
||||||
|
data: res.data.totalCounts
|
||||||
|
}]
|
||||||
|
});
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
console.warn('文章统计接口返回异常:', res);
|
||||||
|
document.querySelector('.stat-card:nth-child(3) .stat-value').textContent = '0';
|
||||||
|
}
|
||||||
|
})
|
||||||
|
.catch(error => {
|
||||||
|
console.error('获取文章统计失败:', error);
|
||||||
|
document.querySelector('.stat-card:nth-child(3) .stat-value').textContent = '0';
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
// // 获取资源统计数据
|
||||||
|
// function getResourcesCounts() {
|
||||||
|
// fetch('{:url("resources/counts")}')
|
||||||
|
// .then(response => response.json())
|
||||||
|
// .then(res => {
|
||||||
|
// // console.log('资源统计接口返回数据:', res);
|
||||||
|
// if (res.code === 0 && res.data) {
|
||||||
|
// // 更新资源总数
|
||||||
|
// document.querySelector('.stat-card:nth-child(4) .stat-value').textContent = res.data.total.toLocaleString();
|
||||||
|
|
||||||
|
// // 更新资源统计图表
|
||||||
|
// if (window.resourceChart) {
|
||||||
|
// window.resourceChart.setOption({
|
||||||
|
// xAxis: {
|
||||||
|
// data: res.data.dates
|
||||||
|
// },
|
||||||
|
// series: [{
|
||||||
|
// name: '新增资源',
|
||||||
|
// data: res.data.counts
|
||||||
|
// }, {
|
||||||
|
// name: '总资源数',
|
||||||
|
// data: res.data.totalCounts
|
||||||
|
// }]
|
||||||
|
// });
|
||||||
|
// }
|
||||||
|
// } else {
|
||||||
|
// console.warn('资源统计接口返回异常:', res);
|
||||||
|
// document.querySelector('.stat-card:nth-child(4) .stat-value').textContent = '0';
|
||||||
|
// }
|
||||||
|
// })
|
||||||
|
// .catch(error => {
|
||||||
|
// console.error('获取资源统计失败:', error);
|
||||||
|
// document.querySelector('.stat-card:nth-child(4) .stat-value').textContent = '0';
|
||||||
|
// });
|
||||||
|
// }
|
||||||
|
|
||||||
|
updateTime();
|
||||||
|
setInterval(updateTime, 1000);
|
||||||
|
|
||||||
|
// 页面加载完成后获取统计数据
|
||||||
|
document.addEventListener('DOMContentLoaded', function() {
|
||||||
|
getUserCounts();
|
||||||
|
getArticleCounts();
|
||||||
|
// getResourcesCounts();
|
||||||
|
});
|
||||||
|
|
||||||
|
// 访问趋势图表
|
||||||
|
function initVisitTrend() {
|
||||||
|
var chart = echarts.init(document.getElementById('visitTrend'));
|
||||||
|
var option = {
|
||||||
|
tooltip: {
|
||||||
|
trigger: 'axis',
|
||||||
|
axisPointer: {
|
||||||
|
type: 'shadow'
|
||||||
|
}
|
||||||
|
},
|
||||||
|
legend: {
|
||||||
|
data: ['访问量', '独立访客']
|
||||||
|
},
|
||||||
|
grid: {
|
||||||
|
left: '3%',
|
||||||
|
right: '4%',
|
||||||
|
bottom: '3%',
|
||||||
|
containLabel: true
|
||||||
|
},
|
||||||
|
xAxis: {
|
||||||
|
type: 'category',
|
||||||
|
data: {$chartData.visitTrend.dates|json_encode|raw},
|
||||||
|
axisLine: {
|
||||||
|
lineStyle: {
|
||||||
|
color: '#e2e8f0'
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
yAxis: {
|
||||||
|
type: 'value',
|
||||||
|
axisLine: {
|
||||||
|
lineStyle: {
|
||||||
|
color: '#e2e8f0'
|
||||||
|
}
|
||||||
|
},
|
||||||
|
splitLine: {
|
||||||
|
lineStyle: {
|
||||||
|
color: '#f1f5f9'
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
series: [{
|
||||||
|
name: '访问量',
|
||||||
|
data: {$chartData.visitTrend.visits|json_encode|raw},
|
||||||
|
type: 'line',
|
||||||
|
smooth: true,
|
||||||
|
areaStyle: {
|
||||||
|
opacity: 0.1
|
||||||
|
},
|
||||||
|
itemStyle: {
|
||||||
|
color: '#3881fd'
|
||||||
|
},
|
||||||
|
lineStyle: {
|
||||||
|
width: 3
|
||||||
|
}
|
||||||
|
}, {
|
||||||
|
name: '独立访客',
|
||||||
|
data: {$chartData.visitTrend.uvs|json_encode|raw},
|
||||||
|
type: 'line',
|
||||||
|
smooth: true,
|
||||||
|
itemStyle: {
|
||||||
|
color: '#10b981'
|
||||||
|
},
|
||||||
|
lineStyle: {
|
||||||
|
width: 3
|
||||||
|
}
|
||||||
|
}]
|
||||||
|
};
|
||||||
|
chart.setOption(option);
|
||||||
|
}
|
||||||
|
|
||||||
|
// 用户增长图表
|
||||||
|
function initUserGrowth() {
|
||||||
|
var chart = echarts.init(document.getElementById('userGrowth'));
|
||||||
|
var option = {
|
||||||
|
tooltip: {
|
||||||
|
trigger: 'axis',
|
||||||
|
axisPointer: {
|
||||||
|
type: 'cross',
|
||||||
|
label: {
|
||||||
|
backgroundColor: '#6a7985'
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
legend: {
|
||||||
|
data: ['新增用户', '总用户数']
|
||||||
|
},
|
||||||
|
grid: {
|
||||||
|
left: '3%',
|
||||||
|
right: '4%',
|
||||||
|
bottom: '3%',
|
||||||
|
containLabel: true
|
||||||
|
},
|
||||||
|
xAxis: {
|
||||||
|
type: 'category',
|
||||||
|
boundaryGap: false,
|
||||||
|
data: {$chartData.userGrowth.dates|json_encode|raw}
|
||||||
|
},
|
||||||
|
yAxis: {
|
||||||
|
type: 'value'
|
||||||
|
},
|
||||||
|
series: [
|
||||||
|
{
|
||||||
|
name: '新增用户',
|
||||||
|
type: 'bar',
|
||||||
|
data: {$chartData.userGrowth.newUsers|json_encode|raw},
|
||||||
|
itemStyle: {
|
||||||
|
color: '#3881fd'
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: '总用户数',
|
||||||
|
type: 'line',
|
||||||
|
smooth: true,
|
||||||
|
data: {$chartData.userGrowth.totalUsers|json_encode|raw},
|
||||||
|
itemStyle: {
|
||||||
|
color: '#10b981'
|
||||||
|
},
|
||||||
|
lineStyle: {
|
||||||
|
width: 3
|
||||||
|
}
|
||||||
|
}
|
||||||
|
]
|
||||||
|
};
|
||||||
|
chart.setOption(option);
|
||||||
|
}
|
||||||
|
|
||||||
|
// 资源统计图表
|
||||||
|
function initResourceStats() {
|
||||||
|
var chart = echarts.init(document.getElementById('resourceStats'));
|
||||||
|
var option = {
|
||||||
|
tooltip: {
|
||||||
|
trigger: 'axis',
|
||||||
|
axisPointer: {
|
||||||
|
type: 'cross',
|
||||||
|
label: {
|
||||||
|
backgroundColor: '#6a7985'
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
legend: {
|
||||||
|
data: ['新增资源', '总资源数', '下载量']
|
||||||
|
},
|
||||||
|
grid: {
|
||||||
|
left: '3%',
|
||||||
|
right: '4%',
|
||||||
|
bottom: '3%',
|
||||||
|
containLabel: true
|
||||||
|
},
|
||||||
|
xAxis: {
|
||||||
|
type: 'category',
|
||||||
|
boundaryGap: false,
|
||||||
|
data: {$chartData.resourceStats.dates|json_encode|raw}
|
||||||
|
},
|
||||||
|
yAxis: {
|
||||||
|
type: 'value'
|
||||||
|
},
|
||||||
|
series: [
|
||||||
|
{
|
||||||
|
name: '新增资源',
|
||||||
|
type: 'bar',
|
||||||
|
data: {$chartData.resourceStats.newResources|json_encode|raw},
|
||||||
|
itemStyle: {
|
||||||
|
color: '#3881fd'
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: '总资源数',
|
||||||
|
type: 'line',
|
||||||
|
smooth: true,
|
||||||
|
data: {$chartData.resourceStats.totalResources|json_encode|raw},
|
||||||
|
itemStyle: {
|
||||||
|
color: '#10b981'
|
||||||
|
},
|
||||||
|
lineStyle: {
|
||||||
|
width: 3
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: '下载量',
|
||||||
|
type: 'line',
|
||||||
|
smooth: true,
|
||||||
|
data: {$chartData.resourceStats.downloads|json_encode|raw},
|
||||||
|
itemStyle: {
|
||||||
|
color: '#f59e0b'
|
||||||
|
},
|
||||||
|
lineStyle: {
|
||||||
|
width: 3
|
||||||
|
}
|
||||||
|
}
|
||||||
|
]
|
||||||
|
};
|
||||||
|
chart.setOption(option);
|
||||||
|
}
|
||||||
|
|
||||||
|
// 文章统计图表
|
||||||
|
function initArticleStats() {
|
||||||
|
var chart = echarts.init(document.getElementById('articleStats'));
|
||||||
|
var option = {
|
||||||
|
tooltip: {
|
||||||
|
trigger: 'axis',
|
||||||
|
axisPointer: {
|
||||||
|
type: 'cross',
|
||||||
|
label: {
|
||||||
|
backgroundColor: '#6a7985'
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
legend: {
|
||||||
|
data: ['新增文章', '总文章数', '浏览量']
|
||||||
|
},
|
||||||
|
grid: {
|
||||||
|
left: '3%',
|
||||||
|
right: '4%',
|
||||||
|
bottom: '3%',
|
||||||
|
containLabel: true
|
||||||
|
},
|
||||||
|
xAxis: {
|
||||||
|
type: 'category',
|
||||||
|
boundaryGap: false,
|
||||||
|
data: {$chartData.articleStats.dates|json_encode|raw}
|
||||||
|
},
|
||||||
|
yAxis: {
|
||||||
|
type: 'value'
|
||||||
|
},
|
||||||
|
series: [
|
||||||
|
{
|
||||||
|
name: '新增文章',
|
||||||
|
type: 'bar',
|
||||||
|
data: {$chartData.articleStats.newArticles|json_encode|raw},
|
||||||
|
itemStyle: {
|
||||||
|
color: '#3881fd'
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: '总文章数',
|
||||||
|
type: 'line',
|
||||||
|
smooth: true,
|
||||||
|
data: {$chartData.articleStats.totalArticles|json_encode|raw},
|
||||||
|
itemStyle: {
|
||||||
|
color: '#10b981'
|
||||||
|
},
|
||||||
|
lineStyle: {
|
||||||
|
width: 3
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: '浏览量',
|
||||||
|
type: 'line',
|
||||||
|
smooth: true,
|
||||||
|
data: {$chartData.articleStats.views|json_encode|raw},
|
||||||
|
itemStyle: {
|
||||||
|
color: '#f59e0b'
|
||||||
|
},
|
||||||
|
lineStyle: {
|
||||||
|
width: 3
|
||||||
|
}
|
||||||
|
}
|
||||||
|
]
|
||||||
|
};
|
||||||
|
chart.setOption(option);
|
||||||
|
}
|
||||||
|
|
||||||
|
// 初始化所有图表
|
||||||
|
document.addEventListener('DOMContentLoaded', function() {
|
||||||
|
// 确保ECharts已加载
|
||||||
|
if (typeof echarts === 'undefined') {
|
||||||
|
console.error('ECharts未加载');
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// 初始化图表
|
||||||
|
try {
|
||||||
|
initVisitTrend();
|
||||||
|
initUserGrowth();
|
||||||
|
// initResourceStats();
|
||||||
|
initArticleStats();
|
||||||
|
|
||||||
|
// 监听窗口大小变化,重绘图表
|
||||||
|
window.addEventListener('resize', function() {
|
||||||
|
var charts = document.querySelectorAll('.chart-container');
|
||||||
|
charts.forEach(function(chart) {
|
||||||
|
var instance = echarts.getInstanceByDom(chart);
|
||||||
|
if (instance) {
|
||||||
|
instance.resize();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
} catch (error) {
|
||||||
|
console.error('初始化图表失败:', error);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
</script><script src="__STATIC__/js/jquery.min.js"></script>
|
||||||
|
<style>
|
||||||
|
.dashboard-container {
|
||||||
|
padding: 24px;
|
||||||
|
font-family: -apple-system, BlinkMacSystemFont, 'Segoe UI', Roboto, 'Helvetica Neue', Arial, sans-serif;
|
||||||
|
/* background-color: #f5f7fa; */
|
||||||
|
/* min-height: calc(100vh - 60px); */
|
||||||
|
}
|
||||||
|
.welcome-header {
|
||||||
|
background: linear-gradient(135deg, #3881fd 0%, #2c5fd9 100%);
|
||||||
|
border-radius: 12px;
|
||||||
|
padding: 30px;
|
||||||
|
color: white;
|
||||||
|
margin-bottom: 24px;
|
||||||
|
box-shadow: 0 4px 20px rgba(56, 129, 253, 0.15);
|
||||||
|
}
|
||||||
|
.welcome-header h1 {
|
||||||
|
font-size: 28px;
|
||||||
|
font-weight: 600;
|
||||||
|
margin-bottom: 8px;
|
||||||
|
}
|
||||||
|
.welcome-header p {
|
||||||
|
font-size: 15px;
|
||||||
|
opacity: 0.9;
|
||||||
|
margin: 0;
|
||||||
|
}
|
||||||
|
.stats-container {
|
||||||
|
display: grid;
|
||||||
|
grid-template-columns: repeat(auto-fit, minmax(240px, 1fr));
|
||||||
|
gap: 24px;
|
||||||
|
margin-bottom: 24px;
|
||||||
|
}
|
||||||
|
.stat-card {
|
||||||
|
background: white;
|
||||||
|
border-radius: 12px;
|
||||||
|
padding: 24px;
|
||||||
|
box-shadow: 0 2px 12px rgba(0, 0, 0, 0.04);
|
||||||
|
transition: all 0.3s ease;
|
||||||
|
position: relative;
|
||||||
|
overflow: hidden;
|
||||||
|
}
|
||||||
|
.stat-card::before {
|
||||||
|
content: '';
|
||||||
|
position: absolute;
|
||||||
|
top: 0;
|
||||||
|
left: 0;
|
||||||
|
width: 4px;
|
||||||
|
height: 100%;
|
||||||
|
background: #3881fd;
|
||||||
|
}
|
||||||
|
.stat-card:hover {
|
||||||
|
transform: translateY(-5px);
|
||||||
|
box-shadow: 0 8px 24px rgba(0, 0, 0, 0.08);
|
||||||
|
}
|
||||||
|
.stat-card .stat-value {
|
||||||
|
font-size: 32px;
|
||||||
|
font-weight: 600;
|
||||||
|
color: #2c3e50;
|
||||||
|
margin: 12px 0;
|
||||||
|
display: flex;
|
||||||
|
align-items: baseline;
|
||||||
|
}
|
||||||
|
.stat-card .stat-title {
|
||||||
|
color: #64748b;
|
||||||
|
font-size: 14px;
|
||||||
|
font-weight: 500;
|
||||||
|
text-transform: uppercase;
|
||||||
|
letter-spacing: 0.5px;
|
||||||
|
}
|
||||||
|
.stat-card .stat-icon {
|
||||||
|
position: absolute;
|
||||||
|
right: 20px;
|
||||||
|
top: 50%;
|
||||||
|
transform: translateY(-50%);
|
||||||
|
font-size: 48px;
|
||||||
|
opacity: 0.1;
|
||||||
|
}
|
||||||
|
.quick-actions {
|
||||||
|
background: white;
|
||||||
|
border-radius: 12px;
|
||||||
|
padding: 24px;
|
||||||
|
box-shadow: 0 2px 12px rgba(0, 0, 0, 0.04);
|
||||||
|
}
|
||||||
|
.quick-actions h2 {
|
||||||
|
color: #1e293b;
|
||||||
|
font-size: 18px;
|
||||||
|
font-weight: 600;
|
||||||
|
margin-bottom: 20px;
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
}
|
||||||
|
.quick-actions h2::before {
|
||||||
|
content: '';
|
||||||
|
display: inline-block;
|
||||||
|
width: 4px;
|
||||||
|
height: 18px;
|
||||||
|
background: #3881fd;
|
||||||
|
margin-right: 8px;
|
||||||
|
border-radius: 2px;
|
||||||
|
}
|
||||||
|
.action-buttons {
|
||||||
|
display: grid;
|
||||||
|
grid-template-columns: repeat(auto-fill, minmax(160px, 1fr));
|
||||||
|
gap: 16px;
|
||||||
|
}
|
||||||
|
.action-button {
|
||||||
|
background: #f8fafc;
|
||||||
|
border: 1px solid #e2e8f0;
|
||||||
|
border-radius: 8px;
|
||||||
|
padding: 12px 16px;
|
||||||
|
color: #1e293b;
|
||||||
|
font-weight: 500;
|
||||||
|
cursor: pointer;
|
||||||
|
transition: all 0.2s ease;
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
text-decoration: none;
|
||||||
|
}
|
||||||
|
.action-button:hover {
|
||||||
|
background: #3881fd;
|
||||||
|
color: white;
|
||||||
|
border-color: #3881fd;
|
||||||
|
transform: translateY(-2px);
|
||||||
|
}
|
||||||
|
.action-button i {
|
||||||
|
margin-right: 8px;
|
||||||
|
}
|
||||||
|
.recent-activity {
|
||||||
|
margin-top: 24px;
|
||||||
|
background: white;
|
||||||
|
border-radius: 12px;
|
||||||
|
padding: 24px;
|
||||||
|
box-shadow: 0 2px 12px rgba(0, 0, 0, 0.04);
|
||||||
|
}
|
||||||
|
.activity-list {
|
||||||
|
margin-top: 16px;
|
||||||
|
}
|
||||||
|
.activity-item {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
padding: 12px 0;
|
||||||
|
border-bottom: 1px solid #f1f5f9;
|
||||||
|
}
|
||||||
|
.activity-item:last-child {
|
||||||
|
border-bottom: none;
|
||||||
|
}
|
||||||
|
.activity-icon {
|
||||||
|
width: 36px;
|
||||||
|
height: 36px;
|
||||||
|
border-radius: 8px;
|
||||||
|
background: #f1f5f9;
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
margin-right: 12px;
|
||||||
|
}
|
||||||
|
.activity-content {
|
||||||
|
flex: 1;
|
||||||
|
}
|
||||||
|
.activity-title {
|
||||||
|
font-weight: 500;
|
||||||
|
color: #9b9b9b;
|
||||||
|
}
|
||||||
|
.activity-time {
|
||||||
|
font-size: 12px;
|
||||||
|
color: #64748b;
|
||||||
|
}
|
||||||
|
.charts-container {
|
||||||
|
display: grid;
|
||||||
|
grid-template-columns: repeat(auto-fit, minmax(400px, 1fr));
|
||||||
|
gap: 24px;
|
||||||
|
margin-top: 24px;
|
||||||
|
}
|
||||||
|
.chart-card {
|
||||||
|
background: white;
|
||||||
|
border-radius: 12px;
|
||||||
|
padding: 24px;
|
||||||
|
box-shadow: 0 2px 12px rgba(0, 0, 0, 0.04);
|
||||||
|
}
|
||||||
|
.chart-card h2 {
|
||||||
|
color: #1e293b;
|
||||||
|
font-size: 18px;
|
||||||
|
font-weight: 600;
|
||||||
|
margin-bottom: 20px;
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
}
|
||||||
|
.chart-card h2::before {
|
||||||
|
content: '';
|
||||||
|
display: inline-block;
|
||||||
|
width: 4px;
|
||||||
|
height: 18px;
|
||||||
|
background: #3881fd;
|
||||||
|
margin-right: 8px;
|
||||||
|
border-radius: 2px;
|
||||||
|
}
|
||||||
|
.chart-container {
|
||||||
|
height: 300px;
|
||||||
|
width: 100%;
|
||||||
|
}
|
||||||
|
</style>
|
||||||
|
|
||||||
|
{include file="public/tail" /}
|
||||||
117
app/admin/view/log/login.php
Normal file
117
app/admin/view/log/login.php
Normal file
@ -0,0 +1,117 @@
|
|||||||
|
{include file="public/header" /}
|
||||||
|
<div class="layui-card">
|
||||||
|
<div class="layui-card-header">
|
||||||
|
<span class="layui-badge layui-bg-blue">登录日志</span>
|
||||||
|
</div>
|
||||||
|
<div class="layui-card-body">
|
||||||
|
<form class="layui-form layui-form-pane" action="">
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<div class="layui-inline">
|
||||||
|
<label class="layui-form-label">用户名</label>
|
||||||
|
<div class="layui-input-inline">
|
||||||
|
<input type="text" name="username" placeholder="请输入用户名" autocomplete="off" class="layui-input">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layui-inline">
|
||||||
|
<label class="layui-form-label">IP地址</label>
|
||||||
|
<div class="layui-input-inline">
|
||||||
|
<input type="text" name="ip" placeholder="请输入IP地址" autocomplete="off" class="layui-input">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layui-inline">
|
||||||
|
<label class="layui-form-label">状态</label>
|
||||||
|
<div class="layui-input-inline">
|
||||||
|
<select name="status">
|
||||||
|
<option value="">全部</option>
|
||||||
|
<option value="1">成功</option>
|
||||||
|
<option value="0">失败</option>
|
||||||
|
</select>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layui-inline">
|
||||||
|
<label class="layui-form-label">时间范围</label>
|
||||||
|
<div class="layui-input-inline" style="width: 300px;">
|
||||||
|
<input type="text" name="time_range" class="layui-input" id="timeRange" placeholder="请选择时间范围">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layui-inline">
|
||||||
|
<button class="layui-btn" lay-submit lay-filter="searchForm">
|
||||||
|
<i class="layui-icon layui-icon-search"></i> 搜索
|
||||||
|
</button>
|
||||||
|
<button type="reset" class="layui-btn layui-btn-primary">重置</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</form>
|
||||||
|
|
||||||
|
<table id="loginLogTable" lay-filter="loginLogTable"></table>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<script src="/static/layui/layui.js"></script>
|
||||||
|
<script>
|
||||||
|
layui.use(['table', 'form', 'laydate'], function(){
|
||||||
|
var table = layui.table;
|
||||||
|
var form = layui.form;
|
||||||
|
var laydate = layui.laydate;
|
||||||
|
|
||||||
|
// 初始化时间范围选择器
|
||||||
|
laydate.render({
|
||||||
|
elem: '#timeRange',
|
||||||
|
type: 'datetime',
|
||||||
|
range: true
|
||||||
|
});
|
||||||
|
|
||||||
|
// 初始化表格
|
||||||
|
table.render({
|
||||||
|
elem: '#loginLogTable',
|
||||||
|
url: '{:url("log/login")}',
|
||||||
|
method: 'post',
|
||||||
|
defaultToolbar: ['filter', 'exports', 'print'],
|
||||||
|
parseData: function(res) {
|
||||||
|
return {
|
||||||
|
"code": res.code === 0 ? 0 : 1,
|
||||||
|
"msg": res.msg,
|
||||||
|
"count": res.count,
|
||||||
|
"data": res.data
|
||||||
|
};
|
||||||
|
},
|
||||||
|
cols: [[
|
||||||
|
{field: 'id', title: 'ID', width: 80, sort: true, align: 'center'},
|
||||||
|
{field: 'username', title: '用户名', align: 'center'},
|
||||||
|
{field: 'ip_address', title: 'IP地址', width: 130, align: 'center'},
|
||||||
|
{field: 'location', title: '登录地点', width: 120, align: 'center'},
|
||||||
|
{field: 'device_type', title: '设备类型', width: 100, align: 'center'},
|
||||||
|
{field: 'user_agent', title: '浏览器', width: 200, align: 'center'},
|
||||||
|
{field: 'login_status', title: '状态', width: 100, align: 'center', templet: function(d){
|
||||||
|
return d.login_status == 1 ?
|
||||||
|
'<span class="layui-badge layui-bg-green">成功</span>' :
|
||||||
|
'<span class="layui-badge layui-bg-red">失败</span>';
|
||||||
|
}},
|
||||||
|
{field: 'failure_reason', title: '失败原因', width: 150, align: 'center'},
|
||||||
|
{field: 'login_time', title: '登录时间', width: 180, sort: true, align: 'center'}
|
||||||
|
]],
|
||||||
|
page: true,
|
||||||
|
limit: 10,
|
||||||
|
limits: [10, 20, 50, 100]
|
||||||
|
});
|
||||||
|
|
||||||
|
// 监听搜索表单提交
|
||||||
|
form.on('submit(searchForm)', function(data){
|
||||||
|
var timeRange = data.field.time_range;
|
||||||
|
if(timeRange){
|
||||||
|
var times = timeRange.split(' - ');
|
||||||
|
data.field.start_time = times[0];
|
||||||
|
data.field.end_time = times[1];
|
||||||
|
}
|
||||||
|
delete data.field.time_range;
|
||||||
|
|
||||||
|
table.reload('loginLogTable', {
|
||||||
|
where: data.field,
|
||||||
|
page: {curr: 1}
|
||||||
|
});
|
||||||
|
return false;
|
||||||
|
});
|
||||||
|
});
|
||||||
|
</script>
|
||||||
|
</body>
|
||||||
|
</html>
|
||||||
173
app/admin/view/log/operation.php
Normal file
173
app/admin/view/log/operation.php
Normal file
@ -0,0 +1,173 @@
|
|||||||
|
{include file="public/header" /}
|
||||||
|
<div class="layui-card">
|
||||||
|
<div class="layui-card-header">
|
||||||
|
<span class="layui-badge layui-bg-blue">操作日志</span>
|
||||||
|
</div>
|
||||||
|
<div class="layui-card-body">
|
||||||
|
<form class="layui-form layui-form-pane" action="">
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<div class="layui-inline">
|
||||||
|
<label class="layui-form-label">用户名</label>
|
||||||
|
<div class="layui-input-inline">
|
||||||
|
<input type="text" name="username" placeholder="请输入用户名" autocomplete="off" class="layui-input">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layui-inline">
|
||||||
|
<label class="layui-form-label">模块</label>
|
||||||
|
<div class="layui-input-inline">
|
||||||
|
<input type="text" name="module" placeholder="请输入模块" autocomplete="off" class="layui-input">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layui-inline">
|
||||||
|
<label class="layui-form-label">操作</label>
|
||||||
|
<div class="layui-input-inline">
|
||||||
|
<input type="text" name="operation" placeholder="请输入操作" autocomplete="off" class="layui-input">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layui-inline">
|
||||||
|
<label class="layui-form-label">状态</label>
|
||||||
|
<div class="layui-input-inline">
|
||||||
|
<select name="status">
|
||||||
|
<option value="">全部</option>
|
||||||
|
<option value="1">成功</option>
|
||||||
|
<option value="0">失败</option>
|
||||||
|
</select>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layui-inline">
|
||||||
|
<label class="layui-form-label">时间范围</label>
|
||||||
|
<div class="layui-input-inline" style="width: 300px;">
|
||||||
|
<input type="text" name="time_range" class="layui-input" id="timeRange" placeholder="请选择时间范围">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layui-inline">
|
||||||
|
<button class="layui-btn" lay-submit lay-filter="searchForm">
|
||||||
|
<i class="layui-icon layui-icon-search"></i> 搜索
|
||||||
|
</button>
|
||||||
|
<button type="reset" class="layui-btn layui-btn-primary">重置</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</form>
|
||||||
|
|
||||||
|
<table id="operationLogTable" lay-filter="operationLogTable"></table>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<script src="/static/layui/layui.js"></script>
|
||||||
|
<script>
|
||||||
|
layui.use(['table', 'form', 'laydate', 'layer'], function(){
|
||||||
|
var table = layui.table;
|
||||||
|
var form = layui.form;
|
||||||
|
var laydate = layui.laydate;
|
||||||
|
var layer = layui.layer;
|
||||||
|
|
||||||
|
// 初始化时间范围选择器
|
||||||
|
laydate.render({
|
||||||
|
elem: '#timeRange',
|
||||||
|
type: 'datetime',
|
||||||
|
range: true
|
||||||
|
});
|
||||||
|
|
||||||
|
// 初始化表格
|
||||||
|
table.render({
|
||||||
|
elem: '#operationLogTable',
|
||||||
|
url: '{:url("log/operation")}',
|
||||||
|
method: 'get',
|
||||||
|
defaultToolbar: ['filter', 'exports', 'print'],
|
||||||
|
parseData: function(res) {
|
||||||
|
return {
|
||||||
|
"code": 0,
|
||||||
|
"msg": res.msg || '获取成功',
|
||||||
|
"count": res.count || 0,
|
||||||
|
"data": res.data || []
|
||||||
|
};
|
||||||
|
},
|
||||||
|
cols: [[
|
||||||
|
{field: 'id', title: 'ID', width: 80, sort: true, align: 'center'},
|
||||||
|
{field: 'username', title: '操作人', width: 120, align: 'center'},
|
||||||
|
{field: 'module', title: '模块', width: 120, align: 'center'},
|
||||||
|
{field: 'operation', title: '操作', width: 150, align: 'center'},
|
||||||
|
{field: 'request_method', title: '请求方法', width: 100, align: 'center'},
|
||||||
|
{field: 'request_url', title: '请求地址', align: 'center'},
|
||||||
|
{field: 'ip_address', title: 'IP地址', width: 120, align: 'center'},
|
||||||
|
{field: 'status', title: '状态', width: 100, align: 'center', templet: function(d){
|
||||||
|
return d.status == 1 ? '<span class="layui-badge layui-bg-green">成功</span>' : '<span class="layui-badge layui-bg-red">失败</span>';
|
||||||
|
}},
|
||||||
|
{field: 'operation_time', title: '操作时间', width: 180, align: 'center'},
|
||||||
|
{field: 'execution_time', title: '执行时间(ms)', width: 120, align: 'center'},
|
||||||
|
{title: '操作', width: 120, toolbar: '#operationBar', fixed: 'right', align: 'center'}
|
||||||
|
]],
|
||||||
|
page: true,
|
||||||
|
limit: 10,
|
||||||
|
limits: [10, 20, 50, 100]
|
||||||
|
});
|
||||||
|
|
||||||
|
// 监听搜索表单提交
|
||||||
|
form.on('submit(searchForm)', function(data){
|
||||||
|
var timeRange = data.field.time_range;
|
||||||
|
if(timeRange){
|
||||||
|
var times = timeRange.split(' - ');
|
||||||
|
data.field.start_time = times[0];
|
||||||
|
data.field.end_time = times[1];
|
||||||
|
}
|
||||||
|
delete data.field.time_range;
|
||||||
|
|
||||||
|
table.reload('operationLogTable', {
|
||||||
|
where: data.field,
|
||||||
|
page: {curr: 1}
|
||||||
|
});
|
||||||
|
return false;
|
||||||
|
});
|
||||||
|
|
||||||
|
// 监听工具条
|
||||||
|
table.on('tool(operationLogTable)', function(obj){
|
||||||
|
var data = obj.data;
|
||||||
|
if(obj.event === 'detail'){
|
||||||
|
// 获取详情
|
||||||
|
$.ajax({
|
||||||
|
url: '{:url("log/getOperationDetail")}',
|
||||||
|
type: 'GET',
|
||||||
|
data: {id: data.id},
|
||||||
|
success: function(res){
|
||||||
|
if(res.code === 0){
|
||||||
|
var detail = res.data;
|
||||||
|
var content = '<div class="layui-card">' +
|
||||||
|
'<div class="layui-card-body">' +
|
||||||
|
'<table class="layui-table" lay-skin="nob">' +
|
||||||
|
'<colgroup><col width="100"><col></colgroup>' +
|
||||||
|
'<tbody>' +
|
||||||
|
'<tr><td>操作人:</td><td>' + detail.username + '</td></tr>' +
|
||||||
|
'<tr><td>模块:</td><td>' + detail.module + '</td></tr>' +
|
||||||
|
'<tr><td>操作:</td><td>' + detail.operation + '</td></tr>' +
|
||||||
|
'<tr><td>请求方法:</td><td>' + detail.request_method + '</td></tr>' +
|
||||||
|
'<tr><td>请求地址:</td><td>' + detail.request_url + '</td></tr>' +
|
||||||
|
'<tr><td>请求参数:</td><td><pre>' + JSON.stringify(detail.request_params, null, 2) + '</pre></td></tr>' +
|
||||||
|
'<tr><td>IP地址:</td><td>' + detail.ip_address + '</td></tr>' +
|
||||||
|
'<tr><td>状态:</td><td>' + (detail.status == 1 ? '成功' : '失败') + '</td></tr>' +
|
||||||
|
'<tr><td>错误信息:</td><td>' + (detail.error_message || '无') + '</td></tr>' +
|
||||||
|
'<tr><td>操作时间:</td><td>' + detail.operation_time + '</td></tr>' +
|
||||||
|
'<tr><td>执行时间:</td><td>' + detail.execution_time + 'ms</td></tr>' +
|
||||||
|
'</tbody></table></div></div>';
|
||||||
|
|
||||||
|
layer.open({
|
||||||
|
type: 1,
|
||||||
|
title: '操作日志详情',
|
||||||
|
area: ['800px', '600px'],
|
||||||
|
content: content
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
layer.msg(res.msg);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<!-- 表格工具栏模板 -->
|
||||||
|
<script type="text/html" id="operationBar">
|
||||||
|
<a class="layui-btn layui-btn-xs" lay-event="detail">详情</a>
|
||||||
|
</script>
|
||||||
|
</body>
|
||||||
|
</html>
|
||||||
130
app/admin/view/login/index.php
Normal file
130
app/admin/view/login/index.php
Normal file
@ -0,0 +1,130 @@
|
|||||||
|
<!DOCTYPE html>
|
||||||
|
<html lang="zh-CN">
|
||||||
|
|
||||||
|
<head>
|
||||||
|
<title>后台管理系统</title>
|
||||||
|
<meta name="renderer" content="yz">
|
||||||
|
<meta http-equiv="X-UA-Compatible" content="IE=edge,chrome=1">
|
||||||
|
<meta name="viewport"
|
||||||
|
content="width=device-width, initial-scale=1.0, minimum-scale=1.0, maximum-scale=1.0, user-scalable=0">
|
||||||
|
<link rel="stylesheet" type="text/css" href="/static/layui/css/layui.css">
|
||||||
|
<link rel="stylesheet" type="text/css" href="/static/css/login.css">
|
||||||
|
<script type="text/javascript" src="/static/layui/layui.js"></script>
|
||||||
|
</head>
|
||||||
|
|
||||||
|
<body>
|
||||||
|
<div class="layadmin-user-login layadmin-user-display-show" id="LAY-user-login">
|
||||||
|
<div class="layadmin-user-login-main">
|
||||||
|
<div class="layadmin-user-login-box layadmin-user-login-header">
|
||||||
|
<img src="/static/images/logo.png" />
|
||||||
|
<h2>后台管理系统</h2>
|
||||||
|
</div>
|
||||||
|
<div class="layadmin-user-login-box layadmin-user-login-body layui-form">
|
||||||
|
<form class="layui-form login-form">
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layadmin-user-login-icon layui-icon layui-icon-username" for="account"></label>
|
||||||
|
<input type="text" id="account" name="account" placeholder="邮箱" class="layui-input"
|
||||||
|
value="">
|
||||||
|
</div>
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layadmin-user-login-icon layui-icon layui-icon-password" for="password"></label>
|
||||||
|
<input type="password" name="password" lay-affix="eye" class="layui-input" placeholder="密码" class="layui-input"
|
||||||
|
value="">
|
||||||
|
</div>
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<div class="layui-row">
|
||||||
|
<div class="layui-col-xs7">
|
||||||
|
<label class="layadmin-user-login-icon layui-icon layui-icon-vercode"
|
||||||
|
for="code"></label>
|
||||||
|
<input type="text" name="code" placeholder="图形验证码" class="layui-input">
|
||||||
|
</div>
|
||||||
|
<div class="layui-col-xs5">
|
||||||
|
<div style="margin-left:10px;">
|
||||||
|
<img src="{:captcha_src()}?t={:time()}" class="layadmin-user-login-codeimg" id="img"
|
||||||
|
onclick="reloadImg()">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layui-form-item" style="margin-bottom: 20px;">
|
||||||
|
<input type="checkbox" name="remember" lay-skin="primary" title="记住密码">
|
||||||
|
</div>
|
||||||
|
</form>
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<button class="layui-btn layui-btn-fluid" onclick="login()">登 陆</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</body>
|
||||||
|
|
||||||
|
</html>
|
||||||
|
<script type="text/javascript">
|
||||||
|
layui.use(['layer', 'form'], function () {
|
||||||
|
var form = layui.form;
|
||||||
|
layer = layui.layer;
|
||||||
|
$ = layui.jquery;
|
||||||
|
// 用户名控件获取焦点
|
||||||
|
$('#account').focus();
|
||||||
|
// 回车登录
|
||||||
|
$('input').keydown(function (e) {
|
||||||
|
if (e.keyCode == 13) {
|
||||||
|
login();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
// 重新生成验证码
|
||||||
|
function reloadImg() {
|
||||||
|
var timestamp = new Date().getTime();
|
||||||
|
$('#img').attr('src', '{:captcha_src()}?t=' + timestamp);
|
||||||
|
}
|
||||||
|
|
||||||
|
// 登录处理函数
|
||||||
|
function login() {
|
||||||
|
var account = $('input[name="account"]').val();
|
||||||
|
var password = $('input[name="password"]').val();
|
||||||
|
var code = $('input[name="code"]').val();
|
||||||
|
var remember = $('input[name="remember"]:checked').val();
|
||||||
|
|
||||||
|
if (!account) {
|
||||||
|
layer.msg('邮箱不能为空');
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!password) {
|
||||||
|
layer.msg('密码不能为空');
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!code) {
|
||||||
|
layer.msg('验证码不能为空');
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
|
||||||
|
$.ajax({
|
||||||
|
type: 'post',
|
||||||
|
url: '{:url("login")}',
|
||||||
|
data: {
|
||||||
|
account: account,
|
||||||
|
password: password,
|
||||||
|
code: code,
|
||||||
|
remember: remember
|
||||||
|
},
|
||||||
|
dataType: 'json',
|
||||||
|
success: function(res) {
|
||||||
|
if (res.code == 0) {
|
||||||
|
layer.msg(res.msg, {icon: 1, time: 1000}, function() {
|
||||||
|
window.location.href = '{:url("Index/index")}';
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
layer.msg(res.msg, {icon: 2});
|
||||||
|
reloadImg();
|
||||||
|
}
|
||||||
|
},
|
||||||
|
error: function() {
|
||||||
|
layer.msg('网络错误,请重试', {icon: 2});
|
||||||
|
reloadImg();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
</script>
|
||||||
79
app/admin/view/login/resetpwd.php
Normal file
79
app/admin/view/login/resetpwd.php
Normal file
@ -0,0 +1,79 @@
|
|||||||
|
<!DOCTYPE html>
|
||||||
|
<html>
|
||||||
|
|
||||||
|
<head>
|
||||||
|
<meta charset="utf-8">
|
||||||
|
<title>重置密码</title>
|
||||||
|
<meta name="renderer" content="yz">
|
||||||
|
<meta http-equiv="X-UA-Compatible" content="IE=edge,chrome=1">
|
||||||
|
<meta name="viewport"
|
||||||
|
content="width=device-width, initial-scale=1.0, minimum-scale=1.0, maximum-scale=1.0, user-scalable=0">
|
||||||
|
<link rel="stylesheet" type="text/css" href="/static/layui/css/layui.css">
|
||||||
|
<link rel="stylesheet" type="text/css" href="/static/css/login.css">
|
||||||
|
<script type="text/javascript" src="/static/layui/layui.js"></script>
|
||||||
|
</head>
|
||||||
|
|
||||||
|
<body>
|
||||||
|
<div class="layadmin-user-login layadmin-user-display-show" id="LAY-user-login">
|
||||||
|
<div class="layadmin-user-login-main">
|
||||||
|
<div class="layadmin-user-login-box layadmin-user-login-header">
|
||||||
|
<h2>重置密码</h2>
|
||||||
|
</div>
|
||||||
|
<div class="layadmin-user-login-box layadmin-user-login-body layui-form">
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layadmin-user-login-icon layui-icon layui-icon-username"
|
||||||
|
for="LAY-user-login-username"></label>
|
||||||
|
<input type="text" name="account" id="LAY-user-login-username" lay-verify="required"
|
||||||
|
placeholder="用户名" class="layui-input">
|
||||||
|
</div>
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<button class="layui-btn layui-btn-fluid" lay-submit
|
||||||
|
lay-filter="LAY-user-reset-submit">确认重置</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<script src="/static/layui/layui.js"></script>
|
||||||
|
<script>
|
||||||
|
layui.use(['form', 'layer'], function () {
|
||||||
|
var $ = layui.$
|
||||||
|
, form = layui.form
|
||||||
|
, layer = layui.layer;
|
||||||
|
|
||||||
|
form.on('submit(LAY-user-reset-submit)', function (obj) {
|
||||||
|
var field = obj.field;
|
||||||
|
|
||||||
|
//请求重置密码
|
||||||
|
$.ajax({
|
||||||
|
url: '/admin/login/resetpwd'
|
||||||
|
, type: 'post'
|
||||||
|
, data: {
|
||||||
|
account: field.account
|
||||||
|
}
|
||||||
|
, success: function (res) {
|
||||||
|
if (res.code === 0) {
|
||||||
|
layer.msg('密码重置成功', {
|
||||||
|
offset: '15px'
|
||||||
|
, icon: 1
|
||||||
|
, time: 1000
|
||||||
|
}, function () {
|
||||||
|
location.href = '/admin/login/index';
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
layer.msg(res.msg, {
|
||||||
|
offset: '15px'
|
||||||
|
, icon: 2
|
||||||
|
, time: 1000
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
return false;
|
||||||
|
});
|
||||||
|
});
|
||||||
|
</script>
|
||||||
|
</body>
|
||||||
|
|
||||||
|
</html>
|
||||||
94
app/admin/view/public/header.php
Normal file
94
app/admin/view/public/header.php
Normal file
@ -0,0 +1,94 @@
|
|||||||
|
<!DOCTYPE html>
|
||||||
|
<html>
|
||||||
|
<head>
|
||||||
|
<title>{$config['web_title']}</title>
|
||||||
|
<meta name="renderer" content="webkit">
|
||||||
|
<meta http-equiv="X-UA-Compatible" content="IE=edge,chrome=1">
|
||||||
|
<meta name="viewport" content="width=device-width, initial-scale=1.0, minimum-scale=1.0, maximum-scale=1.0, user-scalable=0">
|
||||||
|
<link rel="stylesheet" type="text/css" href="/static/layui/css/layui.css" media="all"/>
|
||||||
|
<link rel="stylesheet" type="text/css" href="__CSS__/moban.css" media="all"/>
|
||||||
|
<link rel="stylesheet" type="text/css" href="__CSS__/wangeditor.css" media="all"/>
|
||||||
|
<style type="text/css">
|
||||||
|
.header span{background:#009688;margin-left:30px;padding:10px;color:#ffffff;}
|
||||||
|
.header div{border-bottom:solid 2px #009688;margin-top: 8px;}
|
||||||
|
.header button{float:right;margin-top:-5px;}
|
||||||
|
.pagination {
|
||||||
|
display: inline-block;
|
||||||
|
padding-left: 0;
|
||||||
|
margin: 20px 0;
|
||||||
|
border-radius: 4px;
|
||||||
|
}
|
||||||
|
.pagination > li {
|
||||||
|
display: inline;
|
||||||
|
}
|
||||||
|
.pagination > li > a,
|
||||||
|
.pagination > li > span {
|
||||||
|
position: relative;
|
||||||
|
float: left;
|
||||||
|
padding: 6px 12px;
|
||||||
|
margin-left: -1px;
|
||||||
|
line-height: 1.42857143;
|
||||||
|
color: #337ab7;
|
||||||
|
text-decoration: none;
|
||||||
|
background-color: #fff;
|
||||||
|
border: 1px solid #ddd;
|
||||||
|
}
|
||||||
|
.pagination > li:first-child > a,
|
||||||
|
.pagination > li:first-child > span {
|
||||||
|
margin-left: 0;
|
||||||
|
border-top-left-radius: 4px;
|
||||||
|
border-bottom-left-radius: 4px;
|
||||||
|
}
|
||||||
|
.pagination > li:last-child > a,
|
||||||
|
.pagination > li:last-child > span {
|
||||||
|
border-top-right-radius: 4px;
|
||||||
|
border-bottom-right-radius: 4px;
|
||||||
|
}
|
||||||
|
.pagination > li > a:hover,
|
||||||
|
.pagination > li > span:hover,
|
||||||
|
.pagination > li > a:focus,
|
||||||
|
.pagination > li > span:focus {
|
||||||
|
z-index: 2;
|
||||||
|
color: #23527c;
|
||||||
|
background-color: #eee;
|
||||||
|
border-color: #ddd;
|
||||||
|
}
|
||||||
|
.pagination > .active > a,
|
||||||
|
.pagination > .active > span,
|
||||||
|
.pagination > .active > a:hover,
|
||||||
|
.pagination > .active > span:hover,
|
||||||
|
.pagination > .active > a:focus,
|
||||||
|
.pagination > .active > span:focus {
|
||||||
|
z-index: 3;
|
||||||
|
color: #fff;
|
||||||
|
cursor: default;
|
||||||
|
background-color: #337ab7;
|
||||||
|
border-color: #337ab7;
|
||||||
|
}
|
||||||
|
.pagination > .disabled > span,
|
||||||
|
.pagination > .disabled > span:hover,
|
||||||
|
.pagination > .disabled > span:focus,
|
||||||
|
.pagination > .disabled > a,
|
||||||
|
.pagination > .disabled > a:hover,
|
||||||
|
.pagination > .disabled > a:focus {
|
||||||
|
color: #777;
|
||||||
|
cursor: not-allowed;
|
||||||
|
background-color: #fff;
|
||||||
|
border-color: #ddd;
|
||||||
|
}
|
||||||
|
.close-img { background: url(/static/images/close_img.png); background-size: 20px 20px; width:20px; height: 20px; position: absolute; right: 5px; top: 5px; z-index: 2;}
|
||||||
|
</style>
|
||||||
|
<script type="text/javascript" src="/static/layui/layui.js"></script>
|
||||||
|
<script type="text/javascript">
|
||||||
|
layui.use(['layer','form','table','laydate','element','upload'],function(){
|
||||||
|
layer = layui.layer; // layui 弹框
|
||||||
|
form = layui.form; // layui form表单
|
||||||
|
table = layui.table; // layui 表格
|
||||||
|
laydate = layui.laydate; // layui 时间框
|
||||||
|
element = layui.element; // layui element
|
||||||
|
upload = layui.upload; // layui 上传
|
||||||
|
$ = layui.jquery; // layui jquery
|
||||||
|
})
|
||||||
|
</script>
|
||||||
|
</head>
|
||||||
|
<body style="padding:10px; box-sizing: border-box;">
|
||||||
30
app/admin/view/public/tail.php
Normal file
30
app/admin/view/public/tail.php
Normal file
@ -0,0 +1,30 @@
|
|||||||
|
</body>
|
||||||
|
</html>
|
||||||
|
<script type="text/javascript">
|
||||||
|
// 显示图片
|
||||||
|
function show_img(obj){
|
||||||
|
var imgurl = $(obj).attr('src');
|
||||||
|
var res = getMousePos();
|
||||||
|
var html = '<div style="background:#fff;position:absolute;width:200px;border:solid 1px #cdcdcd;border-radius:6px;padding:2px;left:'+res.x+'px;top:'+res.y+'px;z-index:1000" id="preview">\
|
||||||
|
<img style="width:100%;border-radius:6px;" src="'+imgurl+'">\
|
||||||
|
</div>';
|
||||||
|
$('body').append(html);
|
||||||
|
}
|
||||||
|
// 隐藏图片
|
||||||
|
function hide_img(){
|
||||||
|
$('#preview').remove();
|
||||||
|
}
|
||||||
|
// 图片位置计算
|
||||||
|
function getMousePos(event) {
|
||||||
|
var e = event || window.event;
|
||||||
|
var scrollX = document.documentElement.scrollLeft || document.body.scrollLeft;
|
||||||
|
var scrollY = document.documentElement.scrollTop || document.body.scrollTop;
|
||||||
|
var x = e.pageX || e.clientX + scrollX;
|
||||||
|
var y = e.pageY || e.clientY + scrollY;
|
||||||
|
return { 'x': x, 'y': y };
|
||||||
|
}
|
||||||
|
// 删除图片
|
||||||
|
function deleteImage(path,obj){
|
||||||
|
$(obj).closest('.upload_pic_li').remove();
|
||||||
|
}
|
||||||
|
</script>
|
||||||
571
app/admin/view/resources/add.php
Normal file
571
app/admin/view/resources/add.php
Normal file
@ -0,0 +1,571 @@
|
|||||||
|
{include file="public/header" /}
|
||||||
|
<div class="config-container">
|
||||||
|
<div class="config-header" style="display:flex;justify-content: space-between;">
|
||||||
|
<div>
|
||||||
|
<span>添加资源</span>
|
||||||
|
</div>
|
||||||
|
<div>
|
||||||
|
<button type="button" class="layui-btn layui-btn-primary layui-btn-sm" onclick="goBack()">
|
||||||
|
<i class="layui-icon layui-icon-return"></i>返回
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<form class="layui-form" action="" method="post">
|
||||||
|
<div class="form-container">
|
||||||
|
<div class="container-left">
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">资源名称</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<input type="text" name="title" required lay-verify="required" placeholder="请输入资源名称"
|
||||||
|
autocomplete="off" class="layui-input" lay-affix="clear">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">分类</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<select name="cate" lay-verify="required" lay-filter="cate">
|
||||||
|
<option value="">请选择分类</option>
|
||||||
|
</select>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">资源编号</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<input type="text" name="number" required lay-verify="required" placeholder="选取分类后系统自动生成"
|
||||||
|
autocomplete="off" class="layui-input" lay-affix="clear" disabled>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">描述</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<textarea name="desc" placeholder="请输入资源描述" class="layui-textarea" lay-affix="clear"></textarea>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">上传者</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<input type="text" name="uploader" required lay-verify="required" placeholder="请输入上传者"
|
||||||
|
autocomplete="off" class="layui-input" lay-affix="clear" value="{$aUser['name']}">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">排序</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<input type="number" name="sort" value="0" class="layui-input" placeholder="数字越大越靠前"
|
||||||
|
lay-affix="clear">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">资源图标</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<button type="button" class="layui-btn" id="upload-btn">
|
||||||
|
<i class="layui-icon layui-icon-upload"></i> 图标上传
|
||||||
|
</button>
|
||||||
|
<div style="width: 120px;">
|
||||||
|
<div class="layui-upload-list">
|
||||||
|
<img class="layui-upload-img" id="upload-img"
|
||||||
|
style="width: 118px; height: 118px;object-fit: cover;">
|
||||||
|
<div id="upload-text"></div>
|
||||||
|
</div>
|
||||||
|
<div class="layui-progress layui-progress-big" lay-showPercent="yes"
|
||||||
|
lay-filter="icon-progress">
|
||||||
|
<div class="layui-progress-bar" lay-percent=""></div>
|
||||||
|
</div>
|
||||||
|
<input type="hidden" name="icon" id="icon" value="">
|
||||||
|
</div>
|
||||||
|
<div class="layui-form-mid layui-word-aux">建议尺寸:250px * 140px</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">图片上传</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<div class="layui-upload">
|
||||||
|
<button type="button" class="layui-btn" id="imageUpload">多图片上传</button>
|
||||||
|
<blockquote class="layui-elem-quote layui-quote-nm" style="margin-top: 10px;">
|
||||||
|
预览图:
|
||||||
|
<div class="layui-upload-list" id="imagePreview"
|
||||||
|
style="display: flex; flex-direction: column;">
|
||||||
|
</div>
|
||||||
|
</blockquote>
|
||||||
|
</div>
|
||||||
|
<div class="layui-progress layui-progress-big" lay-showPercent="yes" lay-filter="image-progress"
|
||||||
|
style="margin-top: 10px;">
|
||||||
|
<div class="layui-progress-bar" lay-percent=""></div>
|
||||||
|
</div>
|
||||||
|
<input type="hidden" name="images" id="images" value="">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">资源文件</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<input type="text" name="fileurl" required placeholder="本地资源地址" autocomplete="off"
|
||||||
|
class="layui-input" value="" style="margin-bottom: 10px;" lay-affix="clear">
|
||||||
|
<div class="layui-upload-drag" style="display: block;" id="ID-upload-demo-drag">
|
||||||
|
<i class="layui-icon layui-icon-upload"></i>
|
||||||
|
<div>点击上传,或将文件拖拽到此处</div>
|
||||||
|
<div class="layui-hide" id="ID-upload-demo-preview">
|
||||||
|
<hr>
|
||||||
|
<div class="file-info">
|
||||||
|
<i class="layui-icon layui-icon-file"></i>
|
||||||
|
<span class="file-name"></span>
|
||||||
|
<span class="file-size"></span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layui-progress layui-progress-big" lay-showPercent="yes" lay-filter="file-progress"
|
||||||
|
style="margin-top: 10px;">
|
||||||
|
<div class="layui-progress-bar" lay-percent=""></div>
|
||||||
|
</div>
|
||||||
|
<input type="hidden" name="file" id="file" value="">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">资源链接</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<input type="text" name="url" required placeholder="百度网盘、115网盘、蓝奏云等" autocomplete="off"
|
||||||
|
class="layui-input" lay-affix="clear">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">分享码</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<input type="text" name="code" required placeholder="请输入分享码" autocomplete="off"
|
||||||
|
class="layui-input" lay-affix="clear">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">解压密码</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<input type="text" name="zipcode" required placeholder="请输入解压密码" autocomplete="off"
|
||||||
|
class="layui-input" lay-affix="clear">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="container-right">
|
||||||
|
<div class="layui-form-item layui-form-text">
|
||||||
|
<label class="layui-form-label">内容</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<div id="editor—wrapper" id="content" name="content" style="border: 1px solid #ccc;">
|
||||||
|
<div id="toolbar-container" style="border-bottom: 1px solid #ccc;"><!-- 工具栏 --></div>
|
||||||
|
<div id="editor-container" style="height: 800px;"><!-- 编辑器 --></div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="layui-form-item" style="margin-top: 80px;">
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<button class="layui-btn" lay-submit lay-filter="formSubmit">立即提交</button>
|
||||||
|
<button type="reset" class="layui-btn layui-btn-primary">重置</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</form>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<script src="/static/js/wangeditor.js"></script>
|
||||||
|
<script>
|
||||||
|
layui.use(['form', 'layer', 'upload', 'element'], function () {
|
||||||
|
var form = layui.form;
|
||||||
|
var layer = layui.layer;
|
||||||
|
var $ = layui.$;
|
||||||
|
var upload = layui.upload;
|
||||||
|
var element = layui.element;
|
||||||
|
|
||||||
|
// 格式化文件大小
|
||||||
|
function formatFileSize(bytes) {
|
||||||
|
if (bytes === 0) return '0 B';
|
||||||
|
const k = 1024;
|
||||||
|
const sizes = ['B', 'KB', 'MB', 'GB', 'TB'];
|
||||||
|
const i = Math.floor(Math.log(bytes) / Math.log(k));
|
||||||
|
return parseFloat((bytes / Math.pow(k, i)).toFixed(2)) + ' ' + sizes[i];
|
||||||
|
}
|
||||||
|
|
||||||
|
// 图标上传
|
||||||
|
var iconUpload = upload.render({
|
||||||
|
elem: '#upload-btn',
|
||||||
|
url: '{:url("index/upload_img")}',
|
||||||
|
before: function (obj) {
|
||||||
|
obj.preview(function (index, file, result) {
|
||||||
|
$('#upload-img').attr('src', result);
|
||||||
|
});
|
||||||
|
element.progress('icon-progress', '0%');
|
||||||
|
layer.msg('图标上传中', { icon: 16, time: 0 });
|
||||||
|
},
|
||||||
|
done: function (res) {
|
||||||
|
if (res.code > 0) {
|
||||||
|
return layer.msg('图标上传失败');
|
||||||
|
}
|
||||||
|
$('#icon').val(res.data);
|
||||||
|
$('#upload-text').html('');
|
||||||
|
layer.msg('图标上传成功', { icon: 1 });
|
||||||
|
},
|
||||||
|
error: function () {
|
||||||
|
var demoText = $('#upload-text');
|
||||||
|
demoText.html('<span style="color: #FF5722;">上传失败</span> <a class="layui-btn layui-btn-xs demo-reload">重试</a>');
|
||||||
|
demoText.find('.demo-reload').on('click', function () {
|
||||||
|
iconUpload.upload();
|
||||||
|
});
|
||||||
|
},
|
||||||
|
progress: function (n, elem, e) {
|
||||||
|
element.progress('icon-progress', n + '%');
|
||||||
|
if (n == 100) {
|
||||||
|
layer.msg('图标上传完毕', { icon: 1 });
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
// 文件上传
|
||||||
|
var fileUpload = upload.render({
|
||||||
|
elem: '#ID-upload-demo-drag',
|
||||||
|
url: '{:url("index/upload_file")}',
|
||||||
|
accept: 'file',
|
||||||
|
exts: 'doc|docx|xls|xlsx|ppt|pptx|pdf|txt|zip|rar|7z',
|
||||||
|
before: function (obj) {
|
||||||
|
obj.preview(function (index, file, result) {
|
||||||
|
$('#ID-upload-demo-preview').show();
|
||||||
|
$('.file-name').text(file.name);
|
||||||
|
$('.file-size').text(formatFileSize(file.size));
|
||||||
|
});
|
||||||
|
element.progress('file-progress', '0%');
|
||||||
|
layer.msg('文件上传中', { icon: 16, time: 0 });
|
||||||
|
},
|
||||||
|
done: function (res) {
|
||||||
|
if (res.code > 0) {
|
||||||
|
return layer.msg('文件上传失败');
|
||||||
|
}
|
||||||
|
$('#file').val(res.data.src);
|
||||||
|
$('input[name="fileurl"]').val(res.data.src);
|
||||||
|
layer.msg('文件上传成功', { icon: 1 });
|
||||||
|
},
|
||||||
|
error: function () {
|
||||||
|
layer.msg('文件上传失败', { icon: 2 });
|
||||||
|
},
|
||||||
|
progress: function (n, elem, e) {
|
||||||
|
element.progress('file-progress', n + '%');
|
||||||
|
if (n == 100) {
|
||||||
|
layer.msg('文件上传完毕', { icon: 1 });
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
// 多图片上传
|
||||||
|
var uploadInst = upload.render({
|
||||||
|
elem: '#imageUpload',
|
||||||
|
url: '{:url("index/upload_img")}',
|
||||||
|
multiple: true,
|
||||||
|
accept: 'images',
|
||||||
|
before: function (obj) {
|
||||||
|
obj.preview(function (index, file, result) {
|
||||||
|
$('#imagePreview').append('<div class="layui-upload-img-item" style="display: inline-block; margin-right: 10px;"><img src="' + result + '" alt="' + file.name + '" style="width: 100px; height: 100px; object-fit: cover;"><p>' + file.name + '</p><button type="button" class="layui-btn layui-btn-xs layui-btn-danger delete-image" style="position: absolute; top: 0; right: 0;">删除</button></div>');
|
||||||
|
});
|
||||||
|
element.progress('image-progress', '0%');
|
||||||
|
layer.msg('图片上传中', { icon: 16, time: 0 });
|
||||||
|
},
|
||||||
|
done: function (res) {
|
||||||
|
if (res.code > 0) {
|
||||||
|
return layer.msg('图片上传失败');
|
||||||
|
}
|
||||||
|
var images = $('#images').val();
|
||||||
|
images = images ? images + ',' + res.data : res.data;
|
||||||
|
$('#images').val(images);
|
||||||
|
layer.msg('图片上传成功', { icon: 1 });
|
||||||
|
},
|
||||||
|
error: function () {
|
||||||
|
var demoText = $('#imagePreview');
|
||||||
|
demoText.append('<span style="color: #FF5722;">上传失败</span> <a class="layui-btn layui-btn-xs demo-reload">重试</a>');
|
||||||
|
demoText.find('.demo-reload').on('click', function () {
|
||||||
|
uploadInst.upload();
|
||||||
|
});
|
||||||
|
},
|
||||||
|
progress: function (n, elem, e) {
|
||||||
|
element.progress('image-progress', n + '%');
|
||||||
|
if (n == 100) {
|
||||||
|
layer.msg('图片上传完毕', { icon: 1 });
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
// 获取分类列表
|
||||||
|
function loadCategories() {
|
||||||
|
var categories = { $categories| json_encode | raw
|
||||||
|
};
|
||||||
|
var html = '<option value="">请选择分类</option>';
|
||||||
|
categories.forEach(function (item) {
|
||||||
|
var disabled = item.cid == 0 ? 'disabled' : '';
|
||||||
|
html += '<option value="' + item.id + '" ' + disabled + '>' + item.name + '</option>';
|
||||||
|
if (item.children && item.children.length > 0) {
|
||||||
|
item.children.forEach(function (child) {
|
||||||
|
html += '<option value="' + child.id + '">├─ ' + child.name + '</option>';
|
||||||
|
});
|
||||||
|
}
|
||||||
|
});
|
||||||
|
$('select[name="cate"]').html(html);
|
||||||
|
form.render('select');
|
||||||
|
window.categoryData = categories;
|
||||||
|
}
|
||||||
|
loadCategories();
|
||||||
|
|
||||||
|
// 递归查找分类信息的函数
|
||||||
|
function findCategory(categories, targetId) {
|
||||||
|
for (let category of categories) {
|
||||||
|
if (category.id == targetId) {
|
||||||
|
return {
|
||||||
|
parent: null,
|
||||||
|
current: category,
|
||||||
|
total: category.total || 0
|
||||||
|
};
|
||||||
|
}
|
||||||
|
if (category.children && category.children.length > 0) {
|
||||||
|
for (let child of category.children) {
|
||||||
|
if (child.id == targetId) {
|
||||||
|
return {
|
||||||
|
parent: category,
|
||||||
|
current: child,
|
||||||
|
total: child.total || 0
|
||||||
|
};
|
||||||
|
}
|
||||||
|
if (child.children && child.children.length > 0) {
|
||||||
|
const result = findCategory([child], targetId);
|
||||||
|
if (result) {
|
||||||
|
return result;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
// 监听分类选择变化
|
||||||
|
form.on('select(cate)', function (data) {
|
||||||
|
var selectedId = data.value;
|
||||||
|
if (!selectedId) {
|
||||||
|
$('input[name="number"]').val('');
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
const categoryInfo = findCategory(window.categoryData, selectedId);
|
||||||
|
if (categoryInfo) {
|
||||||
|
var nextNumber = categoryInfo.total + 1;
|
||||||
|
var numberStr = nextNumber.toString().padStart(5, '0');
|
||||||
|
var resourceNumber = categoryInfo.parent ? categoryInfo.parent.number + categoryInfo.current.number : categoryInfo.current.number;
|
||||||
|
resourceNumber += numberStr;
|
||||||
|
$('input[name="number"]').val(resourceNumber);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
// 配置 wangeditor 编辑器
|
||||||
|
const { createEditor, createToolbar } = window.wangEditor;
|
||||||
|
const editorConfig = {
|
||||||
|
MENU_CONF: {},
|
||||||
|
placeholder: '请输入内容...',
|
||||||
|
onChange(editor) {
|
||||||
|
const html = editor.getHtml();
|
||||||
|
},
|
||||||
|
};
|
||||||
|
editorConfig.MENU_CONF['uploadImage'] = {
|
||||||
|
server: '{:url("index/upload_img")}',
|
||||||
|
fieldName: 'file',
|
||||||
|
maxFileSize: 50 * 1024 * 1024,
|
||||||
|
maxNumberOfFiles: 10,
|
||||||
|
allowedFileTypes: ['image/*'],
|
||||||
|
meta: { token: 'xxx' },
|
||||||
|
metaWithUrl: true,
|
||||||
|
headers: { Accept: 'text/x-json' },
|
||||||
|
timeout: 30 * 1000,
|
||||||
|
onBeforeUpload(file) {
|
||||||
|
console.log('准备上传图片', file);
|
||||||
|
return file;
|
||||||
|
},
|
||||||
|
onProgress(progress) {
|
||||||
|
console.log('上传进度', progress);
|
||||||
|
},
|
||||||
|
onSuccess(file, res) {
|
||||||
|
console.log('上传成功', file, res);
|
||||||
|
},
|
||||||
|
onFailed(file, res) {
|
||||||
|
layer.msg('上传失败:' + res.msg, { icon: 2 });
|
||||||
|
console.log('上传失败', file, res);
|
||||||
|
},
|
||||||
|
onError(file, err, res) {
|
||||||
|
layer.msg('上传出错:' + err.message, { icon: 2 });
|
||||||
|
console.error('上传出错', file, err, res);
|
||||||
|
},
|
||||||
|
customInsert(res, insertFn) {
|
||||||
|
if (res.code === 0 && res.url) {
|
||||||
|
let imageUrl = res.url;
|
||||||
|
if (!imageUrl.startsWith('http')) {
|
||||||
|
imageUrl = 'https://' + imageUrl;
|
||||||
|
}
|
||||||
|
imageUrl = imageUrl.replace(/^https?:\/\/[^\/]+\/admin\/resources\//, 'https://www.yunzer.cn/');
|
||||||
|
insertFn(imageUrl);
|
||||||
|
} else {
|
||||||
|
layer.msg('图片上传失败:' + (res.msg || '未知错误'), { icon: 2 });
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
const editor = createEditor({
|
||||||
|
selector: '#editor-container',
|
||||||
|
html: '<p><br></p>',
|
||||||
|
config: editorConfig,
|
||||||
|
mode: 'default',
|
||||||
|
});
|
||||||
|
const toolbar = createToolbar({
|
||||||
|
editor,
|
||||||
|
selector: '#toolbar-container',
|
||||||
|
config: {},
|
||||||
|
mode: 'default',
|
||||||
|
});
|
||||||
|
|
||||||
|
// 表单提交
|
||||||
|
form.on('submit(formSubmit)', function (data) {
|
||||||
|
var content = editor.getHtml();
|
||||||
|
if (!content || content === '<p><br></p>') {
|
||||||
|
layer.msg('请输入文章内容', { icon: 2 });
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
var loadIndex = layer.load(2);
|
||||||
|
data.field.content = content;
|
||||||
|
$.ajax({
|
||||||
|
url: '{:url("resources/add")}',
|
||||||
|
type: 'POST',
|
||||||
|
data: data.field,
|
||||||
|
success: function (res) {
|
||||||
|
layer.close(loadIndex);
|
||||||
|
if (res.code == 0) {
|
||||||
|
layer.msg(res.msg, { icon: 1 });
|
||||||
|
setTimeout(function () {
|
||||||
|
window.location.href = '{:url("resources/lists")}';
|
||||||
|
}, 1000);
|
||||||
|
} else {
|
||||||
|
layer.msg(res.msg, { icon: 2 });
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
return false;
|
||||||
|
});
|
||||||
|
|
||||||
|
// 重置按钮点击事件
|
||||||
|
$('button[type="reset"]').on('click', function () {
|
||||||
|
loadCategories();
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
//返回资源列表
|
||||||
|
function goBack() {
|
||||||
|
window.location.href = '{:url("resources/lists")}';
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<!-- wangeditor编辑器脚本 -->
|
||||||
|
<script>
|
||||||
|
const { createEditor, createToolbar } = window.wangEditor
|
||||||
|
|
||||||
|
const editorConfig = {
|
||||||
|
MENU_CONF: {},
|
||||||
|
placeholder: '请输入内容...',
|
||||||
|
onChange(editor) {
|
||||||
|
const html = editor.getHtml()
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
// 配置图片上传
|
||||||
|
editorConfig.MENU_CONF['uploadImage'] = {
|
||||||
|
server: '{:url("index/upload_img")}',
|
||||||
|
fieldName: 'file',
|
||||||
|
maxFileSize: 50 * 1024 * 1024, // 50M
|
||||||
|
maxNumberOfFiles: 10,
|
||||||
|
allowedFileTypes: ['image/*'],
|
||||||
|
meta: {
|
||||||
|
token: 'xxx'
|
||||||
|
},
|
||||||
|
metaWithUrl: true,
|
||||||
|
headers: {
|
||||||
|
Accept: 'text/x-json'
|
||||||
|
},
|
||||||
|
timeout: 30 * 1000, // 30s
|
||||||
|
|
||||||
|
onBeforeUpload(file) {
|
||||||
|
console.log('准备上传图片', file)
|
||||||
|
return file
|
||||||
|
},
|
||||||
|
onProgress(progress) {
|
||||||
|
console.log('上传进度', progress)
|
||||||
|
},
|
||||||
|
onSuccess(file, res) {
|
||||||
|
console.log('上传成功', file, res)
|
||||||
|
},
|
||||||
|
onFailed(file, res) {
|
||||||
|
layer.msg('上传失败:' + res.msg, { icon: 2 })
|
||||||
|
console.log('上传失败', file, res)
|
||||||
|
},
|
||||||
|
onError(file, err, res) {
|
||||||
|
layer.msg('上传出错:' + err.message, { icon: 2 })
|
||||||
|
console.error('上传出错', file, err, res)
|
||||||
|
},
|
||||||
|
customInsert(res, insertFn) {
|
||||||
|
// 只使用返回的url字段,并确保使用完整的URL
|
||||||
|
if (res.code === 0 && res.url) {
|
||||||
|
// 如果URL不是以http开头,添加https://
|
||||||
|
let imageUrl = res.url;
|
||||||
|
if (!imageUrl.startsWith('http')) {
|
||||||
|
imageUrl = 'https://' + imageUrl;
|
||||||
|
}
|
||||||
|
// 移除可能存在的重复域名和路径
|
||||||
|
imageUrl = imageUrl.replace(/^https?:\/\/[^\/]+\/admin\/resources\//, 'https://www.yunzer.cn/');
|
||||||
|
insertFn(imageUrl);
|
||||||
|
} else {
|
||||||
|
layer.msg('图片上传失败:' + (res.msg || '未知错误'), { icon: 2 });
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const editor = createEditor({
|
||||||
|
selector: '#editor-container',
|
||||||
|
html: '<p><br></p>',
|
||||||
|
config: editorConfig,
|
||||||
|
mode: 'default', // or 'simple'
|
||||||
|
})
|
||||||
|
|
||||||
|
const toolbarConfig = {}
|
||||||
|
|
||||||
|
const toolbar = createToolbar({
|
||||||
|
editor,
|
||||||
|
selector: '#toolbar-container',
|
||||||
|
config: toolbarConfig,
|
||||||
|
mode: 'default', // or 'simple'
|
||||||
|
})
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
//返回资源列表
|
||||||
|
function goBack() {
|
||||||
|
window.location.href = '{:url("resources/lists")}';
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<style>
|
||||||
|
.form-container {
|
||||||
|
display: flex;
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
.container-left {
|
||||||
|
width: 35%;
|
||||||
|
}
|
||||||
|
|
||||||
|
.container-right {
|
||||||
|
width: 65%;
|
||||||
|
}
|
||||||
|
</style>
|
||||||
533
app/admin/view/resources/cate.php
Normal file
533
app/admin/view/resources/cate.php
Normal file
@ -0,0 +1,533 @@
|
|||||||
|
{include file="public/header" /}
|
||||||
|
<div class="config-container">
|
||||||
|
<!-- 页面头部样式 -->
|
||||||
|
<div class="config-header" style="display: flex;flex-direction: column;flex-wrap: wrap;align-items: flex-start;">
|
||||||
|
<div class="maintitle">
|
||||||
|
<i class="layui-icon layui-icon-app"></i>
|
||||||
|
<span>资源分类管理</span>
|
||||||
|
</div>
|
||||||
|
<div style="display: flex;align-items: flex-start;flex-direction: column;gap: 15px;margin-bottom: 10px;">
|
||||||
|
<div>
|
||||||
|
<button type="button" class="layui-btn layui-btn-normal" onclick="add()">
|
||||||
|
<i class="layui-icon layui-icon-add-1"></i>添加分类
|
||||||
|
</button>
|
||||||
|
<button type="button" class="layui-btn layui-btn-primary layui-border-blue" onclick="refresh()">
|
||||||
|
<i class="layui-icon layui-icon-refresh"></i>刷新
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<!-- 主要内容区 -->
|
||||||
|
<div class="main-content">
|
||||||
|
<div class="layui-row layui-col-space20">
|
||||||
|
<!-- 左侧分类列表 -->
|
||||||
|
<div class="layui-col-md7">
|
||||||
|
<div class="layui-card">
|
||||||
|
<div class="layui-card-header">
|
||||||
|
<span>分类列表</span>
|
||||||
|
<small class="text-muted">支持两级分类结构</small>
|
||||||
|
</div>
|
||||||
|
<div class="layui-card-body">
|
||||||
|
<div id="categoryList" class="category-tree"></div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<!-- 右侧分类信息 -->
|
||||||
|
<div class="layui-col-md5">
|
||||||
|
<div class="layui-card">
|
||||||
|
<div class="layui-card-header">
|
||||||
|
<span>分类信息</span>
|
||||||
|
</div>
|
||||||
|
<div class="layui-card-body">
|
||||||
|
<!-- 默认提示 -->
|
||||||
|
<div id="defaultTip" class="empty-tip">
|
||||||
|
<i class="layui-icon layui-icon-face-surprised"></i>
|
||||||
|
<p>请选择左侧分类或点击新增按钮</p>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<!-- 分类表单 -->
|
||||||
|
<form class="layui-form category-form" lay-filter="categoryForm" style="display: none;">
|
||||||
|
<input type="hidden" name="id" id="categoryId">
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">分类编号</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<input type="text" name="number" required lay-verify="required" placeholder="请输入分类编号"
|
||||||
|
autocomplete="off" class="layui-input">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">分类名称</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<input type="text" name="name" required lay-verify="required" placeholder="请输入分类名称"
|
||||||
|
autocomplete="off" class="layui-input">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">父级分类</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<select name="cid" lay-verify="required">
|
||||||
|
<option value="0">顶级分类</option>
|
||||||
|
</select>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">分类图标</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<div class="layui-upload-drag" id="uploadImage">
|
||||||
|
<i class="layui-icon layui-icon-upload"></i>
|
||||||
|
<p>点击上传或拖拽图片至此处</p>
|
||||||
|
<div class="layui-hide" id="uploadPreview">
|
||||||
|
<hr>
|
||||||
|
<img src="" alt="分类图标" style="max-width: 100%">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<input type="hidden" name="icon" id="icon">
|
||||||
|
<div class="layui-form-mid layui-word-aux">建议尺寸:250px * 140px</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">排序</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<input type="number" name="sort" value="0" class="layui-input"
|
||||||
|
placeholder="数字越大越靠前">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">状态</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<input type="radio" name="status" value="1" title="正常" checked>
|
||||||
|
<input type="radio" name="status" value="0" title="禁用">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="layui-form-item form-actions">
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<button class="layui-btn" lay-submit lay-filter="saveCategory">保存</button>
|
||||||
|
<button type="reset" class="layui-btn layui-btn-primary">重置</button>
|
||||||
|
<button type="button" class="layui-btn layui-btn-danger" id="deleteBtn"
|
||||||
|
style="display: none;">删除</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</form>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<style>
|
||||||
|
/* 页面整体样式 */
|
||||||
|
.config-container {
|
||||||
|
padding: 15px;
|
||||||
|
/* background: #f2f2f2; */
|
||||||
|
}
|
||||||
|
|
||||||
|
.layui-col-md7 .layui-btn-primary {
|
||||||
|
border-color: #d2d2d2;
|
||||||
|
background: 0 0;
|
||||||
|
color: #5f5f5f
|
||||||
|
}
|
||||||
|
|
||||||
|
.layui-col-md7 .layui-btn-primary:hover {
|
||||||
|
background-color: #1e9fff;
|
||||||
|
color: #efefef
|
||||||
|
}
|
||||||
|
|
||||||
|
/* 页面头部样式 */
|
||||||
|
.page-header {
|
||||||
|
display: flex;
|
||||||
|
justify-content: space-between;
|
||||||
|
align-items: center;
|
||||||
|
margin-bottom: 15px;
|
||||||
|
padding: 10px 15px;
|
||||||
|
background: #fff;
|
||||||
|
border-radius: 2px;
|
||||||
|
box-shadow: 0 1px 2px rgba(0, 0, 0, 0.05);
|
||||||
|
}
|
||||||
|
|
||||||
|
.header-title {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
font-size: 16px;
|
||||||
|
font-weight: 500;
|
||||||
|
}
|
||||||
|
|
||||||
|
.header-title .layui-icon {
|
||||||
|
margin-right: 8px;
|
||||||
|
font-size: 18px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.header-actions .layui-btn {
|
||||||
|
margin-left: 8px;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* 主要内容区样式 */
|
||||||
|
.main-content {
|
||||||
|
min-height: calc(100vh - 170px);
|
||||||
|
}
|
||||||
|
|
||||||
|
.layui-card {
|
||||||
|
margin-bottom: 0;
|
||||||
|
border-radius: 8px;
|
||||||
|
box-shadow: 0 3px 6px rgba(0, 0, 0, 0.1);
|
||||||
|
}
|
||||||
|
|
||||||
|
.layui-card-header {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
height: auto;
|
||||||
|
padding: 12px 15px;
|
||||||
|
font-size: 15px;
|
||||||
|
font-weight: 500;
|
||||||
|
}
|
||||||
|
|
||||||
|
.text-muted {
|
||||||
|
margin-left: 10px;
|
||||||
|
font-size: 12px;
|
||||||
|
font-weight: normal;
|
||||||
|
color: #999;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* 分类树样式 */
|
||||||
|
.category-tree {
|
||||||
|
padding: 10px 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.category-item {
|
||||||
|
margin: 5px 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.category-header {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: space-between;
|
||||||
|
padding: 10px 15px;
|
||||||
|
background-color: #f8f8f8;
|
||||||
|
border-radius: 2px;
|
||||||
|
transition: all 0.3s;
|
||||||
|
cursor: pointer;
|
||||||
|
}
|
||||||
|
|
||||||
|
.category-header:hover {
|
||||||
|
background-color: #f2f2f2;
|
||||||
|
}
|
||||||
|
|
||||||
|
.category-header.active {
|
||||||
|
background-color: #e6f7ff;
|
||||||
|
border-right: 3px solid #1890ff;
|
||||||
|
}
|
||||||
|
|
||||||
|
.category-name {
|
||||||
|
flex: 1;
|
||||||
|
font-size: 14px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.category-actions {
|
||||||
|
/* opacity: 0; */
|
||||||
|
transition: opacity 0.3s;
|
||||||
|
}
|
||||||
|
|
||||||
|
.category-header:hover .category-actions {
|
||||||
|
opacity: 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
.category-children {
|
||||||
|
margin-left: 20px;
|
||||||
|
padding-left: 15px;
|
||||||
|
border-left: 1px dashed #e6e6e6;
|
||||||
|
}
|
||||||
|
|
||||||
|
.add-child {
|
||||||
|
padding: 3px 8px;
|
||||||
|
font-size: 12px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.add-child .layui-icon {
|
||||||
|
font-size: 12px;
|
||||||
|
margin-right: 3px;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* 空状态提示 */
|
||||||
|
.empty-tip {
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
padding: 40px 0;
|
||||||
|
color: #999;
|
||||||
|
}
|
||||||
|
|
||||||
|
.empty-tip .layui-icon {
|
||||||
|
font-size: 32px;
|
||||||
|
margin-bottom: 10px;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* 表单样式优化 */
|
||||||
|
.category-form {
|
||||||
|
padding: 10px 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.layui-form-label {
|
||||||
|
width: 100px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.layui-input-block {
|
||||||
|
margin-left: 130px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.form-actions {
|
||||||
|
margin-top: 30px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.layui-upload-drag {
|
||||||
|
padding: 20px;
|
||||||
|
border: 1px dashed #e2e2e2;
|
||||||
|
background-color: #fff;
|
||||||
|
text-align: center;
|
||||||
|
cursor: pointer;
|
||||||
|
}
|
||||||
|
|
||||||
|
.layui-upload-drag:hover {
|
||||||
|
border-color: #009688;
|
||||||
|
}
|
||||||
|
|
||||||
|
.layui-upload-drag img {
|
||||||
|
max-width: 100%;
|
||||||
|
margin: 10px 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.layui-btn-xs {
|
||||||
|
height: 30px;
|
||||||
|
line-height: 25px;
|
||||||
|
font-size: 12px;
|
||||||
|
}
|
||||||
|
</style>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
// 定义全局变量和函数
|
||||||
|
var categoryManager = {
|
||||||
|
init: function () {
|
||||||
|
this.initLayui();
|
||||||
|
},
|
||||||
|
|
||||||
|
initLayui: function () {
|
||||||
|
var that = this;
|
||||||
|
layui.use(['layer', 'form', 'upload'], function () {
|
||||||
|
var layer = layui.layer;
|
||||||
|
var form = layui.form;
|
||||||
|
var upload = layui.upload;
|
||||||
|
var $ = layui.jquery;
|
||||||
|
|
||||||
|
// 初始化分类列表
|
||||||
|
that.initCategoryList = function () {
|
||||||
|
$.ajax({
|
||||||
|
url: '/admin/resources/cate',
|
||||||
|
type: 'POST',
|
||||||
|
success: function (res) {
|
||||||
|
if (res.code === 0) {
|
||||||
|
var html = '';
|
||||||
|
res.data.forEach(function (item) {
|
||||||
|
html += that.renderCategory(item);
|
||||||
|
});
|
||||||
|
$('#categoryList').html(html);
|
||||||
|
that.bindEvents();
|
||||||
|
} else {
|
||||||
|
layer.msg('获取分类数据失败', { icon: 2 });
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
// 渲染分类项
|
||||||
|
that.renderCategory = function (category, level = 0) {
|
||||||
|
var html = '<div class="category-item" data-id="' + category.id + '">';
|
||||||
|
html += '<div class="category-header">';
|
||||||
|
html += '<div class="category-name">' + category.title + '</div>';
|
||||||
|
if (level === 0) {
|
||||||
|
html += '<div class="category-actions">';
|
||||||
|
html += '<button type="button" class="layui-btn layui-btn-primary layui-btn-xs add-child">';
|
||||||
|
html += '<i class="layui-icon layui-icon-add-1"></i>添加子分类</button>';
|
||||||
|
html += '</div>';
|
||||||
|
}
|
||||||
|
html += '</div>';
|
||||||
|
|
||||||
|
if (category.children && category.children.length > 0) {
|
||||||
|
html += '<div class="category-children">';
|
||||||
|
category.children.forEach(function (child) {
|
||||||
|
html += that.renderCategory(child, level + 1);
|
||||||
|
});
|
||||||
|
html += '</div>';
|
||||||
|
}
|
||||||
|
html += '</div>';
|
||||||
|
return html;
|
||||||
|
};
|
||||||
|
|
||||||
|
// 绑定事件
|
||||||
|
that.bindEvents = function () {
|
||||||
|
// 点击分类项加载编辑信息
|
||||||
|
$('.category-header').off('click').on('click', function (e) {
|
||||||
|
if (!$(e.target).closest('.add-child').length) {
|
||||||
|
var id = $(this).closest('.category-item').data('id');
|
||||||
|
that.loadCategoryInfo(id);
|
||||||
|
|
||||||
|
// 添加选中效果
|
||||||
|
$('.category-header').removeClass('active');
|
||||||
|
$(this).addClass('active');
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
// 添加子分类
|
||||||
|
$('.add-child').off('click').on('click', function (e) {
|
||||||
|
e.stopPropagation();
|
||||||
|
var parentId = $(this).closest('.category-item').data('id');
|
||||||
|
that.showCategoryForm(parentId);
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
// 加载分类信息
|
||||||
|
that.loadCategoryInfo = function (id) {
|
||||||
|
$.get('/admin/resources/cateedit?id=' + id, function (res) {
|
||||||
|
if (res.code === 0) {
|
||||||
|
that.showCategoryForm(0, res.data);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
// 显示分类表单
|
||||||
|
that.showCategoryForm = function (parentId = 0, data = null) {
|
||||||
|
$('#defaultTip').hide();
|
||||||
|
$('.category-form').show();
|
||||||
|
|
||||||
|
// 重置表单
|
||||||
|
form.val('categoryForm', {
|
||||||
|
id: data ? data.info.id : '',
|
||||||
|
name: data ? data.info.name : '',
|
||||||
|
number: data ? data.info.number : '',
|
||||||
|
cid: data ? data.info.cid : parentId,
|
||||||
|
sort: data ? data.info.sort : 0,
|
||||||
|
status: data ? data.info.status : 1
|
||||||
|
});
|
||||||
|
|
||||||
|
// 更新父级分类选项
|
||||||
|
var $select = $('select[name="cid"]');
|
||||||
|
$select.empty().append('<option value="0">顶级分类</option>');
|
||||||
|
// 获取所有分类作为父级选项
|
||||||
|
$.ajax({
|
||||||
|
url: '/admin/resources/cate',
|
||||||
|
type: 'POST',
|
||||||
|
async: false,
|
||||||
|
success: function (res) {
|
||||||
|
if (res.code === 0) {
|
||||||
|
// 当前编辑的分类ID
|
||||||
|
var currentId = data ? data.info.id : 0;
|
||||||
|
// 递归构建分类选项
|
||||||
|
function buildOptions(categories, level) {
|
||||||
|
categories.forEach(function (category) {
|
||||||
|
// 不能选择自己或自己的子分类作为父级
|
||||||
|
if (category.id != currentId) {
|
||||||
|
var prefix = new Array(level + 1).join('├─ ');
|
||||||
|
$select.append('<option value="' + category.id + '">' + prefix + category.title + '</option>'); if (category.children && category.children.length > 0) { buildOptions(category.children, level + 1); }
|
||||||
|
}
|
||||||
|
});
|
||||||
|
} buildOptions(res.data, 0);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
// 设置选中的父级分类
|
||||||
|
if (data && data.info.cid) {
|
||||||
|
$select.val(data.info.cid);
|
||||||
|
}
|
||||||
|
|
||||||
|
// 更新图标预览
|
||||||
|
if (data && data.info.icon) {
|
||||||
|
$('#uploadPreview').removeClass('layui-hide').find('img').attr('src', data.info.icon);
|
||||||
|
$('#icon').val(data.info.icon);
|
||||||
|
} else {
|
||||||
|
$('#uploadPreview').addClass('layui-hide').find('img').attr('src', '');
|
||||||
|
$('#icon').val('');
|
||||||
|
}
|
||||||
|
|
||||||
|
// 显示/隐藏删除按钮
|
||||||
|
$('#deleteBtn')[data ? 'show' : 'hide']();
|
||||||
|
|
||||||
|
form.render();
|
||||||
|
};
|
||||||
|
|
||||||
|
// 初始化上传组件
|
||||||
|
upload.render({
|
||||||
|
elem: '#uploadImage',
|
||||||
|
url: '/admin/index/upload_img',
|
||||||
|
accept: 'images',
|
||||||
|
acceptMime: 'image/*',
|
||||||
|
done: function (res) {
|
||||||
|
if (res.code === 0) {
|
||||||
|
$('#uploadPreview').removeClass('layui-hide').find('img').attr('src', res.data);
|
||||||
|
$('#icon').val(res.data);
|
||||||
|
layer.msg('上传成功');
|
||||||
|
} else {
|
||||||
|
layer.msg('上传失败');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
// 监听表单提交
|
||||||
|
form.on('submit(saveCategory)', function (data) {
|
||||||
|
var url = data.field.id ? '/admin/resources/cateedit' : '/admin/resources/cateadd';
|
||||||
|
$.post(url, data.field, function (res) {
|
||||||
|
if (res.code === 0) {
|
||||||
|
layer.msg(res.msg, { icon: 1 });
|
||||||
|
that.initCategoryList();
|
||||||
|
} else {
|
||||||
|
layer.msg(res.msg, { icon: 2 });
|
||||||
|
}
|
||||||
|
});
|
||||||
|
return false;
|
||||||
|
});
|
||||||
|
|
||||||
|
// 监听删除按钮
|
||||||
|
$('#deleteBtn').on('click', function () {
|
||||||
|
var id = $('#categoryId').val();
|
||||||
|
if (!id) return;
|
||||||
|
|
||||||
|
layer.confirm('确定要删除该分类吗?', function (index) {
|
||||||
|
$.post('/admin/resources/catedel', { id: id }, function (res) {
|
||||||
|
if (res.code === 0) {
|
||||||
|
layer.msg(res.msg, { icon: 1 });
|
||||||
|
that.initCategoryList();
|
||||||
|
$('#defaultTip').show();
|
||||||
|
$('.category-form').hide();
|
||||||
|
} else {
|
||||||
|
layer.msg(res.msg, { icon: 2 });
|
||||||
|
}
|
||||||
|
});
|
||||||
|
layer.close(index);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
// 初始化页面
|
||||||
|
that.initCategoryList();
|
||||||
|
});
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
// 初始化
|
||||||
|
categoryManager.init();
|
||||||
|
|
||||||
|
// 新增分类
|
||||||
|
function add() {
|
||||||
|
categoryManager.showCategoryForm();
|
||||||
|
}
|
||||||
|
|
||||||
|
// 刷新列表
|
||||||
|
function refresh() {
|
||||||
|
categoryManager.initCategoryList();
|
||||||
|
}
|
||||||
|
</script>
|
||||||
872
app/admin/view/resources/edit.php
Normal file
872
app/admin/view/resources/edit.php
Normal file
@ -0,0 +1,872 @@
|
|||||||
|
{include file="public/header" /}
|
||||||
|
<div class="config-container">
|
||||||
|
<div class="config-header" style="display:flex;justify-content: space-between;">
|
||||||
|
<div>
|
||||||
|
<span>编辑资源</span>
|
||||||
|
</div>
|
||||||
|
<div>
|
||||||
|
<button type="button" class="layui-btn layui-btn-primary layui-btn-sm" onclick="goBack()">
|
||||||
|
<i class="layui-icon layui-icon-return"></i>返回
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<form class="layui-form" action="" method="post">
|
||||||
|
<input type="hidden" name="id" value="{$resource.id|default=''}">
|
||||||
|
<div class="form-container">
|
||||||
|
<div class="container-left">
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">资源名称</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<input type="text" name="title" required lay-verify="required" placeholder="请输入资源名称"
|
||||||
|
autocomplete="off" class="layui-input" value="{$resource.title|default=''}"
|
||||||
|
lay-affix="clear">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">分类</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<select name="cate" lay-verify="required" lay-filter="cate">
|
||||||
|
<option value="">请选择分类</option>
|
||||||
|
</select>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">资源编号</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<input type="text" name="number" required lay-verify="required" placeholder="请输入分类编号"
|
||||||
|
autocomplete="off" class="layui-input" value="{$resource.number|default=''}"
|
||||||
|
lay-affix="clear" disabled>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">描述</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<textarea name="desc" placeholder="请输入资源描述" class="layui-textarea"
|
||||||
|
lay-affix="clear">{$resource.desc|default=''}</textarea>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">上传者</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<input type="text" name="uploader" required lay-verify="required" placeholder="请输入上传者"
|
||||||
|
autocomplete="off" class="layui-input" value="{$resource.uploader|default=''}"
|
||||||
|
lay-affix="clear">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">排序</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<input type="number" name="sort" value="{$resource.sort|default='0'}" class="layui-input"
|
||||||
|
placeholder="数字越大越靠前" lay-affix="clear">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">资源图标</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<button type="button" class="layui-btn" id="upload-btn">
|
||||||
|
<i class="layui-icon layui-icon-upload"></i> 图标上传
|
||||||
|
</button>
|
||||||
|
<div style="width: 120px;">
|
||||||
|
<div class="layui-upload-list">
|
||||||
|
<img class="layui-upload-img" id="upload-img"
|
||||||
|
style="width: 118px; height: 118px;object-fit: cover;">
|
||||||
|
<div id="upload-text"></div>
|
||||||
|
</div>
|
||||||
|
<div class="layui-progress layui-progress-big" lay-showPercent="yes"
|
||||||
|
lay-filter="icon-progress">
|
||||||
|
<div class="layui-progress-bar" lay-percent=""></div>
|
||||||
|
</div>
|
||||||
|
<input type="hidden" name="icon" id="icon" value="">
|
||||||
|
</div>
|
||||||
|
<div class="layui-form-mid layui-word-aux">建议尺寸:250px * 140px</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">图片上传</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<div class="image-upload-container">
|
||||||
|
<button type="button" class="btn btn-primary" id="imageUpload">
|
||||||
|
<i class="fas fa-upload"></i> 多图片上传
|
||||||
|
</button>
|
||||||
|
<div class="image-preview-container" id="imagePreview">
|
||||||
|
{if condition="isset($resource.images) && !empty($resource.images)"}
|
||||||
|
<div class="image-list-table">
|
||||||
|
<table class="layui-table">
|
||||||
|
<thead>
|
||||||
|
<tr>
|
||||||
|
<th width="100" align="center">缩略图</th>
|
||||||
|
<th width="40" align="center">操作</th>
|
||||||
|
</tr>
|
||||||
|
</thead>
|
||||||
|
<tbody>
|
||||||
|
{php}
|
||||||
|
// 处理图片字符串,正确拆分逗号分隔的路径
|
||||||
|
$imageArray = [];
|
||||||
|
if (isset($resource['images']) && !empty($resource['images'])) {
|
||||||
|
$imageArray = explode(',', $resource['images']);
|
||||||
|
$imageArray = array_map('trim', $imageArray); // 去除每个路径的前后空格
|
||||||
|
$imageArray = array_filter($imageArray); // 过滤空值
|
||||||
|
}
|
||||||
|
{/php}
|
||||||
|
{volist name="imageArray" id="image"}
|
||||||
|
<tr>
|
||||||
|
<td>
|
||||||
|
<div class="image-thumbnail" onclick="previewImage('<?php echo (strpos($image, 'http') === 0 ? $image : request()->domain() . $image); ?>')">
|
||||||
|
<img src="<?php echo (strpos($image, 'http') === 0 ? $image : request()->domain() . $image); ?>" alt="缩略图">
|
||||||
|
</div>
|
||||||
|
</td>
|
||||||
|
<td>
|
||||||
|
<button type="button" class="layui-btn layui-btn-danger layui-btn-xs delete-image" data-src="{$image}">
|
||||||
|
<i class="fas fa-trash"></i> 删除
|
||||||
|
</button>
|
||||||
|
</td>
|
||||||
|
</tr>
|
||||||
|
{/volist}
|
||||||
|
</tbody>
|
||||||
|
</table>
|
||||||
|
</div>
|
||||||
|
{/if}
|
||||||
|
</div>
|
||||||
|
<div class="upload-progress" id="imageProgress" style="display: none;">
|
||||||
|
<div class="progress-bar" role="progressbar" style="width: 0%"></div>
|
||||||
|
</div>
|
||||||
|
<input type="hidden" name="images" id="images" value="{$resource.images|default=''}">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">资源文件</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<input type="text" name="fileurl" required placeholder="本地资源地址" autocomplete="off"
|
||||||
|
class="layui-input" value="{$resource.fileurl|default=''}" style="margin-bottom: 10px;"
|
||||||
|
lay-affix="clear">
|
||||||
|
<div class="layui-upload-drag" style="display: block;" id="ID-upload-demo-drag">
|
||||||
|
<i class="layui-icon layui-icon-upload"></i>
|
||||||
|
<div>点击上传,或将文件拖拽到此处</div>
|
||||||
|
<div class="layui-hide" id="ID-upload-demo-preview">
|
||||||
|
<hr>
|
||||||
|
<div class="file-info">
|
||||||
|
<i class="layui-icon layui-icon-file"></i>
|
||||||
|
<span class="file-name"></span>
|
||||||
|
<span class="file-size"></span>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layui-progress layui-progress-big" lay-showPercent="yes" lay-filter="file-progress"
|
||||||
|
style="margin-top: 10px;">
|
||||||
|
<div class="layui-progress-bar" lay-percent=""></div>
|
||||||
|
</div>
|
||||||
|
<input type="hidden" name="file" id="file" value="">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">资源链接</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<input type="text" name="url" required placeholder="百度网盘、115网盘、蓝奏云等" autocomplete="off"
|
||||||
|
class="layui-input" value="{$resource.url|default=''}" lay-affix="clear">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">分享码</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<input type="text" name="code" required placeholder="请输入分享码" autocomplete="off"
|
||||||
|
class="layui-input" value="{$resource.code|default=''}" lay-affix="clear">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">解压密码</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<input type="text" name="zipcode" required placeholder="请输入解压密码" autocomplete="off"
|
||||||
|
class="layui-input" value="{$resource.zipcode|default=''}" lay-affix="clear">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="container-right">
|
||||||
|
<div class="layui-form-item layui-form-text">
|
||||||
|
<label class="layui-form-label">内容</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<div id="editor—wrapper" id="content" name="content" style="border: 1px solid #ccc;">
|
||||||
|
<div id="toolbar-container" style="border-bottom: 1px solid #ccc;"><!-- 工具栏 --></div>
|
||||||
|
<div id="editor-container" style="height: 800px;"><!-- 编辑器 --></div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="layui-form-item" style="margin-top: 80px;">
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<button class="layui-btn" lay-submit lay-filter="formSubmit">立即提交</button>
|
||||||
|
<button type="reset" class="layui-btn layui-btn-primary">重置</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
</form>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<script src="/static/js/wangeditor.js"></script>
|
||||||
|
<script>
|
||||||
|
layui.use(['form', 'layer'], function () {
|
||||||
|
var form = layui.form;
|
||||||
|
var layer = layui.layer;
|
||||||
|
var $ = layui.$;
|
||||||
|
var upload = layui.upload;
|
||||||
|
var element = layui.element;
|
||||||
|
|
||||||
|
// 获取资源详情
|
||||||
|
var resourceId = $('input[name="id"]').val();
|
||||||
|
var resourceData = null;
|
||||||
|
|
||||||
|
if (resourceId) {
|
||||||
|
$.get('{:url("resources/get")}', { id: resourceId }, function (res) {
|
||||||
|
if (res.code == 0) {
|
||||||
|
resourceData = res.data;
|
||||||
|
console.log('Resource data:', resourceData); // 调试输出
|
||||||
|
|
||||||
|
// 设置表单值
|
||||||
|
$('input[name="title"]').val(resourceData.title || '');
|
||||||
|
$('select[name="cate"]').val(resourceData.cate || '');
|
||||||
|
form.render('select'); // 重新渲染select以显示选中值
|
||||||
|
$('textarea[name="desc"]').val(resourceData.desc || '');
|
||||||
|
$('input[name="uploader"]').val(resourceData.uploader || '');
|
||||||
|
$('input[name="url"]').val(resourceData.url || '');
|
||||||
|
$('input[name="code"]').val(resourceData.code || '');
|
||||||
|
$('input[name="sort"]').val(resourceData.sort || '0');
|
||||||
|
$('input[name="icon"]').val(resourceData.icon || '');
|
||||||
|
$('input[name="file"]').val(resourceData.file || '');
|
||||||
|
|
||||||
|
// 设置图标预览
|
||||||
|
if (resourceData.icon) {
|
||||||
|
$('#upload-img').attr('src', resourceData.icon);
|
||||||
|
}
|
||||||
|
|
||||||
|
// 设置文件预览
|
||||||
|
if (resourceData.file) {
|
||||||
|
$('#ID-upload-demo-preview').show();
|
||||||
|
$('.file-name').text(resourceData.file_name || '已上传文件');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
// 图标上传
|
||||||
|
var iconUpload = upload.render({
|
||||||
|
elem: '#upload-btn',
|
||||||
|
url: '{:url("index/upload_img")}',
|
||||||
|
before: function (obj) {
|
||||||
|
obj.preview(function (index, file, result) {
|
||||||
|
$('#upload-img').attr('src', result);
|
||||||
|
});
|
||||||
|
element.progress('icon-progress', '0%');
|
||||||
|
layer.msg('图标上传中', { icon: 16, time: 0 });
|
||||||
|
},
|
||||||
|
done: function (res) {
|
||||||
|
if (res.code > 0) {
|
||||||
|
return layer.msg('图标上传失败');
|
||||||
|
}
|
||||||
|
$('#icon').val(res.data);
|
||||||
|
$('#upload-text').html('');
|
||||||
|
layer.msg('图标上传成功', { icon: 1 });
|
||||||
|
},
|
||||||
|
error: function () {
|
||||||
|
var demoText = $('#upload-text');
|
||||||
|
demoText.html('<span style="color: #FF5722;">上传失败</span> <a class="layui-btn layui-btn-xs demo-reload">重试</a>');
|
||||||
|
demoText.find('.demo-reload').on('click', function () {
|
||||||
|
iconUpload.upload();
|
||||||
|
});
|
||||||
|
},
|
||||||
|
progress: function (n, elem, e) {
|
||||||
|
element.progress('icon-progress', n + '%');
|
||||||
|
if (n == 100) {
|
||||||
|
layer.msg('图标上传完毕', { icon: 1 });
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
// 多图片上传
|
||||||
|
var uploadInst = upload.render({
|
||||||
|
elem: '#imageUpload',
|
||||||
|
url: '{:url("index/upload_img")}',
|
||||||
|
multiple: true,
|
||||||
|
accept: 'images',
|
||||||
|
before: function (obj) {
|
||||||
|
obj.preview(function (index, file, result) {
|
||||||
|
$('#imagePreview').append('<div class="layui-upload-img-item" data-src="' + result + '"><img src="' + result + '" alt="' + file.name + '" style="width: 100px; height: 100px; object-fit: cover;"><p>' + file.name + '</p><button type="button" class="layui-btn layui-btn-xs layui-btn-danger delete-image" style="position: absolute; top: 0; right: 0;">删除</button></div>');
|
||||||
|
});
|
||||||
|
element.progress('image-progress', '0%');
|
||||||
|
layer.msg('图片上传中', { icon: 16, time: 0 });
|
||||||
|
},
|
||||||
|
done: function (res) {
|
||||||
|
if (res.code > 0) {
|
||||||
|
return layer.msg('图片上传失败');
|
||||||
|
}
|
||||||
|
var images = $('#images').val().split(',');
|
||||||
|
if (res.data) {
|
||||||
|
images.push(res.data);
|
||||||
|
}
|
||||||
|
$('#images').val(images.filter(Boolean).join(','));
|
||||||
|
layer.msg('图片上传成功', { icon: 1 });
|
||||||
|
},
|
||||||
|
error: function () {
|
||||||
|
var demoText = $('#imagePreview');
|
||||||
|
demoText.append('<span style="color: #FF5722;">上传失败</span> <a class="layui-btn layui-btn-xs demo-reload">重试</a>');
|
||||||
|
demoText.find('.demo-reload').on('click', function () {
|
||||||
|
uploadInst.upload();
|
||||||
|
});
|
||||||
|
},
|
||||||
|
progress: function (n, elem, e) {
|
||||||
|
element.progress('image-progress', n + '%');
|
||||||
|
if (n == 100) {
|
||||||
|
layer.msg('图片上传完毕', { icon: 1 });
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
// 删除图片功能
|
||||||
|
$('#imagePreview').on('click', '.delete-image', function () {
|
||||||
|
var $item = $(this).closest('.layui-upload-img-item');
|
||||||
|
var imageSrc = $item.data('src');
|
||||||
|
var images = $('#images').val().split(',');
|
||||||
|
var index = images.indexOf(imageSrc);
|
||||||
|
if (index > -1) {
|
||||||
|
images.splice(index, 1);
|
||||||
|
}
|
||||||
|
$('#images').val(images.join(','));
|
||||||
|
$item.remove();
|
||||||
|
});
|
||||||
|
|
||||||
|
// 文件上传
|
||||||
|
var fileUpload = upload.render({
|
||||||
|
elem: '#ID-upload-demo-drag',
|
||||||
|
url: '{:url("index/upload_file")}',
|
||||||
|
accept: 'file',
|
||||||
|
exts: 'doc|docx|xls|xlsx|ppt|pptx|pdf|txt|zip|rar|7z',
|
||||||
|
before: function (obj) {
|
||||||
|
obj.preview(function (index, file, result) {
|
||||||
|
$('#ID-upload-demo-preview').show();
|
||||||
|
$('.file-name').text(file.name);
|
||||||
|
$('.file-size').text(formatFileSize(file.size));
|
||||||
|
});
|
||||||
|
element.progress('file-progress', '0%');
|
||||||
|
layer.msg('文件上传中', { icon: 16, time: 0 });
|
||||||
|
},
|
||||||
|
done: function (res) {
|
||||||
|
if (res.code > 0) {
|
||||||
|
return layer.msg('文件上传失败');
|
||||||
|
}
|
||||||
|
$('#file').val(res.data.src);
|
||||||
|
$('input[name="fileurl"]').val(res.data.src);
|
||||||
|
layer.msg('文件上传成功', { icon: 1 });
|
||||||
|
},
|
||||||
|
uploadError: function () {
|
||||||
|
layer.msg('文件上传失败', { icon: 2 });
|
||||||
|
},
|
||||||
|
progress: function (n, elem, e) {
|
||||||
|
element.progress('file-progress', n + '%');
|
||||||
|
if (n == 100) {
|
||||||
|
layer.msg('文件上传完毕', { icon: 1 });
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
// 格式化文件大小
|
||||||
|
function formatFileSize(bytes) {
|
||||||
|
if (bytes === 0) return '0 B';
|
||||||
|
const k = 1024;
|
||||||
|
const sizes = ['B', 'KB', 'MB', 'GB', 'TB'];
|
||||||
|
const i = Math.floor(Math.log(bytes) / Math.log(k));
|
||||||
|
return parseFloat((bytes / Math.pow(k, i)).toFixed(2)) + ' ' + sizes[i];
|
||||||
|
}
|
||||||
|
|
||||||
|
// 获取分类列表
|
||||||
|
$.get('{:url("resources/getcate")}', function (res) {
|
||||||
|
if (res.code == 0) {
|
||||||
|
// 存储分类数据供后续使用
|
||||||
|
window.categoryData = res.data;
|
||||||
|
|
||||||
|
var html = '<option value="">请选择分类</option>';
|
||||||
|
res.data.forEach(function (item) {
|
||||||
|
html += '<option value="' + item.id + '">' + item.name + '</option>';
|
||||||
|
if (item.children && item.children.length > 0) {
|
||||||
|
item.children.forEach(function (child) {
|
||||||
|
html += '<option value="' + child.id + '">├─ ' + child.name + '</option>';
|
||||||
|
});
|
||||||
|
}
|
||||||
|
});
|
||||||
|
$('select[name="cate"]').html(html);
|
||||||
|
|
||||||
|
// 如果有资源数据,设置分类值
|
||||||
|
if (resourceData && resourceData.cate) {
|
||||||
|
console.log('Setting cate value:', resourceData.cate); // 调试输出
|
||||||
|
setTimeout(function () {
|
||||||
|
$('select[name="cate"]').val(resourceData.cate);
|
||||||
|
form.render('select');
|
||||||
|
}, 100);
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
layer.msg(res.msg, { icon: 2 });
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
// 监听分类选择变化
|
||||||
|
form.on('select(cate)', function (data) {
|
||||||
|
var selectedId = data.value;
|
||||||
|
if (!selectedId) {
|
||||||
|
$('input[name="number"]').val('');
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
// 查找选中的分类信息
|
||||||
|
var parentCategory = null;
|
||||||
|
var childCategory = null;
|
||||||
|
|
||||||
|
window.categoryData.forEach(function (parent) {
|
||||||
|
if (parent.children) {
|
||||||
|
parent.children.forEach(function (child) {
|
||||||
|
if (child.id == selectedId) {
|
||||||
|
parentCategory = parent;
|
||||||
|
childCategory = child;
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
if (parentCategory && childCategory) {
|
||||||
|
// 生成资源编号
|
||||||
|
var total = childCategory.total || 0;
|
||||||
|
// 判断是否是初始化时的分类
|
||||||
|
var isInitialCategory = resourceData && resourceData.cate == selectedId;
|
||||||
|
var nextNumber = isInitialCategory ? total : total + 1;
|
||||||
|
var numberStr = nextNumber.toString().padStart(5, '0');
|
||||||
|
var resourceNumber = parentCategory.number + childCategory.number + numberStr;
|
||||||
|
|
||||||
|
// 设置资源编号
|
||||||
|
$('input[name="number"]').val(resourceNumber);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
// 表单提交
|
||||||
|
form.on('submit(formSubmit)', function (data) {
|
||||||
|
// 获取编辑器内容
|
||||||
|
var content = editor.getHtml();
|
||||||
|
var loadIndex = layer.load(2);
|
||||||
|
data.field.content = content;
|
||||||
|
|
||||||
|
$.ajax({
|
||||||
|
url: '{:url("resources/edit")}',
|
||||||
|
type: 'POST',
|
||||||
|
data: data.field,
|
||||||
|
success: function (res) {
|
||||||
|
layer.close(loadIndex);
|
||||||
|
if (res.code == 0) {
|
||||||
|
layer.msg(res.msg, { icon: 1 });
|
||||||
|
setTimeout(function () {
|
||||||
|
window.location.href = '{:url("resources/lists")}';
|
||||||
|
}, 1000);
|
||||||
|
} else {
|
||||||
|
layer.msg(res.msg, { icon: 2 });
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
return false;
|
||||||
|
});
|
||||||
|
|
||||||
|
// 重置按钮点击事件
|
||||||
|
$('button[type="reset"]').on('click', function () {
|
||||||
|
// 重新加载分类列表
|
||||||
|
$.get('{:url("resources/getcate")}', function (res) {
|
||||||
|
if (res.code == 0) {
|
||||||
|
var html = '<option value="">请选择分类</option>';
|
||||||
|
res.data.forEach(function (item) {
|
||||||
|
html += '<option value="' + item.id + '">' + item.name + '</option>';
|
||||||
|
if (item.children && item.children.length > 0) {
|
||||||
|
item.children.forEach(function (child) {
|
||||||
|
html += '<option value="' + child.id + '">├─ ' + child.name + '</option>';
|
||||||
|
});
|
||||||
|
}
|
||||||
|
});
|
||||||
|
$('select[name="cate"]').html(html);
|
||||||
|
|
||||||
|
// 如果有资源数据,设置分类值
|
||||||
|
if (resourceData && resourceData.cate) {
|
||||||
|
setTimeout(function () {
|
||||||
|
$('select[name="cate"]').val(resourceData.cate);
|
||||||
|
form.render('select');
|
||||||
|
}, 100);
|
||||||
|
} else {
|
||||||
|
form.render('select');
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
layer.msg(res.msg, { icon: 2 });
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<!-- wangeditor编辑器脚本 -->
|
||||||
|
<script>
|
||||||
|
const { createEditor, createToolbar } = window.wangEditor
|
||||||
|
|
||||||
|
const editorConfig = {
|
||||||
|
MENU_CONF: {},
|
||||||
|
placeholder: '请输入内容...',
|
||||||
|
onChange(editor) {
|
||||||
|
const html = editor.getHtml()
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
// 配置图片上传
|
||||||
|
editorConfig.MENU_CONF['uploadImage'] = {
|
||||||
|
server: '{:url("index/upload_img")}',
|
||||||
|
fieldName: 'file',
|
||||||
|
maxFileSize: 50 * 1024 * 1024, // 50M
|
||||||
|
maxNumberOfFiles: 10,
|
||||||
|
allowedFileTypes: ['image/*'],
|
||||||
|
meta: {
|
||||||
|
token: 'xxx'
|
||||||
|
},
|
||||||
|
metaWithUrl: true,
|
||||||
|
headers: {
|
||||||
|
Accept: 'text/x-json'
|
||||||
|
},
|
||||||
|
timeout: 30 * 1000, // 30s
|
||||||
|
|
||||||
|
onBeforeUpload(file) {
|
||||||
|
console.log('准备上传图片', file)
|
||||||
|
return file
|
||||||
|
},
|
||||||
|
onProgress(progress) {
|
||||||
|
console.log('上传进度', progress)
|
||||||
|
},
|
||||||
|
onSuccess(file, res) {
|
||||||
|
console.log('上传成功', file, res)
|
||||||
|
},
|
||||||
|
onFailed(file, res) {
|
||||||
|
layer.msg('上传失败:' + res.msg, { icon: 2 })
|
||||||
|
console.log('上传失败', file, res)
|
||||||
|
},
|
||||||
|
onError(file, err, res) {
|
||||||
|
layer.msg('上传出错:' + err.message, { icon: 2 })
|
||||||
|
console.error('上传出错', file, err, res)
|
||||||
|
},
|
||||||
|
customInsert(res, insertFn) {
|
||||||
|
// 只使用返回的url字段,并确保使用完整的URL
|
||||||
|
if (res.code === 0 && res.url) {
|
||||||
|
// 如果URL不是以http开头,添加https://
|
||||||
|
let imageUrl = res.url;
|
||||||
|
if (!imageUrl.startsWith('http')) {
|
||||||
|
imageUrl = 'https://' + imageUrl;
|
||||||
|
}
|
||||||
|
// 移除可能存在的重复域名和路径
|
||||||
|
imageUrl = imageUrl.replace(/^https?:\/\/[^\/]+\/admin\/resources\//, 'https://www.yunzer.cn/');
|
||||||
|
insertFn(imageUrl);
|
||||||
|
} else {
|
||||||
|
layer.msg('图片上传失败:' + (res.msg || '未知错误'), { icon: 2 });
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const editor = createEditor({
|
||||||
|
selector: '#editor-container',
|
||||||
|
html: `{$resource.content|raw|default=''}`,
|
||||||
|
config: editorConfig,
|
||||||
|
mode: 'default', // or 'simple'
|
||||||
|
})
|
||||||
|
|
||||||
|
const toolbarConfig = {}
|
||||||
|
|
||||||
|
const toolbar = createToolbar({
|
||||||
|
editor,
|
||||||
|
selector: '#toolbar-container',
|
||||||
|
config: toolbarConfig,
|
||||||
|
mode: 'default', // or 'simple'
|
||||||
|
})
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
//返回资源列表
|
||||||
|
function goBack() {
|
||||||
|
window.location.href = '{:url("resources/lists")}';
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
document.addEventListener('DOMContentLoaded', function () {
|
||||||
|
const imageUpload = document.getElementById('imageUpload');
|
||||||
|
const imagePreview = document.getElementById('imagePreview');
|
||||||
|
const imageProgress = document.getElementById('imageProgress');
|
||||||
|
const progressBar = imageProgress.querySelector('.progress-bar');
|
||||||
|
const imagesInput = document.getElementById('images');
|
||||||
|
|
||||||
|
// 处理图片上传
|
||||||
|
imageUpload.addEventListener('click', function () {
|
||||||
|
const input = document.createElement('input');
|
||||||
|
input.type = 'file';
|
||||||
|
input.multiple = true;
|
||||||
|
input.accept = 'image/*';
|
||||||
|
|
||||||
|
input.onchange = function (e) {
|
||||||
|
const files = e.target.files;
|
||||||
|
if (files.length === 0) return;
|
||||||
|
|
||||||
|
imageProgress.style.display = 'block';
|
||||||
|
let uploadedCount = 0;
|
||||||
|
|
||||||
|
Array.from(files).forEach(file => {
|
||||||
|
const formData = new FormData();
|
||||||
|
formData.append('file', file);
|
||||||
|
|
||||||
|
fetch('{:url("index/upload_img")}', {
|
||||||
|
method: 'POST',
|
||||||
|
body: formData
|
||||||
|
})
|
||||||
|
.then(response => response.json())
|
||||||
|
.then(res => {
|
||||||
|
if (res.code === 0) {
|
||||||
|
addImagePreview(res.data);
|
||||||
|
updateImagesInput();
|
||||||
|
} else {
|
||||||
|
alert('上传失败:' + res.msg);
|
||||||
|
}
|
||||||
|
})
|
||||||
|
.catch(error => {
|
||||||
|
console.error('上传错误:', error);
|
||||||
|
alert('上传出错');
|
||||||
|
})
|
||||||
|
.finally(() => {
|
||||||
|
uploadedCount++;
|
||||||
|
const progress = (uploadedCount / files.length) * 100;
|
||||||
|
progressBar.style.width = progress + '%';
|
||||||
|
|
||||||
|
if (uploadedCount === files.length) {
|
||||||
|
setTimeout(() => {
|
||||||
|
imageProgress.style.display = 'none';
|
||||||
|
progressBar.style.width = '0%';
|
||||||
|
}, 500);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
input.click();
|
||||||
|
});
|
||||||
|
|
||||||
|
// 添加图片预览
|
||||||
|
function addImagePreview(imageUrl) {
|
||||||
|
const div = document.createElement('div');
|
||||||
|
div.className = 'image-preview-item';
|
||||||
|
div.dataset.src = imageUrl;
|
||||||
|
|
||||||
|
div.innerHTML = `
|
||||||
|
<img src="${imageUrl}" alt="已上传图片">
|
||||||
|
<div class="image-preview-overlay">
|
||||||
|
<button type="button" class="btn btn-danger btn-sm delete-image">
|
||||||
|
<i class="fas fa-trash"></i>
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
<p class="image-filename">${imageUrl.split('/').pop()}</p>
|
||||||
|
`;
|
||||||
|
|
||||||
|
imagePreview.appendChild(div);
|
||||||
|
}
|
||||||
|
|
||||||
|
// 更新隐藏输入框的值
|
||||||
|
function updateImagesInput() {
|
||||||
|
const images = Array.from(imagePreview.querySelectorAll('.image-preview-item'))
|
||||||
|
.map(item => item.dataset.src);
|
||||||
|
imagesInput.value = images.join(',');
|
||||||
|
}
|
||||||
|
|
||||||
|
// 删除图片
|
||||||
|
imagePreview.addEventListener('click', function (e) {
|
||||||
|
if (e.target.closest('.delete-image')) {
|
||||||
|
const item = e.target.closest('.image-preview-item');
|
||||||
|
item.remove();
|
||||||
|
updateImagesInput();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
// 图片预览功能
|
||||||
|
window.previewImage = function(imageUrl) {
|
||||||
|
layer.photos({
|
||||||
|
photos: {
|
||||||
|
"data": [{
|
||||||
|
"src": imageUrl,
|
||||||
|
"alt": "图片预览"
|
||||||
|
}]
|
||||||
|
},
|
||||||
|
anim: 5
|
||||||
|
});
|
||||||
|
};
|
||||||
|
});
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<style>
|
||||||
|
.image-upload-container {
|
||||||
|
margin: 20px 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.image-preview-container {
|
||||||
|
display: grid;
|
||||||
|
grid-template-columns: repeat(auto-fill, minmax(150px, 1fr));
|
||||||
|
gap: 15px;
|
||||||
|
margin-top: 15px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.image-preview-item {
|
||||||
|
position: relative;
|
||||||
|
border-radius: 8px;
|
||||||
|
overflow: hidden;
|
||||||
|
box-shadow: 0 2px 4px rgba(0, 0, 0, 0.1);
|
||||||
|
}
|
||||||
|
|
||||||
|
.image-preview-item img {
|
||||||
|
width: 100%;
|
||||||
|
height: 150px;
|
||||||
|
object-fit: cover;
|
||||||
|
display: block;
|
||||||
|
}
|
||||||
|
|
||||||
|
.image-preview-overlay {
|
||||||
|
position: absolute;
|
||||||
|
top: 0;
|
||||||
|
left: 0;
|
||||||
|
right: 0;
|
||||||
|
bottom: 0;
|
||||||
|
background: rgba(0, 0, 0, 0.5);
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
opacity: 0;
|
||||||
|
transition: opacity 0.3s;
|
||||||
|
z-index: 1;
|
||||||
|
/* 添加 z-index */
|
||||||
|
}
|
||||||
|
|
||||||
|
.image-preview-item:hover .image-preview-overlay {
|
||||||
|
opacity: 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
.image-filename {
|
||||||
|
margin: 5px 0;
|
||||||
|
font-size: 0.9em;
|
||||||
|
text-align: center;
|
||||||
|
word-break: break-all;
|
||||||
|
}
|
||||||
|
|
||||||
|
.upload-progress {
|
||||||
|
margin-top: 10px;
|
||||||
|
height: 4px;
|
||||||
|
background: #f0f0f0;
|
||||||
|
border-radius: 2px;
|
||||||
|
overflow: hidden;
|
||||||
|
}
|
||||||
|
|
||||||
|
.progress-bar {
|
||||||
|
height: 100%;
|
||||||
|
background: #007bff;
|
||||||
|
transition: width 0.3s ease;
|
||||||
|
}
|
||||||
|
|
||||||
|
.btn {
|
||||||
|
padding: 8px 16px;
|
||||||
|
border: none;
|
||||||
|
border-radius: 4px;
|
||||||
|
cursor: pointer;
|
||||||
|
font-size: 14px;
|
||||||
|
transition: all 0.3s;
|
||||||
|
}
|
||||||
|
|
||||||
|
.btn-primary {
|
||||||
|
background: #007bff;
|
||||||
|
color: white;
|
||||||
|
}
|
||||||
|
|
||||||
|
.btn-primary:hover {
|
||||||
|
background: #0056b3;
|
||||||
|
}
|
||||||
|
|
||||||
|
.btn-danger {
|
||||||
|
background: #dc3545;
|
||||||
|
color: white;
|
||||||
|
}
|
||||||
|
|
||||||
|
.btn-danger:hover {
|
||||||
|
background: #c82333;
|
||||||
|
}
|
||||||
|
|
||||||
|
.btn-sm {
|
||||||
|
padding: 4px 8px;
|
||||||
|
font-size: 12px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.delete-image {
|
||||||
|
/* background: #dc3545; */
|
||||||
|
/* color: white; */
|
||||||
|
border: none;
|
||||||
|
border-radius: 4px;
|
||||||
|
padding: 8px 12px;
|
||||||
|
cursor: pointer;
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
transition: background-color 0.3s;
|
||||||
|
z-index: 2;
|
||||||
|
/* 确保按钮在悬停层之上 */
|
||||||
|
}
|
||||||
|
|
||||||
|
.delete-image i {
|
||||||
|
font-size: 40px;
|
||||||
|
margin-right: 4px;
|
||||||
|
/* 添加图标右边距 */
|
||||||
|
}
|
||||||
|
|
||||||
|
.form-container {
|
||||||
|
display: flex;
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
.container-left {
|
||||||
|
width: 35%;
|
||||||
|
}
|
||||||
|
|
||||||
|
.container-right {
|
||||||
|
width: 65%;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* 图片列表表格样式 */
|
||||||
|
.image-list-table {
|
||||||
|
margin-top: 15px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.image-thumbnail {
|
||||||
|
width: 60px;
|
||||||
|
height: 60px;
|
||||||
|
border-radius: 4px;
|
||||||
|
overflow: hidden;
|
||||||
|
cursor: pointer;
|
||||||
|
transition: transform 0.2s;
|
||||||
|
}
|
||||||
|
|
||||||
|
.image-thumbnail:hover {
|
||||||
|
transform: scale(1.05);
|
||||||
|
}
|
||||||
|
|
||||||
|
.image-thumbnail img {
|
||||||
|
width: 100%;
|
||||||
|
height: 100%;
|
||||||
|
object-fit: cover;
|
||||||
|
}
|
||||||
|
</style>
|
||||||
207
app/admin/view/resources/lists.php
Normal file
207
app/admin/view/resources/lists.php
Normal file
@ -0,0 +1,207 @@
|
|||||||
|
{include file="public/header" /}
|
||||||
|
<div class="config-container">
|
||||||
|
<!-- 页面头部样式 -->
|
||||||
|
<div class="config-header" style="display: flex;flex-direction: column;flex-wrap: wrap;align-items: flex-start;">
|
||||||
|
<div class="maintitle">
|
||||||
|
<i class="layui-icon layui-icon-file"></i>
|
||||||
|
<span>资源列表</span>
|
||||||
|
</div>
|
||||||
|
<div style="display: flex;align-items: flex-start;flex-direction: column;gap: 15px;margin-bottom: 10px;">
|
||||||
|
<div class="shaixuan">
|
||||||
|
<label>筛选:</label>
|
||||||
|
<div class="layui-form" style="display: flex; gap: 10px;">
|
||||||
|
<div class="layui-input-inline">
|
||||||
|
<select id="categoryFilter" lay-filter="categoryFilter" lay-verify="">
|
||||||
|
<option value="">全部分类</option>
|
||||||
|
{volist name="categories" id="category"}
|
||||||
|
<optgroup label="{$category.name}">
|
||||||
|
{volist name="category.children" id="subCategory"}
|
||||||
|
<option value="{$subCategory.id}">{$subCategory.name}</option>
|
||||||
|
{/volist}
|
||||||
|
</optgroup>
|
||||||
|
{/volist}
|
||||||
|
</select>
|
||||||
|
</div>
|
||||||
|
<div class="layui-input-inline">
|
||||||
|
<input type="text" id="nameSearch" placeholder="搜索资源名称" class="layui-input">
|
||||||
|
</div>
|
||||||
|
<div class="layui-input-inline">
|
||||||
|
<input type="text" id="uploaderSearch" placeholder="搜索上传者" class="layui-input">
|
||||||
|
</div>
|
||||||
|
<button type="button" class="layui-btn layui-btn-normal" onclick="doSearch()">
|
||||||
|
<i class="layui-icon layui-icon-search"></i>搜索
|
||||||
|
</button>
|
||||||
|
<button type="button" class="layui-btn layui-btn-primary layui-border-blue" onclick="doRefresh()">
|
||||||
|
<i class="layui-icon layui-icon-refresh"></i>重置
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div>
|
||||||
|
<button type="button" class="layui-btn layui-btn-normal" onclick="add()">
|
||||||
|
<i class="layui-icon layui-icon-add-1"></i>添加资源
|
||||||
|
</button>
|
||||||
|
<button type="button" class="layui-btn layui-btn-primary layui-border-blue" onclick="refresh()">
|
||||||
|
<i class="layui-icon layui-icon-refresh"></i>刷新
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<table id="resourceTable" lay-filter="resourceTable"></table>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<script type="text/html" id="iconTemplate">
|
||||||
|
{{# if(d.icon){ }}
|
||||||
|
<img src="{{ d.icon }}" style="max-width: 50px; max-height: 50px;">
|
||||||
|
{{# } }}
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<script type="text/html" id="statusTemplate">
|
||||||
|
{{# if(d.status == '0'){ }}
|
||||||
|
<span style="color:red;">未审核</span>
|
||||||
|
{{# } else if(d.status == '1'){ }}
|
||||||
|
<span style="color:green;">已审核</span>
|
||||||
|
{{# } }}
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<script type="text/html" id="operationBar">
|
||||||
|
<button type="button" class="layui-btn layui-btn-normal layui-btn-xs" lay-event="edit">
|
||||||
|
<i class="layui-icon layui-icon-edit"></i>编辑
|
||||||
|
</button>
|
||||||
|
<button type="button" class="layui-btn layui-btn-primary layui-btn-xs" lay-event="del">
|
||||||
|
<i class="layui-icon layui-icon-delete"></i>删除
|
||||||
|
</button>
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<script type="text/javascript">
|
||||||
|
layui.use(['layer', 'form', 'table'], function () {
|
||||||
|
var layer = layui.layer;
|
||||||
|
var $ = layui.jquery;
|
||||||
|
var form = layui.form;
|
||||||
|
var table = layui.table;
|
||||||
|
|
||||||
|
// 初始化表格
|
||||||
|
table.render({
|
||||||
|
elem: '#resourceTable',
|
||||||
|
url: '/admin/resources/lists',
|
||||||
|
method: 'post',
|
||||||
|
cols: [[
|
||||||
|
{ field: 'id', title: 'ID', align: 'center', width: 80 },
|
||||||
|
{ field: 'number', title: '资源编号', width: 100 },
|
||||||
|
{ field: 'title', title: '资源名称' },
|
||||||
|
{ field: 'cate', title: '分类', align: 'center', width: 120 },
|
||||||
|
{ field: 'icon', title: '图标', templet: '#iconTemplate', align: 'center', width: 100 },
|
||||||
|
{ field: 'uploader', title: '上传者', align: 'center', width: 100 },
|
||||||
|
{ field: 'desc', title: '描述', width: 200 },
|
||||||
|
{ field: 'status', title: '状态', templet: '#statusTemplate', align: 'center', width: 80 },
|
||||||
|
{ title: '操作', toolbar: '#operationBar', align: 'center', width: 150, fixed: 'right' }
|
||||||
|
]],
|
||||||
|
page: true,
|
||||||
|
limit: 10,
|
||||||
|
limits: [10, 50, 100],
|
||||||
|
//height: 'full-220'
|
||||||
|
});
|
||||||
|
|
||||||
|
// 监听工具条事件
|
||||||
|
table.on('tool(resourceTable)', function (obj) {
|
||||||
|
var data = obj.data;
|
||||||
|
if (obj.event === 'edit') {
|
||||||
|
edit(data.id);
|
||||||
|
} else if (obj.event === 'del') {
|
||||||
|
del(data.id);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
// 监听分类筛选变化
|
||||||
|
form.on('select(categoryFilter)', function (data) {
|
||||||
|
filterByCategory(data.value);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
function filterByCategory(categoryId) {
|
||||||
|
reloadTable();
|
||||||
|
}
|
||||||
|
|
||||||
|
function doSearch() {
|
||||||
|
var nameKeyword = $('#nameSearch').val().trim();
|
||||||
|
var uploaderKeyword = $('#uploaderSearch').val().trim();
|
||||||
|
|
||||||
|
if (!nameKeyword && !uploaderKeyword && !$('#categoryFilter').val()) {
|
||||||
|
layer.msg('请输入搜索条件', { icon: 0 });
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
reloadTable();
|
||||||
|
}
|
||||||
|
|
||||||
|
function doRefresh() {
|
||||||
|
// 清空搜索条件
|
||||||
|
$('#nameSearch').val('');
|
||||||
|
$('#uploaderSearch').val('');
|
||||||
|
$('#categoryFilter').val('');
|
||||||
|
layui.form.render('select'); // 重新渲染select
|
||||||
|
|
||||||
|
// 重新加载表格,不带任何筛选条件
|
||||||
|
layui.table.reload('resourceTable', {
|
||||||
|
where: {},
|
||||||
|
page: {
|
||||||
|
curr: 1
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
layer.msg('已重置', { icon: 1 });
|
||||||
|
}
|
||||||
|
|
||||||
|
function reloadTable() {
|
||||||
|
var categoryId = $('#categoryFilter').val();
|
||||||
|
var categoryName = '';
|
||||||
|
if (categoryId) {
|
||||||
|
var selectedOption = $('#categoryFilter option[value="' + categoryId + '"]');
|
||||||
|
if (selectedOption.length > 0) {
|
||||||
|
categoryName = selectedOption.text().trim();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
var nameKeyword = $('#nameSearch').val().trim();
|
||||||
|
var uploaderKeyword = $('#uploaderSearch').val().trim();
|
||||||
|
|
||||||
|
layui.table.reload('resourceTable', {
|
||||||
|
where: {
|
||||||
|
category: categoryName,
|
||||||
|
name: nameKeyword,
|
||||||
|
uploader: uploaderKeyword
|
||||||
|
},
|
||||||
|
page: {
|
||||||
|
curr: 1
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
function add() {
|
||||||
|
window.location.href = '/admin/resources/add';
|
||||||
|
}
|
||||||
|
|
||||||
|
function edit(id) {
|
||||||
|
window.location.href = '/admin/resources/edit?id=' + id;
|
||||||
|
}
|
||||||
|
|
||||||
|
function del(id) {
|
||||||
|
layer.confirm('确定要删除该资源吗?', {
|
||||||
|
btn: ['确定', '取消']
|
||||||
|
}, function () {
|
||||||
|
$.post('/admin/resources/delete', { id: id }, function (res) {
|
||||||
|
if (res.code == 0) {
|
||||||
|
layer.msg(res.msg, { icon: 1 });
|
||||||
|
setTimeout(function () {
|
||||||
|
layui.table.reload('resourceTable');
|
||||||
|
}, 1000);
|
||||||
|
} else {
|
||||||
|
layer.msg(res.msg, { icon: 2 });
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
function refresh() {
|
||||||
|
layui.table.reload('resourceTable');
|
||||||
|
}
|
||||||
|
</script>
|
||||||
94
app/admin/view/yunzer/buttonadd.php
Normal file
94
app/admin/view/yunzer/buttonadd.php
Normal file
@ -0,0 +1,94 @@
|
|||||||
|
{include file="public/header" /}
|
||||||
|
<form class="layui-form">
|
||||||
|
<input type="hidden" name="smid" value="{$smid}">
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">按钮名</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<input type="text" class="layui-input" name="label" placeholder="请输入按钮名">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">图标</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<select name="icon_class" lay-verify="required" lay-search="">
|
||||||
|
<option value="">请输入layui图标</option>
|
||||||
|
{foreach($iconfont as $icon_v)}
|
||||||
|
<option value="{$icon_v['icon_css']}">{$icon_v['icon_css']} {$icon_v['icon_name']}</option>
|
||||||
|
{/foreach}
|
||||||
|
</select>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">类型</label>
|
||||||
|
<div class="layui-input-inline">
|
||||||
|
<select name="type" lay-filter="type">
|
||||||
|
<option value='1' selected>内部代码</option>
|
||||||
|
<option value='2'>超链接</option>
|
||||||
|
</select>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layui-form-item" id="src1">
|
||||||
|
<label class="layui-form-label">内部代码</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<input type="text" class="layui-input" name="src1" placeholder="请输入:模块/方法">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layui-form-item" style="display:none;" id="src2">
|
||||||
|
<label class="layui-form-label">链接地址</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<input type="text" class="layui-input" name="src2" placeholder="请输入http网址">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">排序</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<input type="number" class="layui-input" name="sort" placeholder="值越大越靠前">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">状态</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<input type="radio" name="status" value="1" title="开启" checked>
|
||||||
|
<input type="radio" name="status" value="0" title="禁用">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</form>
|
||||||
|
<div class="layui-form-item" style="margin-top:10px;">
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<button type="button" class="layui-btn" onclick="save()">保存</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<script type="text/javascript">
|
||||||
|
layui.use(['layer','form'],function(){
|
||||||
|
var form = layui.form;
|
||||||
|
layer = layui.layer;
|
||||||
|
$ = layui.jquery;
|
||||||
|
|
||||||
|
form.on('select(type)', function(data){
|
||||||
|
if(data.value == 1){
|
||||||
|
$("#src1").show();
|
||||||
|
$("#src2").hide();
|
||||||
|
$("#src3").hide();
|
||||||
|
}else if(data.value == 2){
|
||||||
|
$("#src1").hide();
|
||||||
|
$("#src2").show();
|
||||||
|
$("#src3").hide();
|
||||||
|
}else{
|
||||||
|
$("#src1").show();
|
||||||
|
$("#src2").hide();
|
||||||
|
$("#src3").hide();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
function save(){
|
||||||
|
$.post('{$config["admin_route"]}yunzer/buttonadd',$('form').serialize(),function(res){
|
||||||
|
if(res.code>0){
|
||||||
|
layer.msg(res.msg,{'icon':2});
|
||||||
|
}else{
|
||||||
|
layer.msg(res.msg,{'icon':1});
|
||||||
|
setTimeout(function(){parent.window.location.reload();},1000);
|
||||||
|
}
|
||||||
|
},'json');
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
{include file="public/tail" /}
|
||||||
98
app/admin/view/yunzer/buttonedit.php
Normal file
98
app/admin/view/yunzer/buttonedit.php
Normal file
@ -0,0 +1,98 @@
|
|||||||
|
{include file="public/header" /}
|
||||||
|
<form class="layui-form">
|
||||||
|
<input type="hidden" name="smid" value="{$lists.smid}">
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">按钮名</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<input type="text" class="layui-input" name="label" placeholder="请输入按钮名" value="{$lists.label}">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">图标</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<select name="icon_class" lay-verify="required" lay-search="">
|
||||||
|
<option value="">请输入layui图标</option>
|
||||||
|
{foreach($iconfont as $icon_v)}
|
||||||
|
<option value="{$icon_v['icon_css']}" {if($icon_v['icon_css'] == $lists['icon_class'])}selected{/if}>{$icon_v['icon_css']} {$icon_v['icon_name']}</option>
|
||||||
|
{/foreach}
|
||||||
|
</select>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">类型</label>
|
||||||
|
<div class="layui-input-inline">
|
||||||
|
<select name="type" lay-filter="type">
|
||||||
|
<option value='1' {$lists.type==1?'selected':''}>内部代码</option>
|
||||||
|
<option value='2' {$lists.type==2?'selected':''}>超链接</option>
|
||||||
|
</select>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layui-form-item" {$lists.type==1?'style="display:none;"':''} id="src1">
|
||||||
|
<label class="layui-form-label">内部代码</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<input type="text" class="layui-input" name="src1" placeholder="请输入:/项目/模块/方法" {if $lists.type==1}value="{$lists['src']}"{/if}>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layui-form-item" {$lists.type==2?'style="display:none;"':''} id="src2">
|
||||||
|
<label class="layui-form-label">链接地址</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<input type="text" class="layui-input" name="src2" placeholder="请输入http网址" {if $lists.type==2}value="{$lists['src']}"{/if}>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">排序</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<input type="number" class="layui-input" name="sort" placeholder="值越大越靠前" value="{$lists.sort}">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">状态</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<input type="radio" name="status" value="1" title="开启" {$lists['status']==1?'checked':''}>
|
||||||
|
<input type="radio" name="status" value="0" title="禁用" {$lists['status']==0?'checked':''}>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</form>
|
||||||
|
<div class="layui-form-item" style="margin-top:10px;">
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<button type="button" class="layui-btn" onclick="save()">保存</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<script type="text/javascript">
|
||||||
|
layui.use(['layer','form'],function(){
|
||||||
|
form = layui.form;
|
||||||
|
layer = layui.layer;
|
||||||
|
$ = layui.jquery;
|
||||||
|
|
||||||
|
type({$lists.type});
|
||||||
|
form.on('select(type)', function(data){
|
||||||
|
type(data.value);
|
||||||
|
});
|
||||||
|
});
|
||||||
|
function type(value){
|
||||||
|
if(value == 1){
|
||||||
|
$("#src1").show();
|
||||||
|
$("#src2").hide();
|
||||||
|
$("#src3").hide();
|
||||||
|
}else if(value == 2){
|
||||||
|
$("#src1").hide();
|
||||||
|
$("#src2").show();
|
||||||
|
$("#src3").hide();
|
||||||
|
}else{
|
||||||
|
$("#src1").show();
|
||||||
|
$("#src2").hide();
|
||||||
|
$("#src3").hide();
|
||||||
|
}
|
||||||
|
}
|
||||||
|
function save(){
|
||||||
|
$.post('{$config["admin_route"]}yunzer/buttonedit',$('form').serialize(),function(res){
|
||||||
|
if(res.code>0){
|
||||||
|
layer.msg(res.msg,{'icon':2});
|
||||||
|
}else{
|
||||||
|
layer.msg(res.msg,{'icon':1});
|
||||||
|
setTimeout(function(){parent.window.location.reload();},1000);
|
||||||
|
}
|
||||||
|
},'json');
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
{include file="public/tail" /}
|
||||||
155
app/admin/view/yunzer/buttoninfo.php
Normal file
155
app/admin/view/yunzer/buttoninfo.php
Normal file
@ -0,0 +1,155 @@
|
|||||||
|
{include file="public/header" /}
|
||||||
|
<div class="layui-card">
|
||||||
|
<div class="layui-card-header">
|
||||||
|
<div class="layui-row">
|
||||||
|
<div class="layui-col-md6">
|
||||||
|
<div class="layui-breadcrumb">
|
||||||
|
<a href="{$config['admin_route']}Yunzer/menuInfo"><i class="layui-icon layui-icon-menu-fill"></i> 上级菜单</a>
|
||||||
|
<a><cite>本级菜单</cite></a>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layui-col-md6" style="text-align: right;">
|
||||||
|
<button type="button" class="layui-btn layui-btn-normal layui-btn-sm" onclick="add({$smid})">
|
||||||
|
<i class="layui-icon layui-icon-add-1"></i> 添加按钮
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layui-card-body">
|
||||||
|
<div class="layui-row layui-col-space15">
|
||||||
|
<div class="layui-col-md3">
|
||||||
|
<div class="layui-card">
|
||||||
|
<div class="layui-card-header"><i class="layui-icon layui-icon-tree"></i> 菜单结构</div>
|
||||||
|
<div class="layui-card-body" id="menuTree">
|
||||||
|
<!-- 树形结构将在这里渲染 -->
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layui-col-md9">
|
||||||
|
<table class="layui-table" lay-skin="line">
|
||||||
|
<thead>
|
||||||
|
<tr>
|
||||||
|
<th>排序</th>
|
||||||
|
<th>类型</th>
|
||||||
|
<th>按钮名</th>
|
||||||
|
<th>图标</th>
|
||||||
|
<th>状态</th>
|
||||||
|
<th>操作</th>
|
||||||
|
</tr>
|
||||||
|
</thead>
|
||||||
|
<tbody>
|
||||||
|
{volist name="lists" id='vo'}
|
||||||
|
<tr>
|
||||||
|
<td><span class="layui-badge layui-bg-gray">{$vo['sort']}</span></td>
|
||||||
|
<td>
|
||||||
|
{if($vo['type']==1)}
|
||||||
|
<span class="layui-badge layui-bg-blue">功能模块</span> {$vo['src']}
|
||||||
|
{elseif($vo['type']==2)/}
|
||||||
|
<span class="layui-badge layui-bg-green">超链接</span> {$vo['src']}
|
||||||
|
{/if}
|
||||||
|
</td>
|
||||||
|
<td><b>{$vo['label']}</b></td>
|
||||||
|
<td><i class="layui-icon {$vo['icon_class']}"></i> {$vo['icon_class']}</td>
|
||||||
|
<td>
|
||||||
|
{if($vo['status']==1)}
|
||||||
|
<span class="layui-badge layui-bg-green">开启</span>
|
||||||
|
{else/}
|
||||||
|
<span class="layui-badge layui-bg-red">禁用</span>
|
||||||
|
{/if}
|
||||||
|
</td>
|
||||||
|
<td>
|
||||||
|
<div class="layui-btn-group">
|
||||||
|
<button type="button" class="layui-btn layui-btn-xs" onclick="edit({$vo.smid})">
|
||||||
|
<i class="layui-icon layui-icon-edit"></i> 编辑
|
||||||
|
</button>
|
||||||
|
<button type="button" class="layui-btn layui-btn-xs layui-btn-danger" onclick="del({$vo.smid})">
|
||||||
|
<i class="layui-icon layui-icon-delete"></i> 删除
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</td>
|
||||||
|
</tr>
|
||||||
|
{/volist}
|
||||||
|
</tbody>
|
||||||
|
</table>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<script type="text/javascript">
|
||||||
|
layui.use(['layer', 'tree'], function(){
|
||||||
|
layer = layui.layer;
|
||||||
|
$ = layui.jquery;
|
||||||
|
var tree = layui.tree;
|
||||||
|
|
||||||
|
// 构建树形结构数据
|
||||||
|
var treeData = [{
|
||||||
|
title: '当前菜单',
|
||||||
|
id: {$smid},
|
||||||
|
spread: true,
|
||||||
|
children: []
|
||||||
|
}];
|
||||||
|
|
||||||
|
// 将按钮数据添加到树中
|
||||||
|
{volist name="lists" id='vo'}
|
||||||
|
treeData[0].children.push({
|
||||||
|
title: '{$vo.label}',
|
||||||
|
id: {$vo.smid},
|
||||||
|
icon: 'layui-icon {$vo.icon_class}'
|
||||||
|
});
|
||||||
|
{/volist}
|
||||||
|
|
||||||
|
// 渲染树形结构
|
||||||
|
tree.render({
|
||||||
|
elem: '#menuTree',
|
||||||
|
data: treeData,
|
||||||
|
showLine: true,
|
||||||
|
click: function(obj){
|
||||||
|
var data = obj.data;
|
||||||
|
if(data.id != {$smid}) {
|
||||||
|
edit(data.id);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
// 添加
|
||||||
|
function add(smid){
|
||||||
|
layer.open({
|
||||||
|
type: 2,
|
||||||
|
title: '添加按钮',
|
||||||
|
shade: 0.3,
|
||||||
|
area: ['550px','550px'],
|
||||||
|
content: '{$config["admin_route"]}yunzer/buttonadd?smid='+smid
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
// 编辑
|
||||||
|
function edit(smid){
|
||||||
|
layer.open({
|
||||||
|
type: 2,
|
||||||
|
title: '编辑按钮',
|
||||||
|
shade: 0.3,
|
||||||
|
area: ['550px','550px'],
|
||||||
|
content: '{$config["admin_route"]}yunzer/buttonedit?smid='+smid
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
// 删除
|
||||||
|
function del(smid){
|
||||||
|
layer.confirm('确定要删除吗?', {
|
||||||
|
icon: 3,
|
||||||
|
btn: ['确定','取消']
|
||||||
|
}, function(){
|
||||||
|
$.post('{$config["admin_route"]}yunzer/buttondel',{'smid':smid},function(res){
|
||||||
|
if(res.code>0){
|
||||||
|
layer.alert(res.msg,{icon:2});
|
||||||
|
}else{
|
||||||
|
layer.msg(res.msg,{icon:1});
|
||||||
|
setTimeout(function(){window.location.reload();},1000);
|
||||||
|
}
|
||||||
|
},'json');
|
||||||
|
});
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
{include file="public/tail" /}
|
||||||
47
app/admin/view/yunzer/configadd.php
Normal file
47
app/admin/view/yunzer/configadd.php
Normal file
@ -0,0 +1,47 @@
|
|||||||
|
{include file="public/header" /}
|
||||||
|
<form class="layui-form">
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">关键词</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<input type="text" class="layui-input" id="config_name" name="config_name" placeholder="请输入关键词,在代码中的key">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">作用</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<input type="text" class="layui-input" id="config_info" name="config_info" placeholder="请输入作用">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">类型</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<select id="config_type" name="config_type" lay-filter="config_type">
|
||||||
|
<option value="1" selected>文本</option>
|
||||||
|
<option value="2">图片</option>
|
||||||
|
<option value="3">富文本</option>
|
||||||
|
</select>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">说明</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<input type="text" class="layui-input" id="config_desc" name="config_desc" placeholder="请输入说明">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">排序</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<input type="number" class="layui-input" id="config_sort" name="config_sort" placeholder="请输入排序,值越大,越靠前" value="0">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">状态</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<select id="config_status" name="config_status" lay-filter="config_status">
|
||||||
|
<option value="1" selected>开启</option>
|
||||||
|
<option value="0">禁用</option>
|
||||||
|
</select>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</form>
|
||||||
|
{include file="public/tail" /}
|
||||||
47
app/admin/view/yunzer/configedit.php
Normal file
47
app/admin/view/yunzer/configedit.php
Normal file
@ -0,0 +1,47 @@
|
|||||||
|
{include file="public/header" /}
|
||||||
|
<form class="layui-form">
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">关键词</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<input type="text" class="layui-input" id="config_name" name="config_name" placeholder="请输入关键词,在代码中的key" value="{$find['config_name']}">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">作用</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<input type="text" class="layui-input" id="config_info" name="config_info" placeholder="请输入作用" value="{$find['config_info']}">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">类型</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<select id="config_type" name="config_type" lay-filter="config_type">
|
||||||
|
<option value="1" {$find['config_type']==1?'selected':''}>文本</option>
|
||||||
|
<option value="2" {$find['config_type']==2?'selected':''}>图片</option>
|
||||||
|
<option value="3" {$find['config_type']==3?'selected':''}>富文本</option>
|
||||||
|
</select>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">说明</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<input type="text" class="layui-input" id="config_desc" name="config_desc" placeholder="请输入说明" value="{$find['config_desc']}">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">排序</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<input type="number" class="layui-input" id="config_sort" name="config_sort" placeholder="请输入排序,值越大,越靠前" value="{$find['config_sort']}">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">状态</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<select id="config_status" name="config_status" lay-filter="config_status">
|
||||||
|
<option value="1" {$find['config_status']==1?'selected':''}>开启</option>
|
||||||
|
<option value="0" {$find['config_status']==0?'selected':''}>禁用</option>
|
||||||
|
</select>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</form>
|
||||||
|
{include file="public/tail" /}
|
||||||
307
app/admin/view/yunzer/configlist.php
Normal file
307
app/admin/view/yunzer/configlist.php
Normal file
@ -0,0 +1,307 @@
|
|||||||
|
{include file="public/header" /}
|
||||||
|
<style>
|
||||||
|
.config-container {
|
||||||
|
padding: 20px;
|
||||||
|
background-color: #fff;
|
||||||
|
border-radius: 8px;
|
||||||
|
/* box-shadow: 0 2px 10px rgba(0, 0, 0, 0.08); */
|
||||||
|
}
|
||||||
|
|
||||||
|
.config-header {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
margin-bottom: 30px;
|
||||||
|
padding-bottom: 15px;
|
||||||
|
border-bottom: 1px solid #eee;
|
||||||
|
}
|
||||||
|
|
||||||
|
.config-header span {
|
||||||
|
font-size: 18px;
|
||||||
|
color: #2c3e50;
|
||||||
|
font-weight: 500;
|
||||||
|
margin-right: 20px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.config-header a {
|
||||||
|
text-decoration: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
.config-header a span {
|
||||||
|
padding: 6px 15px;
|
||||||
|
background: #f8f9fa;
|
||||||
|
border-radius: 4px;
|
||||||
|
color: #606266;
|
||||||
|
font-size: 14px;
|
||||||
|
transition: all 0.3s;
|
||||||
|
}
|
||||||
|
|
||||||
|
.config-header a span:hover {
|
||||||
|
background: #e9ecef;
|
||||||
|
color: #409EFF;
|
||||||
|
}
|
||||||
|
|
||||||
|
.action-buttons {
|
||||||
|
display: flex;
|
||||||
|
gap: 8px;
|
||||||
|
margin-bottom: 20px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.layui-btn {
|
||||||
|
border-radius: 4px;
|
||||||
|
box-shadow: 0 2px 5px rgba(0, 0, 0, 0.1);
|
||||||
|
transition: all 0.3s;
|
||||||
|
height: 32px;
|
||||||
|
line-height: 32px;
|
||||||
|
padding: 0 16px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.layui-btn:hover {
|
||||||
|
transform: translateY(-1px);
|
||||||
|
box-shadow: 0 4px 8px rgba(0, 0, 0, 0.15);
|
||||||
|
}
|
||||||
|
|
||||||
|
.layui-btn .layui-icon {
|
||||||
|
margin-right: 4px;
|
||||||
|
font-size: 14px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.layui-table {
|
||||||
|
margin: 15px 0;
|
||||||
|
border-radius: 4px;
|
||||||
|
overflow: hidden;
|
||||||
|
}
|
||||||
|
|
||||||
|
.layui-table thead tr {
|
||||||
|
background-color: #fafafa;
|
||||||
|
}
|
||||||
|
|
||||||
|
.layui-table tbody tr:hover {
|
||||||
|
background-color: #f5f7fa;
|
||||||
|
}
|
||||||
|
|
||||||
|
.layui-table-cell {
|
||||||
|
height: 40px;
|
||||||
|
line-height: 40px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.layui-badge {
|
||||||
|
padding: 4px 8px;
|
||||||
|
border-radius: 3px;
|
||||||
|
font-weight: 500;
|
||||||
|
height: auto;
|
||||||
|
}
|
||||||
|
|
||||||
|
.layui-badge.layui-bg-green {
|
||||||
|
background-color: #67C23A !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
.layui-badge.layui-bg-gray {
|
||||||
|
background-color: #909399 !important;
|
||||||
|
}
|
||||||
|
|
||||||
|
.layui-layer {
|
||||||
|
border-radius: 8px;
|
||||||
|
overflow: hidden;
|
||||||
|
}
|
||||||
|
|
||||||
|
.layui-textarea {
|
||||||
|
width: 80% !important;
|
||||||
|
}
|
||||||
|
</style>
|
||||||
|
|
||||||
|
<div class="config-container">
|
||||||
|
<div class="config-header">
|
||||||
|
<span>站点管理</span>
|
||||||
|
<!-- <a href="{$config['admin_route']}Yunzer/configvalue">
|
||||||
|
<span>站点配置</span>
|
||||||
|
</a> -->
|
||||||
|
</div>
|
||||||
|
|
||||||
|
|
||||||
|
<div class="layui-card-body" style="padding: 0;">
|
||||||
|
<div class="action-buttons">
|
||||||
|
<button type="button" class="layui-btn layui-btn-normal" id="btn-add">
|
||||||
|
<i class="layui-icon layui-icon-add-1"></i>添加
|
||||||
|
</button>
|
||||||
|
<button type="button" class="layui-btn layui-btn-normal" id="btn-edit">
|
||||||
|
<i class="layui-icon layui-icon-edit"></i>修改
|
||||||
|
</button>
|
||||||
|
<button type="button" class="layui-btn layui-btn-danger" id="btn-del">
|
||||||
|
<i class="layui-icon layui-icon-delete"></i>删除
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
<table class="layui-table" id="lists" lay-filter="lists"></table>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<script type="text/javascript">
|
||||||
|
layui.use(['layer', 'table', 'element'], function () {
|
||||||
|
var table = layui.table;
|
||||||
|
var layer = layui.layer;
|
||||||
|
var element = layui.element;
|
||||||
|
var $ = layui.jquery;
|
||||||
|
|
||||||
|
// 初始化表格
|
||||||
|
function initTable() {
|
||||||
|
table.render({
|
||||||
|
elem: '#lists',
|
||||||
|
method: 'post',
|
||||||
|
url: "{$config['admin_route']}yunzer/configlist",
|
||||||
|
title: '配置列表',
|
||||||
|
page: true,
|
||||||
|
skin: 'line',
|
||||||
|
even: true,
|
||||||
|
size: 'lg',
|
||||||
|
cols: [[
|
||||||
|
{ type: 'radio', fixed: 'left', width: 50 },
|
||||||
|
{ field: 'config_id', width: 80, title: 'ID', sort: true, align: 'center' },
|
||||||
|
{ field: 'config_sort', width: 120, title: '排序', sort: true, align: 'center' },
|
||||||
|
{ field: 'config_name', width: 150, title: '关键词' },
|
||||||
|
{ field: 'config_info', width: 150, title: '作用' },
|
||||||
|
{ field: 'config_desc', minWidth: 300, title: '说明' },
|
||||||
|
{
|
||||||
|
field: 'config_status', width: 100, title: '状态', align: 'center', templet: function (res) {
|
||||||
|
if (res.config_status == 1) {
|
||||||
|
return '<span class="layui-badge layui-bg-green">开启</span>';
|
||||||
|
} else {
|
||||||
|
return '<span class="layui-badge layui-bg-gray">关闭</span>';
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
]],
|
||||||
|
limit: 15,
|
||||||
|
limits: [15, 30, 50, 100],
|
||||||
|
height: 'full-180'
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
// 加载表格数据
|
||||||
|
initTable();
|
||||||
|
|
||||||
|
// 行点击事件
|
||||||
|
table.on('row(lists)', function (obj) {
|
||||||
|
obj.tr.addClass('layui-table-click').siblings().removeClass('layui-table-click');
|
||||||
|
obj.tr.find("div.layui-unselect.layui-form-radio")[0].click();
|
||||||
|
});
|
||||||
|
|
||||||
|
// 添加按钮事件
|
||||||
|
$('#btn-add').on('click', function () {
|
||||||
|
layer.open({
|
||||||
|
title: '<i class="layui-icon layui-icon-add-1"></i> 添加配置',
|
||||||
|
type: 2,
|
||||||
|
content: '{$config["admin_route"]}yunzer/configadd',
|
||||||
|
maxmin: true,
|
||||||
|
area: ['700px', '550px'],
|
||||||
|
skin: 'layui-layer-molv',
|
||||||
|
btn: ['确定', '取消'],
|
||||||
|
yes: function (index, layero) {
|
||||||
|
var form = layero.find('iframe')[0].contentWindow;
|
||||||
|
var button = {
|
||||||
|
config_name: form.config_name.value,
|
||||||
|
config_info: form.config_info.value,
|
||||||
|
config_type: form.config_type.value,
|
||||||
|
config_desc: form.config_desc.value,
|
||||||
|
config_sort: form.config_sort.value,
|
||||||
|
config_status: form.config_status.value
|
||||||
|
};
|
||||||
|
|
||||||
|
var loadIndex = layer.load(1, {
|
||||||
|
shade: [0.1, '#fff']
|
||||||
|
});
|
||||||
|
|
||||||
|
$.post('{$config["admin_route"]}yunzer/configadd', button, function (res) {
|
||||||
|
layer.close(loadIndex);
|
||||||
|
if (res.code > 0) {
|
||||||
|
layer.msg(res.msg, { icon: 2, time: 2000 });
|
||||||
|
} else {
|
||||||
|
layer.msg(res.msg, { icon: 1, time: 1000 });
|
||||||
|
initTable();
|
||||||
|
layer.close(index);
|
||||||
|
}
|
||||||
|
}, 'json');
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
// 编辑按钮事件
|
||||||
|
$('#btn-edit').on('click', function () {
|
||||||
|
var checkStatus = table.checkStatus('lists');
|
||||||
|
var data = checkStatus.data;
|
||||||
|
|
||||||
|
if (data.length === 0) {
|
||||||
|
layer.msg('请选择一条数据进行编辑', { icon: 0 });
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
layer.open({
|
||||||
|
title: '<i class="layui-icon layui-icon-edit"></i> 修改配置',
|
||||||
|
type: 2,
|
||||||
|
content: '{$config["admin_route"]}yunzer/configedit?config_id=' + data[0].config_id,
|
||||||
|
maxmin: true,
|
||||||
|
area: ['700px', '550px'],
|
||||||
|
skin: 'layui-layer-molv',
|
||||||
|
btn: ['确定', '取消'],
|
||||||
|
yes: function (index, layero) {
|
||||||
|
var form = layero.find('iframe')[0].contentWindow;
|
||||||
|
var button = {
|
||||||
|
config_id: data[0].config_id,
|
||||||
|
config_name: form.config_name.value,
|
||||||
|
config_info: form.config_info.value,
|
||||||
|
config_type: form.config_type.value,
|
||||||
|
config_desc: form.config_desc.value,
|
||||||
|
config_sort: form.config_sort.value,
|
||||||
|
config_status: form.config_status.value
|
||||||
|
};
|
||||||
|
|
||||||
|
var loadIndex = layer.load(1, {
|
||||||
|
shade: [0.1, '#fff']
|
||||||
|
});
|
||||||
|
|
||||||
|
$.post('{$config["admin_route"]}yunzer/configedit', button, function (res) {
|
||||||
|
layer.close(loadIndex);
|
||||||
|
if (res.code > 0) {
|
||||||
|
layer.msg(res.msg, { icon: 2, time: 2000 });
|
||||||
|
} else {
|
||||||
|
layer.msg(res.msg, { icon: 1, time: 1000 });
|
||||||
|
initTable();
|
||||||
|
layer.close(index);
|
||||||
|
}
|
||||||
|
}, 'json');
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
// 删除按钮事件
|
||||||
|
$('#btn-del').on('click', function () {
|
||||||
|
var checkStatus = table.checkStatus('lists');
|
||||||
|
var data = checkStatus.data;
|
||||||
|
|
||||||
|
if (data.length === 0) {
|
||||||
|
layer.msg('请选择一条数据进行删除', { icon: 0 });
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
layer.confirm('确定要删除该配置吗?', {
|
||||||
|
icon: 3,
|
||||||
|
title: '删除确认',
|
||||||
|
skin: 'layui-layer-molv',
|
||||||
|
btn: ['确定', '取消']
|
||||||
|
}, function () {
|
||||||
|
var loadIndex = layer.load(1, {
|
||||||
|
shade: [0.1, '#fff']
|
||||||
|
});
|
||||||
|
|
||||||
|
$.post('{$config["admin_route"]}yunzer/configdel', { 'config_id': data[0].config_id }, function (res) {
|
||||||
|
layer.close(loadIndex);
|
||||||
|
if (res.code > 0) {
|
||||||
|
layer.msg(res.msg, { icon: 2, time: 2000 });
|
||||||
|
} else {
|
||||||
|
layer.msg(res.msg, { icon: 1, time: 1000 });
|
||||||
|
initTable();
|
||||||
|
}
|
||||||
|
}, 'json');
|
||||||
|
});
|
||||||
|
});
|
||||||
|
});
|
||||||
|
</script>
|
||||||
|
{include file="public/tail" /}
|
||||||
222
app/admin/view/yunzer/configvalue.php
Normal file
222
app/admin/view/yunzer/configvalue.php
Normal file
@ -0,0 +1,222 @@
|
|||||||
|
{include file="public/header" /}
|
||||||
|
<style>
|
||||||
|
.config-container {
|
||||||
|
padding: 20px;
|
||||||
|
background-color: #fff;
|
||||||
|
border-radius: 8px;
|
||||||
|
/* box-shadow: 0 2px 10px rgba(0, 0, 0, 0.08); */
|
||||||
|
}
|
||||||
|
|
||||||
|
.config-header {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
margin-bottom: 30px;
|
||||||
|
padding-bottom: 15px;
|
||||||
|
border-bottom: 1px solid #eee;
|
||||||
|
}
|
||||||
|
|
||||||
|
.config-header span {
|
||||||
|
font-size: 18px;
|
||||||
|
color: #2c3e50;
|
||||||
|
font-weight: 500;
|
||||||
|
margin-right: 20px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.config-header a {
|
||||||
|
text-decoration: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
.config-header a span {
|
||||||
|
padding: 6px 15px;
|
||||||
|
background: #f8f9fa;
|
||||||
|
border-radius: 4px;
|
||||||
|
color: #606266;
|
||||||
|
font-size: 14px;
|
||||||
|
transition: all 0.3s;
|
||||||
|
}
|
||||||
|
|
||||||
|
.config-header a span:hover {
|
||||||
|
background: #e9ecef;
|
||||||
|
color: #409EFF;
|
||||||
|
}
|
||||||
|
|
||||||
|
.layui-form-item {
|
||||||
|
margin-bottom: 25px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.layui-form-label {
|
||||||
|
color: #606266;
|
||||||
|
font-weight: 500;
|
||||||
|
}
|
||||||
|
|
||||||
|
.layui-input {
|
||||||
|
border-radius: 4px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.layui-input:focus {
|
||||||
|
border-color: #409EFF;
|
||||||
|
}
|
||||||
|
|
||||||
|
.layui-textarea {
|
||||||
|
border-radius: 4px;
|
||||||
|
min-height: 100px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.layui-word-aux {
|
||||||
|
color: #909399;
|
||||||
|
}
|
||||||
|
|
||||||
|
.layui-btn {
|
||||||
|
border-radius: 4px;
|
||||||
|
box-shadow: 0 2px 5px rgba(0, 0, 0, 0.1);
|
||||||
|
transition: all 0.3s;
|
||||||
|
}
|
||||||
|
|
||||||
|
.layui-btn:hover {
|
||||||
|
transform: translateY(-1px);
|
||||||
|
box-shadow: 0 4px 8px rgba(0, 0, 0, 0.15);
|
||||||
|
}
|
||||||
|
|
||||||
|
.upload-preview {
|
||||||
|
margin-top: 10px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.upload-preview img {
|
||||||
|
border-radius: 4px;
|
||||||
|
box-shadow: 0 2px 5px rgba(0, 0, 0, 0.1);
|
||||||
|
transition: all 0.3s;
|
||||||
|
}
|
||||||
|
|
||||||
|
.upload-preview img:hover {
|
||||||
|
transform: scale(1.02);
|
||||||
|
}
|
||||||
|
|
||||||
|
.save-button-container {
|
||||||
|
text-align: center;
|
||||||
|
margin-top: 40px;
|
||||||
|
padding-top: 20px;
|
||||||
|
border-top: 1px solid #eee;
|
||||||
|
}
|
||||||
|
|
||||||
|
.save-button-container .layui-btn {
|
||||||
|
padding: 0 50px;
|
||||||
|
height: 40px;
|
||||||
|
line-height: 40px;
|
||||||
|
font-size: 15px;
|
||||||
|
}
|
||||||
|
</style>
|
||||||
|
|
||||||
|
<div class="config-container">
|
||||||
|
<div class="config-header">
|
||||||
|
<span>站点配置</span>
|
||||||
|
<!-- <a href="{$config['admin_route']}Yunzer/configlist">
|
||||||
|
<span>站点管理</span>
|
||||||
|
</a> -->
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<form class="layui-form">
|
||||||
|
{foreach($lists as $lists_v)}
|
||||||
|
{if($lists_v['config_type'] == 1)}
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">{$lists_v['config_info']}</label>
|
||||||
|
<div class="layui-input-inline" style="width: 350px;">
|
||||||
|
<input type="text" class="layui-input" id="{$lists_v['config_name']}" name="{$lists_v['config_name']}"
|
||||||
|
placeholder="{$lists_v['config_desc']}" value="{$lists_v['config_value']}">
|
||||||
|
</div>
|
||||||
|
<div class="layui-form-mid layui-word-aux">{$lists_v['config_desc']}</div>
|
||||||
|
</div>
|
||||||
|
{elseif($lists_v['config_type'] == 2)/}
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">{$lists_v['config_info']}</label>
|
||||||
|
<div class="layui-input-inline">
|
||||||
|
<button type="button" class="layui-btn layui-btn-normal" id="{$lists_v['config_name']}">
|
||||||
|
<i class="layui-icon layui-icon-upload"></i> 上传图片
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
<div class="layui-form-mid layui-word-aux">{$lists_v['config_desc']}</div>
|
||||||
|
</div>
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">{$lists_v['config_info']}预览</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<div class="layui-upload-list upload-preview" id="yulan_{$lists_v['config_name']}" style="width:90%;">
|
||||||
|
{if !empty($lists_v['config_value'])}
|
||||||
|
<div class="upload_pic_li" style="position:relative;width:120px;margin:0 10px 10px 0">
|
||||||
|
<img style="background-color: #efefef;height:120px;object-fit:cover;"
|
||||||
|
src="{$lists_v['config_value']}" class="layui-upload-img" onmouseover="show_img(this)"
|
||||||
|
onmouseleave="hide_img()">
|
||||||
|
<input type="hidden" name="{$lists_v['config_name']}" value="{$lists_v['config_value']}" />
|
||||||
|
</div>
|
||||||
|
{/if}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<script type="text/javascript">
|
||||||
|
layui.use(['layer', 'upload'], function () {
|
||||||
|
layer = layui.layer;
|
||||||
|
upload = layui.upload;
|
||||||
|
$ = layui.jquery;
|
||||||
|
|
||||||
|
upload.render({
|
||||||
|
elem: "#{$lists_v['config_name']}",
|
||||||
|
url: "{$config['admin_route']}index/upload_img",
|
||||||
|
multiple: true,
|
||||||
|
done: function (res, title) {
|
||||||
|
if (res.code > 0) {
|
||||||
|
return layer.msg("上传失败," + res.msg, { icon: 2 });
|
||||||
|
} else {
|
||||||
|
$("#yulan_{$lists_v['config_name']}").html('<div class="upload_pic_li" style="position:relative;width:120px;margin:0 10px 10px 0"><img style="width:120px;height:120px;object-fit:cover;" src="' + res.url + '" class="layui-upload-img" onmouseover="show_img(this)" onmouseleave="hide_img()"><input type="hidden" name="{$lists_v['config_name']}" value="' + res.data + '"/></div>');
|
||||||
|
layer.msg("上传成功", { icon: 1 });
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
})
|
||||||
|
</script>
|
||||||
|
{elseif($lists_v['config_type'] == 3)/}
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">{$lists_v['config_info']}</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<textarea class="layui-textarea" id="{$lists_v['config_name']}" name="{$lists_v['config_name']}"
|
||||||
|
placeholder="{$lists_v['config_desc']}">{$lists_v['config_value']}</textarea>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
{elseif($lists_v['config_type'] == 4)/}
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">{$lists_v['config_info']}</label>
|
||||||
|
<div class="layui-input-inline">
|
||||||
|
<input type="checkbox" name="{$lists_v['config_name']}_checkbox" lay-skin="switch" lay-text="开启|关闭" {if
|
||||||
|
$lists_v['config_value']=='1' }checked{/if}>
|
||||||
|
<input type="hidden" name="{$lists_v['config_name']}"
|
||||||
|
value="{if $lists_v['config_value'] == '1'}1{else}0{/if}">
|
||||||
|
</div>
|
||||||
|
<div class="layui-form-mid layui-word-aux">{$lists_v['config_desc']}</div>
|
||||||
|
</div>
|
||||||
|
{/if}
|
||||||
|
{/foreach}
|
||||||
|
</form>
|
||||||
|
|
||||||
|
<div class="save-button-container">
|
||||||
|
<button type="button" class="layui-btn layui-btn-normal" onclick="save()">
|
||||||
|
<i class="layui-icon layui-icon-ok"></i> 保存配置
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<script type="text/javascript">
|
||||||
|
function save() {
|
||||||
|
var loadIndex = layer.load(1, {
|
||||||
|
shade: [0.1, '#fff']
|
||||||
|
});
|
||||||
|
$.post('{$config["admin_route"]}yunzer/configvalue', $('form').serialize(), function (res) {
|
||||||
|
layer.close(loadIndex);
|
||||||
|
if (res.code > 0) {
|
||||||
|
layer.msg(res.msg, { 'icon': 2 });
|
||||||
|
} else {
|
||||||
|
layer.msg(res.msg, { 'icon': 1 });
|
||||||
|
setTimeout(function () {
|
||||||
|
history.go(0);
|
||||||
|
}, 1000);
|
||||||
|
}
|
||||||
|
}, 'json');
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
{include file="public/tail" /}
|
||||||
105
app/admin/view/yunzer/mailconfig.php
Normal file
105
app/admin/view/yunzer/mailconfig.php
Normal file
@ -0,0 +1,105 @@
|
|||||||
|
{include file="public/header" /}
|
||||||
|
<div class="layui-form" lay-filter="mailConfigForm" style="padding: 30px;">
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">SMTP服务器</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<input type="text" name="smtp_host" value="smtp.qq.com" placeholder="请输入SMTP服务器地址" class="layui-input">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">SMTP端口</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<input type="text" name="smtp_port" value="465" placeholder="请输入SMTP端口" class="layui-input">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">发件人邮箱</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<input type="text" name="smtp_email" placeholder="请输入发件人邮箱" class="layui-input">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">邮箱密码</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<input type="password" name="smtp_password" placeholder="请输入邮箱密码或授权码" class="layui-input">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">发件人名称</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<input type="text" name="smtp_name" value="" placeholder="请输入发件人名称" class="layui-input">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<button class="layui-btn" lay-submit lay-filter="saveMailConfig">保存配置</button>
|
||||||
|
<button type="button" class="layui-btn layui-btn-primary" lay-submit lay-filter="testMailConfig">测试发送</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
layui.use(['form', 'layer'], function(){
|
||||||
|
var form = layui.form;
|
||||||
|
var layer = layui.layer;
|
||||||
|
|
||||||
|
// 加载配置
|
||||||
|
$.get('/admin/yunzer/getMailConfig', function(res){
|
||||||
|
if(res.code === 0){
|
||||||
|
form.val('mailConfigForm', res.data);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
// 保存配置
|
||||||
|
form.on('submit(saveMailConfig)', function(data){
|
||||||
|
$.post('/admin/yunzer/mailconfig', data.field, function(res){
|
||||||
|
if(res.code === 1){
|
||||||
|
layer.msg('保存成功');
|
||||||
|
}else{
|
||||||
|
layer.msg(res.msg);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
return false;
|
||||||
|
});
|
||||||
|
|
||||||
|
// 测试发送
|
||||||
|
form.on('submit(testMailConfig)', function(data){
|
||||||
|
layer.prompt({
|
||||||
|
formType: 0,
|
||||||
|
title: '请输入测试邮箱地址',
|
||||||
|
area: ['300px', '150px']
|
||||||
|
}, function(value, index){
|
||||||
|
$.post('/admin/yunzer/testMailConfig', {
|
||||||
|
email: value,
|
||||||
|
config: data.field
|
||||||
|
}, function(res){
|
||||||
|
if(res.code === 1){
|
||||||
|
layer.msg('发送成功');
|
||||||
|
}else{
|
||||||
|
layer.msg(res.msg);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
layer.close(index);
|
||||||
|
});
|
||||||
|
return false;
|
||||||
|
});
|
||||||
|
});
|
||||||
|
</script>
|
||||||
|
<style>
|
||||||
|
.layui-form-item{
|
||||||
|
display: flex;
|
||||||
|
}
|
||||||
|
.layui-form-label{
|
||||||
|
width: 90px !important;
|
||||||
|
}
|
||||||
|
.layui-input-block{
|
||||||
|
width: 100% !important;
|
||||||
|
margin-left: 0 !important;
|
||||||
|
}
|
||||||
|
</style>
|
||||||
|
{include file="public/tail" /}
|
||||||
95
app/admin/view/yunzer/menuadd.php
Normal file
95
app/admin/view/yunzer/menuadd.php
Normal file
@ -0,0 +1,95 @@
|
|||||||
|
{include file="public/header" /}
|
||||||
|
<form class="layui-form">
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">菜单名</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<input type="text" class="layui-input" name="label" placeholder="请输入菜单名">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">图标</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<select name="icon_class" lay-verify="required" lay-search="">
|
||||||
|
<option value="">请输入layui图标</option>
|
||||||
|
{foreach($iconfont as $icon_v)}
|
||||||
|
<option value="{$icon_v['icon_css']}">{$icon_v['icon_css']} {$icon_v['icon_name']}</option>
|
||||||
|
{/foreach}
|
||||||
|
</select>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">类型</label>
|
||||||
|
<div class="layui-input-inline">
|
||||||
|
<select name="type" lay-filter="type">
|
||||||
|
<option value='0' selected>顶级菜单</option>
|
||||||
|
<option value='1'>内部代码</option>
|
||||||
|
<option value='2'>超链接</option>
|
||||||
|
</select>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layui-form-item" style="display:none;" id="src1">
|
||||||
|
<label class="layui-form-label">内部代码</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<input type="text" class="layui-input" name="src1" placeholder="请输入:模块/方法">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layui-form-item" style="display:none;" id="src2">
|
||||||
|
<label class="layui-form-label">链接地址</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<input type="text" class="layui-input" name="src2" placeholder="请输入http网址">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">排序</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<input type="number" class="layui-input" name="sort" placeholder="值越大越靠前">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">状态</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<input type="radio" name="status" value="1" title="开启" checked>
|
||||||
|
<input type="radio" name="status" value="0" title="禁用">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</form>
|
||||||
|
<div class="layui-form-item" style="margin-top:10px;">
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<button type="button" class="layui-btn" onclick="save()">保存</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<script type="text/javascript">
|
||||||
|
layui.use(['layer','form'],function(){
|
||||||
|
var form = layui.form;
|
||||||
|
layer = layui.layer;
|
||||||
|
$ = layui.jquery;
|
||||||
|
|
||||||
|
form.on('select(type)', function(data){
|
||||||
|
if(data.value == 1){
|
||||||
|
$("#src1").show();
|
||||||
|
$("#src2").hide();
|
||||||
|
$("#src3").hide();
|
||||||
|
}else if(data.value == 2){
|
||||||
|
$("#src1").hide();
|
||||||
|
$("#src2").show();
|
||||||
|
$("#src3").hide();
|
||||||
|
}else{
|
||||||
|
$("#src1").hide();
|
||||||
|
$("#src2").hide();
|
||||||
|
$("#src3").hide();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
function save(){
|
||||||
|
$.post('{$config["admin_route"]}yunzer/menuadd',$('form').serialize(),function(res){
|
||||||
|
if(res.code>0){
|
||||||
|
layer.msg(res.msg,{'icon':2});
|
||||||
|
}else{
|
||||||
|
layer.msg(res.msg,{'icon':1});
|
||||||
|
setTimeout(function(){parent.window.location.reload();},1000);
|
||||||
|
}
|
||||||
|
},'json');
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
{include file="public/tail" /}
|
||||||
185
app/admin/view/yunzer/menuedit.php
Normal file
185
app/admin/view/yunzer/menuedit.php
Normal file
@ -0,0 +1,185 @@
|
|||||||
|
{include file="public/header" /}
|
||||||
|
<form class="layui-form">
|
||||||
|
<input type="hidden" name="smid" value="{$lists.smid}">
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">菜单名</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<input type="text" class="layui-input" name="label" placeholder="请输入菜单名" value="{$lists['label']}">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">图标</label>
|
||||||
|
<div class="layui-input-block" style="position:relative;">
|
||||||
|
<input type="text" readonly class="layui-input" name="icon_class" id="icon_class_input"
|
||||||
|
value="{$lists['icon_class']|default=''}" placeholder="请选择图标" autocomplete="off"
|
||||||
|
style="background:#fff;cursor:pointer;">
|
||||||
|
<div class="icon-dropdown" id="iconDropdown"
|
||||||
|
style="display:none;position:absolute;z-index:9999;width:400px;max-height:320px;overflow:auto;background:#fff;border:1px solid #eee;border-radius:6px;box-shadow:0 2px 8px rgba(0,0,0,0.08);padding:12px;">
|
||||||
|
<input type="text" class="layui-input" id="iconSearchInput" placeholder="搜索图标名称或class"
|
||||||
|
style="margin-bottom:10px;">
|
||||||
|
<div class="icon-card-list">
|
||||||
|
{foreach $iconfont as $icon_v}
|
||||||
|
<div class="icon-card{if($icon_v['icon_css'] == $lists['icon_class'])} selected{/if}"
|
||||||
|
data-icon="{$icon_v['icon_css']}" data-name="{$icon_v['icon_name']}"
|
||||||
|
data-css="{$icon_v['icon_css']}">
|
||||||
|
<i class="layui-icon {$icon_v['icon_css']}" style="font-size:24px;"></i>
|
||||||
|
<div class="icon-name">{$icon_v['icon_name']}</div>
|
||||||
|
</div>
|
||||||
|
{/foreach}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">排序</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<input type="number" class="layui-input" name="sort" placeholder="值越大越靠前" value="{$lists['sort']}">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">状态</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<input type="radio" name="status" value="1" title="开启" {$lists['status']==1?'checked':''}>
|
||||||
|
<input type="radio" name="status" value="0" title="禁用" {$lists['status']==0?'checked':''}>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</form>
|
||||||
|
<div class="layui-form-item" style="margin-top:10px;">
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<button type="button" class="layui-btn" onclick="save()">保存</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<script>
|
||||||
|
layui.use(['form', 'jquery', 'layer'], function () {
|
||||||
|
var $ = layui.jquery;
|
||||||
|
var form = layui.form;
|
||||||
|
var layer = layui.layer;
|
||||||
|
var $input = $('#icon_class_input');
|
||||||
|
var $dropdown = $('#iconDropdown');
|
||||||
|
var $search = $('#iconSearchInput');
|
||||||
|
var $cards = $dropdown.find('.icon-card');
|
||||||
|
// 显示下拉
|
||||||
|
$input.on('focus click', function (e) {
|
||||||
|
var offset = $input.offset();
|
||||||
|
$dropdown.css({
|
||||||
|
left: '-15px',
|
||||||
|
top: '40px',
|
||||||
|
display: 'block'
|
||||||
|
});
|
||||||
|
$search.val('').trigger('input');
|
||||||
|
});
|
||||||
|
// 选中卡片
|
||||||
|
$dropdown.on('click', '.icon-card', function () {
|
||||||
|
$cards.removeClass('selected');
|
||||||
|
$(this).addClass('selected');
|
||||||
|
$input.val($(this).data('icon'));
|
||||||
|
$dropdown.hide();
|
||||||
|
});
|
||||||
|
// 搜索
|
||||||
|
$search.on('input', function () {
|
||||||
|
var val = $(this).val().toLowerCase();
|
||||||
|
$cards.each(function () {
|
||||||
|
var name = $(this).data('name').toLowerCase();
|
||||||
|
var css = $(this).data('css').toLowerCase();
|
||||||
|
if (name.indexOf(val) !== -1 || css.indexOf(val) !== -1) {
|
||||||
|
$(this).show();
|
||||||
|
} else {
|
||||||
|
$(this).hide();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
// 点击外部关闭
|
||||||
|
$(document).on('mousedown', function (e) {
|
||||||
|
if (!$(e.target).closest('.icon-dropdown').length && !$(e.target).is($input)) {
|
||||||
|
$dropdown.hide();
|
||||||
|
}
|
||||||
|
});
|
||||||
|
// 回显
|
||||||
|
var selected = $input.val();
|
||||||
|
if (selected) {
|
||||||
|
$cards.each(function () {
|
||||||
|
if ($(this).data('icon') == selected) {
|
||||||
|
$(this).addClass('selected');
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
//保存
|
||||||
|
window.save = function () {
|
||||||
|
var data = {};
|
||||||
|
// 获取表单数据
|
||||||
|
data.smid = $('input[name="smid"]').val();
|
||||||
|
data.label = $('input[name="label"]').val();
|
||||||
|
data.icon_class = $('input[name="icon_class"]').val();
|
||||||
|
data.sort = $('input[name="sort"]').val();
|
||||||
|
data.status = $('input[name="status"]:checked').val();
|
||||||
|
|
||||||
|
// 简单校验
|
||||||
|
if (!data.label) {
|
||||||
|
layer.msg('菜单名不能为空', { icon: 2 });
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
$.post(
|
||||||
|
'{$config["admin_route"]}yunzer/menuedit',
|
||||||
|
data,
|
||||||
|
function (res) {
|
||||||
|
if (res.code == 0) {
|
||||||
|
layer.msg(res.msg, { icon: 1 });
|
||||||
|
// 1秒后关闭弹窗并刷新父页面
|
||||||
|
setTimeout(function () {
|
||||||
|
var index = parent.layer.getFrameIndex(window.name);
|
||||||
|
parent.layer.close(index);
|
||||||
|
parent.location.reload();
|
||||||
|
}, 1000);
|
||||||
|
} else {
|
||||||
|
layer.alert(res.msg, { icon: 2 });
|
||||||
|
}
|
||||||
|
},
|
||||||
|
'json'
|
||||||
|
);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
</script>
|
||||||
|
<style>
|
||||||
|
.icon-card-list {
|
||||||
|
display: flex;
|
||||||
|
flex-wrap: wrap;
|
||||||
|
gap: 12px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.icon-card {
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
width: 52px;
|
||||||
|
height: 52px;
|
||||||
|
border: 1px solid #eee;
|
||||||
|
border-radius: 6px;
|
||||||
|
cursor: pointer;
|
||||||
|
transition: border 0.2s, box-shadow 0.2s;
|
||||||
|
background: #fafbfc;
|
||||||
|
font-size: 12px;
|
||||||
|
color: #666;
|
||||||
|
position: relative;
|
||||||
|
}
|
||||||
|
|
||||||
|
.icon-card.selected {
|
||||||
|
border: 2px solid #409eff;
|
||||||
|
box-shadow: 0 2px 8px rgba(64, 158, 255, 0.10);
|
||||||
|
color: #409eff;
|
||||||
|
}
|
||||||
|
|
||||||
|
.icon-card .icon-name {
|
||||||
|
margin-top: 4px;
|
||||||
|
font-size: 9px;
|
||||||
|
text-align: center;
|
||||||
|
word-break: break-all;
|
||||||
|
}
|
||||||
|
|
||||||
|
.icon-dropdown {
|
||||||
|
min-width: 320px;
|
||||||
|
max-width: 600px;
|
||||||
|
}
|
||||||
|
</style>
|
||||||
|
{include file="public/tail" /}
|
||||||
120
app/admin/view/yunzer/menuinfo.php
Normal file
120
app/admin/view/yunzer/menuinfo.php
Normal file
@ -0,0 +1,120 @@
|
|||||||
|
{include file="public/header" /}
|
||||||
|
<div class="config-container">
|
||||||
|
<!-- 页面头部样式 -->
|
||||||
|
<div class="config-header" style="display: flex;flex-direction: column;flex-wrap: wrap;align-items: flex-start;">
|
||||||
|
<div class="maintitle">
|
||||||
|
<i class="layui-icon layui-icon-picture"></i>
|
||||||
|
<span>菜单管理</span>
|
||||||
|
</div>
|
||||||
|
<div style="display: flex;align-items: flex-start;flex-direction: column;gap: 15px;margin-bottom: 10px;">
|
||||||
|
<div>
|
||||||
|
<button class="layui-btn layui-bg-blue" onclick="add()">
|
||||||
|
<i class="layui-icon layui-icon-add-1"></i>添加菜单
|
||||||
|
</button>
|
||||||
|
<button type="button" class="layui-btn layui-btn-primary layui-border-blue" onclick="refresh()">
|
||||||
|
<i class="layui-icon layui-icon-refresh"></i>刷新
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div id="test7" class="demo-tree"></div>
|
||||||
|
<table class="layui-table">
|
||||||
|
<thead>
|
||||||
|
<tr>
|
||||||
|
<th style="text-align:center">类型</th>
|
||||||
|
<th style="text-align:center">菜单名</th>
|
||||||
|
<th style="text-align:center">图标</th>
|
||||||
|
<th style="text-align:center">排序</th>
|
||||||
|
<th style="text-align:center">状态</th>
|
||||||
|
<th style="text-align:center">操作</th>
|
||||||
|
</tr>
|
||||||
|
</thead>
|
||||||
|
<tbody>
|
||||||
|
{volist name="lists" id='vo'}
|
||||||
|
<tr>
|
||||||
|
<td align="center">
|
||||||
|
{if($vo['type']==1)}
|
||||||
|
功能模块:{$vo['src']}
|
||||||
|
{elseif($vo['type']==2)/}
|
||||||
|
超链接:{$vo['src']}
|
||||||
|
{else/}
|
||||||
|
顶级
|
||||||
|
{/if}
|
||||||
|
</td>
|
||||||
|
<td align="center">{$vo['label']}</td>
|
||||||
|
<td align="center"><i class="layui-icon {$vo['icon_class']}"></i> {$vo['icon_class']}</td>
|
||||||
|
<td align="center">{$vo['sort']}</td>
|
||||||
|
<td align="center">{$vo['status']==1?'开启':'<span style="color:red;">禁用</span>'}</td>
|
||||||
|
<td align="center">
|
||||||
|
{if($vo['type'] == 0)}
|
||||||
|
<button type="button" class="layui-btn layui-btn-xs layui-btn-normal"
|
||||||
|
onclick="buttonInfo({$vo.smid})">下级菜单</button>
|
||||||
|
{/if}
|
||||||
|
<button type="button" class="layui-btn layui-btn-xs layui-btn-primary" onclick="edit({$vo.smid})">
|
||||||
|
<i class="layui-icon layui-icon-edit"></i>编辑
|
||||||
|
</button>
|
||||||
|
<button type="button" class="layui-btn layui-btn-xs layui-btn-primary" onclick="del({$vo.smid})">
|
||||||
|
<i class="layui-icon layui-icon-delete"></i>删除
|
||||||
|
</button>
|
||||||
|
</td>
|
||||||
|
</tr>
|
||||||
|
{/volist}
|
||||||
|
</tbody>
|
||||||
|
</table>
|
||||||
|
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<script type="text/javascript">
|
||||||
|
layui.use(['layer'], function () {
|
||||||
|
layer = layui.layer;
|
||||||
|
$ = layui.jquery;
|
||||||
|
});
|
||||||
|
// 添加
|
||||||
|
function add() {
|
||||||
|
layer.open({
|
||||||
|
type: 2,
|
||||||
|
title: '添加菜单',
|
||||||
|
shade: 0.3,
|
||||||
|
area: ['550px', '550px'],
|
||||||
|
content: '{$config["admin_route"]}yunzer/menuadd'
|
||||||
|
});
|
||||||
|
}
|
||||||
|
// 编辑
|
||||||
|
function edit(smid) {
|
||||||
|
layer.open({
|
||||||
|
type: 2,
|
||||||
|
title: '编辑菜单',
|
||||||
|
shade: 0.3,
|
||||||
|
area: ['550px', '550px'],
|
||||||
|
content: '{$config["admin_route"]}yunzer/menuedit?smid=' + smid
|
||||||
|
});
|
||||||
|
}
|
||||||
|
// 删除
|
||||||
|
function del(smid) {
|
||||||
|
layer.confirm('确定要删除吗?', {
|
||||||
|
icon: 3,
|
||||||
|
btn: ['确定', '取消']
|
||||||
|
}, function () {
|
||||||
|
$.post('{$config["admin_route"]}yunzer/menudel', { 'smid': smid }, function (res) {
|
||||||
|
if (res.code > 0) {
|
||||||
|
layer.alert(res.msg, { icon: 2 });
|
||||||
|
} else {
|
||||||
|
layer.msg(res.msg, { icon: 1 });
|
||||||
|
setTimeout(function () { window.location.reload(); }, 1000);
|
||||||
|
}
|
||||||
|
}, 'json');
|
||||||
|
});
|
||||||
|
}
|
||||||
|
// 按钮列表
|
||||||
|
function buttonInfo(smid) {
|
||||||
|
window.location.href = "{$config['admin_route']}yunzer/buttoninfo?smid=" + smid;
|
||||||
|
}
|
||||||
|
|
||||||
|
//刷新列表
|
||||||
|
function refresh() {
|
||||||
|
window.location.reload();
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
|
||||||
|
{include file="public/tail" /}
|
||||||
109
app/admin/view/yunzeradmin/admininfo.php
Normal file
109
app/admin/view/yunzeradmin/admininfo.php
Normal file
@ -0,0 +1,109 @@
|
|||||||
|
{include file="public/header" /}
|
||||||
|
<form class="layui-form" style="padding:0 20%;">
|
||||||
|
<div class="layui-tab">
|
||||||
|
<ul class="layui-tab-title">
|
||||||
|
<li class="layui-this">个人信息</li>
|
||||||
|
<li>密码信息</li>
|
||||||
|
<li>其它信息</li>
|
||||||
|
</ul>
|
||||||
|
<div class="layui-tab-content">
|
||||||
|
<div class="layui-tab-item layui-show">
|
||||||
|
<div style="padding:15px;">
|
||||||
|
<label class="layui-form-label">账户</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<div class="layui-form-label" style="color:#3881fd;text-align:left;width:100%;">{$aUser['account']}</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div style="padding:15px;">
|
||||||
|
<label class="layui-form-label">说明</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<div class="layui-form-label" style="width:100%;text-align:left;color:#999;">账户用于登录系统的账号,不能修改</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div style="padding:15px;">
|
||||||
|
<label class="layui-form-label">姓名</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<input type="text" class="layui-input" name="name" placeholder="请输入真实姓名" value="{$aUser['name']}">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div style="padding:15px;">
|
||||||
|
<label class="layui-form-label">手机号</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<input type="text" class="layui-input" name="phone" placeholder="请输入手机号" value="{$aUser['phone']}">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div style="padding:15px;">
|
||||||
|
<label class="layui-form-label">QQ号</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<input type="text" class="layui-input" name="qq" placeholder="请输入QQ号" value="{$aUser['qq']}">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div style="padding:15px;">
|
||||||
|
<label class="layui-form-label">性别</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<input type="radio" name="sex" value="1" title="男" {$aUser['status']==1?'checked':''}>
|
||||||
|
<input type="radio" name="sex" value="2" title="女" {$aUser['status']==2?'checked':''}>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layui-tab-item">
|
||||||
|
<div style="padding:15px;">
|
||||||
|
<label class="layui-form-label">老密码</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<input type="text" class="layui-input" name="old_pw" placeholder="请输入老密码,不修改不填写">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div style="padding:15px;">
|
||||||
|
<label class="layui-form-label">新密码</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<input type="text" class="layui-input" name="new_pw" placeholder="请输入新密码,不修改不填写">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layui-tab-item">
|
||||||
|
<div style="padding:15px;">
|
||||||
|
<label class="layui-form-label">登陆次数</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<div class="layui-form-label" style="text-align:left;width:100%;">{$aUser['login_count']}次</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div style="padding:15px;">
|
||||||
|
<label class="layui-form-label">注册时间</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<div class="layui-form-label" style="text-align:left;width:100%;">{:date('Y-m-d H:i:s',$aUser['create_time'])}</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div style="padding:15px;">
|
||||||
|
<label class="layui-form-label">登陆时间</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<div class="layui-form-label" style="text-align:left;width:100%;">{:date('Y-m-d H:i:s',$aUser['update_time'])}</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</form>
|
||||||
|
<div class="layui-form-item" style="margin-top:10px;">
|
||||||
|
<div class="layui-input-block" style="text-align:center;">
|
||||||
|
<button type="button" class="layui-btn" onclick="save()">保存</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<script type="text/javascript">
|
||||||
|
layui.use(['layer','form','element'],function(){
|
||||||
|
var form = layui.form;
|
||||||
|
element = layui.element;
|
||||||
|
layer = layui.layer;
|
||||||
|
$ = layui.jquery;
|
||||||
|
});
|
||||||
|
function save(){
|
||||||
|
$.post("{$config['admin_route']}yunzeradmin/admininfo",$('form').serialize(),function(res){
|
||||||
|
if(res.code>0){
|
||||||
|
layer.msg(res.msg,{'icon':2});
|
||||||
|
}else{
|
||||||
|
layer.msg(res.msg,{'icon':1});
|
||||||
|
setTimeout(function(){parent.window.location.reload();},1000);
|
||||||
|
}
|
||||||
|
},'json');
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
{include file="public/tail" /}
|
||||||
249
app/admin/view/yunzeradmin/banner.php
Normal file
249
app/admin/view/yunzeradmin/banner.php
Normal file
@ -0,0 +1,249 @@
|
|||||||
|
{include file="public/header" /}
|
||||||
|
<!-- 主体内容 -->
|
||||||
|
<div class="config-container">
|
||||||
|
<!-- 页面头部样式 -->
|
||||||
|
<div class="config-header" style="display: flex;flex-direction: column;flex-wrap: wrap;align-items: flex-start;">
|
||||||
|
<div class="maintitle">
|
||||||
|
<i class="layui-icon layui-icon-picture"></i>
|
||||||
|
<span>Banner管理</span>
|
||||||
|
</div>
|
||||||
|
<div style="display: flex;align-items: flex-start;flex-direction: column;gap: 15px;margin-bottom: 10px;">
|
||||||
|
<div>
|
||||||
|
<button class="layui-btn layui-bg-blue" id="addBanner">
|
||||||
|
<i class="layui-icon layui-icon-add-1"></i>添加Banner
|
||||||
|
</button>
|
||||||
|
<button type="button" class="layui-btn layui-btn-primary layui-border-blue" onclick="refresh()">
|
||||||
|
<i class="layui-icon layui-icon-refresh"></i>刷新
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="layui-fluid">
|
||||||
|
<div class="layui-card">
|
||||||
|
<div class="layui-card-body">
|
||||||
|
<table id="bannerTable" lay-filter="bannerTable"></table>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<!-- Banner表单 -->
|
||||||
|
<div id="bannerForm" style="display: none;">
|
||||||
|
<form class="layui-form banner-form">
|
||||||
|
<input type="hidden" name="id">
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">标题</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<input type="text" name="title" required lay-verify="required" placeholder="请输入标题"
|
||||||
|
class="layui-input">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">图片</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<div class="layui-upload-drag" id="uploadImage">
|
||||||
|
<i class="layui-icon layui-icon-upload"></i>
|
||||||
|
<p>点击上传或拖拽图片至此处</p>
|
||||||
|
<div id="uploadPreview" style="display: none;">
|
||||||
|
<img src="" alt="Banner图片" class="banner-upload-preview">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<input type="hidden" name="image" id="imageInput">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">链接</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<input type="text" name="url" placeholder="请输入链接地址" class="layui-input">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">排序</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<input type="number" name="sort" value="0" class="layui-input" placeholder="数字越大越靠前">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">状态</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<input type="radio" name="status" value="1" title="启用" checked>
|
||||||
|
<input type="radio" name="status" value="0" title="禁用">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<button class="layui-btn" lay-submit lay-filter="saveBanner">保存</button>
|
||||||
|
<button type="reset" class="layui-btn layui-btn-primary">重置</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</form>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
layui.use(['table', 'form', 'upload', 'layer'], function () {
|
||||||
|
var table = layui.table,
|
||||||
|
form = layui.form,
|
||||||
|
upload = layui.upload,
|
||||||
|
layer = layui.layer;
|
||||||
|
|
||||||
|
// 表格列配置
|
||||||
|
var tableColumns = [[
|
||||||
|
{ field: 'id', title: 'ID', width: 80, align: 'center' },
|
||||||
|
{ field: 'title', title: '标题', align: 'center' },
|
||||||
|
{
|
||||||
|
field: 'image', title: '图片', width: 180, align: 'center', templet: function (d) {
|
||||||
|
return '<img src="' + d.image + '" class="banner-preview-img" onclick="previewImage(\'' + d.image + '\')">';
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{ field: 'url', title: '链接', align: 'center', width: 300 },
|
||||||
|
{ field: 'sort', title: '排序', align: 'center', width: 100, sort: true },
|
||||||
|
{ field: 'create_time', title: '创建时间', align: 'center', width: 180, sort: true },
|
||||||
|
{
|
||||||
|
title: '操作', align: 'center', width: 240, templet: function (d) {
|
||||||
|
return '<a class="layui-btn layui-btn-xs" lay-event="edit">编辑</a>' +
|
||||||
|
'<a class="layui-btn layui-btn-danger layui-btn-xs" lay-event="del">删除</a>';
|
||||||
|
}
|
||||||
|
}
|
||||||
|
]];
|
||||||
|
|
||||||
|
// 初始化表格
|
||||||
|
table.render({
|
||||||
|
elem: '#bannerTable',
|
||||||
|
url: '/admin/yunzeradmin/bannerlist',
|
||||||
|
method: 'get',
|
||||||
|
page: true,
|
||||||
|
cols: tableColumns,
|
||||||
|
limit: 10,
|
||||||
|
limits: [10, 20, 30, 50],
|
||||||
|
text: { none: '暂无相关数据' }
|
||||||
|
});
|
||||||
|
|
||||||
|
// 上传组件配置
|
||||||
|
var uploadConfig = {
|
||||||
|
elem: '#uploadImage',
|
||||||
|
url: '/admin/upload/image',
|
||||||
|
accept: 'images',
|
||||||
|
acceptMime: 'image/*',
|
||||||
|
done: function (res) {
|
||||||
|
if (res.code === 0) {
|
||||||
|
$('#uploadPreview').show().find('img').attr('src', res.data.url);
|
||||||
|
$('#imageInput').val(res.data.url);
|
||||||
|
layer.msg('上传成功');
|
||||||
|
} else {
|
||||||
|
layer.msg('上传失败');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
// 初始化上传组件
|
||||||
|
upload.render(uploadConfig);
|
||||||
|
|
||||||
|
// 监听表格工具条事件
|
||||||
|
table.on('tool(bannerTable)', function (obj) {
|
||||||
|
var data = obj.data;
|
||||||
|
if (obj.event === 'edit') {
|
||||||
|
showBannerForm(data);
|
||||||
|
} else if (obj.event === 'del') {
|
||||||
|
layer.confirm('确定删除此Banner?', function (index) {
|
||||||
|
deleteBanner(data.id, obj);
|
||||||
|
layer.close(index);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
// 添加Banner按钮点击事件
|
||||||
|
$('#addBanner').on('click', function () {
|
||||||
|
showBannerForm();
|
||||||
|
});
|
||||||
|
|
||||||
|
// 监听表单提交
|
||||||
|
form.on('submit(saveBanner)', function (data) {
|
||||||
|
var url = data.field.id ? '/admin/yunzeradmin/banneredit' : '/admin/yunzeradmin/banneradd';
|
||||||
|
$.post(url, data.field, function (res) {
|
||||||
|
if (res.code === 0) {
|
||||||
|
layer.closeAll('page');
|
||||||
|
table.reload('bannerTable');
|
||||||
|
layer.msg('保存成功');
|
||||||
|
} else {
|
||||||
|
layer.msg(res.msg);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
return false;
|
||||||
|
});
|
||||||
|
|
||||||
|
// 显示Banner表单
|
||||||
|
function showBannerForm(data) {
|
||||||
|
layer.open({
|
||||||
|
type: 1,
|
||||||
|
title: data ? '编辑Banner' : '添加Banner',
|
||||||
|
content: $('#bannerForm'),
|
||||||
|
area: ['800px', '600px'],
|
||||||
|
success: function (layero) {
|
||||||
|
form.render();
|
||||||
|
if (data) {
|
||||||
|
form.val('bannerForm', data);
|
||||||
|
if (data.image) {
|
||||||
|
$('#uploadPreview').show().find('img').attr('src', data.image);
|
||||||
|
$('#imageInput').val(data.image);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
// 删除Banner
|
||||||
|
function deleteBanner(id, obj) {
|
||||||
|
$.post('/admin/yunzeradmin/bannerdel', { id: id }, function (res) {
|
||||||
|
if (res.code === 0) {
|
||||||
|
obj.del();
|
||||||
|
layer.msg('删除成功');
|
||||||
|
} else {
|
||||||
|
layer.msg(res.msg);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
// 预览图片
|
||||||
|
function previewImage(url) {
|
||||||
|
layer.photos({
|
||||||
|
photos: {
|
||||||
|
"data": [{
|
||||||
|
"src": url
|
||||||
|
}]
|
||||||
|
}
|
||||||
|
});
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
<!-- 页面样式 -->
|
||||||
|
<style>
|
||||||
|
.layui-table-cell {
|
||||||
|
height: 40px;
|
||||||
|
line-height: 40px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.layui-table-cell img {
|
||||||
|
height: 36px;
|
||||||
|
width: auto;
|
||||||
|
max-width: 80px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.banner-preview-img {
|
||||||
|
max-width: 80px;
|
||||||
|
cursor: pointer;
|
||||||
|
}
|
||||||
|
|
||||||
|
.banner-form {
|
||||||
|
padding: 20px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.banner-upload-preview {
|
||||||
|
max-width: 100%;
|
||||||
|
margin-top: 10px;
|
||||||
|
}
|
||||||
|
</style>
|
||||||
231
app/admin/view/yunzeradmin/contentpush.php
Normal file
231
app/admin/view/yunzeradmin/contentpush.php
Normal file
@ -0,0 +1,231 @@
|
|||||||
|
{include file="public/header" /}
|
||||||
|
<div class="config-container">
|
||||||
|
<!-- 页面头部样式 -->
|
||||||
|
<div class="config-header" style="display: flex;flex-direction: column;flex-wrap: wrap;align-items: flex-start;">
|
||||||
|
<div class="maintitle">
|
||||||
|
<i class="layui-icon layui-icon-notice"></i>
|
||||||
|
<span>内容推送列表</span>
|
||||||
|
</div>
|
||||||
|
<div style="display: flex;align-items: flex-start;flex-direction: column;gap: 15px;margin-bottom: 10px;">
|
||||||
|
<div>
|
||||||
|
<!-- <button class="layui-btn layui-bg-blue" onclick="add()">
|
||||||
|
<i class="layui-icon layui-icon-add-1"></i>添加推送
|
||||||
|
</button> -->
|
||||||
|
<button class="layui-btn layui-bg-blue" onclick="setting()">
|
||||||
|
<i class="layui-icon layui-icon-set-fill"></i>推送配置
|
||||||
|
</button>
|
||||||
|
<button type="button" class="layui-btn layui-btn-primary layui-border-green" onclick="pushSelected()">
|
||||||
|
<i class="layui-icon layui-icon-release"></i>推送
|
||||||
|
</button>
|
||||||
|
<button type="button" class="layui-btn layui-btn-primary layui-border-blue" onclick="refresh()">
|
||||||
|
<i class="layui-icon layui-icon-refresh"></i>刷新
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<table id="contentPushTable" lay-filter="contentPushTable"></table>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<script type="text/javascript">
|
||||||
|
layui.use(['table', 'layer'], function () {
|
||||||
|
var table = layui.table;
|
||||||
|
var layer = layui.layer;
|
||||||
|
var $ = layui.jquery;
|
||||||
|
|
||||||
|
// 初始化表格
|
||||||
|
table.render({
|
||||||
|
elem: '#contentPushTable',
|
||||||
|
url: '{$config["admin_route"]}yunzeradmin/contentpushlist',
|
||||||
|
page: true,
|
||||||
|
cols: [[
|
||||||
|
{ type: 'checkbox', fixed: 'left' }, // 多选全选
|
||||||
|
// {field: 'id', title: 'ID', width: 80, sort: true},
|
||||||
|
{
|
||||||
|
field: 'type', title: '类型', width: 80, templet: function (d) {
|
||||||
|
return d.type == 'article' ? '文章' : '资源';
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{ field: 'title', title: '标题' },
|
||||||
|
{
|
||||||
|
field: 'url', title: '链接', templet: function (d) {
|
||||||
|
if (d.type == 'article') {
|
||||||
|
return '<a href="https://www.yunzer.cn/index/articles/detail?id=' + d.id + '" target="_blank">https://www.yunzer.cn/index/articles/detail?id=' + d.id + '</a>';
|
||||||
|
} else if (d.type == 'resource') {
|
||||||
|
return '<a href="https://www.yunzer.cn/index/resources/detail?id=' + d.id + '" target="_blank">https://www.yunzer.cn/index/resources/detail?id=' + d.id + '</a>';
|
||||||
|
} else {
|
||||||
|
return '';
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
field: 'push', title: '推送状态', width: 100, templet: function (d) {
|
||||||
|
return d.push == 1 ?
|
||||||
|
'<span class="layui-badge layui-bg-green">已推送</span>' :
|
||||||
|
'<span class="layui-badge">未推送</span>';
|
||||||
|
}
|
||||||
|
},
|
||||||
|
// {field: 'sort', title: '排序', width: 100, sort: true},
|
||||||
|
{ field: 'create_time', title: '创建时间', width: 180 },
|
||||||
|
// {title: '操作', width: 200, toolbar: '#tableBar', fixed: 'right'}
|
||||||
|
]],
|
||||||
|
limit: 10,
|
||||||
|
limits: [10, 20, 30, 50]
|
||||||
|
});
|
||||||
|
|
||||||
|
// 监听工具条
|
||||||
|
table.on('tool(contentPushTable)', function (obj) {
|
||||||
|
var data = obj.data;
|
||||||
|
if (obj.event === 'edit') {
|
||||||
|
edit(data.id);
|
||||||
|
} else if (obj.event === 'del') {
|
||||||
|
del(data.id);
|
||||||
|
} else if (obj.event === 'status') {
|
||||||
|
changeStatus(data.id, data.status);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
// 添加
|
||||||
|
function add() {
|
||||||
|
layer.open({
|
||||||
|
type: 2,
|
||||||
|
title: '添加内容推送',
|
||||||
|
shade: 0.3,
|
||||||
|
area: ['800px', '600px'],
|
||||||
|
content: "{$config['admin_route']}yunzeradmin/contentpushadd"
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
// 编辑
|
||||||
|
function edit(id) {
|
||||||
|
layer.open({
|
||||||
|
type: 2,
|
||||||
|
title: '编辑内容推送',
|
||||||
|
shade: 0.3,
|
||||||
|
area: ['800px', '600px'],
|
||||||
|
content: "{$config['admin_route']}yunzeradmin/contentpushedit?id=" + id
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
// 删除
|
||||||
|
function del(id) {
|
||||||
|
layer.confirm('确定要删除这条推送吗?', {
|
||||||
|
icon: 3,
|
||||||
|
btn: ['确定', '取消']
|
||||||
|
}, function () {
|
||||||
|
$.post("{$config['admin_route']}yunzeradmin/contentpushdel", { 'id': id }, function (res) {
|
||||||
|
if (res.code > 0) {
|
||||||
|
layer.msg(res.msg, { icon: 2 });
|
||||||
|
} else {
|
||||||
|
layer.msg(res.msg, { icon: 1 });
|
||||||
|
layui.table.reload('contentPushTable');
|
||||||
|
}
|
||||||
|
}, 'json');
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
// 修改状态
|
||||||
|
function changeStatus(id, status) {
|
||||||
|
var newStatus = status == 1 ? 0 : 1;
|
||||||
|
$.post("{$config['admin_route']}yunzeradmin/contentpushstatus", {
|
||||||
|
'id': id,
|
||||||
|
'status': newStatus
|
||||||
|
}, function (res) {
|
||||||
|
if (res.code > 0) {
|
||||||
|
layer.msg(res.msg, { icon: 2 });
|
||||||
|
} else {
|
||||||
|
layer.msg(res.msg, { icon: 1 });
|
||||||
|
layui.table.reload('contentPushTable');
|
||||||
|
}
|
||||||
|
}, 'json');
|
||||||
|
}
|
||||||
|
|
||||||
|
//配置
|
||||||
|
function setting() {
|
||||||
|
layer.open({
|
||||||
|
type: 2,
|
||||||
|
title: '推送配置',
|
||||||
|
shade: 0.3,
|
||||||
|
area: ['1000px', '800px'],
|
||||||
|
content: "{$config['admin_route']}yunzeradmin/contentpushsetting"
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
//推送数据
|
||||||
|
function pushSelected() {
|
||||||
|
var checkStatus = layui.table.checkStatus('contentPushTable');
|
||||||
|
var data = checkStatus.data;
|
||||||
|
if (data.length === 0) {
|
||||||
|
layer.msg('请先选择要推送的数据', { icon: 2 });
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
$.getJSON("{$config['admin_route']}yunzeradmin/contentpushsetting", function (res) {
|
||||||
|
if (res.code !== 0 || !res.data || res.data.length === 0) {
|
||||||
|
layer.msg('暂无可用推送平台');
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
var html = '<div style="padding:10px;"><table class="layui-table"><thead><tr><th>ID</th><th>平台名称</th><th>类型标识</th></tr></thead><tbody>';
|
||||||
|
res.data.forEach(function (item) {
|
||||||
|
if (item.status == 1) {
|
||||||
|
html += '<tr style="cursor:pointer" data-type="' + item.platformType + '" data-title="' + item.title.replace(/'/g, "\\'") + '">';
|
||||||
|
html += '<td>' + item.id + '</td>';
|
||||||
|
html += '<td>' + item.title + '</td>';
|
||||||
|
html += '<td>' + item.platformType + '</td>';
|
||||||
|
html += '</tr>';
|
||||||
|
}
|
||||||
|
});
|
||||||
|
html += '</tbody></table></div>';
|
||||||
|
|
||||||
|
layer.open({
|
||||||
|
type: 1,
|
||||||
|
title: '选择推送平台',
|
||||||
|
area: ['500px', '350px'],
|
||||||
|
content: html,
|
||||||
|
success: function (layero, index) {
|
||||||
|
// 事件委托,避免全局变量
|
||||||
|
$(layero).find('tr[data-type]').on('click', function () {
|
||||||
|
var platformType = $(this).data('type');
|
||||||
|
var platformTitle = $(this).data('title');
|
||||||
|
layer.close(index);
|
||||||
|
|
||||||
|
var urls = [];
|
||||||
|
data.forEach(function (item) {
|
||||||
|
urls.push('https://www.yunzer.cn/index/' + (item.type === 'resource' ? 'resources' : 'articles') + '/detail?id=' + item.id);
|
||||||
|
});
|
||||||
|
|
||||||
|
$.ajax({
|
||||||
|
url: "{$config['admin_route']}yunzeradmin/gopush",
|
||||||
|
type: "POST",
|
||||||
|
contentType: "application/json",
|
||||||
|
data: JSON.stringify({
|
||||||
|
platformType: platformType,
|
||||||
|
urls: urls
|
||||||
|
}),
|
||||||
|
dataType: "json",
|
||||||
|
success: function (res) {
|
||||||
|
layer.msg(res.msg, { icon: res.code === 0 ? 1 : 2, time: 5000 });
|
||||||
|
layui.table.reload('contentPushTable');
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
// 刷新列表
|
||||||
|
function refresh() {
|
||||||
|
layui.table.reload('contentPushTable');
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<!-- 表格操作列模板 -->
|
||||||
|
<!-- <script type="text/html" id="tableBar">
|
||||||
|
<a class="layui-btn layui-btn-xs" lay-event="edit">编辑</a>
|
||||||
|
<a class="layui-btn layui-btn-xs layui-btn-danger" lay-event="del">删除</a>
|
||||||
|
</script> -->
|
||||||
|
|
||||||
|
{include file="public/tail" /}
|
||||||
174
app/admin/view/yunzeradmin/contentpushadd.php
Normal file
174
app/admin/view/yunzeradmin/contentpushadd.php
Normal file
@ -0,0 +1,174 @@
|
|||||||
|
{include file="public/header" /}
|
||||||
|
<div class="config-container">
|
||||||
|
<form class="layui-form" action="{$config['admin_route']}yunzeradmin/contentpushsave" method="post"
|
||||||
|
lay-filter="contentPushForm">
|
||||||
|
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">推送平台</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<select name="push_platform" id="pushPlatformSelect" required lay-verify="required">
|
||||||
|
<option value="">请选择推送平台</option>
|
||||||
|
</select>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">推送分类</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<select name="pushcate" required lay-verify="required">
|
||||||
|
<option value="article">文章分类</option>
|
||||||
|
<option value="resource">资源分类</option>
|
||||||
|
</select>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">推送内容</label>
|
||||||
|
<div class="layui-input-block" style="display: flex; align-items: center;">
|
||||||
|
<input type="text" name="title" id="pushTitleInput" required lay-verify="required" placeholder="请输入推送标题"
|
||||||
|
autocomplete="off" class="layui-input" style="flex:1;">
|
||||||
|
<button type="button" class="layui-btn layui-btn-primary" id="selectPushContent"
|
||||||
|
style="margin-left: 8px;">
|
||||||
|
<i class="layui-icon layui-icon-search"></i>
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">跳转链接</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<input type="text" name="url" placeholder="请输入跳转链接" autocomplete="off" class="layui-input">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<!-- <div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">推送图片</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<input type="text" name="image" id="image" placeholder="请上传推送图片" autocomplete="off" class="layui-input">
|
||||||
|
<button type="button" class="layui-btn" id="uploadImage">
|
||||||
|
<i class="layui-icon"></i>上传图片
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</div> -->
|
||||||
|
|
||||||
|
<!-- <div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">推送类型</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<input type="radio" name="type" value="1" title="普通推送" checked>
|
||||||
|
<input type="radio" name="type" value="2" title="重要推送">
|
||||||
|
</div>
|
||||||
|
</div> -->
|
||||||
|
|
||||||
|
<!-- <div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">状态</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<input type="radio" name="status" value="1" title="启用" checked>
|
||||||
|
<input type="radio" name="status" value="0" title="禁用">
|
||||||
|
</div>
|
||||||
|
</div> -->
|
||||||
|
|
||||||
|
<!-- <div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">排序</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<input type="number" name="sort" value="0" placeholder="请输入排序值" autocomplete="off" class="layui-input">
|
||||||
|
</div>
|
||||||
|
</div> -->
|
||||||
|
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<button class="layui-btn" lay-submit lay-filter="contentPushForm">立即提交</button>
|
||||||
|
<button type="reset" class="layui-btn layui-btn-primary">重置</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</form>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
layui.use(['layer', 'jquery', 'form', 'upload'], function () {
|
||||||
|
var $ = layui.jquery,
|
||||||
|
layer = layui.layer,
|
||||||
|
form = layui.form,
|
||||||
|
upload = layui.upload;
|
||||||
|
|
||||||
|
// 动态加载推送平台
|
||||||
|
$.getJSON("{$config['admin_route']}yunzeradmin/contentpushsetting", function (res) {
|
||||||
|
if (res.data.length > 0) {
|
||||||
|
var $select = $('#pushPlatformSelect');
|
||||||
|
res.data.forEach(function (item) {
|
||||||
|
if (item.status == 1) {
|
||||||
|
$select.append(
|
||||||
|
$('<option>', {
|
||||||
|
value: item.id,
|
||||||
|
text: item.title
|
||||||
|
})
|
||||||
|
);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
form.render('select');
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
// 选择推送内容
|
||||||
|
$('#selectPushContent').on('click', function () {
|
||||||
|
var pushCate = $('select[name="pushcate"]').val();
|
||||||
|
|
||||||
|
// 弹窗选择内容
|
||||||
|
layer.open({
|
||||||
|
type: 2,
|
||||||
|
title: '选择推送内容',
|
||||||
|
area: ['800px', '500px'],
|
||||||
|
content: "{$config['admin_route']}yunzeradmin/selectpushcontent?pushcate=" + pushCate,
|
||||||
|
success: function (layero, index) {
|
||||||
|
// 可在弹窗页面通过父页面回调选中内容
|
||||||
|
window.setPushContent = function (title, url) {
|
||||||
|
$('#pushTitleInput').val(title);
|
||||||
|
// 如果需要同步设置跳转链接
|
||||||
|
if (url) {
|
||||||
|
$('input[name="url"]').val(url);
|
||||||
|
}
|
||||||
|
layer.close(index);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
// 图片上传
|
||||||
|
upload.render({
|
||||||
|
elem: '#uploadImage',
|
||||||
|
url: '{$config["admin_route"]}yunzeradmin/upload',
|
||||||
|
accept: 'images',
|
||||||
|
acceptMime: 'image/*',
|
||||||
|
done: function (res) {
|
||||||
|
if (res.code == 0) {
|
||||||
|
$('#image').val(res.data.src);
|
||||||
|
layer.msg('上传成功');
|
||||||
|
} else {
|
||||||
|
layer.msg('上传失败');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
// 表单提交
|
||||||
|
form.on('submit(contentPushForm)', function (data) {
|
||||||
|
$.ajax({
|
||||||
|
url: data.form.action,
|
||||||
|
type: 'POST',
|
||||||
|
data: data.field,
|
||||||
|
success: function (res) {
|
||||||
|
if (res.code == 0) {
|
||||||
|
layer.msg(res.msg, { icon: 1 }, function () {
|
||||||
|
var index = parent.layer.getFrameIndex(window.name);
|
||||||
|
parent.layer.close(index);
|
||||||
|
parent.layui.table.reload('contentPushTable');
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
layer.msg(res.msg, { icon: 2 });
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
return false;
|
||||||
|
});
|
||||||
|
});
|
||||||
|
</script>
|
||||||
|
|
||||||
|
{include file="public/tail" /}
|
||||||
108
app/admin/view/yunzeradmin/contentpushedit.php
Normal file
108
app/admin/view/yunzeradmin/contentpushedit.php
Normal file
@ -0,0 +1,108 @@
|
|||||||
|
{include file="public/header" /}
|
||||||
|
<div class="config-container">
|
||||||
|
<form class="layui-form" action="{$config['admin_route']}yunzeradmin/contentpushsave" method="post" lay-filter="contentPushForm">
|
||||||
|
<input type="hidden" name="id" value="{$info.id}">
|
||||||
|
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">推送标题</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<input type="text" name="title" value="{$info.title}" required lay-verify="required" placeholder="请输入推送标题" autocomplete="off" class="layui-input">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">推送内容</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<textarea name="content" placeholder="请输入推送内容" class="layui-textarea" required lay-verify="required">{$info.content}</textarea>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">推送图片</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<input type="text" name="image" id="image" value="{$info.image}" placeholder="请上传推送图片" autocomplete="off" class="layui-input">
|
||||||
|
<button type="button" class="layui-btn" id="uploadImage">
|
||||||
|
<i class="layui-icon"></i>上传图片
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">跳转链接</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<input type="text" name="url" value="{$info.url}" placeholder="请输入跳转链接" autocomplete="off" class="layui-input">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">推送类型</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<input type="radio" name="type" value="1" title="普通推送" {if $info.type==1}checked{/if}>
|
||||||
|
<input type="radio" name="type" value="2" title="重要推送" {if $info.type==2}checked{/if}>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">状态</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<input type="radio" name="status" value="1" title="启用" {if $info.status==1}checked{/if}>
|
||||||
|
<input type="radio" name="status" value="0" title="禁用" {if $info.status==0}checked{/if}>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<button class="layui-btn" lay-submit lay-filter="contentPushForm">保存</button>
|
||||||
|
<button type="reset" class="layui-btn layui-btn-primary">重置</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</form>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<script type="text/javascript">
|
||||||
|
layui.use(['form', 'upload', 'layer'], function(){
|
||||||
|
var form = layui.form;
|
||||||
|
var upload = layui.upload;
|
||||||
|
var layer = layui.layer;
|
||||||
|
var $ = layui.jquery;
|
||||||
|
|
||||||
|
// 图片上传
|
||||||
|
upload.render({
|
||||||
|
elem: '#uploadImage',
|
||||||
|
url: '{$config["admin_route"]}yunzeradmin/upload',
|
||||||
|
accept: 'images',
|
||||||
|
acceptMime: 'image/*',
|
||||||
|
done: function(res){
|
||||||
|
if(res.code == 0){
|
||||||
|
$('#image').val(res.data.src);
|
||||||
|
layer.msg('上传成功');
|
||||||
|
} else {
|
||||||
|
layer.msg('上传失败');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
// 表单提交
|
||||||
|
form.on('submit(contentPushForm)', function(data){
|
||||||
|
$.ajax({
|
||||||
|
url: data.form.action,
|
||||||
|
type: 'POST',
|
||||||
|
data: data.field,
|
||||||
|
success: function(res){
|
||||||
|
if(res.code == 0){
|
||||||
|
layer.msg(res.msg, {icon: 1}, function(){
|
||||||
|
var index = parent.layer.getFrameIndex(window.name);
|
||||||
|
parent.layer.close(index);
|
||||||
|
parent.layui.table.reload('contentPushTable');
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
layer.msg(res.msg, {icon: 2});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
return false;
|
||||||
|
});
|
||||||
|
});
|
||||||
|
</script>
|
||||||
|
|
||||||
|
{include file="public/tail" /}
|
||||||
109
app/admin/view/yunzeradmin/contentpushsetting.php
Normal file
109
app/admin/view/yunzeradmin/contentpushsetting.php
Normal file
@ -0,0 +1,109 @@
|
|||||||
|
{include file="public/header" /}
|
||||||
|
<div class="config-container">
|
||||||
|
<div class="config-header" style="display: flex;flex-direction: column;flex-wrap: wrap;align-items: flex-start;">
|
||||||
|
<div class="maintitle">
|
||||||
|
<i class="layui-icon layui-icon-set-fill"></i>
|
||||||
|
<span>推送配置列表</span>
|
||||||
|
</div>
|
||||||
|
<div style="display: flex;align-items: flex-start;flex-direction: column;gap: 15px;margin-bottom: 10px;">
|
||||||
|
<div>
|
||||||
|
<button class="layui-btn layui-bg-blue" onclick="addSetting()">
|
||||||
|
<i class="layui-icon layui-icon-add-1"></i>添加配置
|
||||||
|
</button>
|
||||||
|
<button type="button" class="layui-btn layui-btn-primary layui-border-blue" onclick="refreshSetting()">
|
||||||
|
<i class="layui-icon layui-icon-refresh"></i>刷新
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<table id="contentPushSettingTable" lay-filter="contentPushSettingTable"></table>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<script type="text/html" id="settingTableBar">
|
||||||
|
<a class="layui-btn layui-btn-xs" lay-event="edit">编辑</a>
|
||||||
|
<a class="layui-btn layui-btn-danger layui-btn-xs" lay-event="del">删除</a>
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<script type="text/javascript">
|
||||||
|
layui.use(['table', 'layer'], function () {
|
||||||
|
var table = layui.table;
|
||||||
|
var layer = layui.layer;
|
||||||
|
var $ = layui.jquery;
|
||||||
|
|
||||||
|
// 渲染表格
|
||||||
|
table.render({
|
||||||
|
elem: '#contentPushSettingTable',
|
||||||
|
url: '{$config["admin_route"]}yunzeradmin/contentpushsetting',
|
||||||
|
page: true,
|
||||||
|
cols: [[
|
||||||
|
{field: 'id', title: 'ID', width: 80, sort: true},
|
||||||
|
{field: 'title', title: '配置标题', width: 200},
|
||||||
|
{field: 'value', title: '配置值', width: 300},
|
||||||
|
{field: 'status', title: '状态', width: 100, templet: function(d){
|
||||||
|
return d.status == 1 ?
|
||||||
|
'<span class="layui-badge layui-bg-green">启用</span>' :
|
||||||
|
'<span class="layui-badge">禁用</span>';
|
||||||
|
}},
|
||||||
|
{field: 'sort', title: '排序', width: 100, sort: true},
|
||||||
|
{field: 'create_time', title: '创建时间', width: 180},
|
||||||
|
{title: '操作', width: 160, toolbar: '#settingTableBar', fixed: 'right'}
|
||||||
|
]],
|
||||||
|
limit: 10,
|
||||||
|
limits: [10, 20, 30, 50]
|
||||||
|
});
|
||||||
|
|
||||||
|
// 工具条事件
|
||||||
|
table.on('tool(contentPushSettingTable)', function(obj){
|
||||||
|
var data = obj.data;
|
||||||
|
if(obj.event === 'edit'){
|
||||||
|
editSetting(data.id);
|
||||||
|
} else if(obj.event === 'del'){
|
||||||
|
layer.confirm('确定要删除该配置吗?', function(index){
|
||||||
|
$.post('{$config["admin_route"]}yunzeradmin/contentpushsettingdel', {id: data.id}, function(res){
|
||||||
|
if(res.code == 0){
|
||||||
|
layer.msg('删除成功', {icon: 1});
|
||||||
|
table.reload('contentPushSettingTable');
|
||||||
|
} else {
|
||||||
|
layer.msg(res.msg, {icon: 2});
|
||||||
|
}
|
||||||
|
});
|
||||||
|
layer.close(index);
|
||||||
|
});
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
// 刷新
|
||||||
|
window.refreshSetting = function(){
|
||||||
|
table.reload('contentPushSettingTable');
|
||||||
|
};
|
||||||
|
|
||||||
|
// 添加
|
||||||
|
window.addSetting = function(){
|
||||||
|
layer.open({
|
||||||
|
type: 2,
|
||||||
|
title: '添加推送配置',
|
||||||
|
area: ['800px', '600px'],
|
||||||
|
content: '{$config["admin_route"]}yunzeradmin/contentpushsettingadd',
|
||||||
|
end: function(){
|
||||||
|
table.reload('contentPushSettingTable');
|
||||||
|
}
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
|
||||||
|
// 编辑
|
||||||
|
window.editSetting = function(id){
|
||||||
|
layer.open({
|
||||||
|
type: 2,
|
||||||
|
title: '编辑推送配置',
|
||||||
|
area: ['600px', '400px'],
|
||||||
|
content: '{$config["admin_route"]}yunzeradmin/contentpushsettingadd?id=' + id,
|
||||||
|
end: function(){
|
||||||
|
table.reload('contentPushSettingTable');
|
||||||
|
}
|
||||||
|
});
|
||||||
|
};
|
||||||
|
});
|
||||||
|
</script>
|
||||||
|
|
||||||
|
{include file="public/tail" /}
|
||||||
75
app/admin/view/yunzeradmin/contentpushsettingadd.php
Normal file
75
app/admin/view/yunzeradmin/contentpushsettingadd.php
Normal file
@ -0,0 +1,75 @@
|
|||||||
|
{include file="public/header" /}
|
||||||
|
<div class="config-container">
|
||||||
|
<form class="layui-form" action="{$config['admin_route']}yunzeradmin/contentpushsettingadd" method="post" lay-filter="contentPushForm">
|
||||||
|
{if isset($info.id)}
|
||||||
|
<input type="hidden" name="id" value="{$info.id}">
|
||||||
|
{/if}
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">配置标题</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<input type="text" name="title" required lay-verify="required" placeholder="请输入配置标题" autocomplete="off" class="layui-input" value="{$info.title|default=''}">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">配置值</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<textarea name="value" placeholder="请输入配置值" class="layui-textarea" required lay-verify="required">{$info.value|default=''}</textarea>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">状态</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<input type="radio" name="status" value="1" title="启用" {if !isset($info.status) || $info.status==1}checked{/if}>
|
||||||
|
<input type="radio" name="status" value="0" title="禁用" {if isset($info.status) && $info.status==0}checked{/if}>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">排序</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<input type="number" name="sort" value="{$info.sort|default='0'}" placeholder="请输入排序值" autocomplete="off" class="layui-input">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<button class="layui-btn" lay-submit lay-filter="contentPushForm">立即提交</button>
|
||||||
|
<button type="reset" class="layui-btn layui-btn-primary">重置</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</form>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<script type="text/javascript">
|
||||||
|
layui.use(['form', 'upload', 'layer'], function(){
|
||||||
|
var form = layui.form;
|
||||||
|
var upload = layui.upload;
|
||||||
|
var layer = layui.layer;
|
||||||
|
var $ = layui.jquery;
|
||||||
|
|
||||||
|
// 表单提交
|
||||||
|
form.on('submit(contentPushForm)', function(data){
|
||||||
|
$.ajax({
|
||||||
|
url: data.form.action,
|
||||||
|
type: 'POST',
|
||||||
|
data: data.field,
|
||||||
|
success: function(res){
|
||||||
|
if(res.code == 0){
|
||||||
|
layer.msg(res.msg, {icon: 1}, function(){
|
||||||
|
var index = parent.layer.getFrameIndex(window.name);
|
||||||
|
parent.layer.close(index);
|
||||||
|
parent.layui.table.reload('contentPushTable');
|
||||||
|
});
|
||||||
|
} else {
|
||||||
|
layer.msg(res.msg, {icon: 2});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
return false;
|
||||||
|
});
|
||||||
|
});
|
||||||
|
</script>
|
||||||
|
|
||||||
|
{include file="public/tail" /}
|
||||||
181
app/admin/view/yunzeradmin/frontuser.php
Normal file
181
app/admin/view/yunzeradmin/frontuser.php
Normal file
@ -0,0 +1,181 @@
|
|||||||
|
{include file="public/header" /}
|
||||||
|
<div class="config-container">
|
||||||
|
<!-- 页面头部样式 -->
|
||||||
|
<div class="config-header" style="display: flex;flex-direction: column;flex-wrap: wrap;align-items: flex-start;">
|
||||||
|
<div class="maintitle">
|
||||||
|
<i class="layui-icon layui-icon-user"></i>
|
||||||
|
<span>用户列表</span>
|
||||||
|
</div>
|
||||||
|
<div style="display: flex;align-items: flex-start;flex-direction: column;gap: 15px;margin-bottom: 10px;">
|
||||||
|
<div>
|
||||||
|
<button class="layui-btn layui-bg-blue" onclick="add()">
|
||||||
|
<i class="layui-icon layui-icon-add-1"></i>添加
|
||||||
|
</button>
|
||||||
|
<button type="button" class="layui-btn layui-btn-primary layui-border-blue" onclick="refresh()">
|
||||||
|
<i class="layui-icon layui-icon-refresh"></i>刷新
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<table id="userTable" lay-filter="userTable"></table>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<script type="text/javascript">
|
||||||
|
layui.use(['table', 'layer'], function () {
|
||||||
|
var table = layui.table;
|
||||||
|
var layer = layui.layer;
|
||||||
|
|
||||||
|
table.render({
|
||||||
|
elem: '#userTable',
|
||||||
|
url: '{$config['admin_route']}yunzeradmin/frontuser', // 数据接口
|
||||||
|
page: true,
|
||||||
|
cols: [[
|
||||||
|
{ field: 'uid', title: '用户ID', width: 80, align: 'center', fixed: 'left' },
|
||||||
|
{
|
||||||
|
field: 'avatar',
|
||||||
|
title: '头像',
|
||||||
|
width: 80,
|
||||||
|
align: 'center',
|
||||||
|
templet: function (d) {
|
||||||
|
if (d.avatar && d.avatar !== '') {
|
||||||
|
return '<img src="' + d.avatar + '" style="width:36px;height:36px;border-radius:50%;object-fit:cover;" />';
|
||||||
|
} else {
|
||||||
|
return '<span style="color:#aaa;">无</span>';
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{ field: 'name', title: '真实姓名', width: 120 },
|
||||||
|
{ field: 'name', title: '真实姓名', width: 120 },
|
||||||
|
{ field: 'account', title: '账户', width: 200 },
|
||||||
|
{ field: 'phone', title: '手机', width: 120 },
|
||||||
|
{ field: 'qq', title: 'QQ', width: 130 },
|
||||||
|
{
|
||||||
|
field: 'wechat',
|
||||||
|
title: '微信',
|
||||||
|
width: 100,
|
||||||
|
templet: function (d) {
|
||||||
|
if (d.openid && d.openid !== '') {
|
||||||
|
return '<span style="color: #52c41a;">已绑定</span>';
|
||||||
|
} else {
|
||||||
|
return '<span style="color: #aaa;">未绑定</span>';
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{ field: 'group_id', title: '角色', width: 200 },
|
||||||
|
{
|
||||||
|
field: 'sex', title: '性别', width: 80, templet: function (d) {
|
||||||
|
if (d.sex == 1) return '男';
|
||||||
|
if (d.sex == 2) return '女';
|
||||||
|
return '未选择';
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
field: 'status',
|
||||||
|
title: '状态',
|
||||||
|
width: 100,
|
||||||
|
templet: function (d) {
|
||||||
|
if (d.status == 1) {
|
||||||
|
return '<span style="color: #52c41a;">开启</span>';
|
||||||
|
} else {
|
||||||
|
return '<span style="color: #f5222d;">禁用</span>';
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{ field: 'login_count', title: '登陆次数', width: 100 },
|
||||||
|
{
|
||||||
|
field: 'update_time',
|
||||||
|
title: '登陆时间',
|
||||||
|
width: 180,
|
||||||
|
templet: function (d) {
|
||||||
|
if (!d.update_time) return '';
|
||||||
|
var t = d.update_time;
|
||||||
|
// 如果是时间戳(数字),转为int
|
||||||
|
if (/^\d+$/.test(t)) t = parseInt(t) * (t.toString().length == 10 ? 1000 : 1);
|
||||||
|
var date = new Date(t);
|
||||||
|
var Y = date.getFullYear();
|
||||||
|
var M = ('0' + (date.getMonth() + 1)).slice(-2);
|
||||||
|
var D = ('0' + date.getDate()).slice(-2);
|
||||||
|
var h = ('0' + date.getHours()).slice(-2);
|
||||||
|
var m = ('0' + date.getMinutes()).slice(-2);
|
||||||
|
var s = ('0' + date.getSeconds()).slice(-2);
|
||||||
|
return Y + '-' + M + '-' + D + ' ' + h + ':' + m + ':' + s;
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{ title: '操作', toolbar: '#operationBar', width: 160, fixed: 'right' }
|
||||||
|
]]
|
||||||
|
});
|
||||||
|
|
||||||
|
// 工具条事件
|
||||||
|
table.on('tool(userTable)', function (obj) {
|
||||||
|
var data = obj.data;
|
||||||
|
if (obj.event === 'edit') {
|
||||||
|
edit(data.uid);
|
||||||
|
} else if (obj.event === 'del') {
|
||||||
|
del(data.uid);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
// 添加
|
||||||
|
function add() {
|
||||||
|
layer.open({
|
||||||
|
type: 2,
|
||||||
|
title: '添加用户',
|
||||||
|
shade: 0.3,
|
||||||
|
area: ['450px', '550px'],
|
||||||
|
content: "{$config['admin_route']}yunzeradmin/frontuseradd"
|
||||||
|
});
|
||||||
|
}
|
||||||
|
// 编辑
|
||||||
|
function edit(uid) {
|
||||||
|
layer.open({
|
||||||
|
type: 2,
|
||||||
|
title: '编辑用户',
|
||||||
|
shade: 0.3,
|
||||||
|
area: ['450px', '550px'],
|
||||||
|
content: "{$config['admin_route']}yunzeradmin/frontuseredit?uid=" + uid
|
||||||
|
});
|
||||||
|
}
|
||||||
|
// 删除
|
||||||
|
function del(uid) {
|
||||||
|
layer.confirm('确定要删除吗?', {
|
||||||
|
icon: 3,
|
||||||
|
btn: ['确定', '取消']
|
||||||
|
}, function () {
|
||||||
|
$.post("{$config['admin_route']}yunzeradmin/frontuserdel", { 'uid': uid }, function (res) {
|
||||||
|
if (res.code > 0) {
|
||||||
|
layer.alert(res.msg, { icon: 2 });
|
||||||
|
} else {
|
||||||
|
layer.msg(res.msg);
|
||||||
|
setTimeout(function () { window.location.reload(); }, 1000);
|
||||||
|
}
|
||||||
|
}, 'json');
|
||||||
|
});
|
||||||
|
}
|
||||||
|
//刷新列表
|
||||||
|
function refresh() {
|
||||||
|
window.location.reload();
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
<script type="text/html" id="operationBar">
|
||||||
|
<button type="button" class="layui-btn layui-btn-primary layui-btn-xs" lay-event="edit">
|
||||||
|
<i class="layui-icon layui-icon-edit"></i>编辑
|
||||||
|
</button>
|
||||||
|
<button type="button" class="layui-btn layui-btn-primary layui-btn-xs" lay-event="del">
|
||||||
|
<i class="layui-icon layui-icon-delete"></i>删除
|
||||||
|
</button>
|
||||||
|
</script>
|
||||||
|
<style>
|
||||||
|
.layui-table-view .layui-table td,
|
||||||
|
.layui-table-view .layui-table th {
|
||||||
|
height: 60px !important;
|
||||||
|
line-height: 60px !important;
|
||||||
|
padding-top: 0;
|
||||||
|
padding-bottom: 0;
|
||||||
|
}
|
||||||
|
.layui-table-cell{
|
||||||
|
height: auto !important;
|
||||||
|
}
|
||||||
|
</style>
|
||||||
|
{include file="public/tail" /}
|
||||||
84
app/admin/view/yunzeradmin/frontuseradd.php
Normal file
84
app/admin/view/yunzeradmin/frontuseradd.php
Normal file
@ -0,0 +1,84 @@
|
|||||||
|
{include file="public/header" /}
|
||||||
|
<form class="layui-form">
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">账户</label>
|
||||||
|
<div class="layui-input-inline">
|
||||||
|
<input type="text" class="layui-input" name="account" placeholder="账户请用邮箱">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">姓名</label>
|
||||||
|
<div class="layui-input-inline">
|
||||||
|
<input type="text" class="layui-input" name="name" placeholder="请输入真实姓名">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">手机号</label>
|
||||||
|
<div class="layui-input-inline">
|
||||||
|
<input type="text" class="layui-input" name="phone" placeholder="请输入手机号">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">QQ号</label>
|
||||||
|
<div class="layui-input-inline">
|
||||||
|
<input type="text" class="layui-input" name="qq" placeholder="请输入QQ号">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">角色</label>
|
||||||
|
<div class="layui-input-inline">
|
||||||
|
<select name="group_id">
|
||||||
|
<option value=0></option>
|
||||||
|
{volist name="group" id="vo"}
|
||||||
|
<option value="{$vo.group_id}">{$vo.group_name}</option>
|
||||||
|
{/volist}
|
||||||
|
</select>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">密码</label>
|
||||||
|
<div class="layui-input-inline">
|
||||||
|
<input type="text" class="layui-input" name="password" placeholder="请输入密码">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">性别</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<input type="radio" name="sex" value="1" title="男" checked="">
|
||||||
|
<input type="radio" name="sex" value="2" title="女">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">状态</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<input type="radio" name="status" value="1" title="开启" checked="">
|
||||||
|
<input type="radio" name="status" value="0" title="禁用">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</form>
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<button type="button" class="layui-btn" onclick="save()">保存</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<script type="text/javascript">
|
||||||
|
layui.use(['layer','form'],function(){
|
||||||
|
form = layui.form;
|
||||||
|
layer = layui.layer;
|
||||||
|
$ = layui.jquery;
|
||||||
|
});
|
||||||
|
// 保存管理员
|
||||||
|
function save(){
|
||||||
|
$.post("{$config['admin_route']}yunzeradmin/useradd",$('form').serialize(),function(res){
|
||||||
|
if(res.code>0){
|
||||||
|
layer.alert(res.msg,{icon:2});
|
||||||
|
}else{
|
||||||
|
layer.msg(res.msg);
|
||||||
|
setTimeout(function(){parent.window.location.reload();},1000);
|
||||||
|
}
|
||||||
|
},'json');
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
{include file="public/tail" /}
|
||||||
76
app/admin/view/yunzeradmin/frontuseredit.php
Normal file
76
app/admin/view/yunzeradmin/frontuseredit.php
Normal file
@ -0,0 +1,76 @@
|
|||||||
|
{include file="public/header" /}
|
||||||
|
<form class="layui-form">
|
||||||
|
<input type="hidden" name="uid" value="{$lists.uid}">
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">账户</label>
|
||||||
|
<div class="layui-input-inline"><input type="text" class="layui-input" value="{$lists.account}" readonly></div>
|
||||||
|
</div>
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">姓 名</label>
|
||||||
|
<div class="layui-input-inline">
|
||||||
|
<input type="text" class="layui-input" name="name" placeholder="请输入真实姓名" value="{$lists.name}">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">手机号</label>
|
||||||
|
<div class="layui-input-inline">
|
||||||
|
<input type="text" class="layui-input" name="phone" placeholder="请输入手机号" value="{$lists.phone}">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">QQ号</label>
|
||||||
|
<div class="layui-input-inline">
|
||||||
|
<input type="text" class="layui-input" name="qq" placeholder="请输入QQ号" value="{$lists.qq}">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">角色</label>
|
||||||
|
<div class="layui-input-inline">
|
||||||
|
<select name="group_id">
|
||||||
|
<option value=0></option>
|
||||||
|
{volist name="group" id="vo"}
|
||||||
|
<option value="{$vo.group_id}" {$vo.group_id == $lists.group_id ? 'selected' : ''}>{$vo.group_name}</option>
|
||||||
|
{/volist}
|
||||||
|
</select>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">性别</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<input type="radio" name="sex" value="1" title="男" {$lists.sex==1?'checked':''}>
|
||||||
|
<input type="radio" name="sex" value="2" title="女" {$lists.sex==2?'checked':''}>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">状态</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<input type="radio" name="status" value="1" title="开启" {$lists.status==1?'checked':''}>
|
||||||
|
<input type="radio" name="status" value="0" title="禁用" {$lists.status==0?'checked':''}>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</form>
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<button class="layui-btn" onclick="save()">保存</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<script type="text/javascript">
|
||||||
|
layui.use(['layer','form'],function(){
|
||||||
|
form = layui.form;
|
||||||
|
layer = layui.layer;
|
||||||
|
$ = layui.jquery;
|
||||||
|
});
|
||||||
|
// 保存管理员
|
||||||
|
function save(){
|
||||||
|
$.post("{$config['admin_route']}yunzeradmin/frontuseredit",$('form').serialize(),function(res){
|
||||||
|
if(res.code>0){
|
||||||
|
layer.alert(res.msg,{icon:2});
|
||||||
|
}else{
|
||||||
|
layer.msg(res.msg);
|
||||||
|
setTimeout(function(){parent.window.location.reload();},1000);
|
||||||
|
}
|
||||||
|
},'json');
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
{include file="public/tail" /}
|
||||||
53
app/admin/view/yunzeradmin/groupadd.php
Normal file
53
app/admin/view/yunzeradmin/groupadd.php
Normal file
@ -0,0 +1,53 @@
|
|||||||
|
{include file="public/header" /}
|
||||||
|
<form class="layui-form">
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">角色名称</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<input type="text" class="layui-input" name="group_name" placeholder="请输入角色名称">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">状态</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<input type="radio" name="status" value="1" title="开启" checked="">
|
||||||
|
<input type="radio" name="status" value="0" title="禁用">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layui-form-itme">
|
||||||
|
<label class="layui-form-label">权限菜单</label>
|
||||||
|
{volist name="menus" id="vo"}
|
||||||
|
<hr>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<input type="checkbox" name="menu[{$vo.smid}]" lay-skin="primary" title="{$vo.label}">
|
||||||
|
<hr>
|
||||||
|
{volist name="vo.children" id="cvo"}
|
||||||
|
<input type="checkbox" name="menu[{$cvo.smid}]" lay-skin="primary" title="{$cvo.label}">
|
||||||
|
{/volist}
|
||||||
|
</div>
|
||||||
|
{/volist}
|
||||||
|
</div>
|
||||||
|
</form>
|
||||||
|
<div class="layui-form-item" style="margin-top:10px;">
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<button type="button" class="layui-btn" onclick="save()">保存</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<script type="text/javascript">
|
||||||
|
layui.use(['layer','form'],function(){
|
||||||
|
var form = layui.form;
|
||||||
|
layer = layui.layer;
|
||||||
|
$ = layui.jquery;
|
||||||
|
});
|
||||||
|
|
||||||
|
function save(){
|
||||||
|
$.post("{$config['admin_route']}yunzeradmin/groupadd",$('form').serialize(),function(res){
|
||||||
|
if(res.code>0){
|
||||||
|
layer.msg(res.msg,{'icon':2});
|
||||||
|
}else{
|
||||||
|
layer.msg(res.msg,{'icon':1});
|
||||||
|
setTimeout(function(){parent.window.location.reload();},1000);
|
||||||
|
}
|
||||||
|
},'json');
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
{include file="public/tail" /}
|
||||||
54
app/admin/view/yunzeradmin/groupedit.php
Normal file
54
app/admin/view/yunzeradmin/groupedit.php
Normal file
@ -0,0 +1,54 @@
|
|||||||
|
{include file="public/header" /}
|
||||||
|
<form class="layui-form">
|
||||||
|
<input type="hidden" name="group_id" value="{$group.group_id}">
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">角色名称</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<input type="text" class="layui-input" name="group_name" placeholder="请输入角色名称" value="{$group.group_name}">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">状态</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<input type="radio" name="status" value="1" title="开启" {$group.status==1?'checked':''}>
|
||||||
|
<input type="radio" name="status" value="0" title="禁用" {$group.status==0?'checked':''}>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layui-form-itme">
|
||||||
|
<label class="layui-form-label">权限菜单</label>
|
||||||
|
{volist name="menus" id="vo"}
|
||||||
|
<hr>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<input type="checkbox" name="menu[{$vo.smid}]" lay-skin="primary" title="{$vo.label}" {:isset($group['rights']) && $group['rights'] && in_array($vo.smid,$group['rights'])?'checked':''}>
|
||||||
|
<hr>
|
||||||
|
{volist name="vo.children" id="cvo"}
|
||||||
|
<input type="checkbox" name="menu[{$cvo.smid}]" lay-skin="primary" title="{$cvo.label}" {:isset($group['rights']) && $group['rights'] && in_array($cvo.smid,$group['rights'])?'checked':''}>
|
||||||
|
{/volist}
|
||||||
|
</div>
|
||||||
|
{/volist}
|
||||||
|
</div>
|
||||||
|
</form>
|
||||||
|
<div class="layui-form-item" style="margin-top:10px;">
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<button type="button" class="layui-btn" onclick="save()">保存</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<script type="text/javascript">
|
||||||
|
layui.use(['layer','form'],function(){
|
||||||
|
var form = layui.form;
|
||||||
|
layer = layui.layer;
|
||||||
|
$ = layui.jquery;
|
||||||
|
});
|
||||||
|
|
||||||
|
function save(){
|
||||||
|
$.post("{$config['admin_route']}yunzeradmin/groupedit",$('form').serialize(),function(res){
|
||||||
|
if(res.code>0){
|
||||||
|
layer.msg(res.msg,{'icon':2});
|
||||||
|
}else{
|
||||||
|
layer.msg(res.msg,{'icon':1});
|
||||||
|
setTimeout(function(){parent.window.location.reload();},1000);
|
||||||
|
}
|
||||||
|
},'json');
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
{include file="public/tail" /}
|
||||||
100
app/admin/view/yunzeradmin/groupinfo.php
Normal file
100
app/admin/view/yunzeradmin/groupinfo.php
Normal file
@ -0,0 +1,100 @@
|
|||||||
|
{include file="public/header" /}
|
||||||
|
<div class="config-container">
|
||||||
|
<!-- 页面头部样式 -->
|
||||||
|
<div class="config-header" style="display: flex;flex-direction: column;flex-wrap: wrap;align-items: flex-start;">
|
||||||
|
<div class="maintitle">
|
||||||
|
<i class="layui-icon layui-icon-group"></i>
|
||||||
|
<span>角色列表</span>
|
||||||
|
</div>
|
||||||
|
<div style="display: flex;align-items: flex-start;flex-direction: column;gap: 15px;margin-bottom: 10px;">
|
||||||
|
<div>
|
||||||
|
<button class="layui-btn layui-bg-blue" onclick="add()">
|
||||||
|
<i class="layui-icon layui-icon-add-1"></i>添加
|
||||||
|
</button>
|
||||||
|
<button type="button" class="layui-btn layui-btn-primary layui-border-blue" onclick="refresh()">
|
||||||
|
<i class="layui-icon layui-icon-refresh"></i>刷新
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<table class="layui-table">
|
||||||
|
<thead>
|
||||||
|
<tr>
|
||||||
|
<th>角色名</th>
|
||||||
|
<th>状态</th>
|
||||||
|
<th>添加时间</th>
|
||||||
|
<th>操作</th>
|
||||||
|
</tr>
|
||||||
|
</thead>
|
||||||
|
<tbody>
|
||||||
|
{volist name="group" id='vo'}
|
||||||
|
<tr>
|
||||||
|
<td>{$vo.group_name}</td>
|
||||||
|
<td>{$vo.status==1?'开启':'<span style="color:red;">禁用</span>'}</td>
|
||||||
|
<td>{:date('Y-m-d H:i:s',$vo.create_time)}</td>
|
||||||
|
<td>
|
||||||
|
<button type="button" class="layui-btn layui-btn-primary layui-btn-xs"
|
||||||
|
onclick="edit({$vo.group_id})">
|
||||||
|
<i class="layui-icon layui-icon-edit"></i>编辑
|
||||||
|
</button>
|
||||||
|
<button type="button" class="layui-btn layui-btn-primary layui-btn-xs"
|
||||||
|
onclick="del({$vo.group_id})">
|
||||||
|
<i class="layui-icon layui-icon-delete"></i>删除
|
||||||
|
</button>
|
||||||
|
</td>
|
||||||
|
</tr>
|
||||||
|
{/volist}
|
||||||
|
</tbody>
|
||||||
|
</table>
|
||||||
|
|
||||||
|
</div>
|
||||||
|
|
||||||
|
|
||||||
|
<script type="text/javascript">
|
||||||
|
layui.use(['layer'], function () {
|
||||||
|
layer = layui.layer;
|
||||||
|
$ = layui.jquery;
|
||||||
|
});
|
||||||
|
// 添加
|
||||||
|
function add() {
|
||||||
|
layer.open({
|
||||||
|
type: 2,
|
||||||
|
title: '添加角色',
|
||||||
|
shade: 0.3,
|
||||||
|
area: ['450px', '550px'],
|
||||||
|
content: "{$config['admin_route']}yunzeradmin/groupadd"
|
||||||
|
});
|
||||||
|
}
|
||||||
|
// 编辑
|
||||||
|
function edit(group_id) {
|
||||||
|
layer.open({
|
||||||
|
type: 2,
|
||||||
|
title: '编辑角色',
|
||||||
|
shade: 0.3,
|
||||||
|
area: ['450px', '550px'],
|
||||||
|
content: "{$config['admin_route']}yunzeradmin/groupedit?group_id=" + group_id
|
||||||
|
});
|
||||||
|
}
|
||||||
|
// 删除
|
||||||
|
function del(group_id) {
|
||||||
|
layer.confirm('确定要删除吗?', {
|
||||||
|
icon: 3,
|
||||||
|
btn: ['确定', '取消']
|
||||||
|
}, function () {
|
||||||
|
$.post("{$config['admin_route']}yunzeradmin/groupdel", { 'group_id': group_id }, function (res) {
|
||||||
|
if (res.code > 0) {
|
||||||
|
layer.alert(res.msg, { icon: 2 });
|
||||||
|
} else {
|
||||||
|
layer.msg(res.msg, { icon: 1 });
|
||||||
|
setTimeout(function () { window.location.reload(); }, 1000);
|
||||||
|
}
|
||||||
|
}, 'json');
|
||||||
|
});
|
||||||
|
}
|
||||||
|
//刷新列表
|
||||||
|
function refresh() {
|
||||||
|
window.location.reload();
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
{include file="public/tail" /}
|
||||||
363
app/admin/view/yunzeradmin/materialcenter.php
Normal file
363
app/admin/view/yunzeradmin/materialcenter.php
Normal file
@ -0,0 +1,363 @@
|
|||||||
|
{include file="public/header" /}
|
||||||
|
<div class="material-tabs">
|
||||||
|
<div class="tabs-header">
|
||||||
|
<div class="tab active" data-type="image">图片</div>
|
||||||
|
<div class="tab" data-type="video">视频</div>
|
||||||
|
<div class="tab" data-type="file">文件</div>
|
||||||
|
</div>
|
||||||
|
<div class="tabs-content">
|
||||||
|
<!-- 图片Tab内容 -->
|
||||||
|
<div class="tab-panel" id="tab-image" style="display:block;">
|
||||||
|
<div class="material-center">
|
||||||
|
<!-- 左侧分组树 -->
|
||||||
|
<div class="sidebar">
|
||||||
|
<ul class="group-list" id="groupList">
|
||||||
|
<li class="active">全部</li>
|
||||||
|
<!-- ... -->
|
||||||
|
</ul>
|
||||||
|
<div class="sidebar-header">
|
||||||
|
<button class="layui-btn layui-btn-normal" onclick="addGroup()">添加分组</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<!-- 右侧内容 -->
|
||||||
|
<div class="main-content">
|
||||||
|
<div class="toolbar">
|
||||||
|
<div class="material-actions">
|
||||||
|
<button class="layui-btn layui-btn-normal" id="uploadBtn">上传</button>
|
||||||
|
<button class="layui-btn layui-btn-danger" id="batchDeleteBtn">批量删除</button>
|
||||||
|
</div>
|
||||||
|
<button class="layui-btn" onclick="moveMaterial()">移动</button>
|
||||||
|
<label class="select-all">
|
||||||
|
<input type="checkbox" id="selectAll" /> 全选
|
||||||
|
</label>
|
||||||
|
</div>
|
||||||
|
<div class="image-grid" id="imageGrid">
|
||||||
|
<div class="image-item">
|
||||||
|
<input type="checkbox" class="img-checkbox" />
|
||||||
|
<img src="https://placekitten.com/120/120" alt="示例图片" />
|
||||||
|
<div class="img-name">示例图片.jpg</div>
|
||||||
|
<div class="img-actions">
|
||||||
|
<button class="layui-btn layui-btn-xs" onclick="renameImg(this)">重命名</button>
|
||||||
|
<button class="layui-btn layui-btn-xs layui-btn-primary" onclick="viewImg(this)">查看</button>
|
||||||
|
<button class="layui-btn layui-btn-xs layui-btn-warm" onclick="copyUrl(this)">地址</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<!-- ... -->
|
||||||
|
</div>
|
||||||
|
<div class="pagination">
|
||||||
|
<button class="layui-btn layui-btn-xs">上一页</button>
|
||||||
|
<span>1 / 10</span>
|
||||||
|
<button class="layui-btn layui-btn-xs">下一页</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<!-- 视频Tab内容 -->
|
||||||
|
<div class="tab-panel" id="tab-video" style="display:none;">
|
||||||
|
<div class="material-center">
|
||||||
|
<div style="padding:40px;text-align:center;color:#bbb;">这里是视频管理内容</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<!-- 文件Tab内容 -->
|
||||||
|
<div class="tab-panel" id="tab-file" style="display:none;">
|
||||||
|
<div class="material-center">
|
||||||
|
<div style="padding:40px;text-align:center;color:#bbb;">这里是文件管理内容</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
// Tab切换
|
||||||
|
document.querySelectorAll('.material-tabs .tab').forEach(function (tab) {
|
||||||
|
tab.onclick = function () {
|
||||||
|
document.querySelectorAll('.material-tabs .tab').forEach(t => t.classList.remove('active'));
|
||||||
|
tab.classList.add('active');
|
||||||
|
var type = tab.getAttribute('data-type');
|
||||||
|
document.querySelectorAll('.tab-panel').forEach(panel => panel.style.display = 'none');
|
||||||
|
document.getElementById('tab-' + type).style.display = 'block';
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
// 分组高亮切换
|
||||||
|
document.getElementById('groupList').addEventListener('click', function (e) {
|
||||||
|
if (e.target.tagName === 'LI' || e.target.closest('li')) {
|
||||||
|
document.querySelectorAll('#groupList li').forEach(li => li.classList.remove('active'));
|
||||||
|
(e.target.tagName === 'LI' ? e.target : e.target.closest('li')).classList.add('active');
|
||||||
|
// TODO: 根据分组加载图片
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
// 其它功能函数(示例)
|
||||||
|
function addGroup() {
|
||||||
|
alert('添加分组功能待实现');
|
||||||
|
}
|
||||||
|
function uploadMaterial() {
|
||||||
|
alert('上传功能待实现');
|
||||||
|
}
|
||||||
|
function batchDelete() {
|
||||||
|
alert('批量删除功能待实现');
|
||||||
|
}
|
||||||
|
function moveMaterial() {
|
||||||
|
alert('移动功能待实现');
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
|
||||||
|
<style>
|
||||||
|
.material-tabs {
|
||||||
|
width: 100%;
|
||||||
|
height: 100vh;
|
||||||
|
background: #f6f7fb;
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
}
|
||||||
|
|
||||||
|
.tabs-header {
|
||||||
|
display: flex;
|
||||||
|
border-bottom: 2px solid #e6e6e6;
|
||||||
|
background: #fff;
|
||||||
|
padding-left: 32px;
|
||||||
|
height: 52px;
|
||||||
|
align-items: center;
|
||||||
|
}
|
||||||
|
|
||||||
|
.tab {
|
||||||
|
padding: 0 32px;
|
||||||
|
height: 52px;
|
||||||
|
line-height: 52px;
|
||||||
|
font-size: 16px;
|
||||||
|
color: #666;
|
||||||
|
cursor: pointer;
|
||||||
|
position: relative;
|
||||||
|
transition: color 0.2s;
|
||||||
|
}
|
||||||
|
|
||||||
|
.tab.active {
|
||||||
|
color: #409eff;
|
||||||
|
font-weight: bold;
|
||||||
|
}
|
||||||
|
|
||||||
|
.tab.active::after {
|
||||||
|
content: '';
|
||||||
|
position: absolute;
|
||||||
|
left: 16px;
|
||||||
|
right: 16px;
|
||||||
|
bottom: 0;
|
||||||
|
height: 3px;
|
||||||
|
background: #409eff;
|
||||||
|
border-radius: 2px 2px 0 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.tabs-content {
|
||||||
|
flex: 1;
|
||||||
|
width: 100%;
|
||||||
|
overflow: auto;
|
||||||
|
background: #f6f7fb;
|
||||||
|
}
|
||||||
|
|
||||||
|
.tab-panel {
|
||||||
|
width: 100%;
|
||||||
|
/* height: 100%; */
|
||||||
|
display: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
.tab-panel:target,
|
||||||
|
.tab-panel.active {
|
||||||
|
display: block;
|
||||||
|
}
|
||||||
|
|
||||||
|
.material-center {
|
||||||
|
display: flex;
|
||||||
|
height: 100%;
|
||||||
|
background: #fafbfc;
|
||||||
|
font-family: "Microsoft YaHei", Arial, sans-serif;
|
||||||
|
}
|
||||||
|
|
||||||
|
.sidebar {
|
||||||
|
width: 220px;
|
||||||
|
background: #fff;
|
||||||
|
border-right: 1px solid #e6e6e6;
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
padding: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.sidebar-header {
|
||||||
|
display: flex;
|
||||||
|
font-size: 16px;
|
||||||
|
font-weight: bold;
|
||||||
|
color: #333;
|
||||||
|
border-top: 1px solid #f0f0f0;
|
||||||
|
justify-content: center;
|
||||||
|
padding: 20px 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.add-group-btn {
|
||||||
|
background: #409eff;
|
||||||
|
color: #fff;
|
||||||
|
border: none;
|
||||||
|
border-radius: 50%;
|
||||||
|
width: 26px;
|
||||||
|
height: 26px;
|
||||||
|
font-size: 18px;
|
||||||
|
cursor: pointer;
|
||||||
|
transition: background 0.2s;
|
||||||
|
}
|
||||||
|
|
||||||
|
.add-group-btn:hover {
|
||||||
|
background: #66b1ff;
|
||||||
|
}
|
||||||
|
|
||||||
|
.group-list {
|
||||||
|
list-style: none;
|
||||||
|
margin: 0;
|
||||||
|
padding: 10px 0 0 0;
|
||||||
|
flex: 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
.group-list li {
|
||||||
|
padding: 10px 24px;
|
||||||
|
color: #444;
|
||||||
|
cursor: pointer;
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
border-radius: 4px 0 0 4px;
|
||||||
|
transition: background 0.2s, color 0.2s;
|
||||||
|
font-size: 15px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.group-list li .iconfont {
|
||||||
|
font-size: 16px;
|
||||||
|
margin-right: 8px;
|
||||||
|
color: #b2b2b2;
|
||||||
|
}
|
||||||
|
|
||||||
|
.group-list li.active,
|
||||||
|
.group-list li:hover {
|
||||||
|
background: #f2f6fc;
|
||||||
|
color: #409eff;
|
||||||
|
}
|
||||||
|
|
||||||
|
.main-content {
|
||||||
|
flex: 1;
|
||||||
|
padding: 32px 36px;
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
min-width: 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.toolbar {
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
gap: 12px;
|
||||||
|
margin-bottom: 18px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.select-all {
|
||||||
|
margin-left: auto;
|
||||||
|
font-size: 14px;
|
||||||
|
color: #888;
|
||||||
|
user-select: none;
|
||||||
|
}
|
||||||
|
|
||||||
|
.image-grid {
|
||||||
|
display: flex;
|
||||||
|
flex-wrap: wrap;
|
||||||
|
gap: 22px;
|
||||||
|
min-height: 320px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.image-item {
|
||||||
|
width: 140px;
|
||||||
|
background: #fff;
|
||||||
|
border: 1px solid #eaeaea;
|
||||||
|
border-radius: 8px;
|
||||||
|
box-shadow: 0 2px 8px rgba(0, 0, 0, 0.04);
|
||||||
|
display: flex;
|
||||||
|
flex-direction: column;
|
||||||
|
align-items: center;
|
||||||
|
padding: 14px 8px 10px 8px;
|
||||||
|
position: relative;
|
||||||
|
cursor: pointer;
|
||||||
|
transition: box-shadow 0.2s, border 0.2s;
|
||||||
|
}
|
||||||
|
|
||||||
|
.image-item:hover {
|
||||||
|
border: 1px solid #409eff;
|
||||||
|
box-shadow: 0 4px 16px rgba(64, 158, 255, 0.10);
|
||||||
|
}
|
||||||
|
|
||||||
|
.image-item img {
|
||||||
|
width: 120px;
|
||||||
|
height: 120px;
|
||||||
|
object-fit: cover;
|
||||||
|
border-radius: 4px;
|
||||||
|
margin-bottom: 8px;
|
||||||
|
background: #f5f7fa;
|
||||||
|
border: 1px solid #f0f0f0;
|
||||||
|
}
|
||||||
|
|
||||||
|
.img-name {
|
||||||
|
font-size: 14px;
|
||||||
|
color: #666;
|
||||||
|
text-align: center;
|
||||||
|
margin-bottom: 6px;
|
||||||
|
word-break: break-all;
|
||||||
|
height: 32px;
|
||||||
|
line-height: 16px;
|
||||||
|
overflow: hidden;
|
||||||
|
}
|
||||||
|
|
||||||
|
.img-checkbox {
|
||||||
|
position: absolute;
|
||||||
|
top: 12px;
|
||||||
|
left: 12px;
|
||||||
|
transform: scale(1.2);
|
||||||
|
}
|
||||||
|
|
||||||
|
.img-actions {
|
||||||
|
display: none;
|
||||||
|
margin-top: 6px;
|
||||||
|
text-align: center;
|
||||||
|
gap: 4px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.image-item.active .img-actions {
|
||||||
|
display: flex;
|
||||||
|
justify-content: center;
|
||||||
|
}
|
||||||
|
|
||||||
|
.img-actions .layui-btn-xs {
|
||||||
|
margin: 0 2px;
|
||||||
|
font-size: 12px;
|
||||||
|
padding: 0 10px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.pagination {
|
||||||
|
margin-top: 28px;
|
||||||
|
text-align: center;
|
||||||
|
font-size: 15px;
|
||||||
|
color: #888;
|
||||||
|
display: flex;
|
||||||
|
align-items: center;
|
||||||
|
justify-content: center;
|
||||||
|
gap: 12px;
|
||||||
|
}
|
||||||
|
|
||||||
|
@media (max-width: 900px) {
|
||||||
|
.main-content {
|
||||||
|
padding: 12px 4px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.image-item {
|
||||||
|
width: 100px;
|
||||||
|
padding: 8px 4px 8px 4px;
|
||||||
|
}
|
||||||
|
|
||||||
|
.image-item img {
|
||||||
|
width: 80px;
|
||||||
|
height: 80px;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
</style>
|
||||||
|
{include file="public/tail" /}
|
||||||
0
app/admin/view/yunzeradmin/materialcenteradd.php
Normal file
0
app/admin/view/yunzeradmin/materialcenteradd.php
Normal file
0
app/admin/view/yunzeradmin/materialcenteredit.php
Normal file
0
app/admin/view/yunzeradmin/materialcenteredit.php
Normal file
29
app/admin/view/yunzeradmin/selectpushcontent.php
Normal file
29
app/admin/view/yunzeradmin/selectpushcontent.php
Normal file
@ -0,0 +1,29 @@
|
|||||||
|
{include file="public/header" /}
|
||||||
|
<div style="padding: 20px;">
|
||||||
|
<table class="layui-table">
|
||||||
|
<thead>
|
||||||
|
<tr>
|
||||||
|
<th>标题</th>
|
||||||
|
</tr>
|
||||||
|
</thead>
|
||||||
|
<tbody>
|
||||||
|
<?php foreach ($data as $item): ?>
|
||||||
|
<tr>
|
||||||
|
<td>
|
||||||
|
<a href="javascript:;" onclick="selectThis('<?php echo addslashes($item['title']); ?>', '<?php echo isset($item['url']) ? addslashes($item['url']) : ''; ?>')">
|
||||||
|
<?php echo htmlspecialchars($item['title']); ?>
|
||||||
|
</a>
|
||||||
|
</td>
|
||||||
|
</tr>
|
||||||
|
<?php endforeach; ?>
|
||||||
|
</tbody>
|
||||||
|
</table>
|
||||||
|
</div>
|
||||||
|
<script>
|
||||||
|
function selectThis(title, url) {
|
||||||
|
if (window.parent && window.parent.setPushContent) {
|
||||||
|
window.parent.setPushContent(title, url || '');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
{include file="public/tail" /}
|
||||||
84
app/admin/view/yunzeradmin/useradd.php
Normal file
84
app/admin/view/yunzeradmin/useradd.php
Normal file
@ -0,0 +1,84 @@
|
|||||||
|
{include file="public/header" /}
|
||||||
|
<form class="layui-form">
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">账户</label>
|
||||||
|
<div class="layui-input-inline">
|
||||||
|
<input type="text" class="layui-input" name="account" placeholder="账户请用邮箱">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">姓名</label>
|
||||||
|
<div class="layui-input-inline">
|
||||||
|
<input type="text" class="layui-input" name="name" placeholder="请输入真实姓名">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">手机号</label>
|
||||||
|
<div class="layui-input-inline">
|
||||||
|
<input type="text" class="layui-input" name="phone" placeholder="请输入手机号">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">QQ号</label>
|
||||||
|
<div class="layui-input-inline">
|
||||||
|
<input type="text" class="layui-input" name="qq" placeholder="请输入QQ号">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">角色</label>
|
||||||
|
<div class="layui-input-inline">
|
||||||
|
<select name="group_id">
|
||||||
|
<option value=0></option>
|
||||||
|
{volist name="group" id="vo"}
|
||||||
|
<option value="{$vo.group_id}">{$vo.group_name}</option>
|
||||||
|
{/volist}
|
||||||
|
</select>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">密码</label>
|
||||||
|
<div class="layui-input-inline">
|
||||||
|
<input type="text" class="layui-input" name="password" placeholder="请输入密码">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">性别</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<input type="radio" name="sex" value="1" title="男" checked="">
|
||||||
|
<input type="radio" name="sex" value="2" title="女">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">状态</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<input type="radio" name="status" value="1" title="开启" checked="">
|
||||||
|
<input type="radio" name="status" value="0" title="禁用">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</form>
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<button type="button" class="layui-btn" onclick="save()">保存</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<script type="text/javascript">
|
||||||
|
layui.use(['layer','form'],function(){
|
||||||
|
form = layui.form;
|
||||||
|
layer = layui.layer;
|
||||||
|
$ = layui.jquery;
|
||||||
|
});
|
||||||
|
// 保存管理员
|
||||||
|
function save(){
|
||||||
|
$.post("{$config['admin_route']}yunzeradmin/useradd",$('form').serialize(),function(res){
|
||||||
|
if(res.code>0){
|
||||||
|
layer.alert(res.msg,{icon:2});
|
||||||
|
}else{
|
||||||
|
layer.msg(res.msg);
|
||||||
|
setTimeout(function(){parent.window.location.reload();},1000);
|
||||||
|
}
|
||||||
|
},'json');
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
{include file="public/tail" /}
|
||||||
76
app/admin/view/yunzeradmin/useredit.php
Normal file
76
app/admin/view/yunzeradmin/useredit.php
Normal file
@ -0,0 +1,76 @@
|
|||||||
|
{include file="public/header" /}
|
||||||
|
<form class="layui-form">
|
||||||
|
<input type="hidden" name="uid" value="{$lists.uid}">
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">账户</label>
|
||||||
|
<div class="layui-input-inline"><input type="text" class="layui-input" value="{$lists.account}" readonly></div>
|
||||||
|
</div>
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">姓 名</label>
|
||||||
|
<div class="layui-input-inline">
|
||||||
|
<input type="text" class="layui-input" name="name" placeholder="请输入真实姓名" value="{$lists.name}">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">手机号</label>
|
||||||
|
<div class="layui-input-inline">
|
||||||
|
<input type="text" class="layui-input" name="phone" placeholder="请输入手机号" value="{$lists.phone}">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">QQ号</label>
|
||||||
|
<div class="layui-input-inline">
|
||||||
|
<input type="text" class="layui-input" name="qq" placeholder="请输入QQ号" value="{$lists.qq}">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">角色</label>
|
||||||
|
<div class="layui-input-inline">
|
||||||
|
<select name="group_id">
|
||||||
|
<option value=0></option>
|
||||||
|
{volist name="group" id="vo"}
|
||||||
|
<option value="{$vo.group_id}" {$vo.group_id == $lists.group_id ? 'selected' : ''}>{$vo.group_name}</option>
|
||||||
|
{/volist}
|
||||||
|
</select>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">性别</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<input type="radio" name="sex" value="1" title="男" {$lists.sex==1?'checked':''}>
|
||||||
|
<input type="radio" name="sex" value="2" title="女" {$lists.sex==2?'checked':''}>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">状态</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<input type="radio" name="status" value="1" title="开启" {$lists.status==1?'checked':''}>
|
||||||
|
<input type="radio" name="status" value="0" title="禁用" {$lists.status==0?'checked':''}>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</form>
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<button class="layui-btn" onclick="save()">保存</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<script type="text/javascript">
|
||||||
|
layui.use(['layer','form'],function(){
|
||||||
|
form = layui.form;
|
||||||
|
layer = layui.layer;
|
||||||
|
$ = layui.jquery;
|
||||||
|
});
|
||||||
|
// 保存管理员
|
||||||
|
function save(){
|
||||||
|
$.post("{$config['admin_route']}yunzeradmin/useredit",$('form').serialize(),function(res){
|
||||||
|
if(res.code>0){
|
||||||
|
layer.alert(res.msg,{icon:2});
|
||||||
|
}else{
|
||||||
|
layer.msg(res.msg);
|
||||||
|
setTimeout(function(){parent.window.location.reload();},1000);
|
||||||
|
}
|
||||||
|
},'json');
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
{include file="public/tail" /}
|
||||||
116
app/admin/view/yunzeradmin/userinfo.php
Normal file
116
app/admin/view/yunzeradmin/userinfo.php
Normal file
@ -0,0 +1,116 @@
|
|||||||
|
{include file="public/header" /}
|
||||||
|
<div class="config-container">
|
||||||
|
<!-- 页面头部样式 -->
|
||||||
|
<div class="config-header" style="display: flex;flex-direction: column;flex-wrap: wrap;align-items: flex-start;">
|
||||||
|
<div class="maintitle">
|
||||||
|
<i class="layui-icon layui-icon-user"></i>
|
||||||
|
<span>用户列表</span>
|
||||||
|
</div>
|
||||||
|
<div style="display: flex;align-items: flex-start;flex-direction: column;gap: 15px;margin-bottom: 10px;">
|
||||||
|
<div>
|
||||||
|
<button class="layui-btn layui-bg-blue" onclick="add()">
|
||||||
|
<i class="layui-icon layui-icon-add-1"></i>添加
|
||||||
|
</button>
|
||||||
|
<button type="button" class="layui-btn layui-btn-primary layui-border-blue" onclick="refresh()">
|
||||||
|
<i class="layui-icon layui-icon-refresh"></i>刷新
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<table class="layui-table">
|
||||||
|
<thead>
|
||||||
|
<tr>
|
||||||
|
<th>真实姓名</th>
|
||||||
|
<th>账户</th>
|
||||||
|
<th>手机</th>
|
||||||
|
<th>QQ</th>
|
||||||
|
<th>角色</th>
|
||||||
|
<th>性别</th>
|
||||||
|
<th>状态</th>
|
||||||
|
<th>登陆次数</th>
|
||||||
|
<th>登陆时间</th>
|
||||||
|
<th>操作</th>
|
||||||
|
</tr>
|
||||||
|
</thead>
|
||||||
|
<tbody>
|
||||||
|
{volist name="lists" id='vo'}
|
||||||
|
<tr>
|
||||||
|
<td>{$vo.name}</td>
|
||||||
|
<td>{$vo.account}</td>
|
||||||
|
<td>{$vo.phone}</td>
|
||||||
|
<td>{$vo.qq}</td>
|
||||||
|
<td>{:isset($group[$vo.group_id])?$group[$vo.group_id]['group_name']:''}</td>
|
||||||
|
<td>
|
||||||
|
{if $vo['sex']==1}
|
||||||
|
<span style="color:red;">男</span>
|
||||||
|
{elseif $vo['sex'] == 2 /}
|
||||||
|
<span style="color:green;">女</span>
|
||||||
|
{else /}
|
||||||
|
未知
|
||||||
|
{/if}
|
||||||
|
</td>
|
||||||
|
<td>{$vo.status==1?'开启':'<span style="color:red;">禁用</span>'}</td>
|
||||||
|
<td>{$vo.login_count}</td>
|
||||||
|
<td>{:date('Y-m-d H:i:s',$vo.update_time)}</td>
|
||||||
|
<td>
|
||||||
|
<button type="button" class="layui-btn layui-btn-primary layui-btn-xs" onclick="edit({$vo.uid})">
|
||||||
|
<i class="layui-icon layui-icon-edit"></i>编辑
|
||||||
|
</button>
|
||||||
|
<button type="button" class="layui-btn layui-btn-primary layui-btn-xs" onclick="del({$vo.uid})">
|
||||||
|
<i class="layui-icon layui-icon-delete"></i>删除
|
||||||
|
</button>
|
||||||
|
</td>
|
||||||
|
</tr>
|
||||||
|
{/volist}
|
||||||
|
</tbody>
|
||||||
|
</table>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<script type="text/javascript">
|
||||||
|
layui.use(['layer'], function () {
|
||||||
|
layer = layui.layer;
|
||||||
|
$ = layui.jquery;
|
||||||
|
});
|
||||||
|
// 添加
|
||||||
|
function add() {
|
||||||
|
layer.open({
|
||||||
|
type: 2,
|
||||||
|
title: '添加管理员',
|
||||||
|
shade: 0.3,
|
||||||
|
area: ['450px', '550px'],
|
||||||
|
content: "{$config['admin_route']}yunzeradmin/useradd"
|
||||||
|
});
|
||||||
|
}
|
||||||
|
// 编辑
|
||||||
|
function edit(uid) {
|
||||||
|
layer.open({
|
||||||
|
type: 2,
|
||||||
|
title: '编辑管理员',
|
||||||
|
shade: 0.3,
|
||||||
|
area: ['450px', '550px'],
|
||||||
|
content: "{$config['admin_route']}yunzeradmin/useredit?uid=" + uid
|
||||||
|
});
|
||||||
|
}
|
||||||
|
// 删除
|
||||||
|
function del(uid) {
|
||||||
|
layer.confirm('确定要删除吗?', {
|
||||||
|
icon: 3,
|
||||||
|
btn: ['确定', '取消']
|
||||||
|
}, function () {
|
||||||
|
$.post("{$config['admin_route']}yunzeradmin/userdel", { 'uid': uid }, function (res) {
|
||||||
|
if (res.code > 0) {
|
||||||
|
layer.alert(res.msg, { icon: 2 });
|
||||||
|
} else {
|
||||||
|
layer.msg(res.msg);
|
||||||
|
setTimeout(function () { window.location.reload(); }, 1000);
|
||||||
|
}
|
||||||
|
}, 'json');
|
||||||
|
});
|
||||||
|
}
|
||||||
|
//刷新列表
|
||||||
|
function refresh() {
|
||||||
|
window.location.reload();
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
{include file="public/tail" /}
|
||||||
82
app/admin/view/yunzertest/icon_list.php
Normal file
82
app/admin/view/yunzertest/icon_list.php
Normal file
@ -0,0 +1,82 @@
|
|||||||
|
{include file="public/header" /}
|
||||||
|
<style type="text/css">
|
||||||
|
ul {
|
||||||
|
display: block;
|
||||||
|
list-style-type: disc;
|
||||||
|
margin-block-start: 1em;
|
||||||
|
margin-block-end: 1em;
|
||||||
|
margin-inline-start: 0px;
|
||||||
|
margin-inline-end: 0px;
|
||||||
|
/* padding-inline-start: 40px; */
|
||||||
|
}
|
||||||
|
.site-doc-icon {
|
||||||
|
margin-bottom: 50px;
|
||||||
|
font-size: 0;
|
||||||
|
}
|
||||||
|
li {
|
||||||
|
list-style: none;
|
||||||
|
}
|
||||||
|
.site-doc-icon li {
|
||||||
|
display: inline-block;
|
||||||
|
vertical-align: middle;
|
||||||
|
width: 132px;
|
||||||
|
height: 105px;
|
||||||
|
line-height: 25px;
|
||||||
|
padding: 20px 0;
|
||||||
|
margin-right: -1px;
|
||||||
|
margin-bottom: -1px;
|
||||||
|
border: 1px solid #e2e2e2;
|
||||||
|
font-size: 14px;
|
||||||
|
text-align: center;
|
||||||
|
color: #666;
|
||||||
|
transition: all .3s;
|
||||||
|
-webkit-transition: all .3s;
|
||||||
|
}
|
||||||
|
.site-doc-icon li .layui-icon {
|
||||||
|
display: inline-block;
|
||||||
|
font-size: 36px;
|
||||||
|
}
|
||||||
|
.site-doc-icon li .doc-icon-name, .site-doc-icon li .doc-icon-code {
|
||||||
|
color: #c2c2c2;
|
||||||
|
}
|
||||||
|
.site-doc-icon li .doc-icon-fontclass {
|
||||||
|
height: 40px;
|
||||||
|
line-height: 20px;
|
||||||
|
padding: 0 5px;
|
||||||
|
font-size: 13px;
|
||||||
|
color: #333;
|
||||||
|
}
|
||||||
|
</style>
|
||||||
|
<div class="header" style="margin-bottom:20px;">
|
||||||
|
<span>图标</span>
|
||||||
|
<div></div>
|
||||||
|
</div>
|
||||||
|
<pre id="pre" class="layui-code">
|
||||||
|
<!-- 图标 -->
|
||||||
|
<i class="layui-icon layui-icon-face-smile"></i>
|
||||||
|
<!-- 图标,设置颜色、大小 -->
|
||||||
|
<i class="layui-icon layui-icon-face-smile" style="font-size:30px;color:#1E9FFF;"></i>
|
||||||
|
<!-- 图标,设置旋转 -->
|
||||||
|
<i class="layui-icon layui-icon-face-smile layui-anim layui-anim-rotate layui-anim-loop"></i>
|
||||||
|
</pre>
|
||||||
|
<ul class="site-doc-icon" style="text-align: center;">
|
||||||
|
{foreach($lists as $lists_v)}
|
||||||
|
<li>
|
||||||
|
<i class="layui-icon {$lists_v['icon_css']} {if($lists_v['icon_name']=='loading')} layui-anim layui-anim-rotate layui-anim-loop{/if}"></i>
|
||||||
|
<div class="doc-icon-name">{$lists_v['icon_name']}</div>
|
||||||
|
<div class="doc-icon-code">{$lists_v['icon_html']}</div>
|
||||||
|
<div class="doc-icon-fontclass">{$lists_v['icon_css']}</div>
|
||||||
|
</li>
|
||||||
|
{/foreach}
|
||||||
|
</ul>
|
||||||
|
<script type="text/javascript">
|
||||||
|
layui.use('code', function(){
|
||||||
|
layui.code({
|
||||||
|
elem:'#pre'
|
||||||
|
,title: '使用方法'
|
||||||
|
,encode: true
|
||||||
|
,skin: 'notepad' //如果要默认风格,不用设定该key。
|
||||||
|
});
|
||||||
|
});
|
||||||
|
</script>
|
||||||
|
{include file="public/tail" /}
|
||||||
136
app/admin/view/yunzertest/test_add.php
Normal file
136
app/admin/view/yunzertest/test_add.php
Normal file
@ -0,0 +1,136 @@
|
|||||||
|
{include file="public/header" /}
|
||||||
|
<form class="layui-form">
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">文本</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<input type="text" class="layui-input" id="test_input" placeholder="请输入文本">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">富文本</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<textarea class="layui-textarea" id="test_rich" placeholder="请输入富文本"></textarea>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">参照</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<select id="test_reference">
|
||||||
|
<option value="1" selected>开启</option>
|
||||||
|
<option value="0">关闭</option>
|
||||||
|
</select>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">时间戳</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<input type="text" class="layui-input" id="test_time">
|
||||||
|
</div>
|
||||||
|
<script type="text/javascript">
|
||||||
|
layui.use('laydate', function(){
|
||||||
|
laydate = layui.laydate;
|
||||||
|
laydate.render({
|
||||||
|
elem: '#test_time',
|
||||||
|
type: 'datetime'
|
||||||
|
});
|
||||||
|
});
|
||||||
|
</script>
|
||||||
|
</div>
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">日期</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<input type="text" class="layui-input" id="test_data">
|
||||||
|
</div>
|
||||||
|
<script type="text/javascript">
|
||||||
|
layui.use('laydate', function(){
|
||||||
|
laydate = layui.laydate;
|
||||||
|
laydate.render({
|
||||||
|
elem: '#test_data'
|
||||||
|
});
|
||||||
|
});
|
||||||
|
</script>
|
||||||
|
</div>
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">日期时间</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<input type="text" class="layui-input" id="test_datatime">
|
||||||
|
</div>
|
||||||
|
<script type="text/javascript">
|
||||||
|
layui.use('laydate', function(){
|
||||||
|
laydate = layui.laydate;
|
||||||
|
laydate.render({
|
||||||
|
elem: '#test_datatime',
|
||||||
|
type: 'datetime'
|
||||||
|
});
|
||||||
|
});
|
||||||
|
</script>
|
||||||
|
</div>
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">图片</label>
|
||||||
|
<div class="layui-input-inline">
|
||||||
|
<button type="button" class="layui-btn layui-btn-fluid" id="up_test_img">上传图片</button>
|
||||||
|
</div>
|
||||||
|
<div class="layui-form-mid layui-word-aux">支持png,jpg,gif</div>
|
||||||
|
</div>
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">图片预览</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<div class="layui-upload-list" id="demo_test_img" style="width:90%;float:left;"></div>
|
||||||
|
</div>
|
||||||
|
<script type="text/javascript">
|
||||||
|
layui.use(['layer','upload'],function(){
|
||||||
|
layer = layui.layer;
|
||||||
|
$ = layui.jquery;
|
||||||
|
upload = layui.upload;
|
||||||
|
upload.render({
|
||||||
|
elem: "#up_test_img",
|
||||||
|
url: "{$config['admin_route']}index/upload_img",
|
||||||
|
multiple: true,
|
||||||
|
done: function(res,title){
|
||||||
|
if(res.code > 0){
|
||||||
|
return layer.msg("上传失败,"+res.msg);
|
||||||
|
}else{
|
||||||
|
$("#demo_test_img").append('<div class="upload_pic_li" style="position:relative; width:100px; float:left; margin:0 10px 10px 0"><img style="width:100px;height:100px;" src="'+ res.url +'" class="layui-upload-img" onmouseover="show_img(this)" onmouseleave="hide_img()"><input type="hidden" name="test_img" value="'+res.data+'"/><a class="close-img" href="javascript:;" onclick="deleteImage(\''+res.data+'\',this);return false;"></a></div>');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
</script>
|
||||||
|
</div>
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">百度富文本</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<textarea id="test_rich_baidu" style="width:100%;" placeholder="请输入百度富文本"></textarea>
|
||||||
|
</div>
|
||||||
|
<script type="text/javascript" charset="utf-8" src="/static/third/kindeditor/kindeditor-all-min.js"></script>
|
||||||
|
<script type="text/javascript" charset="utf-8" src="/static/third/kindeditor/lang/zh-CN.js"></script>
|
||||||
|
<script type="text/javascript">
|
||||||
|
KindEditor.ready(function(K) {
|
||||||
|
window.editor = K.create('#test_rich_baidu',{
|
||||||
|
items : [
|
||||||
|
'source', '|', 'undo', 'redo', '|', 'preview', 'print', 'template', 'code', 'cut', 'copy', 'paste',
|
||||||
|
'plainpaste', 'wordpaste', '|', 'justifyleft', 'justifycenter', 'justifyright',
|
||||||
|
'justifyfull', 'insertorderedlist', 'insertunorderedlist', 'indent', 'outdent', 'subscript',
|
||||||
|
'superscript', 'clearhtml', 'quickformat', 'selectall', '|', 'fullscreen', '/',
|
||||||
|
'formatblock', 'fontname', 'fontsize', '|', 'forecolor', 'hilitecolor', 'bold',
|
||||||
|
'italic', 'underline', 'strikethrough', 'lineheight', 'removeformat', '|', 'image', 'multiimage',
|
||||||
|
'flash', 'media', 'insertfile', 'table', 'hr', 'emoticons', 'baidumap', 'pagebreak',
|
||||||
|
'anchor', 'link', 'unlink', '|', 'about'
|
||||||
|
],
|
||||||
|
minHeight:'350px',
|
||||||
|
allowFileManager : true,
|
||||||
|
uploadJson : "{$config['admin_route']}index/upload_imgs_kin",
|
||||||
|
afterUpload: function(){this.sync();}, //图片上传后,将上传内容同步到textarea中
|
||||||
|
afterBlur: function(){this.sync();}, //失去焦点时,将上传内容同步到textarea中
|
||||||
|
});
|
||||||
|
});
|
||||||
|
</script>
|
||||||
|
</div>
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">网址链接</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<input type="text" class="layui-input" id="test_url" placeholder="请输入网址链接">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</form>
|
||||||
|
{include file="public/tail" /}
|
||||||
144
app/admin/view/yunzertest/test_edit.php
Normal file
144
app/admin/view/yunzertest/test_edit.php
Normal file
@ -0,0 +1,144 @@
|
|||||||
|
{include file="public/header" /}
|
||||||
|
<form class="layui-form">
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">文本</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<input type="text" class="layui-input" id="test_input" placeholder="请输入文本" value="{$find['test_input']}">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">富文本</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<textarea class="layui-textarea" id="test_rich" placeholder="请输入富文本">{$find['test_rich']}</textarea>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">参照</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<select id="test_reference">
|
||||||
|
<option value="1" {$find['test_reference']==1?'selected':''}>开启</option>
|
||||||
|
<option value="0" {$find['test_reference']==0?'selected':''}>关闭</option>
|
||||||
|
</select>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">时间戳</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<input type="text" class="layui-input" id="test_time" value="{$find['test_time']}">
|
||||||
|
</div>
|
||||||
|
<script type="text/javascript">
|
||||||
|
layui.use('laydate', function(){
|
||||||
|
laydate = layui.laydate;
|
||||||
|
laydate.render({
|
||||||
|
elem: '#test_time',
|
||||||
|
type: 'datetime'
|
||||||
|
});
|
||||||
|
});
|
||||||
|
</script>
|
||||||
|
</div>
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">日期</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<input type="text" class="layui-input" id="test_data" value="{$find['test_data']}">
|
||||||
|
</div>
|
||||||
|
<script type="text/javascript">
|
||||||
|
layui.use('laydate', function(){
|
||||||
|
laydate = layui.laydate;
|
||||||
|
laydate.render({
|
||||||
|
elem: '#test_data'
|
||||||
|
});
|
||||||
|
});
|
||||||
|
</script>
|
||||||
|
</div>
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">日期时间</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<input type="text" class="layui-input" id="test_datatime" value="{$find['test_datatime']}">
|
||||||
|
</div>
|
||||||
|
<script type="text/javascript">
|
||||||
|
layui.use('laydate', function(){
|
||||||
|
laydate = layui.laydate;
|
||||||
|
laydate.render({
|
||||||
|
elem: '#test_datatime',
|
||||||
|
type: 'datetime'
|
||||||
|
});
|
||||||
|
});
|
||||||
|
</script>
|
||||||
|
</div>
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">图片</label>
|
||||||
|
<div class="layui-input-inline">
|
||||||
|
<button type="button" class="layui-btn layui-btn-fluid" id="up_test_img">上传图片</button>
|
||||||
|
</div>
|
||||||
|
<div class="layui-form-mid layui-word-aux">支持png,jpg,gif</div>
|
||||||
|
</div>
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">图片预览</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<div class="layui-upload-list" id="demo_test_img" style="width:90%;float:left;">
|
||||||
|
{if(!empty($find['test_img']))}
|
||||||
|
<div class="upload_pic_li" style="position:relative; width:100px; float:left; margin:0 10px 10px 0">
|
||||||
|
<img style="width:100px;height:100px;" src="{$find['test_img']}" class="layui-upload-img" onmouseover="show_img(this)" onmouseleave="hide_img()">
|
||||||
|
<input type="hidden" name="test_img" value="{$find['test_img']}"/>
|
||||||
|
<a class="close-img" href="javascript:;" onclick="deleteImage('{$find[\'test_img\']}',this);return false;"></a>
|
||||||
|
</div>
|
||||||
|
{/if}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<script type="text/javascript">
|
||||||
|
layui.use(['layer','upload'],function(){
|
||||||
|
layer = layui.layer;
|
||||||
|
$ = layui.jquery;
|
||||||
|
upload = layui.upload;
|
||||||
|
upload.render({
|
||||||
|
elem: "#up_test_img",
|
||||||
|
url: "{$config['admin_route']}index/upload_img",
|
||||||
|
multiple: true,
|
||||||
|
done: function(res,title){
|
||||||
|
if(res.code > 0){
|
||||||
|
return layer.msg("上传失败,"+res.msg);
|
||||||
|
}else{
|
||||||
|
$("#demo_test_img").append('<div class="upload_pic_li" style="position:relative; width:100px; float:left; margin:0 10px 10px 0"><img style="width:100px;height:100px;" src="'+ res.url +'" class="layui-upload-img" onmouseover="show_img(this)" onmouseleave="hide_img()"><input type="hidden" name="test_img" value="'+res.data+'"/><a class="close-img" href="javascript:;" onclick="deleteImage(\''+res.data+'\',this);return false;"></a></div>');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
|
</script>
|
||||||
|
</div>
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">百度富文本</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<textarea id="test_rich_baidu" style="width:100%;" placeholder="请输入百度富文本">{$find['test_rich_baidu']}</textarea>
|
||||||
|
</div>
|
||||||
|
<script type="text/javascript" charset="utf-8" src="/static/third/kindeditor/kindeditor-all-min.js"></script>
|
||||||
|
<script type="text/javascript" charset="utf-8" src="/static/third/kindeditor/lang/zh-CN.js"></script>
|
||||||
|
<script type="text/javascript">
|
||||||
|
KindEditor.ready(function(K) {
|
||||||
|
window.editor = K.create('#test_rich_baidu',{
|
||||||
|
items : [
|
||||||
|
'source', '|', 'undo', 'redo', '|', 'preview', 'print', 'template', 'code', 'cut', 'copy', 'paste',
|
||||||
|
'plainpaste', 'wordpaste', '|', 'justifyleft', 'justifycenter', 'justifyright',
|
||||||
|
'justifyfull', 'insertorderedlist', 'insertunorderedlist', 'indent', 'outdent', 'subscript',
|
||||||
|
'superscript', 'clearhtml', 'quickformat', 'selectall', '|', 'fullscreen', '/',
|
||||||
|
'formatblock', 'fontname', 'fontsize', '|', 'forecolor', 'hilitecolor', 'bold',
|
||||||
|
'italic', 'underline', 'strikethrough', 'lineheight', 'removeformat', '|', 'image', 'multiimage',
|
||||||
|
'flash', 'media', 'insertfile', 'table', 'hr', 'emoticons', 'baidumap', 'pagebreak',
|
||||||
|
'anchor', 'link', 'unlink', '|', 'about'
|
||||||
|
],
|
||||||
|
minHeight:'350px',
|
||||||
|
allowFileManager : true,
|
||||||
|
uploadJson : "{$config['admin_route']}index/upload_imgs_kin",
|
||||||
|
afterUpload: function(){this.sync();}, //图片上传后,将上传内容同步到textarea中
|
||||||
|
afterBlur: function(){this.sync();}, //失去焦点时,将上传内容同步到textarea中
|
||||||
|
});
|
||||||
|
});
|
||||||
|
</script>
|
||||||
|
</div>
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">网址链接</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<input type="text" class="layui-input" id="test_url" placeholder="请输入网址链接" value="{$find['test_url']}">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</form>
|
||||||
|
{include file="public/tail" /}
|
||||||
193
app/admin/view/yunzertest/test_list.php
Normal file
193
app/admin/view/yunzertest/test_list.php
Normal file
@ -0,0 +1,193 @@
|
|||||||
|
{include file="public/header" /}
|
||||||
|
<div class="header">
|
||||||
|
<span>演示列表-方法渲染</span>
|
||||||
|
<div></div>
|
||||||
|
</div>
|
||||||
|
<div class="layui-form layui-card-header layuiadmin-card-header-auto" lay-filter="layadmin-useradmin-formlist" style="height:100%;">
|
||||||
|
<div class="layui-form-item" id="search"></div>
|
||||||
|
</div>
|
||||||
|
<script type="text/html" id="toolbar">
|
||||||
|
<div class="layui-btn-container">
|
||||||
|
<div class="layui-btn-group">
|
||||||
|
<button type="button" class="layui-btn layui-btn-primary" lay-event="add">
|
||||||
|
<i class="layui-icon layui-icon-add-1"></i>添加
|
||||||
|
</button>
|
||||||
|
<button type="button" class="layui-btn layui-btn-primary" lay-event="edit">
|
||||||
|
<i class="layui-icon layui-icon-edit"></i>修改
|
||||||
|
</button>
|
||||||
|
<button type="button" class="layui-btn layui-btn-primary" lay-event="del">
|
||||||
|
<i class="layui-icon layui-icon-delete"></i>删除
|
||||||
|
</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</script>
|
||||||
|
<table class="layui-hide" id="datalist" lay-filter="datalist"></table>
|
||||||
|
<script type="text/javascript">
|
||||||
|
layui.use(['layer','table'],function(){
|
||||||
|
table = layui.table;
|
||||||
|
layer = layui.layer;
|
||||||
|
$ = layui.jquery;
|
||||||
|
get_data();
|
||||||
|
|
||||||
|
table.on('row(datalist)',function(obj){
|
||||||
|
obj.tr.addClass('layui-table-click').siblings().removeClass('layui-table-click');
|
||||||
|
obj.tr.find("div.layui-unselect.layui-form-radio")[0].click();
|
||||||
|
});
|
||||||
|
table.on('toolbar(datalist)',function(obj){
|
||||||
|
var checkStatus = table.checkStatus(obj.config.id);
|
||||||
|
var data = checkStatus.data;
|
||||||
|
var datas = obj.data;
|
||||||
|
var layEvent = obj.event;
|
||||||
|
var tr = obj.tr;
|
||||||
|
|
||||||
|
switch(obj.event){
|
||||||
|
case 'add':
|
||||||
|
layer.full(layer.open({
|
||||||
|
title: '添加',
|
||||||
|
type: 2,
|
||||||
|
content: "{$config['admin_route']}yunzertest/test_add",
|
||||||
|
maxmin: true,
|
||||||
|
area: ['70%','70%'],
|
||||||
|
btn: ['确定','关闭'],
|
||||||
|
yes: function(index,layero){
|
||||||
|
var form = layero.find('iframe')[0].contentWindow;
|
||||||
|
var body = layer.getChildFrame('body', index);
|
||||||
|
var test_img =[];
|
||||||
|
body.find("input[name='test_img']").each(function(){
|
||||||
|
test_img.push($(this).val());
|
||||||
|
})
|
||||||
|
var dataj = {
|
||||||
|
test_input : form.test_input.value,
|
||||||
|
test_reference : form.test_reference.value,
|
||||||
|
test_time : form.test_time.value,
|
||||||
|
test_data : form.test_data.value,
|
||||||
|
test_datatime : form.test_datatime.value,
|
||||||
|
test_img : test_img,
|
||||||
|
test_rich : form.test_rich.value,
|
||||||
|
test_rich_baidu : form.test_rich_baidu.value,
|
||||||
|
test_url : form.test_url.value
|
||||||
|
};
|
||||||
|
$.post('{$config["admin_route"]}yunzertest/test_add',dataj,function(res){
|
||||||
|
if(res.code>0){
|
||||||
|
layer.msg(res.msg,{'icon':2});
|
||||||
|
}else{
|
||||||
|
layer.msg(res.msg,{'icon':1});
|
||||||
|
get_data();
|
||||||
|
layer.close(index);
|
||||||
|
}
|
||||||
|
},'json');
|
||||||
|
}
|
||||||
|
}))
|
||||||
|
break;
|
||||||
|
case 'edit':
|
||||||
|
if(!data[0]){
|
||||||
|
layer.msg('请选择一条数据',{'icon':2});
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
layer.full(layer.open({
|
||||||
|
title: '修改',
|
||||||
|
type: 2,
|
||||||
|
content: "{$config['admin_route']}yunzertest/test_edit?test_id="+data[0].test_id,
|
||||||
|
maxmin: true,
|
||||||
|
area: ['70%','70%'],
|
||||||
|
btn: ['确定','关闭'],
|
||||||
|
yes: function(index,layero){
|
||||||
|
var form = layero.find('iframe')[0].contentWindow;
|
||||||
|
var body = layer.getChildFrame('body', index);
|
||||||
|
|
||||||
|
var test_img =[];
|
||||||
|
body.find("input[name='test_img']").each(function(){
|
||||||
|
test_img.push($(this).val());
|
||||||
|
})
|
||||||
|
var dataj = {
|
||||||
|
test_id : data[0].test_id,
|
||||||
|
test_input : form.test_input.value,
|
||||||
|
test_reference : form.test_reference.value,
|
||||||
|
test_time : form.test_time.value,
|
||||||
|
test_data : form.test_data.value,
|
||||||
|
test_datatime : form.test_datatime.value,
|
||||||
|
test_img : test_img,
|
||||||
|
test_rich : form.test_rich.value,
|
||||||
|
test_rich_baidu : form.test_rich_baidu.value,
|
||||||
|
test_url : form.test_url.value
|
||||||
|
};
|
||||||
|
$.post('{$config["admin_route"]}yunzertest/test_edit',dataj,function(res){
|
||||||
|
if(res.code>0){
|
||||||
|
layer.msg(res.msg,{'icon':2});
|
||||||
|
}else{
|
||||||
|
layer.msg(res.msg,{'icon':1});
|
||||||
|
get_data();
|
||||||
|
layer.close(index);
|
||||||
|
}
|
||||||
|
},'json');
|
||||||
|
}
|
||||||
|
}))
|
||||||
|
break;
|
||||||
|
case 'del':
|
||||||
|
if(!data[0]){
|
||||||
|
layer.msg('请选择一条数据',{'icon':2});
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
layer.confirm('确定要删除吗?',{
|
||||||
|
icon:3,
|
||||||
|
btn: ['确定','取消']
|
||||||
|
},function(){
|
||||||
|
$.post('{$config["admin_route"]}yunzertest/test_del',{'test_id':data[0].test_id},function(res){
|
||||||
|
if(res.code>0){
|
||||||
|
layer.alert(res.msg,{icon:2});
|
||||||
|
}else{
|
||||||
|
layer.msg(res.msg);
|
||||||
|
setTimeout(function(){window.location.reload();},1000);
|
||||||
|
}
|
||||||
|
},'json');
|
||||||
|
});
|
||||||
|
break;
|
||||||
|
};
|
||||||
|
});
|
||||||
|
|
||||||
|
function get_data(){
|
||||||
|
table.render({
|
||||||
|
elem: '#datalist',
|
||||||
|
method : 'post'
|
||||||
|
,url:"{$config['admin_route']}yunzertest/test_list"
|
||||||
|
,toolbar: '#toolbar'
|
||||||
|
,defaultToolbar: ['filter', 'print', 'exports', {
|
||||||
|
title: '提示'
|
||||||
|
,layEvent: 'LAYTABLE_TIPS'
|
||||||
|
,icon: 'layui-icon-tips'
|
||||||
|
}]
|
||||||
|
,title: '演示列表'
|
||||||
|
,page: true
|
||||||
|
,cols: [[
|
||||||
|
{type:'checkbox'},
|
||||||
|
{type:'radio'}
|
||||||
|
,{field:'test_id', width:50, title:'ID'}
|
||||||
|
,{field:'test_input', width:100, title: '文本'}
|
||||||
|
,{field:'test_rich', width:100, title: '富文本'}
|
||||||
|
,{field:'test_rich_baidu', width:120, title: '百度文本'}
|
||||||
|
,{field:'test_img', width:100, title: '图片',templet:function(res){
|
||||||
|
return '<img style="width:30px;height:30px;" onmouseover="show_img(this)" onmouseleave="hide_img()" src="'+res.test_img+'"';
|
||||||
|
}}
|
||||||
|
,{field:'test_reference', width:100, title: '参照',templet:function(res){
|
||||||
|
if(res.test_reference == 1){
|
||||||
|
return '<span style="color:green;">开启</span>';
|
||||||
|
}else{
|
||||||
|
return '<span style="color:grey;">关闭</span>';
|
||||||
|
}
|
||||||
|
}}
|
||||||
|
,{field:'test_time', width:100, title: '时间戳'}
|
||||||
|
,{field:'test_data', width:100, title: '日期'}
|
||||||
|
,{field:'test_datatime', width:150, title: '日期时间'}
|
||||||
|
,{field:'test_url', width:150, title: '网址链接',templet:function(res){
|
||||||
|
return '<a href="'+res.test_url+'" target="_blank">'+res.test_url+'</a>';
|
||||||
|
}}
|
||||||
|
]]
|
||||||
|
,data:[]
|
||||||
|
,limit :10
|
||||||
|
,limits:[10,20,30]
|
||||||
|
,height: 'full-100'
|
||||||
|
});
|
||||||
|
}
|
||||||
|
});
|
||||||
|
</script>
|
||||||
|
{include file="public/tail" /}
|
||||||
140
app/admin/view/yunzertest/test_static_add.php
Normal file
140
app/admin/view/yunzertest/test_static_add.php
Normal file
@ -0,0 +1,140 @@
|
|||||||
|
{include file="public/header" /}
|
||||||
|
<form class="layui-form">
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">文本</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<input type="text" class="layui-input" id="test_input" name="test_input" placeholder="请输入文本">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">富文本</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<textarea class="layui-textarea" id="test_rich" name="test_rich" placeholder="请输入富文本"></textarea>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">参照</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<select id="test_reference">
|
||||||
|
<option value="1" selected>开启</option>
|
||||||
|
<option value="0">关闭</option>
|
||||||
|
</select>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">时间戳</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<input type="text" class="layui-input" id="test_time" name="test_time">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">日期</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<input type="text" class="layui-input" id="test_data" name="test_data">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">日期时间</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<input type="text" class="layui-input" id="test_datatime" name="test_datatime">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">图片</label>
|
||||||
|
<div class="layui-input-inline">
|
||||||
|
<button type="button" class="layui-btn layui-btn-fluid" id="up_test_img">上传图片</button>
|
||||||
|
</div>
|
||||||
|
<div class="layui-form-mid layui-word-aux">支持png,jpg,gif</div>
|
||||||
|
</div>
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">图片预览</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<div class="layui-upload-list" id="demo_test_img" style="width:90%;float:left;"></div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">百度富文本</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<textarea id="test_rich_baidu" name="test_rich_baidu" style="width:100%;" placeholder="请输入百度富文本"></textarea>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">网址链接</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<input type="text" class="layui-input" id="test_url" name="test_url" placeholder="请输入网址链接">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</form>
|
||||||
|
<div class="layui-form-item" style="margin-top:10px;">
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<button type="button" class="layui-btn" onclick="save()">保存</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<script type="text/javascript" charset="utf-8" src="/static/third/kindeditor/kindeditor-all-min.js"></script>
|
||||||
|
<script type="text/javascript" charset="utf-8" src="/static/third/kindeditor/lang/zh-CN.js"></script>
|
||||||
|
<script type="text/javascript">
|
||||||
|
layui.use(['layer','form','upload','laydate'],function(){
|
||||||
|
form = layui.form;
|
||||||
|
layer = layui.layer;
|
||||||
|
$ = layui.jquery;
|
||||||
|
|
||||||
|
upload = layui.upload;
|
||||||
|
upload.render({
|
||||||
|
elem: "#up_test_img",
|
||||||
|
url: "{$config['admin_route']}index/upload_img",
|
||||||
|
multiple: true,
|
||||||
|
done: function(res,title){
|
||||||
|
if(res.code > 0){
|
||||||
|
return layer.msg("上传失败,"+res.msg);
|
||||||
|
}else{
|
||||||
|
$("#demo_test_img").append('<div class="upload_pic_li" style="position:relative; width:100px; float:left; margin:0 10px 10px 0"><img style="width:100px;height:100px;" src="'+ res.url +'" class="layui-upload-img" onmouseover="show_img(this)" onmouseleave="hide_img()"><input type="hidden" name="test_img" value="'+res.data+'"/><a class="close-img" href="javascript:;" onclick="deleteImage(\''+res.data+'\',this);return false;"></a></div>');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
laydate = layui.laydate;
|
||||||
|
laydate.render({
|
||||||
|
elem: '#test_time',
|
||||||
|
type: 'datetime'
|
||||||
|
});
|
||||||
|
laydate.render({
|
||||||
|
elem: '#test_data'
|
||||||
|
});
|
||||||
|
laydate.render({
|
||||||
|
elem: '#test_datatime',
|
||||||
|
type: 'datetime'
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
KindEditor.ready(function(K) {
|
||||||
|
window.editor = K.create('#test_rich_baidu',{
|
||||||
|
items : [
|
||||||
|
'source', '|', 'undo', 'redo', '|', 'preview', 'print', 'template', 'code', 'cut', 'copy', 'paste',
|
||||||
|
'plainpaste', 'wordpaste', '|', 'justifyleft', 'justifycenter', 'justifyright',
|
||||||
|
'justifyfull', 'insertorderedlist', 'insertunorderedlist', 'indent', 'outdent', 'subscript',
|
||||||
|
'superscript', 'clearhtml', 'quickformat', 'selectall', '|', 'fullscreen', '/',
|
||||||
|
'formatblock', 'fontname', 'fontsize', '|', 'forecolor', 'hilitecolor', 'bold',
|
||||||
|
'italic', 'underline', 'strikethrough', 'lineheight', 'removeformat', '|', 'image', 'multiimage',
|
||||||
|
'flash', 'media', 'insertfile', 'table', 'hr', 'emoticons', 'baidumap', 'pagebreak',
|
||||||
|
'anchor', 'link', 'unlink', '|', 'about'
|
||||||
|
],
|
||||||
|
minHeight:'350px',
|
||||||
|
allowFileManager : true,
|
||||||
|
uploadJson : "{$config['admin_route']}index/upload_imgs_kin",
|
||||||
|
afterUpload: function(){this.sync();}, //图片上传后,将上传内容同步到textarea中
|
||||||
|
afterBlur: function(){this.sync();}, //失去焦点时,将上传内容同步到textarea中
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
function save(){
|
||||||
|
$.post("{$config['admin_route']}Yunzertest/test_static_add",$('form').serialize(),function(res){
|
||||||
|
if(res.code>0){
|
||||||
|
layer.msg(res.msg,{'icon':2});
|
||||||
|
}else{
|
||||||
|
layer.msg(res.msg,{'icon':1});
|
||||||
|
setTimeout(function(){parent.window.location.reload();},1000);
|
||||||
|
}
|
||||||
|
},'json');
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
{include file="public/tail" /}
|
||||||
149
app/admin/view/yunzertest/test_static_edit.php
Normal file
149
app/admin/view/yunzertest/test_static_edit.php
Normal file
@ -0,0 +1,149 @@
|
|||||||
|
{include file="public/header" /}
|
||||||
|
<form class="layui-form">
|
||||||
|
<input type="hidden" name="test_id" value="{$find['test_id']}">
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">文本</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<input type="text" class="layui-input" id="test_input" placeholder="请输入文本" value="{$find['test_input']}">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">富文本</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<textarea class="layui-textarea" id="test_rich" placeholder="请输入富文本">{$find['test_rich']}</textarea>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">参照</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<select id="test_reference">
|
||||||
|
<option value="1" {$find['test_reference']==1?'selected':''}>开启</option>
|
||||||
|
<option value="0" {$find['test_reference']==0?'selected':''}>关闭</option>
|
||||||
|
</select>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">时间戳</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<input type="text" class="layui-input" id="test_time" value="{$find['test_time']}">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">日期</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<input type="text" class="layui-input" id="test_data" value="{$find['test_data']}">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">日期时间</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<input type="text" class="layui-input" id="test_datatime" value="{$find['test_datatime']}">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">图片</label>
|
||||||
|
<div class="layui-input-inline">
|
||||||
|
<button type="button" class="layui-btn layui-btn-fluid" id="up_test_img">上传图片</button>
|
||||||
|
</div>
|
||||||
|
<div class="layui-form-mid layui-word-aux">支持png,jpg,gif</div>
|
||||||
|
</div>
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">图片预览</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<div class="layui-upload-list" id="demo_test_img" style="width:90%;float:left;">
|
||||||
|
{if(!empty($find['test_img']))}
|
||||||
|
<div class="upload_pic_li" style="position:relative; width:100px; float:left; margin:0 10px 10px 0">
|
||||||
|
<img style="width:100px;height:100px;" src="{$find['test_img']}" class="layui-upload-img" onmouseover="show_img(this)" onmouseleave="hide_img()">
|
||||||
|
<input type="hidden" name="test_img" value="{$find['test_img']}"/>
|
||||||
|
<a class="close-img" href="javascript:;" onclick="deleteImage('{$find[\'test_img\']}',this);return false;"></a>
|
||||||
|
</div>
|
||||||
|
{/if}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">百度富文本</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<textarea id="test_rich_baidu" style="width:100%;" placeholder="请输入百度富文本">{$find['test_rich_baidu']}</textarea>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div class="layui-form-item">
|
||||||
|
<label class="layui-form-label">网址链接</label>
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<input type="text" class="layui-input" id="test_url" placeholder="请输入网址链接" value="{$find['test_url']}">
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</form>
|
||||||
|
<div class="layui-form-item" style="margin-top:10px;">
|
||||||
|
<div class="layui-input-block">
|
||||||
|
<button type="button" class="layui-btn" onclick="save()">保存</button>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<script type="text/javascript" charset="utf-8" src="/static/third/kindeditor/kindeditor-all-min.js"></script>
|
||||||
|
<script type="text/javascript" charset="utf-8" src="/static/third/kindeditor/lang/zh-CN.js"></script>
|
||||||
|
<script type="text/javascript">
|
||||||
|
layui.use(['layer','form','upload','laydate'],function(){
|
||||||
|
form = layui.form;
|
||||||
|
layer = layui.layer;
|
||||||
|
$ = layui.jquery;
|
||||||
|
|
||||||
|
upload = layui.upload;
|
||||||
|
upload.render({
|
||||||
|
elem: "#up_test_img",
|
||||||
|
url: "{$config['admin_route']}index/upload_img",
|
||||||
|
multiple: true,
|
||||||
|
done: function(res,title){
|
||||||
|
if(res.code > 0){
|
||||||
|
return layer.msg("上传失败,"+res.msg);
|
||||||
|
}else{
|
||||||
|
$("#demo_test_img").append('<div class="upload_pic_li" style="position:relative; width:100px; float:left; margin:0 10px 10px 0"><img style="width:100px;height:100px;" src="'+ res.url +'" class="layui-upload-img" onmouseover="show_img(this)" onmouseleave="hide_img()"><input type="hidden" name="test_img" value="'+res.data+'"/><a class="close-img" href="javascript:;" onclick="deleteImage(\''+res.data+'\',this);return false;"></a></div>');
|
||||||
|
}
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
laydate = layui.laydate;
|
||||||
|
laydate.render({
|
||||||
|
elem: '#test_time',
|
||||||
|
type: 'datetime'
|
||||||
|
});
|
||||||
|
laydate.render({
|
||||||
|
elem: '#test_data'
|
||||||
|
});
|
||||||
|
laydate.render({
|
||||||
|
elem: '#test_datatime',
|
||||||
|
type: 'datetime'
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
KindEditor.ready(function(K) {
|
||||||
|
window.editor = K.create('#test_rich_baidu',{
|
||||||
|
items : [
|
||||||
|
'source', '|', 'undo', 'redo', '|', 'preview', 'print', 'template', 'code', 'cut', 'copy', 'paste',
|
||||||
|
'plainpaste', 'wordpaste', '|', 'justifyleft', 'justifycenter', 'justifyright',
|
||||||
|
'justifyfull', 'insertorderedlist', 'insertunorderedlist', 'indent', 'outdent', 'subscript',
|
||||||
|
'superscript', 'clearhtml', 'quickformat', 'selectall', '|', 'fullscreen', '/',
|
||||||
|
'formatblock', 'fontname', 'fontsize', '|', 'forecolor', 'hilitecolor', 'bold',
|
||||||
|
'italic', 'underline', 'strikethrough', 'lineheight', 'removeformat', '|', 'image', 'multiimage',
|
||||||
|
'flash', 'media', 'insertfile', 'table', 'hr', 'emoticons', 'baidumap', 'pagebreak',
|
||||||
|
'anchor', 'link', 'unlink', '|', 'about'
|
||||||
|
],
|
||||||
|
minHeight:'350px',
|
||||||
|
allowFileManager : true,
|
||||||
|
uploadJson : "{$config['admin_route']}index/upload_imgs_kin",
|
||||||
|
afterUpload: function(){this.sync();}, //图片上传后,将上传内容同步到textarea中
|
||||||
|
afterBlur: function(){this.sync();}, //失去焦点时,将上传内容同步到textarea中
|
||||||
|
});
|
||||||
|
});
|
||||||
|
|
||||||
|
function save(){
|
||||||
|
$.post("{$config['admin_route']}Yunzertest/test_static_edit",$('form').serialize(),function(res){
|
||||||
|
if(res.code>0){
|
||||||
|
layer.msg(res.msg,{'icon':2});
|
||||||
|
}else{
|
||||||
|
layer.msg(res.msg,{'icon':1});
|
||||||
|
setTimeout(function(){parent.window.location.reload();},1000);
|
||||||
|
}
|
||||||
|
},'json');
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
{include file="public/tail" /}
|
||||||
104
app/admin/view/yunzertest/test_static_list.php
Normal file
104
app/admin/view/yunzertest/test_static_list.php
Normal file
@ -0,0 +1,104 @@
|
|||||||
|
{include file="public/header" /}
|
||||||
|
<div class="header">
|
||||||
|
<span>演示列表-静态表格</span>
|
||||||
|
<button type="button" class="layui-btn layui-btn-primary layui-btn-sm" onclick="add()">
|
||||||
|
<i class="layui-icon layui-icon-add-1"></i>添加
|
||||||
|
</button>
|
||||||
|
<div></div>
|
||||||
|
</div>
|
||||||
|
<table class="layui-table">
|
||||||
|
<thead>
|
||||||
|
<tr>
|
||||||
|
<th>ID</th>
|
||||||
|
<th>文本</th>
|
||||||
|
<th>富文本</th>
|
||||||
|
<th>百度文本</th>
|
||||||
|
<th>图片</th>
|
||||||
|
<th>参照</th>
|
||||||
|
<th>时间戳</th>
|
||||||
|
<th>日期</th>
|
||||||
|
<th>日期时间</th>
|
||||||
|
<th>网址链接</th>
|
||||||
|
<th>操作</th>
|
||||||
|
</tr>
|
||||||
|
</thead>
|
||||||
|
<tbody>
|
||||||
|
{volist name="lists" id='vo'}
|
||||||
|
<tr>
|
||||||
|
<td>{$vo['test_id']}</td>
|
||||||
|
<td>{$vo['test_input']}</td>
|
||||||
|
<td>{$vo['test_rich']}</td>
|
||||||
|
<td>{$vo['test_rich_baidu']}</td>
|
||||||
|
<td>
|
||||||
|
<img style="width:30px;height:30px;" onmouseover="show_img(this)" onmouseleave="hide_img()" src="{$vo['test_img']}">
|
||||||
|
</td>
|
||||||
|
<td>
|
||||||
|
{if($vo['test_reference'] == 1)}
|
||||||
|
<span style="color:green;">开启</span>
|
||||||
|
{else/}
|
||||||
|
<span style="color:grey;">关闭</span>
|
||||||
|
{/if}
|
||||||
|
</td>
|
||||||
|
<td>{:date('Y-m-d H:i:s',$vo.test_time)}</td>
|
||||||
|
<td>{$vo['test_data']}</td>
|
||||||
|
<td>{$vo['test_datatime']}</td>
|
||||||
|
<td><a href="{$vo['test_url']}" target="_blank">{$vo['test_url']}</a></td>
|
||||||
|
<td>
|
||||||
|
<button type="button" class="layui-btn layui-btn-primary layui-btn-xs" onclick="edit({$vo.test_id})">
|
||||||
|
<i class="layui-icon layui-icon-edit"></i>编辑
|
||||||
|
</button>
|
||||||
|
<button type="button" class="layui-btn layui-btn-primary layui-btn-xs" onclick="del({$vo.test_id})">
|
||||||
|
<i class="layui-icon layui-icon-delete"></i>删除
|
||||||
|
</button>
|
||||||
|
</td>
|
||||||
|
</tr>
|
||||||
|
{/volist}
|
||||||
|
</tbody>
|
||||||
|
</table>
|
||||||
|
<div>{$lists|raw}</div>
|
||||||
|
<script type="text/javascript">
|
||||||
|
layui.use(['layer'],function(){
|
||||||
|
layer = layui.layer;
|
||||||
|
$ = layui.jquery;
|
||||||
|
});
|
||||||
|
// 添加
|
||||||
|
function add(){
|
||||||
|
layer.full(
|
||||||
|
layer.open({
|
||||||
|
type: 2,
|
||||||
|
title: '添加',
|
||||||
|
shade: 0.3,
|
||||||
|
maxmin: true,
|
||||||
|
area: ['450px','550px'],
|
||||||
|
content: "{$config['admin_route']}Yunzertest/test_static_add"
|
||||||
|
})
|
||||||
|
);
|
||||||
|
}
|
||||||
|
// 编辑
|
||||||
|
function edit(test_id){
|
||||||
|
layer.full(layer.open({
|
||||||
|
type: 2,
|
||||||
|
title: '编辑',
|
||||||
|
shade: 0.3,
|
||||||
|
area: ['450px','550px'],
|
||||||
|
content: "{$config['admin_route']}Yunzertest/test_static_edit?test_id="+test_id
|
||||||
|
}));
|
||||||
|
}
|
||||||
|
// 删除
|
||||||
|
function del(test_id){
|
||||||
|
layer.confirm('确定要删除吗?', {
|
||||||
|
icon:3,
|
||||||
|
btn: ['确定','取消']
|
||||||
|
}, function(){
|
||||||
|
$.post("{$config['admin_route']}Yunzertest/test_del",{'test_id':test_id},function(res){
|
||||||
|
if(res.code>0){
|
||||||
|
layer.alert(res.msg,{icon:2});
|
||||||
|
}else{
|
||||||
|
layer.msg(res.msg,{icon:1});
|
||||||
|
setTimeout(function(){window.location.reload();},1000);
|
||||||
|
}
|
||||||
|
},'json');
|
||||||
|
});
|
||||||
|
}
|
||||||
|
</script>
|
||||||
|
{include file="public/tail" /}
|
||||||
77
app/common.php
Normal file
77
app/common.php
Normal file
@ -0,0 +1,77 @@
|
|||||||
|
<?php
|
||||||
|
/**
|
||||||
|
* 商业使用授权协议
|
||||||
|
*
|
||||||
|
* Copyright (c) 2025 [云泽网]. 保留所有权利.
|
||||||
|
*
|
||||||
|
* 本软件仅供评估使用。任何商业用途必须获得书面授权许可。
|
||||||
|
* 未经授权商业使用本软件属于侵权行为,将承担法律责任。
|
||||||
|
*
|
||||||
|
* 授权购买请联系: 357099073@qq.com
|
||||||
|
* 官方网站: https://www.yunzer.cn
|
||||||
|
*
|
||||||
|
* 评估用户须知:
|
||||||
|
* 1. 禁止移除版权声明
|
||||||
|
* 2. 禁止用于生产环境
|
||||||
|
* 3. 禁止转售或分发
|
||||||
|
*/
|
||||||
|
|
||||||
|
/*
|
||||||
|
* 递归处理菜单数据,解决新菜单(菜单展示时,使用)
|
||||||
|
*/
|
||||||
|
function gettreeitemsnew($items){
|
||||||
|
$tree = array();
|
||||||
|
foreach ($items as $item) {
|
||||||
|
if(isset($items[$item['parent_id']])){
|
||||||
|
$items[$item['parent_id']]['children'][] = &$items[$item['smid']];
|
||||||
|
}else{
|
||||||
|
$tree[] = &$items[$item['smid']];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return $tree;
|
||||||
|
}
|
||||||
|
/**
|
||||||
|
* 递归删除目录和文件
|
||||||
|
* @param string $dir_name
|
||||||
|
* @return bool
|
||||||
|
*/
|
||||||
|
function delete_dir_file($dir_name){
|
||||||
|
$result = false;
|
||||||
|
if (is_dir($dir_name)) {
|
||||||
|
if ($handle = opendir($dir_name)) {
|
||||||
|
while (false !== ($item = readdir($handle))) {
|
||||||
|
if ($item != '.' && $item != '..') {
|
||||||
|
if (is_dir($dir_name . DIRECTORY_SEPARATOR . $item)) {
|
||||||
|
delete_dir_file($dir_name . DIRECTORY_SEPARATOR . $item);
|
||||||
|
} else {
|
||||||
|
unlink($dir_name . DIRECTORY_SEPARATOR . $item);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
closedir($handle);
|
||||||
|
if (rmdir($dir_name)) {
|
||||||
|
$result = true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return $result;
|
||||||
|
}
|
||||||
|
/**
|
||||||
|
* 递归处理菜单数据
|
||||||
|
*/
|
||||||
|
function formatMenus($items){
|
||||||
|
foreach($items as &$item){
|
||||||
|
if(isset($item['children'])){
|
||||||
|
$item['children'] = array_values($item['children']);
|
||||||
|
formatMenus($item['children']);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
$items = array_values($items);
|
||||||
|
return $items;
|
||||||
|
}
|
||||||
|
/**
|
||||||
|
* 在文本里加反斜杠
|
||||||
|
*/
|
||||||
|
function enSql($str){
|
||||||
|
return addslashes($str);
|
||||||
|
}
|
||||||
Some files were not shown because too many files have changed in this diff Show More
Loading…
x
Reference in New Issue
Block a user