mirror of
https://gitee.com/zoujingli/ThinkAdmin.git
synced 2025-04-06 03:58:04 +08:00
Compare commits
No commits in common. "v6.1.69" and "v6" have entirely different histories.
23
.env.example
Normal file
23
.env.example
Normal file
@ -0,0 +1,23 @@
|
||||
# 数据配置
|
||||
DB_TYPE=sqlite
|
||||
DB_MYSQL_HOST=thinkadmin.top
|
||||
DB_MYSQL_PORT=3306
|
||||
DB_MYSQL_PREFIX=
|
||||
DB_MYSQL_USERNAME=root
|
||||
DB_MYSQL_DATABASE=admin
|
||||
DB_MYSQL_CHARSET=utf8mb4
|
||||
DB_MYSQL_PASSWORD=
|
||||
|
||||
# 缓存配置
|
||||
CACHE_TYPE=file
|
||||
CACHE_REDIS_HOST=127.0.0.1
|
||||
CACHE_REDIS_PORT=6379
|
||||
CACHE_REDIS_SELECT=
|
||||
CACHE_REDIS_PASSWORD=
|
||||
|
||||
# 会话配置
|
||||
SESSION_TYPE=file
|
||||
SESSION_NAME=ssid
|
||||
SESSION_STORE=
|
||||
SESSION_EXPIRE=7200
|
||||
SESSION_PREFIX=
|
65
.github/workflows/release.yml
vendored
Normal file
65
.github/workflows/release.yml
vendored
Normal file
@ -0,0 +1,65 @@
|
||||
on:
|
||||
push:
|
||||
# Sequence of patterns matched against refs/tags
|
||||
tags:
|
||||
- 'v*' # Push events to matching v*, i.e. v1.0, v20.15.10
|
||||
|
||||
name: Create Release
|
||||
permissions: write-all
|
||||
|
||||
jobs:
|
||||
release:
|
||||
runs-on: ubuntu-latest
|
||||
steps:
|
||||
- name: Checkout code
|
||||
uses: actions/checkout@v4
|
||||
with:
|
||||
fetch-depth: 0
|
||||
|
||||
- name: Setup Node.js
|
||||
uses: actions/setup-node@v4
|
||||
with:
|
||||
node-version: 18
|
||||
|
||||
- name: Install dependencies
|
||||
run: npm install -g gen-git-log
|
||||
|
||||
- name: Find Last Tag
|
||||
id: last_tag
|
||||
run: |
|
||||
|
||||
# 获取所有标签,按版本排序(降序)
|
||||
Tags=$(git tag --list --sort=-version:refname)
|
||||
|
||||
# 获取最新的标签(即列表中的第一个)
|
||||
LATEST_TAG=$(echo "$Tags" | awk 'NR==1 {print $1; exit}')
|
||||
|
||||
# 获取倒数第二个标签(如果存在)
|
||||
if [[ -n "$Tags" ]]; then
|
||||
# 使用 tail 获取除了最后一个标签之外的所有标签,然后用 head 获取第一个
|
||||
SECOND_LATEST_TAG=$(echo "$Tags" | tail -n +2 | head -n 1)
|
||||
else
|
||||
SECOND_LATEST_TAG=""
|
||||
fi
|
||||
|
||||
# 设置输出变量
|
||||
echo "::set-output name=tag_last::${LATEST_TAG:-v1.0.0}"
|
||||
echo "::set-output name=tag_second::${SECOND_LATEST_TAG:-v1.0.0}"
|
||||
|
||||
- name: Generate Release Notes
|
||||
run: |
|
||||
rm -rf log
|
||||
newTag=${{ steps.last_tag.outputs.tag_last }}
|
||||
git-log -m tag -f -S ${{ steps.last_tag.outputs.tag_second }} -v ${newTag#v}
|
||||
|
||||
- name: Create Release
|
||||
id: create_release
|
||||
uses: actions/create-release@v1
|
||||
env:
|
||||
GITHUB_TOKEN: ${{ secrets.GITHUB_TOKEN }}
|
||||
with:
|
||||
tag_name: ${{ steps.last_tag.outputs.tag_last }}
|
||||
release_name: Release ${{ steps.last_tag.outputs.tag_last }}
|
||||
body_path: log/${{steps.last_tag.outputs.tag_last}}.md
|
||||
draft: false
|
||||
prerelease: false
|
24
.gitignore
vendored
24
.gitignore
vendored
@ -13,38 +13,22 @@
|
||||
/composer.lock
|
||||
!composer.json
|
||||
|
||||
### 屏蔽环境文件
|
||||
/404.html
|
||||
/.user.ini
|
||||
/index.html
|
||||
/public/upload
|
||||
/public/404.html
|
||||
/public/.htaccess
|
||||
/public/.user.ini
|
||||
/public/index.html
|
||||
/public/favicon.ico
|
||||
/database/sqlite.db
|
||||
|
||||
### 屏蔽插件文件
|
||||
/app/admin
|
||||
/app/wechat
|
||||
|
||||
### 屏蔽配置文件
|
||||
/config/app.php
|
||||
/config/cookie.php
|
||||
/config/lang.php
|
||||
/config/log.php
|
||||
/config/route.php
|
||||
/config/session.php
|
||||
/config/view.php
|
||||
|
||||
/public/static/plugs
|
||||
/public/static/theme
|
||||
/public/static/admin.js
|
||||
/public/static/login.js
|
||||
/public/static/theme/css/_*.css*
|
||||
/public/static/theme/css/node_modules
|
||||
/public/static/theme/css/package-lock.json
|
||||
|
||||
### 屏蔽数据库脚本
|
||||
/database/migrations/*_install_*.php
|
||||
/database/migrations/*_upgrade_*.php
|
||||
!database/migrations/*_install_table.php
|
||||
!database/migrations/*_install*_table.php
|
||||
!database/migrations/*_install_package.php
|
||||
|
69
app/admin/Service.php
Normal file
69
app/admin/Service.php
Normal file
@ -0,0 +1,69 @@
|
||||
<?php
|
||||
|
||||
// +----------------------------------------------------------------------
|
||||
// | Admin Plugin for ThinkAdmin
|
||||
// +----------------------------------------------------------------------
|
||||
// | 版权所有 2014~2024 ThinkAdmin [ thinkadmin.top ]
|
||||
// +----------------------------------------------------------------------
|
||||
// | 官方网站: https://thinkadmin.top
|
||||
// +----------------------------------------------------------------------
|
||||
// | 开源协议 ( https://mit-license.org )
|
||||
// | 免责声明 ( https://thinkadmin.top/disclaimer )
|
||||
// +----------------------------------------------------------------------
|
||||
// | gitee 代码仓库:https://gitee.com/zoujingli/think-plugs-admin
|
||||
// | github 代码仓库:https://github.com/zoujingli/think-plugs-admin
|
||||
// +----------------------------------------------------------------------
|
||||
|
||||
declare(strict_types=1);
|
||||
|
||||
namespace app\admin;
|
||||
|
||||
use think\admin\Plugin;
|
||||
|
||||
/**
|
||||
* 插件服务注册
|
||||
* @class Service
|
||||
* @package app\admin
|
||||
*/
|
||||
class Service extends Plugin
|
||||
{
|
||||
/**
|
||||
* 定义插件名称
|
||||
* @var string
|
||||
*/
|
||||
protected $appName = '系统管理';
|
||||
|
||||
/**
|
||||
* 定义安装包名
|
||||
* @var string
|
||||
*/
|
||||
protected $package = 'zoujingli/think-plugs-admin';
|
||||
|
||||
/**
|
||||
* 定义插件中心菜单
|
||||
* @return array
|
||||
*/
|
||||
public static function menu(): array
|
||||
{
|
||||
return [
|
||||
[
|
||||
'name' => '系统配置',
|
||||
'subs' => [
|
||||
['name' => '系统参数配置', 'icon' => 'layui-icon layui-icon-set', 'node' => 'admin/config/index'],
|
||||
['name' => '系统任务管理', 'icon' => 'layui-icon layui-icon-log', 'node' => 'admin/queue/index'],
|
||||
['name' => '系统日志管理', 'icon' => 'layui-icon layui-icon-form', 'node' => 'admin/oplog/index'],
|
||||
['name' => '数据字典管理', 'icon' => 'layui-icon layui-icon-code-circle', 'node' => 'admin/base/index'],
|
||||
['name' => '系统文件管理', 'icon' => 'layui-icon layui-icon-carousel', 'node' => 'admin/file/index'],
|
||||
['name' => '系统菜单管理', 'icon' => 'layui-icon layui-icon-layouts', 'node' => 'admin/menu/index'],
|
||||
],
|
||||
],
|
||||
[
|
||||
'name' => '权限管理',
|
||||
'subs' => [
|
||||
['name' => '系统权限管理', 'icon' => 'layui-icon layui-icon-vercode', 'node' => 'admin/auth/index'],
|
||||
['name' => '系统用户管理', 'icon' => 'layui-icon layui-icon-username', 'node' => 'admin/user/index'],
|
||||
],
|
||||
],
|
||||
];
|
||||
}
|
||||
}
|
134
app/admin/controller/Auth.php
Normal file
134
app/admin/controller/Auth.php
Normal file
@ -0,0 +1,134 @@
|
||||
<?php
|
||||
|
||||
// +----------------------------------------------------------------------
|
||||
// | Admin Plugin for ThinkAdmin
|
||||
// +----------------------------------------------------------------------
|
||||
// | 版权所有 2014~2024 ThinkAdmin [ thinkadmin.top ]
|
||||
// +----------------------------------------------------------------------
|
||||
// | 官方网站: https://thinkadmin.top
|
||||
// +----------------------------------------------------------------------
|
||||
// | 开源协议 ( https://mit-license.org )
|
||||
// | 免责声明 ( https://thinkadmin.top/disclaimer )
|
||||
// +----------------------------------------------------------------------
|
||||
// | gitee 代码仓库:https://gitee.com/zoujingli/think-plugs-admin
|
||||
// | github 代码仓库:https://github.com/zoujingli/think-plugs-admin
|
||||
// +----------------------------------------------------------------------
|
||||
|
||||
declare(strict_types=1);
|
||||
|
||||
namespace app\admin\controller;
|
||||
|
||||
use think\admin\Controller;
|
||||
use think\admin\helper\QueryHelper;
|
||||
use think\admin\model\SystemAuth;
|
||||
use think\admin\model\SystemNode;
|
||||
use think\admin\Plugin;
|
||||
use think\admin\service\AdminService;
|
||||
|
||||
/**
|
||||
* 系统权限管理
|
||||
* @class Auth
|
||||
* @package app\admin\controller
|
||||
*/
|
||||
class Auth extends Controller
|
||||
{
|
||||
/**
|
||||
* 系统权限管理
|
||||
* @auth true
|
||||
* @menu true
|
||||
* @throws \think\db\exception\DataNotFoundException
|
||||
* @throws \think\db\exception\DbException
|
||||
* @throws \think\db\exception\ModelNotFoundException
|
||||
*/
|
||||
public function index()
|
||||
{
|
||||
SystemAuth::mQuery()->layTable(function () {
|
||||
$this->title = '系统权限管理';
|
||||
}, static function (QueryHelper $query) {
|
||||
$query->like('title,desc')->equal('status,utype')->dateBetween('create_at');
|
||||
});
|
||||
}
|
||||
|
||||
/**
|
||||
* 修改权限状态
|
||||
* @auth true
|
||||
*/
|
||||
public function state()
|
||||
{
|
||||
SystemAuth::mSave($this->_vali([
|
||||
'status.in:0,1' => '状态值范围异常!',
|
||||
'status.require' => '状态值不能为空!',
|
||||
]));
|
||||
}
|
||||
|
||||
/**
|
||||
* 删除系统权限
|
||||
* @auth true
|
||||
*/
|
||||
public function remove()
|
||||
{
|
||||
SystemAuth::mDelete();
|
||||
}
|
||||
|
||||
|
||||
/**
|
||||
* 添加系统权限
|
||||
* @auth true
|
||||
*/
|
||||
public function add()
|
||||
{
|
||||
SystemAuth::mForm('form');
|
||||
}
|
||||
|
||||
/**
|
||||
* 编辑系统权限
|
||||
* @auth true
|
||||
*/
|
||||
public function edit()
|
||||
{
|
||||
SystemAuth::mForm('form');
|
||||
}
|
||||
|
||||
/**
|
||||
* 表单后置数据处理
|
||||
* @param array $data
|
||||
*/
|
||||
protected function _form_filter(array $data)
|
||||
{
|
||||
if ($this->request->isGet()) {
|
||||
$this->title = empty($data['title']) ? "添加访问授权" : "编辑【{$data['title']}】授权";
|
||||
} elseif ($this->request->post('action') === 'json') {
|
||||
if ($this->app->isDebug()) AdminService::clear();
|
||||
$ztree = AdminService::getTree(empty($data['id']) ? [] : SystemNode::mk()->where(['auth' => $data['id']])->column('node'));
|
||||
usort($ztree, static function ($a, $b) {
|
||||
if (explode('-', $a['node'])[0] !== explode('-', $b['node'])[0]) {
|
||||
if (stripos($a['node'], 'plugin-') === 0) return 1;
|
||||
}
|
||||
return $a['node'] === $b['node'] ? 0 : ($a['node'] > $b['node'] ? 1 : -1);
|
||||
});
|
||||
[$ps, $cs] = [Plugin::get(), (array)$this->app->config->get('app.app_names', [])];
|
||||
foreach ($ztree as &$n) $n['title'] = lang($cs[$n['node']] ?? (($ps[$n['node']] ?? [])['name'] ?? $n['title']));
|
||||
$this->success('获取权限节点成功!', $ztree);
|
||||
} elseif (empty($data['nodes'])) {
|
||||
$this->error('未配置功能节点!');
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* 节点更新处理
|
||||
* @param boolean $state
|
||||
* @param array $post
|
||||
* @return void
|
||||
*/
|
||||
protected function _form_result(bool $state, array $post)
|
||||
{
|
||||
if ($state && $this->request->post('action') === 'save') {
|
||||
[$map, $data] = [['auth' => $post['id']], []];
|
||||
foreach ($post['nodes'] ?? [] as $node) $data[] = $map + ['node' => $node];
|
||||
SystemNode::mk()->where($map)->delete();
|
||||
count($data) > 0 && SystemNode::mk()->insertAll($data);
|
||||
sysoplog('系统权限管理', "配置系统权限[{$map['auth']}]授权成功");
|
||||
$this->success('权限修改成功!', 'javascript:history.back()');
|
||||
}
|
||||
}
|
||||
}
|
113
app/admin/controller/Base.php
Normal file
113
app/admin/controller/Base.php
Normal file
@ -0,0 +1,113 @@
|
||||
<?php
|
||||
|
||||
// +----------------------------------------------------------------------
|
||||
// | Admin Plugin for ThinkAdmin
|
||||
// +----------------------------------------------------------------------
|
||||
// | 版权所有 2014~2024 ThinkAdmin [ thinkadmin.top ]
|
||||
// +----------------------------------------------------------------------
|
||||
// | 官方网站: https://thinkadmin.top
|
||||
// +----------------------------------------------------------------------
|
||||
// | 开源协议 ( https://mit-license.org )
|
||||
// | 免责声明 ( https://thinkadmin.top/disclaimer )
|
||||
// +----------------------------------------------------------------------
|
||||
// | gitee 代码仓库:https://gitee.com/zoujingli/think-plugs-admin
|
||||
// | github 代码仓库:https://github.com/zoujingli/think-plugs-admin
|
||||
// +----------------------------------------------------------------------
|
||||
|
||||
declare(strict_types=1);
|
||||
|
||||
namespace app\admin\controller;
|
||||
|
||||
use think\admin\Controller;
|
||||
use think\admin\helper\QueryHelper;
|
||||
use think\admin\model\SystemBase;
|
||||
|
||||
/**
|
||||
* 数据字典管理
|
||||
* @class Base
|
||||
* @package app\admin\controller
|
||||
*/
|
||||
class Base extends Controller
|
||||
{
|
||||
/**
|
||||
* 数据字典管理
|
||||
* @auth true
|
||||
* @menu true
|
||||
* @throws \think\db\exception\DataNotFoundException
|
||||
* @throws \think\db\exception\DbException
|
||||
* @throws \think\db\exception\ModelNotFoundException
|
||||
*/
|
||||
public function index()
|
||||
{
|
||||
SystemBase::mQuery()->layTable(function () {
|
||||
$this->title = '数据字典管理';
|
||||
$this->types = SystemBase::types();
|
||||
$this->type = $this->get['type'] ?? ($this->types[0] ?? '-');
|
||||
}, static function (QueryHelper $query) {
|
||||
$query->where(['deleted' => 0])->equal('type');
|
||||
$query->like('code,name,status')->dateBetween('create_at');
|
||||
});
|
||||
}
|
||||
|
||||
/**
|
||||
* 添加数据字典
|
||||
* @auth true
|
||||
*/
|
||||
public function add()
|
||||
{
|
||||
SystemBase::mForm('form');
|
||||
}
|
||||
|
||||
/**
|
||||
* 编辑数据字典
|
||||
* @auth true
|
||||
*/
|
||||
public function edit()
|
||||
{
|
||||
SystemBase::mForm('form');
|
||||
}
|
||||
|
||||
/**
|
||||
* 表单数据处理
|
||||
* @param array $data
|
||||
* @throws \think\db\exception\DbException
|
||||
*/
|
||||
protected function _form_filter(array &$data)
|
||||
{
|
||||
if ($this->request->isGet()) {
|
||||
$this->types = SystemBase::types();
|
||||
$this->types[] = '--- ' . lang('新增类型') . ' ---';
|
||||
$this->type = $this->get['type'] ?? ($this->types[0] ?? '-');
|
||||
} else {
|
||||
$map = [];
|
||||
$map[] = ['deleted', '=', 0];
|
||||
$map[] = ['code', '=', $data['code']];
|
||||
$map[] = ['type', '=', $data['type']];
|
||||
$map[] = ['id', '<>', $data['id'] ?? 0];
|
||||
if (SystemBase::mk()->where($map)->count() > 0) {
|
||||
$this->error("数据编码已经存在!");
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* 修改数据状态
|
||||
* @auth true
|
||||
*/
|
||||
public function state()
|
||||
{
|
||||
SystemBase::mSave($this->_vali([
|
||||
'status.in:0,1' => '状态值范围异常!',
|
||||
'status.require' => '状态值不能为空!',
|
||||
]));
|
||||
}
|
||||
|
||||
/**
|
||||
* 删除数据记录
|
||||
* @auth true
|
||||
*/
|
||||
public function remove()
|
||||
{
|
||||
SystemBase::mDelete();
|
||||
}
|
||||
}
|
146
app/admin/controller/Config.php
Normal file
146
app/admin/controller/Config.php
Normal file
@ -0,0 +1,146 @@
|
||||
<?php
|
||||
|
||||
// +----------------------------------------------------------------------
|
||||
// | Admin Plugin for ThinkAdmin
|
||||
// +----------------------------------------------------------------------
|
||||
// | 版权所有 2014~2024 ThinkAdmin [ thinkadmin.top ]
|
||||
// +----------------------------------------------------------------------
|
||||
// | 官方网站: https://thinkadmin.top
|
||||
// +----------------------------------------------------------------------
|
||||
// | 开源协议 ( https://mit-license.org )
|
||||
// | 免责声明 ( https://thinkadmin.top/disclaimer )
|
||||
// +----------------------------------------------------------------------
|
||||
// | gitee 代码仓库:https://gitee.com/zoujingli/think-plugs-admin
|
||||
// | github 代码仓库:https://github.com/zoujingli/think-plugs-admin
|
||||
// +----------------------------------------------------------------------
|
||||
|
||||
declare(strict_types=1);
|
||||
|
||||
namespace app\admin\controller;
|
||||
|
||||
use think\admin\Controller;
|
||||
use think\admin\Plugin;
|
||||
use think\admin\service\AdminService;
|
||||
use think\admin\service\ModuleService;
|
||||
use think\admin\service\RuntimeService;
|
||||
use think\admin\service\SystemService;
|
||||
use think\admin\Storage;
|
||||
use think\admin\storage\AliossStorage;
|
||||
use think\admin\storage\QiniuStorage;
|
||||
use think\admin\storage\TxcosStorage;
|
||||
|
||||
/**
|
||||
* 系统参数配置
|
||||
* @class Config
|
||||
* @package app\admin\controller
|
||||
*/
|
||||
class Config extends Controller
|
||||
{
|
||||
const themes = [
|
||||
'default' => '默认色0',
|
||||
'white' => '简约白0',
|
||||
'red-1' => '玫瑰红1',
|
||||
'blue-1' => '深空蓝1',
|
||||
'green-1' => '小草绿1',
|
||||
'black-1' => '经典黑1',
|
||||
'red-2' => '玫瑰红2',
|
||||
'blue-2' => '深空蓝2',
|
||||
'green-2' => '小草绿2',
|
||||
'black-2' => '经典黑2',
|
||||
];
|
||||
|
||||
/**
|
||||
* 系统参数配置
|
||||
* @auth true
|
||||
* @menu true
|
||||
*/
|
||||
public function index()
|
||||
{
|
||||
$this->title = '系统参数配置';
|
||||
$this->files = Storage::types();
|
||||
$this->plugins = Plugin::get(null, true);
|
||||
$this->issuper = AdminService::isSuper();
|
||||
$this->systemid = ModuleService::getRunVar('uni');
|
||||
$this->framework = ModuleService::getLibrarys('topthink/framework');
|
||||
$this->thinkadmin = ModuleService::getLibrarys('zoujingli/think-library');
|
||||
if (AdminService::isSuper() && $this->app->session->get('user.password') === md5('admin')) {
|
||||
$url = url('admin/index/pass', ['id' => AdminService::getUserId()]);
|
||||
$this->showErrorMessage = lang("超级管理员账号的密码未修改,建议立即<a data-modal='%s'>修改密码</a>!", [$url]);
|
||||
}
|
||||
uasort($this->plugins, static function ($a, $b) {
|
||||
if ($a['space'] === $b['space']) return 0;
|
||||
return $a['space'] > $b['space'] ? 1 : -1;
|
||||
});
|
||||
$this->fetch();
|
||||
}
|
||||
|
||||
/**
|
||||
* 修改系统参数
|
||||
* @auth true
|
||||
* @throws \think\admin\Exception
|
||||
*/
|
||||
public function system()
|
||||
{
|
||||
if ($this->request->isGet()) {
|
||||
$this->title = '修改系统参数';
|
||||
$this->themes = static::themes;
|
||||
$this->fetch();
|
||||
} else {
|
||||
$post = $this->request->post();
|
||||
// 修改网站后台入口路径
|
||||
if (!empty($post['xpath'])) {
|
||||
if (!preg_match('/^[a-zA-Z_][a-zA-Z0-9_]*$/', $post['xpath'])) {
|
||||
$this->error('后台入口格式错误!');
|
||||
}
|
||||
if ($post['xpath'] !== 'admin') {
|
||||
if (is_dir(syspath("app/{$post['xpath']}")) || !empty(Plugin::get($post['xpath']))) {
|
||||
$this->error(lang('已存在 %s 应用!', [$post['xpath']]));
|
||||
}
|
||||
}
|
||||
RuntimeService::set(null, [$post['xpath'] => 'admin']);
|
||||
}
|
||||
// 修改网站 ICON 图标,替换 public/favicon.ico
|
||||
if (preg_match('#^https?://#', $post['site_icon'] ?? '')) try {
|
||||
SystemService::setFavicon($post['site_icon'] ?? '');
|
||||
} catch (\Exception $exception) {
|
||||
trace_file($exception);
|
||||
}
|
||||
// 数据数据到系统配置表
|
||||
foreach ($post as $k => $v) sysconf($k, $v);
|
||||
sysoplog('系统配置管理', "修改系统参数成功");
|
||||
$this->success('数据保存成功!', admuri('admin/config/index'));
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* 修改文件存储
|
||||
* @auth true
|
||||
* @throws \think\admin\Exception
|
||||
*/
|
||||
public function storage()
|
||||
{
|
||||
$this->_applyFormToken();
|
||||
if ($this->request->isGet()) {
|
||||
$this->type = input('type', 'local');
|
||||
if ($this->type === 'alioss') {
|
||||
$this->points = AliossStorage::region();
|
||||
} elseif ($this->type === 'qiniu') {
|
||||
$this->points = QiniuStorage::region();
|
||||
} elseif ($this->type === 'txcos') {
|
||||
$this->points = TxcosStorage::region();
|
||||
}
|
||||
$this->fetch("storage-{$this->type}");
|
||||
} else {
|
||||
$post = $this->request->post();
|
||||
if (!empty($post['storage']['allow_exts'])) {
|
||||
$deny = ['sh', 'asp', 'bat', 'cmd', 'exe', 'php'];
|
||||
$exts = array_unique(str2arr(strtolower($post['storage']['allow_exts'])));
|
||||
if (count(array_intersect($deny, $exts)) > 0) $this->error('禁止上传可执行的文件!');
|
||||
$post['storage']['allow_exts'] = join(',', $exts);
|
||||
}
|
||||
foreach ($post as $name => $value) sysconf($name, $value);
|
||||
sysoplog('系统配置管理', "修改系统存储参数");
|
||||
$this->success('修改文件存储成功!');
|
||||
}
|
||||
}
|
||||
}
|
116
app/admin/controller/File.php
Normal file
116
app/admin/controller/File.php
Normal file
@ -0,0 +1,116 @@
|
||||
<?php
|
||||
|
||||
// +----------------------------------------------------------------------
|
||||
// | Admin Plugin for ThinkAdmin
|
||||
// +----------------------------------------------------------------------
|
||||
// | 版权所有 2014~2024 ThinkAdmin [ thinkadmin.top ]
|
||||
// +----------------------------------------------------------------------
|
||||
// | 官方网站: https://thinkadmin.top
|
||||
// +----------------------------------------------------------------------
|
||||
// | 开源协议 ( https://mit-license.org )
|
||||
// | 免责声明 ( https://thinkadmin.top/disclaimer )
|
||||
// +----------------------------------------------------------------------
|
||||
// | gitee 代码仓库:https://gitee.com/zoujingli/think-plugs-admin
|
||||
// | github 代码仓库:https://github.com/zoujingli/think-plugs-admin
|
||||
// +----------------------------------------------------------------------
|
||||
|
||||
declare(strict_types=1);
|
||||
|
||||
namespace app\admin\controller;
|
||||
|
||||
use think\admin\Controller;
|
||||
use think\admin\helper\QueryHelper;
|
||||
use think\admin\model\SystemFile;
|
||||
use think\admin\service\AdminService;
|
||||
use think\admin\Storage;
|
||||
|
||||
/**
|
||||
* 系统文件管理
|
||||
* @class File
|
||||
* @package app\admin\controller
|
||||
*/
|
||||
class File extends Controller
|
||||
{
|
||||
/**
|
||||
* 存储类型
|
||||
* @var array
|
||||
*/
|
||||
protected $types;
|
||||
|
||||
/**
|
||||
* 控制器初始化
|
||||
* @return void
|
||||
*/
|
||||
protected function initialize()
|
||||
{
|
||||
$this->types = Storage::types();
|
||||
}
|
||||
|
||||
/**
|
||||
* 系统文件管理
|
||||
* @auth true
|
||||
* @menu true
|
||||
* @throws \think\db\exception\DataNotFoundException
|
||||
* @throws \think\db\exception\DbException
|
||||
* @throws \think\db\exception\ModelNotFoundException
|
||||
*/
|
||||
public function index()
|
||||
{
|
||||
SystemFile::mQuery()->layTable(function () {
|
||||
$this->title = '系统文件管理';
|
||||
$this->xexts = SystemFile::mk()->distinct()->column('xext');
|
||||
}, static function (QueryHelper $query) {
|
||||
$query->like('name,hash,xext')->equal('type')->dateBetween('create_at');
|
||||
$query->where(['issafe' => 0, 'status' => 2, 'uuid' => AdminService::getUserId()]);
|
||||
});
|
||||
}
|
||||
|
||||
/**
|
||||
* 数据列表处理
|
||||
* @param array $data
|
||||
* @return void
|
||||
*/
|
||||
protected function _page_filter(array &$data)
|
||||
{
|
||||
foreach ($data as &$vo) {
|
||||
$vo['ctype'] = $this->types[$vo['type']] ?? $vo['type'];
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* 编辑系统文件
|
||||
* @auth true
|
||||
* @return void
|
||||
*/
|
||||
public function edit()
|
||||
{
|
||||
SystemFile::mForm('form');
|
||||
}
|
||||
|
||||
/**
|
||||
* 删除系统文件
|
||||
* @auth true
|
||||
* @return void
|
||||
*/
|
||||
public function remove()
|
||||
{
|
||||
if (!AdminService::isSuper()) {
|
||||
$where = ['uuid' => AdminService::getUserId()];
|
||||
}
|
||||
SystemFile::mDelete('', $where ?? []);
|
||||
}
|
||||
|
||||
/**
|
||||
* 清理重复文件
|
||||
* @auth true
|
||||
* @return void
|
||||
* @throws \think\db\exception\DbException
|
||||
*/
|
||||
public function distinct()
|
||||
{
|
||||
$map = ['issafe' => 0, 'uuid' => AdminService::getUserId()];
|
||||
$subQuery = SystemFile::mk()->fieldRaw('MAX(id) AS id')->where($map)->group('type, xkey')->buildSql();
|
||||
SystemFile::mk()->where($map)->whereRaw("id NOT IN ({$subQuery})")->delete();
|
||||
$this->success('清理重复文件成功!');
|
||||
}
|
||||
}
|
155
app/admin/controller/Index.php
Normal file
155
app/admin/controller/Index.php
Normal file
@ -0,0 +1,155 @@
|
||||
<?php
|
||||
|
||||
// +----------------------------------------------------------------------
|
||||
// | Admin Plugin for ThinkAdmin
|
||||
// +----------------------------------------------------------------------
|
||||
// | 版权所有 2014~2024 ThinkAdmin [ thinkadmin.top ]
|
||||
// +----------------------------------------------------------------------
|
||||
// | 官方网站: https://thinkadmin.top
|
||||
// +----------------------------------------------------------------------
|
||||
// | 开源协议 ( https://mit-license.org )
|
||||
// | 免责声明 ( https://thinkadmin.top/disclaimer )
|
||||
// +----------------------------------------------------------------------
|
||||
// | gitee 代码仓库:https://gitee.com/zoujingli/think-plugs-admin
|
||||
// | github 代码仓库:https://github.com/zoujingli/think-plugs-admin
|
||||
// +----------------------------------------------------------------------
|
||||
|
||||
namespace app\admin\controller;
|
||||
|
||||
use think\admin\Controller;
|
||||
use think\admin\model\SystemUser;
|
||||
use think\admin\service\AdminService;
|
||||
use think\admin\service\MenuService;
|
||||
|
||||
/**
|
||||
* 后台界面入口
|
||||
* @class Index
|
||||
* @package app\admin\controller
|
||||
*/
|
||||
class Index extends Controller
|
||||
{
|
||||
/**
|
||||
* 显示后台首页
|
||||
* @throws \think\admin\Exception
|
||||
* @throws \think\db\exception\DataNotFoundException
|
||||
* @throws \think\db\exception\DbException
|
||||
* @throws \think\db\exception\ModelNotFoundException
|
||||
*/
|
||||
public function index()
|
||||
{
|
||||
/*! 根据运行模式刷新权限 */
|
||||
AdminService::apply($this->app->isDebug());
|
||||
/*! 读取当前用户权限菜单树 */
|
||||
$this->menus = MenuService::getTree();
|
||||
/*! 判断当前用户的登录状态 */
|
||||
$this->login = AdminService::isLogin();
|
||||
/*! 菜单为空且未登录跳转到登录页 */
|
||||
if (empty($this->menus) && empty($this->login)) {
|
||||
$this->redirect(sysuri('admin/login/index'));
|
||||
} else {
|
||||
$this->title = '系统管理后台';
|
||||
$this->super = AdminService::isSuper();
|
||||
$this->theme = AdminService::getUserTheme();
|
||||
$this->fetch();
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* 后台主题切换
|
||||
* @login true
|
||||
* @return void
|
||||
* @throws \think\admin\Exception
|
||||
*/
|
||||
public function theme()
|
||||
{
|
||||
if ($this->request->isGet()) {
|
||||
$this->theme = AdminService::getUserTheme();
|
||||
$this->themes = Config::themes;
|
||||
$this->fetch();
|
||||
} else {
|
||||
$data = $this->_vali(['site_theme.require' => '主题名称不能为空!']);
|
||||
if (AdminService::setUserTheme($data['site_theme'])) {
|
||||
$this->success('主题配置保存成功!');
|
||||
} else {
|
||||
$this->error('主题配置保存失败!');
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* 修改用户资料
|
||||
* @login true
|
||||
*/
|
||||
public function info()
|
||||
{
|
||||
$id = $this->request->param('id');
|
||||
if (AdminService::getUserId() == intval($id)) {
|
||||
SystemUser::mForm('user/form', 'id', [], ['id' => $id]);
|
||||
} else {
|
||||
$this->error('只能修改自己的资料!');
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* 资料修改表单处理
|
||||
* @param array $data
|
||||
*/
|
||||
protected function _info_form_filter(array &$data)
|
||||
{
|
||||
if ($this->request->isPost()) {
|
||||
unset($data['username'], $data['authorize']);
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* 资料修改结果处理
|
||||
* @param bool $status
|
||||
*/
|
||||
protected function _info_form_result(bool $status)
|
||||
{
|
||||
if ($status) {
|
||||
$this->success('用户资料修改成功!', 'javascript:location.reload()');
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* 修改当前用户密码
|
||||
* @login true
|
||||
* @throws \think\db\exception\DataNotFoundException
|
||||
* @throws \think\db\exception\DbException
|
||||
* @throws \think\db\exception\ModelNotFoundException
|
||||
*/
|
||||
public function pass()
|
||||
{
|
||||
$id = $this->request->param('id');
|
||||
if (AdminService::getUserId() !== intval($id)) {
|
||||
$this->error('禁止修改他人密码!');
|
||||
}
|
||||
if ($this->app->request->isGet()) {
|
||||
$this->verify = true;
|
||||
SystemUser::mForm('user/pass', 'id', [], ['id' => $id]);
|
||||
} else {
|
||||
$data = $this->_vali([
|
||||
'password.require' => '登录密码不能为空!',
|
||||
'repassword.require' => '重复密码不能为空!',
|
||||
'oldpassword.require' => '旧的密码不能为空!',
|
||||
'password.confirm:repassword' => '两次输入的密码不一致!',
|
||||
]);
|
||||
$user = SystemUser::mk()->find($id);
|
||||
if (empty($user)) $this->error('用户不存在!');
|
||||
if (md5($data['oldpassword']) !== $user['password']) {
|
||||
$this->error('旧密码验证失败,请重新输入!');
|
||||
}
|
||||
if ($user->save(['password' => md5($data['password'])])) {
|
||||
sysoplog('系统用户管理', "修改用户[{$user['id']}]密码成功");
|
||||
// 修改密码同步事件处理
|
||||
$this->app->event->trigger('PluginAdminChangePassword', [
|
||||
'uuid' => intval($user['id']), 'pass' => $data['password']
|
||||
]);
|
||||
$this->success('密码修改成功,下次请使用新密码登录!', '');
|
||||
} else {
|
||||
$this->error('密码修改失败,请稍候再试!');
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
137
app/admin/controller/Login.php
Normal file
137
app/admin/controller/Login.php
Normal file
@ -0,0 +1,137 @@
|
||||
<?php
|
||||
|
||||
// +----------------------------------------------------------------------
|
||||
// | Admin Plugin for ThinkAdmin
|
||||
// +----------------------------------------------------------------------
|
||||
// | 版权所有 2014~2024 ThinkAdmin [ thinkadmin.top ]
|
||||
// +----------------------------------------------------------------------
|
||||
// | 官方网站: https://thinkadmin.top
|
||||
// +----------------------------------------------------------------------
|
||||
// | 开源协议 ( https://mit-license.org )
|
||||
// | 免责声明 ( https://thinkadmin.top/disclaimer )
|
||||
// +----------------------------------------------------------------------
|
||||
// | gitee 代码仓库:https://gitee.com/zoujingli/think-plugs-admin
|
||||
// | github 代码仓库:https://github.com/zoujingli/think-plugs-admin
|
||||
// +----------------------------------------------------------------------
|
||||
|
||||
declare(strict_types=1);
|
||||
|
||||
namespace app\admin\controller;
|
||||
|
||||
use think\admin\Controller;
|
||||
use think\admin\extend\CodeExtend;
|
||||
use think\admin\model\SystemUser;
|
||||
use think\admin\service\AdminService;
|
||||
use think\admin\service\CaptchaService;
|
||||
use think\admin\service\RuntimeService;
|
||||
use think\admin\service\SystemService;
|
||||
|
||||
/**
|
||||
* 用户登录管理
|
||||
* @class Login
|
||||
* @package app\admin\controller
|
||||
*/
|
||||
class Login extends Controller
|
||||
{
|
||||
|
||||
/**
|
||||
* 后台登录入口
|
||||
* @return void
|
||||
* @throws \think\admin\Exception
|
||||
*/
|
||||
public function index()
|
||||
{
|
||||
if ($this->app->request->isGet()) {
|
||||
if (AdminService::isLogin()) {
|
||||
$this->redirect(sysuri('admin/index/index'));
|
||||
} else {
|
||||
// 加载登录模板
|
||||
$this->title = '系统登录';
|
||||
// 登录验证令牌
|
||||
$this->captchaType = 'LoginCaptcha';
|
||||
$this->captchaToken = CodeExtend::uuid();
|
||||
// 当前运行模式
|
||||
$this->runtimeMode = RuntimeService::check();
|
||||
// 后台背景处理
|
||||
$images = str2arr(sysconf('login_image|raw') ?: '', '|');
|
||||
if (empty($images)) $images = [
|
||||
SystemService::uri('/static/theme/img/login/bg1.jpg'),
|
||||
SystemService::uri('/static/theme/img/login/bg2.jpg'),
|
||||
];
|
||||
$this->loginStyle = sprintf('style="background-image:url(%s)" data-bg-transition="%s"', $images[0], join(',', $images));
|
||||
// 更新后台主域名,用于部分无法获取域名的场景调用
|
||||
if ($this->request->domain() !== sysconf('base.site_host|raw')) {
|
||||
sysconf('base.site_host', $this->request->domain());
|
||||
}
|
||||
$this->fetch();
|
||||
}
|
||||
} else {
|
||||
$data = $this->_vali([
|
||||
'username.require' => '登录账号不能为空!',
|
||||
'username.min:4' => '账号不能少于4位字符!',
|
||||
'password.require' => '登录密码不能为空!',
|
||||
'password.min:4' => '密码不能少于4位字符!',
|
||||
'verify.require' => '图形验证码不能为空!',
|
||||
'uniqid.require' => '图形验证标识不能为空!',
|
||||
]);
|
||||
if (!CaptchaService::instance()->check($data['verify'], $data['uniqid'])) {
|
||||
$this->error('图形验证码验证失败,请重新输入!');
|
||||
}
|
||||
/*! 用户信息验证 */
|
||||
$map = ['username' => $data['username'], 'is_deleted' => 0];
|
||||
$user = SystemUser::mk()->where($map)->findOrEmpty();
|
||||
if ($user->isEmpty()) {
|
||||
$this->app->session->set('LoginInputSessionError', true);
|
||||
$this->error('登录账号或密码错误,请重新输入!');
|
||||
}
|
||||
if (empty($user['status'])) {
|
||||
$this->app->session->set('LoginInputSessionError', true);
|
||||
$this->error('账号已经被禁用,请联系管理员!');
|
||||
}
|
||||
if (md5("{$user['password']}{$data['uniqid']}") !== $data['password']) {
|
||||
$this->app->session->set('LoginInputSessionError', true);
|
||||
$this->error('登录账号或密码错误,请重新输入!');
|
||||
}
|
||||
$user->hidden(['sort', 'status', 'password', 'is_deleted']);
|
||||
$this->app->session->set('user', $user->toArray());
|
||||
$this->app->session->delete('LoginInputSessionError');
|
||||
// 更新登录次数
|
||||
$user->where(['id' => $user->getAttr('id')])->inc('login_num')->update([
|
||||
'login_at' => date('Y-m-d H:i:s'), 'login_ip' => $this->app->request->ip(),
|
||||
]);
|
||||
// 刷新用户权限
|
||||
AdminService::apply(true);
|
||||
sysoplog('系统用户登录', '登录系统后台成功');
|
||||
$this->success('登录成功', sysuri('admin/index/index'));
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* 生成验证码
|
||||
* @return void
|
||||
*/
|
||||
public function captcha()
|
||||
{
|
||||
$input = $this->_vali([
|
||||
'type.require' => '类型不能为空!',
|
||||
'token.require' => '标识不能为空!',
|
||||
]);
|
||||
$image = CaptchaService::instance()->initialize();
|
||||
$captcha = ['image' => $image->getData(), 'uniqid' => $image->getUniqid()];
|
||||
// 未发生异常时,直接返回验证码内容
|
||||
if (!$this->app->session->get('LoginInputSessionError')) {
|
||||
$captcha['code'] = $image->getCode();
|
||||
}
|
||||
$this->success('生成验证码成功', $captcha);
|
||||
}
|
||||
|
||||
/**
|
||||
* 退出登录
|
||||
* @return void
|
||||
*/
|
||||
public function out()
|
||||
{
|
||||
$this->app->session->destroy();
|
||||
$this->success('退出登录成功!', sysuri('admin/login/index'));
|
||||
}
|
||||
}
|
161
app/admin/controller/Menu.php
Normal file
161
app/admin/controller/Menu.php
Normal file
@ -0,0 +1,161 @@
|
||||
<?php
|
||||
|
||||
// +----------------------------------------------------------------------
|
||||
// | Admin Plugin for ThinkAdmin
|
||||
// +----------------------------------------------------------------------
|
||||
// | 版权所有 2014~2024 ThinkAdmin [ thinkadmin.top ]
|
||||
// +----------------------------------------------------------------------
|
||||
// | 官方网站: https://thinkadmin.top
|
||||
// +----------------------------------------------------------------------
|
||||
// | 开源协议 ( https://mit-license.org )
|
||||
// | 免责声明 ( https://thinkadmin.top/disclaimer )
|
||||
// +----------------------------------------------------------------------
|
||||
// | gitee 代码仓库:https://gitee.com/zoujingli/think-plugs-admin
|
||||
// | github 代码仓库:https://github.com/zoujingli/think-plugs-admin
|
||||
// +----------------------------------------------------------------------
|
||||
|
||||
declare(strict_types=1);
|
||||
|
||||
namespace app\admin\controller;
|
||||
|
||||
use think\admin\Controller;
|
||||
use think\admin\extend\DataExtend;
|
||||
use think\admin\model\SystemMenu;
|
||||
use think\admin\service\AdminService;
|
||||
use think\admin\service\MenuService;
|
||||
use think\admin\service\NodeService;
|
||||
|
||||
/**
|
||||
* 系统菜单管理
|
||||
* @class Menu
|
||||
* @package app\admin\controller
|
||||
*/
|
||||
class Menu extends Controller
|
||||
{
|
||||
/**
|
||||
* 系统菜单管理
|
||||
* @auth true
|
||||
* @menu true
|
||||
* @throws \think\db\exception\DataNotFoundException
|
||||
* @throws \think\db\exception\DbException
|
||||
* @throws \think\db\exception\ModelNotFoundException
|
||||
*/
|
||||
public function index()
|
||||
{
|
||||
$this->title = '系统菜单管理';
|
||||
$this->type = $this->get['type'] ?? 'index';
|
||||
// 获取顶级菜单ID
|
||||
$this->pid = $this->get['pid'] ?? '';
|
||||
|
||||
// 查询顶级菜单集合
|
||||
$this->menupList = SystemMenu::mk()->where(['pid' => 0, 'status' => 1])->order('sort desc,id asc')->column('id,pid,title', 'id');
|
||||
|
||||
SystemMenu::mQuery()->layTable();
|
||||
}
|
||||
|
||||
/**
|
||||
* 列表数据处理
|
||||
* @param array $data
|
||||
*/
|
||||
protected function _index_page_filter(array &$data)
|
||||
{
|
||||
$data = DataExtend::arr2tree($data);
|
||||
// 回收站过滤有效菜单
|
||||
if ($this->type === 'recycle') foreach ($data as $k1 => &$p1) {
|
||||
if (!empty($p1['sub'])) foreach ($p1['sub'] as $k2 => &$p2) {
|
||||
if (!empty($p2['sub'])) foreach ($p2['sub'] as $k3 => $p3) {
|
||||
if ($p3['status'] > 0) unset($p2['sub'][$k3]);
|
||||
}
|
||||
if (empty($p2['sub']) && ($p2['url'] === '#' or $p2['status'] > 0)) unset($p1['sub'][$k2]);
|
||||
}
|
||||
if (empty($p1['sub']) && ($p1['url'] === '#' or $p1['status'] > 0)) unset($data[$k1]);
|
||||
}
|
||||
// 菜单数据树数据变平化
|
||||
$data = DataExtend::arr2table($data);
|
||||
|
||||
// 过滤非当前顶级菜单的下级菜单,并重新索引数组
|
||||
if ($this->type === 'index' && $this->pid) {
|
||||
$data = array_values(array_filter($data, function ($item) {
|
||||
return strpos($item['spp'], ",{$this->pid},") !== false;
|
||||
}));
|
||||
}
|
||||
|
||||
foreach ($data as &$vo) {
|
||||
if ($vo['url'] !== '#' && !preg_match('/^(https?:)?(\/\/|\\\\)/i', $vo['url'])) {
|
||||
$vo['url'] = trim(url($vo['url']) . ($vo['params'] ? "?{$vo['params']}" : ''), '\\/');
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* 添加系统菜单
|
||||
* @auth true
|
||||
*/
|
||||
public function add()
|
||||
{
|
||||
$this->_applyFormToken();
|
||||
SystemMenu::mForm('form');
|
||||
}
|
||||
|
||||
/**
|
||||
* 编辑系统菜单
|
||||
* @auth true
|
||||
*/
|
||||
public function edit()
|
||||
{
|
||||
$this->_applyFormToken();
|
||||
SystemMenu::mForm('form');
|
||||
}
|
||||
|
||||
/**
|
||||
* 表单数据处理
|
||||
* @param array $vo
|
||||
*/
|
||||
protected function _form_filter(array &$vo)
|
||||
{
|
||||
if ($this->request->isGet()) {
|
||||
$debug = $this->app->isDebug();
|
||||
/* 清理权限节点 */
|
||||
$debug && AdminService::clear();
|
||||
/* 读取系统功能节点 */
|
||||
$this->auths = [];
|
||||
$this->nodes = MenuService::getList($debug);
|
||||
foreach (NodeService::getMethods($debug) as $node => $item) {
|
||||
if ($item['isauth'] && substr_count($node, '/') >= 2) {
|
||||
$this->auths[] = ['node' => $node, 'title' => $item['title']];
|
||||
}
|
||||
}
|
||||
/* 选择自己上级菜单 */
|
||||
$vo['pid'] = $vo['pid'] ?? input('pid', '0');
|
||||
/* 列出可选上级菜单 */
|
||||
$menus = SystemMenu::mk()->order('sort desc,id asc')->column('id,pid,icon,url,node,title,params', 'id');
|
||||
$this->menus = DataExtend::arr2table(array_merge($menus, [['id' => '0', 'pid' => '-1', 'url' => '#', 'title' => '顶部菜单']]));
|
||||
if (isset($vo['id'])) foreach ($this->menus as $menu) if ($menu['id'] === $vo['id']) $vo = $menu;
|
||||
foreach ($this->menus as $key => $menu) if ($menu['spt'] >= 3 || $menu['url'] !== '#') unset($this->menus[$key]);
|
||||
if (isset($vo['spt']) && isset($vo['spc']) && in_array($vo['spt'], [1, 2]) && $vo['spc'] > 0) {
|
||||
foreach ($this->menus as $key => $menu) if ($vo['spt'] <= $menu['spt']) unset($this->menus[$key]);
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* 修改菜单状态
|
||||
* @auth true
|
||||
*/
|
||||
public function state()
|
||||
{
|
||||
SystemMenu::mSave($this->_vali([
|
||||
'status.in:0,1' => '状态值范围异常!',
|
||||
'status.require' => '状态值不能为空!',
|
||||
]));
|
||||
}
|
||||
|
||||
/**
|
||||
* 删除系统菜单
|
||||
* @auth true
|
||||
*/
|
||||
public function remove()
|
||||
{
|
||||
SystemMenu::mDelete();
|
||||
}
|
||||
}
|
95
app/admin/controller/Oplog.php
Normal file
95
app/admin/controller/Oplog.php
Normal file
@ -0,0 +1,95 @@
|
||||
<?php
|
||||
|
||||
// +----------------------------------------------------------------------
|
||||
// | Admin Plugin for ThinkAdmin
|
||||
// +----------------------------------------------------------------------
|
||||
// | 版权所有 2014~2024 ThinkAdmin [ thinkadmin.top ]
|
||||
// +----------------------------------------------------------------------
|
||||
// | 官方网站: https://thinkadmin.top
|
||||
// +----------------------------------------------------------------------
|
||||
// | 开源协议 ( https://mit-license.org )
|
||||
// | 免责声明 ( https://thinkadmin.top/disclaimer )
|
||||
// +----------------------------------------------------------------------
|
||||
// | gitee 代码仓库:https://gitee.com/zoujingli/think-plugs-admin
|
||||
// | github 代码仓库:https://github.com/zoujingli/think-plugs-admin
|
||||
// +----------------------------------------------------------------------
|
||||
|
||||
declare(strict_types=1);
|
||||
|
||||
namespace app\admin\controller;
|
||||
|
||||
use Ip2Region;
|
||||
use think\admin\Controller;
|
||||
use think\admin\helper\QueryHelper;
|
||||
use think\admin\model\SystemOplog;
|
||||
use think\exception\HttpResponseException;
|
||||
|
||||
/**
|
||||
* 系统日志管理
|
||||
* @class Oplog
|
||||
* @package app\admin\controller
|
||||
*/
|
||||
class Oplog extends Controller
|
||||
{
|
||||
/**
|
||||
* 系统日志管理
|
||||
* @auth true
|
||||
* @menu true
|
||||
* @throws \think\db\exception\DataNotFoundException
|
||||
* @throws \think\db\exception\DbException
|
||||
* @throws \think\db\exception\ModelNotFoundException
|
||||
*/
|
||||
public function index()
|
||||
{
|
||||
SystemOplog::mQuery()->layTable(function () {
|
||||
$this->title = '系统日志管理';
|
||||
$columns = SystemOplog::mk()->column('action,username', 'id');
|
||||
$this->users = array_unique(array_column($columns, 'username'));
|
||||
$this->actions = array_unique(array_column($columns, 'action'));
|
||||
}, static function (QueryHelper $query) {
|
||||
$query->dateBetween('create_at')->equal('username,action')->like('content,geoip,node');
|
||||
});
|
||||
}
|
||||
|
||||
/**
|
||||
* 列表数据处理
|
||||
* @param array $data
|
||||
* @throws \Exception
|
||||
*/
|
||||
protected function _index_page_filter(array &$data)
|
||||
{
|
||||
$region = new Ip2Region();
|
||||
foreach ($data as &$vo) try {
|
||||
$vo['geoisp'] = $region->simple($vo['geoip']);
|
||||
} catch (\Exception $exception) {
|
||||
$vo['geoip'] = $exception->getMessage();
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* 清理系统日志
|
||||
* @auth true
|
||||
*/
|
||||
public function clear()
|
||||
{
|
||||
try {
|
||||
SystemOplog::mQuery()->empty();
|
||||
sysoplog('系统运维管理', '成功清理所有日志');
|
||||
$this->success('日志清理成功!');
|
||||
} catch (HttpResponseException $exception) {
|
||||
throw $exception;
|
||||
} catch (\Exception $exception) {
|
||||
trace_file($exception);
|
||||
$this->error(lang("日志清理失败,%s", [$exception->getMessage()]));
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* 删除系统日志
|
||||
* @auth true
|
||||
*/
|
||||
public function remove()
|
||||
{
|
||||
SystemOplog::mDelete();
|
||||
}
|
||||
}
|
117
app/admin/controller/Queue.php
Normal file
117
app/admin/controller/Queue.php
Normal file
@ -0,0 +1,117 @@
|
||||
<?php
|
||||
|
||||
// +----------------------------------------------------------------------
|
||||
// | Admin Plugin for ThinkAdmin
|
||||
// +----------------------------------------------------------------------
|
||||
// | 版权所有 2014~2024 ThinkAdmin [ thinkadmin.top ]
|
||||
// +----------------------------------------------------------------------
|
||||
// | 官方网站: https://thinkadmin.top
|
||||
// +----------------------------------------------------------------------
|
||||
// | 开源协议 ( https://mit-license.org )
|
||||
// | 免责声明 ( https://thinkadmin.top/disclaimer )
|
||||
// +----------------------------------------------------------------------
|
||||
// | gitee 代码仓库:https://gitee.com/zoujingli/think-plugs-admin
|
||||
// | github 代码仓库:https://github.com/zoujingli/think-plugs-admin
|
||||
// +----------------------------------------------------------------------
|
||||
|
||||
declare(strict_types=1);
|
||||
|
||||
namespace app\admin\controller;
|
||||
|
||||
use think\admin\Controller;
|
||||
use think\admin\helper\QueryHelper;
|
||||
use think\admin\model\SystemQueue;
|
||||
use think\admin\service\AdminService;
|
||||
use think\admin\service\ProcessService;
|
||||
use think\admin\service\QueueService;
|
||||
use think\exception\HttpResponseException;
|
||||
|
||||
/**
|
||||
* 系统任务管理
|
||||
* @class Queue
|
||||
* @package app\admin\controller
|
||||
*/
|
||||
class Queue extends Controller
|
||||
{
|
||||
/**
|
||||
* 系统任务管理
|
||||
* @auth true
|
||||
* @menu true
|
||||
* @throws \think\db\exception\DataNotFoundException
|
||||
* @throws \think\db\exception\DbException
|
||||
* @throws \think\db\exception\ModelNotFoundException
|
||||
*/
|
||||
public function index()
|
||||
{
|
||||
SystemQueue::mQuery()->layTable(function () {
|
||||
$this->title = '系统任务管理';
|
||||
$this->iswin = ProcessService::iswin();
|
||||
if ($this->super = AdminService::isSuper()) {
|
||||
$this->command = ProcessService::think('xadmin:queue start');
|
||||
if (!$this->iswin && !empty($_SERVER['USER'])) {
|
||||
$this->command = "sudo -u {$_SERVER['USER']} {$this->command}";
|
||||
}
|
||||
}
|
||||
}, static function (QueryHelper $query) {
|
||||
$query->equal('status')->like('code|title#title,command');
|
||||
$query->timeBetween('enter_time,exec_time')->dateBetween('create_at');
|
||||
});
|
||||
}
|
||||
|
||||
/**
|
||||
* 分页数据回调处理
|
||||
* @param array $data
|
||||
* @param array $result
|
||||
* @return void
|
||||
* @throws \think\db\exception\DataNotFoundException
|
||||
* @throws \think\db\exception\DbException
|
||||
* @throws \think\db\exception\ModelNotFoundException
|
||||
*/
|
||||
protected function _index_page_filter(array $data, array &$result)
|
||||
{
|
||||
$result['extra'] = ['dos' => 0, 'pre' => 0, 'oks' => 0, 'ers' => 0];
|
||||
SystemQueue::mk()->field('status,count(1) count')->group('status')->select()->map(static function ($item) use (&$result) {
|
||||
if (intval($item['status']) === 1) $result['extra']['pre'] = $item['count'];
|
||||
if (intval($item['status']) === 2) $result['extra']['dos'] = $item['count'];
|
||||
if (intval($item['status']) === 3) $result['extra']['oks'] = $item['count'];
|
||||
if (intval($item['status']) === 4) $result['extra']['ers'] = $item['count'];
|
||||
});
|
||||
}
|
||||
|
||||
/**
|
||||
* 重启系统任务
|
||||
* @auth true
|
||||
*/
|
||||
public function redo()
|
||||
{
|
||||
try {
|
||||
$data = $this->_vali(['code.require' => '任务编号不能为空!']);
|
||||
$queue = QueueService::instance()->initialize($data['code'])->reset();
|
||||
$queue->progress(1, '>>> 任务重置成功 <<<', '0.00');
|
||||
$this->success('任务重置成功!', $queue->code);
|
||||
} catch (HttpResponseException $exception) {
|
||||
throw $exception;
|
||||
} catch (\Exception $exception) {
|
||||
trace_file($exception);
|
||||
$this->error($exception->getMessage());
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* 清理运行数据
|
||||
* @auth true
|
||||
*/
|
||||
public function clean()
|
||||
{
|
||||
$this->_queue('定时清理系统运行数据', "xadmin:queue clean", 0, [], 0, 3600);
|
||||
}
|
||||
|
||||
/**
|
||||
* 删除系统任务
|
||||
* @auth true
|
||||
*/
|
||||
public function remove()
|
||||
{
|
||||
SystemQueue::mDelete();
|
||||
}
|
||||
}
|
182
app/admin/controller/User.php
Normal file
182
app/admin/controller/User.php
Normal file
@ -0,0 +1,182 @@
|
||||
<?php
|
||||
|
||||
// +----------------------------------------------------------------------
|
||||
// | Admin Plugin for ThinkAdmin
|
||||
// +----------------------------------------------------------------------
|
||||
// | 版权所有 2014~2024 ThinkAdmin [ thinkadmin.top ]
|
||||
// +----------------------------------------------------------------------
|
||||
// | 官方网站: https://thinkadmin.top
|
||||
// +----------------------------------------------------------------------
|
||||
// | 开源协议 ( https://mit-license.org )
|
||||
// | 免责声明 ( https://thinkadmin.top/disclaimer )
|
||||
// +----------------------------------------------------------------------
|
||||
// | gitee 代码仓库:https://gitee.com/zoujingli/think-plugs-admin
|
||||
// | github 代码仓库:https://github.com/zoujingli/think-plugs-admin
|
||||
// +----------------------------------------------------------------------
|
||||
|
||||
namespace app\admin\controller;
|
||||
|
||||
use think\admin\Controller;
|
||||
use think\admin\helper\QueryHelper;
|
||||
use think\admin\model\SystemAuth;
|
||||
use think\admin\model\SystemBase;
|
||||
use think\admin\model\SystemUser;
|
||||
use think\admin\service\AdminService;
|
||||
|
||||
/**
|
||||
* 系统用户管理
|
||||
* @class User
|
||||
* @package app\admin\controller
|
||||
*/
|
||||
class User extends Controller
|
||||
{
|
||||
/**
|
||||
* 系统用户管理
|
||||
* @auth true
|
||||
* @menu true
|
||||
* @throws \think\db\exception\DataNotFoundException
|
||||
* @throws \think\db\exception\DbException
|
||||
* @throws \think\db\exception\ModelNotFoundException
|
||||
*/
|
||||
public function index()
|
||||
{
|
||||
$this->type = $this->get['type'] ?? 'index';
|
||||
SystemUser::mQuery()->layTable(function () {
|
||||
$this->title = '系统用户管理';
|
||||
$this->bases = SystemBase::items('身份权限');
|
||||
}, function (QueryHelper $query) {
|
||||
|
||||
// 加载对应数据列表
|
||||
$query->where(['is_deleted' => 0, 'status' => intval($this->type === 'index')]);
|
||||
|
||||
// 关联用户身份资料
|
||||
/** @var \think\model\Relation|\think\db\Query $query */
|
||||
$query->with(['userinfo' => static function ($query) {
|
||||
$query->field('code,name,content');
|
||||
}]);
|
||||
|
||||
// 数据列表搜索过滤
|
||||
$query->equal('status,usertype')->dateBetween('login_at,create_at');
|
||||
$query->like('username|nickname#username,contact_phone#phone,contact_mail#mail');
|
||||
});
|
||||
}
|
||||
|
||||
/**
|
||||
* 添加系统用户
|
||||
* @auth true
|
||||
*/
|
||||
public function add()
|
||||
{
|
||||
SystemUser::mForm('form');
|
||||
}
|
||||
|
||||
/**
|
||||
* 编辑系统用户
|
||||
* @auth true
|
||||
*/
|
||||
public function edit()
|
||||
{
|
||||
SystemUser::mForm('form');
|
||||
}
|
||||
|
||||
/**
|
||||
* 修改用户密码
|
||||
* @auth true
|
||||
*/
|
||||
public function pass()
|
||||
{
|
||||
$this->_applyFormToken();
|
||||
if ($this->request->isGet()) {
|
||||
$this->verify = false;
|
||||
SystemUser::mForm('pass');
|
||||
} else {
|
||||
$data = $this->_vali([
|
||||
'id.require' => '用户ID不能为空!',
|
||||
'password.require' => '登录密码不能为空!',
|
||||
'repassword.require' => '重复密码不能为空!',
|
||||
'repassword.confirm:password' => '两次输入的密码不一致!',
|
||||
]);
|
||||
$user = SystemUser::mk()->findOrEmpty($data['id']);
|
||||
if ($user->isExists() && $user->save(['password' => md5($data['password'])])) {
|
||||
// 修改密码同步事件处理
|
||||
$this->app->event->trigger('PluginAdminChangePassword', [
|
||||
'uuid' => $data['id'], 'pass' => $data['password']
|
||||
]);
|
||||
sysoplog('系统用户管理', "修改用户[{$data['id']}]密码成功");
|
||||
$this->success('密码修改成功,请使用新密码登录!', '');
|
||||
} else {
|
||||
$this->error('密码修改失败,请稍候再试!');
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* 表单数据处理
|
||||
* @param array $data
|
||||
* @throws \think\db\exception\DataNotFoundException
|
||||
* @throws \think\db\exception\DbException
|
||||
* @throws \think\db\exception\ModelNotFoundException
|
||||
*/
|
||||
protected function _form_filter(array &$data)
|
||||
{
|
||||
if ($this->request->isPost()) {
|
||||
// 检查资料是否完整
|
||||
empty($data['username']) && $this->error('登录账号不能为空!');
|
||||
if ($data['username'] !== AdminService::getSuperName()) {
|
||||
empty($data['authorize']) && $this->error('未配置权限!');
|
||||
}
|
||||
// 处理上传的权限格式
|
||||
$data['authorize'] = arr2str($data['authorize'] ?? []);
|
||||
if (empty($data['id'])) {
|
||||
// 检查账号是否重复
|
||||
$map = ['username' => $data['username'], 'is_deleted' => 0];
|
||||
if (SystemUser::mk()->where($map)->count() > 0) {
|
||||
$this->error("账号已经存在,请使用其它账号!");
|
||||
}
|
||||
// 新添加的用户密码与账号相同
|
||||
$data['password'] = md5($data['username']);
|
||||
} else {
|
||||
unset($data['username']);
|
||||
}
|
||||
} else {
|
||||
// 权限绑定处理
|
||||
$data['authorize'] = str2arr($data['authorize'] ?? '');
|
||||
$this->auths = SystemAuth::items();
|
||||
$this->bases = SystemBase::items('身份权限');
|
||||
$this->super = AdminService::getSuperName();
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* 修改用户状态
|
||||
* @auth true
|
||||
*/
|
||||
public function state()
|
||||
{
|
||||
$this->_checkInput();
|
||||
SystemUser::mSave($this->_vali([
|
||||
'status.in:0,1' => '状态值范围异常!',
|
||||
'status.require' => '状态值不能为空!',
|
||||
]));
|
||||
}
|
||||
|
||||
/**
|
||||
* 删除系统用户
|
||||
* @auth true
|
||||
*/
|
||||
public function remove()
|
||||
{
|
||||
$this->_checkInput();
|
||||
SystemUser::mDelete();
|
||||
}
|
||||
|
||||
/**
|
||||
* 检查输入变量
|
||||
*/
|
||||
private function _checkInput()
|
||||
{
|
||||
if (in_array('10000', str2arr(input('id', '')))) {
|
||||
$this->error('系统超级账号禁止删除!');
|
||||
}
|
||||
}
|
||||
}
|
103
app/admin/controller/api/Plugs.php
Normal file
103
app/admin/controller/api/Plugs.php
Normal file
@ -0,0 +1,103 @@
|
||||
<?php
|
||||
|
||||
// +----------------------------------------------------------------------
|
||||
// | Admin Plugin for ThinkAdmin
|
||||
// +----------------------------------------------------------------------
|
||||
// | 版权所有 2014~2024 ThinkAdmin [ thinkadmin.top ]
|
||||
// +----------------------------------------------------------------------
|
||||
// | 官方网站: https://thinkadmin.top
|
||||
// +----------------------------------------------------------------------
|
||||
// | 开源协议 ( https://mit-license.org )
|
||||
// | 免责声明 ( https://thinkadmin.top/disclaimer )
|
||||
// +----------------------------------------------------------------------
|
||||
// | gitee 代码仓库:https://gitee.com/zoujingli/think-plugs-admin
|
||||
// | github 代码仓库:https://github.com/zoujingli/think-plugs-admin
|
||||
// +----------------------------------------------------------------------
|
||||
|
||||
declare(strict_types=1);
|
||||
|
||||
namespace app\admin\controller\api;
|
||||
|
||||
use think\admin\Controller;
|
||||
use think\admin\service\AdminService;
|
||||
use think\Response;
|
||||
|
||||
/**
|
||||
* 扩展插件管理
|
||||
* @class Plugs
|
||||
* @package app\admin\controller\api
|
||||
*/
|
||||
class Plugs extends Controller
|
||||
{
|
||||
|
||||
/**
|
||||
* 图标选择器
|
||||
* @login true
|
||||
*/
|
||||
public function icon()
|
||||
{
|
||||
$this->title = '图标选择器';
|
||||
// 读取 layui 字体图标
|
||||
if (empty($this->layuiIcons = $this->app->cache->get('LayuiIcons', []))) {
|
||||
$style = file_get_contents(syspath('public/static/plugs/layui/css/layui.css'));
|
||||
if (preg_match_all('#\.(layui-icon-[\w-]+):#', $style, $matches)) {
|
||||
if (count($this->layuiIcons = $matches[1]) > 0) {
|
||||
$this->app->cache->set('LayuiIcons', $this->layuiIcons, 60);
|
||||
}
|
||||
}
|
||||
}
|
||||
// 读取 ThinkAdmin 字体图标
|
||||
if (empty($this->thinkIcons = $this->app->cache->get('ThinkAdminSelfIcons', []))) {
|
||||
$style = file_get_contents(syspath('public/static/theme/css/iconfont.css'));
|
||||
if (preg_match_all('#\.(iconfont-[\w-]+):#', $style, $matches)) {
|
||||
if (count($this->thinkIcons = $matches[1]) > 0) {
|
||||
$this->app->cache->set('ThinkAdminSelfIcons', $this->thinkIcons, 60);
|
||||
}
|
||||
}
|
||||
}
|
||||
// 读取 extra 自定义字体图标
|
||||
if (empty($this->extraIcons = $this->app->cache->get('ThinkAdminExtraIcons', []))) {
|
||||
$extraIconPath = syspath('public/static/extra/icon/iconfont.css');
|
||||
if (file_exists($extraIconPath)) {
|
||||
$style = file_get_contents($extraIconPath);
|
||||
if (preg_match_all('#\.(iconfont-[\w-]+):#', $style, $matches)) {
|
||||
if (count($this->extraIcons = $matches[1]) > 0) {
|
||||
$this->app->cache->set('ThinkAdminExtraIcons', $this->extraIcons, 60);
|
||||
}
|
||||
}
|
||||
}
|
||||
}
|
||||
$this->field = $this->app->request->get('field', 'icon');
|
||||
$this->fetch(dirname(__DIR__, 2) . '/view/api/icon.html');
|
||||
}
|
||||
|
||||
/**
|
||||
* 前端脚本变量
|
||||
* @return \think\Response
|
||||
* @throws \think\admin\Exception
|
||||
*/
|
||||
public function script(): Response
|
||||
{
|
||||
$token = $this->request->get('uptoken', '');
|
||||
$domain = boolval(AdminService::withUploadUnid($token));
|
||||
return response(join("\r\n", [
|
||||
sprintf("window.taDebug = %s;", $this->app->isDebug() ? 'true' : 'false'),
|
||||
sprintf("window.taAdmin = '%s';", sysuri('admin/index/index', [], false, $domain)),
|
||||
sprintf("window.taEditor = '%s';", sysconf('base.editor|raw') ?: 'ckeditor4'),
|
||||
]))->contentType('application/javascript');
|
||||
}
|
||||
|
||||
/**
|
||||
* 优化数据库
|
||||
* @login true
|
||||
*/
|
||||
public function optimize()
|
||||
{
|
||||
if (AdminService::isSuper()) {
|
||||
sysoplog('系统运维管理', '创建数据库优化任务');
|
||||
$this->_queue('优化数据库所有数据表', 'xadmin:database optimize');
|
||||
} else {
|
||||
$this->error('请使用超管账号操作!');
|
||||
}
|
||||
}
|
||||
}
|
118
app/admin/controller/api/Queue.php
Normal file
118
app/admin/controller/api/Queue.php
Normal file
@ -0,0 +1,118 @@
|
||||
<?php
|
||||
|
||||
// +----------------------------------------------------------------------
|
||||
// | Admin Plugin for ThinkAdmin
|
||||
// +----------------------------------------------------------------------
|
||||
// | 版权所有 2014~2024 ThinkAdmin [ thinkadmin.top ]
|
||||
// +----------------------------------------------------------------------
|
||||
// | 官方网站: https://thinkadmin.top
|
||||
// +----------------------------------------------------------------------
|
||||
// | 开源协议 ( https://mit-license.org )
|
||||
// | 免责声明 ( https://thinkadmin.top/disclaimer )
|
||||
// +----------------------------------------------------------------------
|
||||
// | gitee 代码仓库:https://gitee.com/zoujingli/think-plugs-admin
|
||||
// | github 代码仓库:https://github.com/zoujingli/think-plugs-admin
|
||||
// +----------------------------------------------------------------------
|
||||
|
||||
declare(strict_types=1);
|
||||
|
||||
namespace app\admin\controller\api;
|
||||
|
||||
use Psr\Log\NullLogger;
|
||||
use think\admin\Controller;
|
||||
use think\admin\model\SystemQueue;
|
||||
use think\admin\service\AdminService;
|
||||
use think\exception\HttpResponseException;
|
||||
|
||||
/**
|
||||
* 任务监听服务管理
|
||||
* @class Queue
|
||||
* @package app\admin\controller\api
|
||||
*/
|
||||
class Queue extends Controller
|
||||
{
|
||||
/**
|
||||
* 停止监听服务
|
||||
* @login true
|
||||
*/
|
||||
public function stop()
|
||||
{
|
||||
if (AdminService::isSuper()) try {
|
||||
$message = $this->app->console->call('xadmin:queue', ['stop'])->fetch();
|
||||
if (stripos($message, 'sent end signal to process')) {
|
||||
sysoplog('系统运维管理', '尝试停止任务监听服务');
|
||||
$this->success('停止任务监听服务成功!');
|
||||
} elseif (stripos($message, 'processes to stop')) {
|
||||
$this->success('没有找到需要停止的服务!');
|
||||
} else {
|
||||
$this->error(nl2br($message));
|
||||
}
|
||||
} catch (HttpResponseException $exception) {
|
||||
throw $exception;
|
||||
} catch (\Exception $exception) {
|
||||
trace_file($exception);
|
||||
$this->error($exception->getMessage());
|
||||
} else {
|
||||
$this->error('请使用超管账号操作!');
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* 启动监听服务
|
||||
* @login true
|
||||
*/
|
||||
public function start()
|
||||
{
|
||||
if (AdminService::isSuper()) try {
|
||||
$message = $this->app->console->call('xadmin:queue', ['start'])->fetch();
|
||||
if (stripos($message, 'daemons started successfully for pid')) {
|
||||
sysoplog('系统运维管理', '尝试启动任务监听服务');
|
||||
$this->success('任务监听服务启动成功!');
|
||||
} elseif (stripos($message, 'daemons already exist for pid')) {
|
||||
$this->success('任务监听服务已经启动!');
|
||||
} else {
|
||||
$this->error(nl2br($message));
|
||||
}
|
||||
} catch (HttpResponseException $exception) {
|
||||
throw $exception;
|
||||
} catch (\Exception $exception) {
|
||||
trace_file($exception);
|
||||
$this->error($exception->getMessage());
|
||||
} else {
|
||||
$this->error('请使用超管账号操作!');
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* 检查监听服务
|
||||
* @login true
|
||||
*/
|
||||
public function status()
|
||||
{
|
||||
if (AdminService::isSuper()) try {
|
||||
$message = $this->app->console->call('xadmin:queue', ['status'])->fetch();
|
||||
if (preg_match('/process.*?\d+.*?running/', $message)) {
|
||||
echo "<span class='color-green pointer' data-tips-text='{$message}'>{$this->app->lang->get('已启动')}</span>";
|
||||
} else {
|
||||
echo "<span class='color-red pointer' data-tips-text='{$message}'>{$this->app->lang->get('未启动')}</span>";
|
||||
}
|
||||
} catch (\Error|\Exception $exception) {
|
||||
echo "<span class='color-red pointer' data-tips-text='{$exception->getMessage()}'>{$this->app->lang->get('异 常')}</span>";
|
||||
} else {
|
||||
$message = lang('只有超级管理员才能操作!');
|
||||
echo "<span class='color-red pointer' data-tips-text='{$message}'>{$this->app->lang->get('无权限')}</span>";
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* 查询任务进度
|
||||
* @login true
|
||||
*/
|
||||
public function progress()
|
||||
{
|
||||
$input = $this->_vali(['code.require' => '任务编号不能为空!']);
|
||||
$this->app->db->setLog(new NullLogger()); /* 关闭数据库请求日志 */
|
||||
$message = SystemQueue::mk()->where($input)->value('message', '');
|
||||
$this->success('获取任务进度成功d!', json_decode($message, true));
|
||||
}
|
||||
}
|
138
app/admin/controller/api/System.php
Normal file
138
app/admin/controller/api/System.php
Normal file
@ -0,0 +1,138 @@
|
||||
<?php
|
||||
|
||||
// +----------------------------------------------------------------------
|
||||
// | Admin Plugin for ThinkAdmin
|
||||
// +----------------------------------------------------------------------
|
||||
// | 版权所有 2014~2024 ThinkAdmin [ thinkadmin.top ]
|
||||
// +----------------------------------------------------------------------
|
||||
// | 官方网站: https://thinkadmin.top
|
||||
// +----------------------------------------------------------------------
|
||||
// | 开源协议 ( https://mit-license.org )
|
||||
// | 免责声明 ( https://thinkadmin.top/disclaimer )
|
||||
// +----------------------------------------------------------------------
|
||||
// | gitee 代码仓库:https://gitee.com/zoujingli/think-plugs-admin
|
||||
// | github 代码仓库:https://github.com/zoujingli/think-plugs-admin
|
||||
// +----------------------------------------------------------------------
|
||||
|
||||
declare(strict_types=1);
|
||||
|
||||
namespace app\admin\controller\api;
|
||||
|
||||
use think\admin\Controller;
|
||||
use think\admin\model\SystemConfig;
|
||||
use think\admin\service\AdminService;
|
||||
use think\admin\service\RuntimeService;
|
||||
use think\exception\HttpResponseException;
|
||||
|
||||
/**
|
||||
* 系统运行管理
|
||||
* @class System
|
||||
* @package app\admin\controller\api
|
||||
*/
|
||||
class System extends Controller
|
||||
{
|
||||
|
||||
/**
|
||||
* 网站压缩发布
|
||||
* @login true
|
||||
*/
|
||||
public function push()
|
||||
{
|
||||
if (AdminService::isSuper()) try {
|
||||
RuntimeService::push() && sysoplog('系统运维管理', '刷新发布运行缓存');
|
||||
$this->success('网站缓存加速成功!', 'javascript:location.reload()');
|
||||
} catch (HttpResponseException $exception) {
|
||||
throw $exception;
|
||||
} catch (\Exception $exception) {
|
||||
trace_file($exception);
|
||||
$this->error($exception->getMessage());
|
||||
} else {
|
||||
$this->error('请使用超管账号操作!');
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* 清理运行缓存
|
||||
* @login true
|
||||
*/
|
||||
public function clear()
|
||||
{
|
||||
if (AdminService::isSuper()) try {
|
||||
RuntimeService::clear() && sysoplog('系统运维管理', '清理网站日志缓存');
|
||||
$this->success('清空日志缓存成功!', 'javascript:location.reload()');
|
||||
} catch (HttpResponseException $exception) {
|
||||
throw $exception;
|
||||
} catch (\Exception $exception) {
|
||||
trace_file($exception);
|
||||
$this->error($exception->getMessage());
|
||||
} else {
|
||||
$this->error('请使用超管账号操作!');
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* 当前运行模式
|
||||
* @login true
|
||||
*/
|
||||
public function debug()
|
||||
{
|
||||
if (AdminService::isSuper()) if (input('state')) {
|
||||
RuntimeService::set('product');
|
||||
sysoplog('系统运维管理', '开发模式切换为生产模式');
|
||||
$this->success('已切换为生产模式!', 'javascript:location.reload()');
|
||||
} else {
|
||||
RuntimeService::set('debug');
|
||||
sysoplog('系统运维管理', '生产模式切换为开发模式');
|
||||
$this->success('已切换为开发模式!', 'javascript:location.reload()');
|
||||
} else {
|
||||
$this->error('请使用超管账号操作!');
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* 修改富文本编辑器
|
||||
* @return void
|
||||
* @throws \think\admin\Exception
|
||||
*/
|
||||
public function editor()
|
||||
{
|
||||
if (AdminService::isSuper()) {
|
||||
$editor = input('editor', 'auto');
|
||||
sysconf('base.editor', $editor);
|
||||
sysoplog('系统运维管理', "切换编辑器为{$editor}");
|
||||
$this->success('已切换后台编辑器!', 'javascript:location.reload()');
|
||||
} else {
|
||||
$this->error('请使用超管账号操作!');
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* 清理系统配置
|
||||
* @login true
|
||||
*/
|
||||
public function config()
|
||||
{
|
||||
if (AdminService::isSuper()) try {
|
||||
[$tmpdata, $newdata] = [[], []];
|
||||
foreach (SystemConfig::mk()->order('type,name asc')->cursor() as $item) {
|
||||
$tmpdata[$item['type']][$item['name']] = $item['value'];
|
||||
}
|
||||
foreach ($tmpdata as $type => $items) foreach ($items as $name => $value) {
|
||||
$newdata[] = ['type' => $type, 'name' => $name, 'value' => $value];
|
||||
}
|
||||
$this->app->db->transaction(static function () use ($newdata) {
|
||||
SystemConfig::mQuery()->empty()->insertAll($newdata);
|
||||
});
|
||||
$this->app->cache->delete('SystemConfig');
|
||||
sysoplog('系统运维管理', '清理系统配置参数');
|
||||
$this->success('清理系统配置成功!', 'javascript:location.reload()');
|
||||
} catch (HttpResponseException $exception) {
|
||||
throw $exception;
|
||||
} catch (\Exception $exception) {
|
||||
trace_file($exception);
|
||||
$this->error($exception->getMessage());
|
||||
} else {
|
||||
$this->error('请使用超管账号操作!');
|
||||
}
|
||||
}
|
||||
}
|
336
app/admin/controller/api/Upload.php
Normal file
336
app/admin/controller/api/Upload.php
Normal file
@ -0,0 +1,336 @@
|
||||
<?php
|
||||
|
||||
// +----------------------------------------------------------------------
|
||||
// | Admin Plugin for ThinkAdmin
|
||||
// +----------------------------------------------------------------------
|
||||
// | 版权所有 2014~2024 ThinkAdmin [ thinkadmin.top ]
|
||||
// +----------------------------------------------------------------------
|
||||
// | 官方网站: https://thinkadmin.top
|
||||
// +----------------------------------------------------------------------
|
||||
// | 开源协议 ( https://mit-license.org )
|
||||
// | 免责声明 ( https://thinkadmin.top/disclaimer )
|
||||
// +----------------------------------------------------------------------
|
||||
// | gitee 代码仓库:https://gitee.com/zoujingli/think-plugs-admin
|
||||
// | github 代码仓库:https://github.com/zoujingli/think-plugs-admin
|
||||
// +----------------------------------------------------------------------
|
||||
|
||||
declare(strict_types=1);
|
||||
|
||||
namespace app\admin\controller\api;
|
||||
|
||||
use think\admin\Controller;
|
||||
use think\admin\helper\QueryHelper;
|
||||
use think\admin\model\SystemFile;
|
||||
use think\admin\service\AdminService;
|
||||
use think\admin\Storage;
|
||||
use think\admin\storage\AliossStorage;
|
||||
use think\admin\storage\AlistStorage;
|
||||
use think\admin\storage\LocalStorage;
|
||||
use think\admin\storage\QiniuStorage;
|
||||
use think\admin\storage\TxcosStorage;
|
||||
use think\admin\storage\UpyunStorage;
|
||||
use think\exception\HttpResponseException;
|
||||
use think\file\UploadedFile;
|
||||
use think\Response;
|
||||
|
||||
/**
|
||||
* 文件上传接口
|
||||
* @class Upload
|
||||
* @package app\admin\controller\api
|
||||
*/
|
||||
class Upload extends Controller
|
||||
{
|
||||
/**
|
||||
* 文件上传脚本
|
||||
* @return Response
|
||||
* @throws \think\admin\Exception
|
||||
*/
|
||||
public function index(): Response
|
||||
{
|
||||
$data = ['exts' => []];
|
||||
[$uuid, $unid, $exts] = $this->initUnid(false);
|
||||
$allows = str2arr(sysconf('storage.allow_exts|raw'));
|
||||
if (empty($uuid) && $unid > 0) $allows = array_intersect($exts, $allows);
|
||||
foreach ($allows as $ext) $data['exts'][$ext] = Storage::mime($ext);
|
||||
$data['exts'] = json_encode($data['exts'], JSON_UNESCAPED_UNICODE);
|
||||
$data['nameType'] = sysconf('storage.name_type|raw') ?: 'xmd5';
|
||||
return view(dirname(__DIR__, 2) . '/view/api/upload.js', $data)->contentType('application/x-javascript');
|
||||
}
|
||||
|
||||
/**
|
||||
* 文件选择器
|
||||
* @throws \think\db\exception\DataNotFoundException
|
||||
* @throws \think\db\exception\DbException
|
||||
* @throws \think\db\exception\ModelNotFoundException
|
||||
*/
|
||||
public function image()
|
||||
{
|
||||
[$uuid, $unid] = $this->initUnid();
|
||||
SystemFile::mQuery()->layTable(function () {
|
||||
$this->title = '文件选择器';
|
||||
}, function (QueryHelper $query) use ($unid, $uuid) {
|
||||
if ($unid && $uuid) $query->where(function ($query) use ($uuid, $unid) {
|
||||
/** @var \think\db\Query $query */
|
||||
$query->whereOr([['uuid', '=', $uuid], ['unid', '=', $unid]]);
|
||||
}); else {
|
||||
$query->where($unid ? ['unid' => $unid] : ['uuid' => $uuid]);
|
||||
}
|
||||
$query->where(['status' => 2, 'issafe' => 0])->in('xext#type');
|
||||
$query->like('name,hash')->dateBetween('create_at')->order('id desc');
|
||||
});
|
||||
}
|
||||
|
||||
/**
|
||||
* 文件上传检查
|
||||
*/
|
||||
public function state()
|
||||
{
|
||||
try {
|
||||
[$uuid, $unid] = $this->initUnid();
|
||||
[$name, $safe] = [input('name'), $this->getSafe()];
|
||||
$data = ['uptype' => $this->getType(), 'safe' => intval($safe), 'key' => input('key')];
|
||||
$file = SystemFile::mk()->data($this->_vali([
|
||||
'xkey.value' => $data['key'],
|
||||
'type.value' => $this->getType(),
|
||||
'uuid.value' => $uuid,
|
||||
'unid.value' => $unid,
|
||||
'name.require' => '名称不能为空!',
|
||||
'hash.require' => '哈希不能为空!',
|
||||
'xext.require' => '后缀不能为空!',
|
||||
'size.require' => '大小不能为空!',
|
||||
'mime.default' => '',
|
||||
'status.value' => 1,
|
||||
]));
|
||||
$mime = $file->getAttr('mime');
|
||||
if (empty($mime)) $file->setAttr('mime', Storage::mime($file->getAttr('xext')));
|
||||
$info = Storage::instance($data['uptype'])->info($data['key'], $safe, $name);
|
||||
if (isset($info['url']) && isset($info['key'])) {
|
||||
$file->save(['xurl' => $info['url'], 'isfast' => 1, 'issafe' => $data['safe']]);
|
||||
$extr = ['id' => $file->id ?? 0, 'url' => $info['url'], 'key' => $info['key']];
|
||||
$this->success('文件已经上传', array_merge($data, $extr), 200);
|
||||
} elseif ('local' === $data['uptype']) {
|
||||
$local = LocalStorage::instance();
|
||||
$data['url'] = $local->url($data['key'], $safe, $name);
|
||||
$data['server'] = $local->upload();
|
||||
} elseif ('qiniu' === $data['uptype']) {
|
||||
$qiniu = QiniuStorage::instance();
|
||||
$data['url'] = $qiniu->url($data['key'], $safe, $name);
|
||||
$data['token'] = $qiniu->token($data['key'], 3600, $name);
|
||||
$data['server'] = $qiniu->upload();
|
||||
} elseif ('alioss' === $data['uptype']) {
|
||||
$alioss = AliossStorage::instance();
|
||||
$token = $alioss->token($data['key'], 3600, $name);
|
||||
$data['url'] = $token['siteurl'];
|
||||
$data['policy'] = $token['policy'];
|
||||
$data['signature'] = $token['signature'];
|
||||
$data['OSSAccessKeyId'] = $token['keyid'];
|
||||
$data['server'] = $alioss->upload();
|
||||
} elseif ('txcos' === $data['uptype']) {
|
||||
$txcos = TxcosStorage::instance();
|
||||
$token = $txcos->token($data['key'], 3600, $name);
|
||||
$data['url'] = $token['siteurl'];
|
||||
$data['q-ak'] = $token['q-ak'];
|
||||
$data['policy'] = $token['policy'];
|
||||
$data['q-key-time'] = $token['q-key-time'];
|
||||
$data['q-signature'] = $token['q-signature'];
|
||||
$data['q-sign-algorithm'] = $token['q-sign-algorithm'];
|
||||
$data['server'] = $txcos->upload();
|
||||
} elseif ('upyun' === $data['uptype']) {
|
||||
$upyun = UpyunStorage::instance();
|
||||
$token = $upyun->token($data['key'], 3600, $name, input('hash', ''));
|
||||
$data['url'] = $token['siteurl'];
|
||||
$data['policy'] = $token['policy'];
|
||||
$data['server'] = $upyun->upload();
|
||||
$data['authorization'] = $token['authorization'];
|
||||
} elseif ('alist' === $data['uptype']) {
|
||||
$alist = AlistStorage::instance();
|
||||
$data['url'] = $alist->url($data['key']);
|
||||
$data['server'] = $alist->upload();
|
||||
$data['filepath'] = $alist->real($data['key']);
|
||||
$data['authorization'] = $alist->token();
|
||||
} else {
|
||||
$this->error('未知的存储引擎!');
|
||||
}
|
||||
$file->save(['xurl' => $data['url'], 'isfast' => 0, 'issafe' => $data['safe']]);
|
||||
$this->success('获取上传授权参数', array_merge($data, ['id' => $file->id ?? 0]), 404);
|
||||
} catch (HttpResponseException $exception) {
|
||||
throw $exception;
|
||||
} catch (\Exception $exception) {
|
||||
$this->error($exception->getMessage());
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* 更新文件状态
|
||||
* @return void
|
||||
*/
|
||||
public function done()
|
||||
{
|
||||
[$uuid, $unid] = $this->initUnid();
|
||||
$data = $this->_vali([
|
||||
'id.require' => '编号不能为空!',
|
||||
'hash.require' => '哈希不能为空!',
|
||||
'uuid.value' => $uuid,
|
||||
'unid.value' => $unid,
|
||||
]);
|
||||
$file = SystemFile::mk()->where($data)->findOrEmpty();
|
||||
if ($file->isEmpty()) $this->error('文件不存在!');
|
||||
if ($file->save(['status' => 2])) {
|
||||
$this->success('更新成功!');
|
||||
} else {
|
||||
$this->error('更新失败!');
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* 文件上传入口
|
||||
* @throws \think\admin\Exception
|
||||
*/
|
||||
public function file()
|
||||
{
|
||||
[$uuid, $unid, $unexts] = $this->initUnid();
|
||||
// 开始处理文件上传
|
||||
$file = $this->getFile();
|
||||
$extension = strtolower($file->getOriginalExtension());
|
||||
$saveFileName = input('key') ?: Storage::name($file->getPathname(), $extension, '', 'md5_file');
|
||||
// 检查文件名称是否合法
|
||||
if (strpos($saveFileName, '..') !== false) {
|
||||
$this->error('文件路径不能出现跳级操作!');
|
||||
}
|
||||
// 检查文件后缀是否被恶意修改
|
||||
if (strtolower(pathinfo(parse_url($saveFileName, PHP_URL_PATH), PATHINFO_EXTENSION)) !== $extension) {
|
||||
$this->error('文件后缀异常,请重新上传文件!');
|
||||
}
|
||||
// 屏蔽禁止上传指定后缀的文件
|
||||
if (!in_array($extension, str2arr(sysconf('storage.allow_exts|raw')))) {
|
||||
$this->error('文件类型受限,请在后台配置规则!');
|
||||
}
|
||||
// 前端用户上传后缀检查处理
|
||||
if (empty($uuid) && $unid > 0 && !in_array($extension, $unexts)) {
|
||||
$this->error('文件类型受限,请上传允许的文件类型!');
|
||||
}
|
||||
if (in_array($extension, ['sh', 'asp', 'bat', 'cmd', 'exe', 'php'])) {
|
||||
$this->error('文件安全保护,禁止上传可执行文件!');
|
||||
}
|
||||
try {
|
||||
$safeMode = $this->getSafe();
|
||||
if (($type = $this->getType()) === 'local') {
|
||||
$local = LocalStorage::instance();
|
||||
$distName = $local->path($saveFileName, $safeMode);
|
||||
if (PHP_SAPI === 'cli') {
|
||||
is_dir(dirname($distName)) || mkdir(dirname($distName), 0777, true);
|
||||
rename($file->getPathname(), $distName);
|
||||
} else {
|
||||
$file->move(dirname($distName), basename($distName));
|
||||
}
|
||||
$info = $local->info($saveFileName, $safeMode, $file->getOriginalName());
|
||||
if (in_array($extension, ['jpg', 'gif', 'png', 'bmp', 'jpeg', 'wbmp'])) {
|
||||
if ($this->imgNotSafe($distName) && $local->del($saveFileName)) {
|
||||
$this->error('图片未通过安全检查!');
|
||||
}
|
||||
[$width, $height] = getimagesize($distName);
|
||||
if (($width < 1 || $height < 1) && $local->del($saveFileName)) {
|
||||
$this->error('读取图片的尺寸失败!');
|
||||
}
|
||||
}
|
||||
} else {
|
||||
$bina = file_get_contents($file->getPathname());
|
||||
$info = Storage::instance($type)->set($saveFileName, $bina, $safeMode, $file->getOriginalName());
|
||||
}
|
||||
if (isset($info['url'])) {
|
||||
$this->success('文件上传成功!', ['url' => $safeMode ? $saveFileName : $info['url']]);
|
||||
} else {
|
||||
$this->error('文件处理失败,请稍候再试!');
|
||||
}
|
||||
} catch (HttpResponseException $exception) {
|
||||
throw $exception;
|
||||
} catch (\Exception $exception) {
|
||||
trace_file($exception);
|
||||
$this->error($exception->getMessage());
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* 获取上传类型
|
||||
* @return boolean
|
||||
*/
|
||||
private function getSafe(): bool
|
||||
{
|
||||
return boolval(input('safe', '0'));
|
||||
}
|
||||
|
||||
/**
|
||||
* 获取上传方式
|
||||
* @return string
|
||||
* @throws \think\admin\Exception
|
||||
*/
|
||||
private function getType(): string
|
||||
{
|
||||
$type = strtolower(input('uptype', ''));
|
||||
if (in_array($type, array_keys(Storage::types()))) {
|
||||
return $type;
|
||||
} else {
|
||||
return strtolower(sysconf('storage.type|raw'));
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* 获取文件对象
|
||||
* @return UploadedFile|void
|
||||
*/
|
||||
private function getFile(): UploadedFile
|
||||
{
|
||||
try {
|
||||
$file = $this->request->file('file');
|
||||
if ($file instanceof UploadedFile) {
|
||||
return $file;
|
||||
} else {
|
||||
$this->error('读取临时文件失败!');
|
||||
}
|
||||
} catch (HttpResponseException $exception) {
|
||||
throw $exception;
|
||||
} catch (\Exception $exception) {
|
||||
trace_file($exception);
|
||||
$this->error(lang($exception->getMessage()));
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* 初始化用户状态
|
||||
* @param boolean $check
|
||||
* @return array
|
||||
*/
|
||||
private function initUnid(bool $check = true): array
|
||||
{
|
||||
$uuid = AdminService::getUserId();
|
||||
[$unid, $exts] = AdminService::withUploadUnid();
|
||||
if ($check && empty($uuid) && empty($unid)) {
|
||||
$this->error('未登录,禁止使用文件上传!');
|
||||
} else {
|
||||
return [$uuid, $unid, $exts];
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* 检查图片是否安全
|
||||
* @param string $filename
|
||||
* @return boolean
|
||||
*/
|
||||
private function imgNotSafe(string $filename): bool
|
||||
{
|
||||
$source = fopen($filename, 'rb');
|
||||
if (($size = filesize($filename)) > 512) {
|
||||
$hexs = bin2hex(fread($source, 512));
|
||||
fseek($source, $size - 512);
|
||||
$hexs .= bin2hex(fread($source, 512));
|
||||
} else {
|
||||
$hexs = bin2hex(fread($source, $size));
|
||||
}
|
||||
if (is_resource($source)) fclose($source);
|
||||
$bins = hex2bin($hexs);
|
||||
/* 匹配十六进制中的 <% ( ) %> 或 <? ( ) ?> 或 <script | /script> */
|
||||
foreach (['<?php ', '<% ', '<script '] as $key) if (stripos($bins, $key) !== false) return true;
|
||||
$result = preg_match("/(3c25.*?28.*?29.*?253e)|(3c3f.*?28.*?29.*?3f3e)|(3C534352495054)|(2F5343524950543E)|(3C736372697074)|(2F7363726970743E)/is", $hexs);
|
||||
return $result === false || $result > 0;
|
||||
}
|
||||
}
|
200
app/admin/lang/en-us.php
Normal file
200
app/admin/lang/en-us.php
Normal file
@ -0,0 +1,200 @@
|
||||
<?php
|
||||
|
||||
// +----------------------------------------------------------------------
|
||||
// | Admin Plugin for ThinkAdmin
|
||||
// +----------------------------------------------------------------------
|
||||
// | 版权所有 2014~2024 ThinkAdmin [ thinkadmin.top ]
|
||||
// +----------------------------------------------------------------------
|
||||
// | 官方网站: https://thinkadmin.top
|
||||
// +----------------------------------------------------------------------
|
||||
// | 开源协议 ( https://mit-license.org )
|
||||
// | 免责声明 ( https://thinkadmin.top/disclaimer )
|
||||
// +----------------------------------------------------------------------
|
||||
// | gitee 代码仓库:https://gitee.com/zoujingli/think-plugs-admin
|
||||
// | github 代码仓库:https://github.com/zoujingli/think-plugs-admin
|
||||
// +----------------------------------------------------------------------
|
||||
|
||||
$extra = [];
|
||||
$extra['开发人员或在功能调试时使用,系统异常时会显示详细的错误信息,同时还会记录操作日志及数据库 SQL 语句信息。'] = 'Developers may use it during functional debugging. When there are system exceptions, detailed error messages will be displayed, and operation logs and database SQL statement information will also be recorded.';
|
||||
$extra['项目正式部署上线后使用,系统异常时统一显示 “%s”,只记录重要的异常日志信息,强烈推荐上线后使用此模式。'] = 'After the project is officially deployed and launched, it will be used. When there are system exceptions, " %s " will be displayed uniformly, and only important exception log information will be recorded. It is strongly recommended to use this mode after launch.';
|
||||
$extra['旧版本编辑器,对浏览器兼容较好,但内容编辑体验稍有不足。'] = 'The old version of the editor is compatible with browsers, but the content editing experience is slightly insufficient.';
|
||||
$extra['新版本编辑器,只支持新特性浏览器,对内容编辑体验较好,推荐使用。'] = 'The new version of the editor only supports the new feature browser and has a good experience in content editing. It is recommended to use it.';
|
||||
$extra['优先使用新版本编辑器,若浏览器不支持新版本时自动降级为旧版本编辑器。'] = 'Priority should be given to using the new version of the editor. If the browser does not support the new version, it will automatically be downgraded to the old version of the editor.';
|
||||
$extra['文件上传到本地服务器的 `static/upload` 目录,不支持大文件上传,占用服务器磁盘空间,访问时消耗服务器带宽流量。'] = 'Uploading files to the `static/upload` directory of the local server does not support uploading large files, occupying server disk space, and consuming server bandwidth traffic during access.';
|
||||
$extra['文件上传到 Alist 存储的服务器或云存储空间,根据服务配置可支持大文件上传,不占用本身服务器空间及服务器带宽流量。'] = 'Files can be uploaded to the Alist storage server or Cloud storage space. According to the service configuration, large file upload can be supported without occupying the server space and server bandwidth traffic.';
|
||||
$extra['文件上传到七牛云存储空间,支持大文件上传,不占用服务器空间及服务器带宽流量,支持 CDN 加速访问,访问量大时推荐使用。'] = 'Files can be uploaded to Qiniu Cloud storage space. It supports large file upload, does not occupy server space and server bandwidth traffic, and supports CDN accelerated access. It is recommended when there is a large amount of access.';
|
||||
$extra['文件上传到又拍云 USS 存储空间,支持大文件上传,不占用服务器空间及服务器带宽流量,支持 CDN 加速访问,访问量大时推荐使用。'] = "Uploading files to Upyun Cloud's USS storage space supports large file uploads without occupying server space or bandwidth traffic. It supports CDN accelerated access and is recommended for high traffic.";
|
||||
$extra['文件上传到阿里云 OSS 存储空间,支持大文件上传,不占用服务器空间及服务器带宽流量,支持 CDN 加速访问,访问量大时推荐使用。'] = "Uploading files to Aliyun Cloud's OSS storage space supports large file uploads without occupying server space or bandwidth traffic. It supports CDN accelerated access and is recommended for high traffic.";
|
||||
$extra['文件上传到腾讯云 COS 存储空间,支持大文件上传,不占用服务器空间及服务器带宽流量,支持 CDN 加速访问,访问量大时推荐使用。'] = "Uploading files to Tencent Cloud's COS storage space supports large file uploads without occupying server space or bandwidth traffic. It supports CDN accelerated access and is recommended for high traffic.";
|
||||
$extra['网站名称及网站图标,将显示在浏览器的标签上。'] = "The website name and icon will be displayed on the browser's label.";
|
||||
$extra['管理程序名称,将显示在后台左上角标题。'] = 'The name of the management program will be displayed in the header in the upper left corner of the background.';
|
||||
$extra['管理程序版本,将显示在后台左上角标题。'] = 'The management program version will be displayed in the top left corner of the background with a title.';
|
||||
$extra['网站版权信息,在后台登录页面显示版本信息并链接到备案到信息备案管理系统。'] = 'Website copyright information is displayed on the backend login page and linked to the information filing management system.';
|
||||
$extra['网站备案号,可以在 %s 查询获取,将显示在登录页面下面。'] = 'The website registration number can be found at %s and will be displayed below the login page.';
|
||||
$extra['公安备案号,可以在 %s 查询获取,将在登录页面下面显示。'] = 'The public security registration number can be obtained by searching at %s and will be displayed below the login page.';
|
||||
$extra['点击可复制【服务启动指令】'] = "Click to copy the 'Service Start Command'";
|
||||
$extra['待处理 %s 个任务,处理中 %s 个任务,已完成 %s 个任务,已失败 %s 个任务。'] = 'There are %s tasks to be processed, %s tasks in progress, %s tasks completed, and %s tasks failed.';
|
||||
$extra['确定要切换到生产模式运行吗?'] = 'Are you sure you want to switch to Production mode?';
|
||||
$extra['确定要切换到开发模式运行吗?'] = 'Are you sure you want to switch to Development mode?';
|
||||
$extra["超级管理员账号的密码未修改,建议立即<a data-modal='%s'>修改密码</a>!"] = "The super administrator password has not been changed. Suggest <a data-modal='%s'>changing password</a>.";
|
||||
|
||||
$extra['等待处理'] = 'Pending';
|
||||
$extra['正在处理'] = 'Processing';
|
||||
$extra['处理完成'] = 'Completed';
|
||||
$extra['处理失败'] = 'Failed';
|
||||
|
||||
$extra['条件搜索'] = 'Search';
|
||||
$extra['批量删除'] = 'Batch Delete';
|
||||
|
||||
$extra['上传进度 %s'] = 'Upload progress %s';
|
||||
$extra['文件上传出错!'] = 'File upload error.';
|
||||
$extra['文件上传失败!'] = 'File upload failed.';
|
||||
$extra['大小超出限制!'] = 'Size exceeds limit.';
|
||||
$extra['文件秒传成功!'] = 'Successfully transmitted the file in seconds.';
|
||||
$extra['上传接口异常!'] = 'Abnormal upload interface.';
|
||||
$extra['文件上传成功!'] = 'File uploaded successfully.';
|
||||
$extra['图片压缩失败!'] = 'Image compression failed.';
|
||||
$extra['无效的文件上传对象!'] = 'Invalid file upload object.';
|
||||
|
||||
return array_merge($extra, [
|
||||
// 系统操作
|
||||
'基本资料' => 'Basic information',
|
||||
'安全设置' => 'Security setting',
|
||||
'缓存加速' => 'Cache acceleration',
|
||||
'清理缓存' => 'Clean cache',
|
||||
'配色方案' => 'Color scheme',
|
||||
'立即登录' => 'Login',
|
||||
'退出登录' => 'Logout',
|
||||
'系统提示:' => 'System Notify: ',
|
||||
'清空日志缓存成功!' => 'Successfully cleared the log cache.',
|
||||
'获取任务进度成功!' => 'Successfully obtained task progress.',
|
||||
'网站缓存加速成功!' => 'Website cache acceleration successful.',
|
||||
'请使用超管账号操作!' => 'Please use a super managed account to operate.',
|
||||
'停止任务监听服务成功!' => 'Successfully stopped task listening service.',
|
||||
'任务监听服务启动成功!' => 'Task monitoring service started successfully.',
|
||||
'任务监听服务已经启动!' => 'The task monitoring service has started.',
|
||||
'没有找到需要停止的服务!' => 'No services found that need to be stopped.',
|
||||
'已切换后台编辑器!' => 'Switched to background editor.',
|
||||
// 其他搜索器提示
|
||||
'请选择登录时间' => 'Please select the Login time',
|
||||
'请选择创建时间' => 'Please select the creation time',
|
||||
'请输入账号或名称' => 'Please enter an account or name',
|
||||
'请输入权限名称' => 'Please enter the permission name',
|
||||
'请输入数据编码' => 'Please enter the data code',
|
||||
'请输入数据名称' => 'Please enter the data name',
|
||||
'请输入文件名称' => 'Please enter the file name',
|
||||
'请输入文件哈希' => 'Please enter the file hash',
|
||||
'请输入操作节点' => 'Please enter the operate node',
|
||||
'请输入操作内容' => 'Please enter the operate content',
|
||||
'请输入访问地址' => 'Please enter the access Geoip',
|
||||
// 系统配置
|
||||
'运行模式' => 'Running Mode',
|
||||
'生产模式' => 'Production mode',
|
||||
'开发模式' => 'Development mode',
|
||||
'以开发模式运行' => 'Running in Development mode',
|
||||
'以生产模式运行' => 'Running in Production mode',
|
||||
'清理无效配置' => 'Clean up Invalid Configurations',
|
||||
'修改系统参数' => 'Modify System Parameters',
|
||||
'清理系统配置成功!' => 'Successfully cleaned.',
|
||||
'自适应模式' => 'Adaptive Mode',
|
||||
'富编辑器' => 'RichText Editor',
|
||||
'存储引擎' => 'Storage Engine',
|
||||
'系统参数' => 'System Parameter',
|
||||
'网站名称' => 'Site Name',
|
||||
'管理程序名称' => 'Program Name',
|
||||
'管理程序版本' => 'Program Version',
|
||||
'公安备案号' => 'Public security registration number',
|
||||
'网站备案号' => 'Website registration number',
|
||||
'网站版权信息' => 'Website copyright information',
|
||||
'系统信息' => 'System Information',
|
||||
'应用插件' => 'Plugin Information',
|
||||
'核心框架' => 'Core Framework',
|
||||
'平台框架' => 'Platform Framework',
|
||||
'操作系统' => 'Operating System',
|
||||
'运行环境' => 'Runtime Environment',
|
||||
'仅开发模式可见' => 'Visible only in Development mode',
|
||||
'仅生产模式可见' => 'Visible only in Production mode',
|
||||
'插件名称' => 'Plugin Name',
|
||||
'应用名称' => 'App Name',
|
||||
'插件包名' => 'Package Name',
|
||||
'插件版本' => 'Plugin Version',
|
||||
'授权协议' => 'License',
|
||||
'文件默认存储方式' => 'Default storage method for file upload',
|
||||
'当前系统配置参数' => 'Current system configuration parameters',
|
||||
'仅超级管理员可配置' => 'Only super administrators can configure',
|
||||
|
||||
// 系统任务管理
|
||||
'优化数据库' => 'Optimize Database',
|
||||
'开启服务' => 'Start Service',
|
||||
'关闭服务' => 'Shutdown Service',
|
||||
'定时清理' => 'Regular cleaning',
|
||||
'服务状态' => 'Service',
|
||||
'任务统计' => 'Total',
|
||||
'编号名称' => 'Name',
|
||||
'任务指令' => 'Command',
|
||||
'任务状态' => 'Status',
|
||||
'计划时间' => 'scheduled time',
|
||||
'任务名称' => 'Name',
|
||||
'检查中' => 'Checking',
|
||||
'任务计划' => 'Scheduled',
|
||||
'重 置' => 'Reset',
|
||||
'日 志' => 'Logs',
|
||||
'异 常' => 'Abnormal',
|
||||
'无权限' => 'Denied',
|
||||
'已启动' => 'Started',
|
||||
'未启动' => 'Stopped',
|
||||
// 数据字典管理
|
||||
'数据编码' => 'Code',
|
||||
'数据名称' => 'Name',
|
||||
'操作账号' => 'User',
|
||||
'操作节点' => "Node",
|
||||
'操作行为' => 'Action',
|
||||
'操作内容' => "Content",
|
||||
'访问地址' => 'Geo IP',
|
||||
'网络服务商' => 'ISP.',
|
||||
'日志清理成功!' => 'Logger Clear Complate.',
|
||||
'成功清理所有日志' => 'Successfully cleared all logs.',
|
||||
// 系统文件管理
|
||||
'文件名称' => 'Name',
|
||||
'文件哈希' => "HASH",
|
||||
'文件大小' => "Size",
|
||||
'文件后缀' => 'Exts',
|
||||
'存储方式' => 'Storage Type',
|
||||
'清理重复' => 'Clear Replace',
|
||||
'上传方式' => 'Upload Type',
|
||||
'查看文件' => 'View',
|
||||
'文件链接' => 'Link',
|
||||
'秒传' => 'Speedy',
|
||||
'普通' => 'Normal',
|
||||
// 系统菜单管理
|
||||
'图 标' => "Icon",
|
||||
'添加菜单' => 'Add',
|
||||
'禁用菜单' => 'Forbid',
|
||||
'激活菜单' => "Resume",
|
||||
'系统菜单' => 'Menus',
|
||||
'菜单名称' => 'Name',
|
||||
'跳转链接' => 'Link',
|
||||
'上级菜单' => 'Parent',
|
||||
'菜单链接' => 'Link',
|
||||
'链接参数' => 'Params',
|
||||
'权限节点' => "Node",
|
||||
'菜单图标' => 'Icon',
|
||||
'选择图标' => 'Select Icon',
|
||||
// 系统权限管理
|
||||
"授 权" => 'Auth',
|
||||
'添加权限' => 'Add',
|
||||
'权限名称' => "Name",
|
||||
'权限描述' => 'Description',
|
||||
'请输入权限描述' => 'Please enter a permission description',
|
||||
// 系统用户管理
|
||||
'账号名称' => 'Username',
|
||||
'添加用户' => 'Add User',
|
||||
'最后登录' => "Last Login Time",
|
||||
'头像' => "Head",
|
||||
'登录账号' => 'Username',
|
||||
'用户名称' => 'Nickname',
|
||||
'登录次数' => 'Login Times',
|
||||
'系统用户' => 'System User',
|
||||
'密 码' => 'Password',
|
||||
'系统用户管理' => 'Users',
|
||||
]);
|
52
app/admin/route/demo.php
Normal file
52
app/admin/route/demo.php
Normal file
@ -0,0 +1,52 @@
|
||||
<?php
|
||||
|
||||
// +----------------------------------------------------------------------
|
||||
// | Admin Plugin for ThinkAdmin
|
||||
// +----------------------------------------------------------------------
|
||||
// | 版权所有 2014~2024 ThinkAdmin [ thinkadmin.top ]
|
||||
// +----------------------------------------------------------------------
|
||||
// | 官方网站: https://thinkadmin.top
|
||||
// +----------------------------------------------------------------------
|
||||
// | 开源协议 ( https://mit-license.org )
|
||||
// | 免责声明 ( https://thinkadmin.top/disclaimer )
|
||||
// +----------------------------------------------------------------------
|
||||
// | gitee 代码仓库:https://gitee.com/zoujingli/think-plugs-admin
|
||||
// | github 代码仓库:https://github.com/zoujingli/think-plugs-admin
|
||||
// +----------------------------------------------------------------------
|
||||
|
||||
use think\admin\Library;
|
||||
use think\admin\service\RuntimeService;
|
||||
|
||||
/*! 演示环境禁止操作路由绑定 */
|
||||
if (RuntimeService::check('demo')) {
|
||||
Library::$sapp->route->post('index/pass', static function () {
|
||||
return json(['code' => 0, 'info' => lang('演示环境禁止修改用户密码!')]);
|
||||
});
|
||||
Library::$sapp->route->post('config/system', static function () {
|
||||
return json(['code' => 0, 'info' => lang('演示环境禁止修改系统配置!')]);
|
||||
});
|
||||
Library::$sapp->route->post('config/storage', static function () {
|
||||
return json(['code' => 0, 'info' => lang('演示环境禁止修改系统配置!')]);
|
||||
});
|
||||
Library::$sapp->route->post('menu', static function () {
|
||||
return json(['code' => 0, 'info' => lang('演示环境禁止给菜单排序!')]);
|
||||
});
|
||||
Library::$sapp->route->post('menu/index', static function () {
|
||||
return json(['code' => 0, 'info' => lang('演示环境禁止给菜单排序!')]);
|
||||
});
|
||||
Library::$sapp->route->post('menu/add', static function () {
|
||||
return json(['code' => 0, 'info' => lang('演示环境禁止添加菜单!')]);
|
||||
});
|
||||
Library::$sapp->route->post('menu/edit', static function () {
|
||||
return json(['code' => 0, 'info' => lang('演示环境禁止编辑菜单!')]);
|
||||
});
|
||||
Library::$sapp->route->post('menu/state', static function () {
|
||||
return json(['code' => 0, 'info' => lang('演示环境禁止禁用菜单!')]);
|
||||
});
|
||||
Library::$sapp->route->post('menu/remove', static function () {
|
||||
return json(['code' => 0, 'info' => lang('演示环境禁止删除菜单!')]);
|
||||
});
|
||||
Library::$sapp->route->post('user/pass', static function () {
|
||||
return json(['code' => 0, 'info' => lang('演示环境禁止修改密码!')]);
|
||||
});
|
||||
}
|
134
app/admin/view/api/icon.html
Normal file
134
app/admin/view/api/icon.html
Normal file
@ -0,0 +1,134 @@
|
||||
<!DOCTYPE html>
|
||||
<html lang="zh-CN">
|
||||
<head>
|
||||
<meta charset="utf-8">
|
||||
<title>{block name="title"}{$title|default=''}{if !empty($title)} · {/if}{:sysconf('site_name')}{/block}</title>
|
||||
<meta name="renderer" content="webkit">
|
||||
<meta name="format-detection" content="telephone=no">
|
||||
<meta name="mobile-web-app-capable" content="yes">
|
||||
<meta http-equiv="X-UA-Compatible" content="IE=edge,chrome=1">
|
||||
<meta name="apple-mobile-web-app-status-bar-style" content="black">
|
||||
<meta name="viewport" content="width=device-width,initial-scale=1,maximum-scale=1">
|
||||
<link rel="stylesheet" href="__ROOT__/static/theme/css/iconfont.css?at={:date('md')}">
|
||||
<link rel="stylesheet" href="__ROOT__/static/plugs/layui/css/layui.css?v={:date('ymd')}">
|
||||
{if file_exists(syspath("public/static/extra/icon/iconfont.css"))}
|
||||
<link rel="stylesheet" href="__ROOT__/static/extra/icon/iconfont.css?at={:date('md')}">
|
||||
{/if}
|
||||
<style>
|
||||
::-webkit-input-placeholder {
|
||||
color: #aaa
|
||||
}
|
||||
|
||||
::-webkit-scrollbar {
|
||||
width: 3px;
|
||||
height: 3px
|
||||
}
|
||||
|
||||
::-webkit-scrollbar-track {
|
||||
background: #ccc
|
||||
}
|
||||
|
||||
::-webkit-scrollbar-thumb {
|
||||
background-color: #666
|
||||
}
|
||||
|
||||
::selection {
|
||||
background-color: #ec494e;
|
||||
color: #fff
|
||||
}
|
||||
|
||||
::-moz-selection {
|
||||
background-color: #ec494e;
|
||||
color: #fff
|
||||
}
|
||||
|
||||
:-webkit-autofill {
|
||||
-webkit-box-shadow: 0 0 0 1000px white inset;
|
||||
-webkit-text-fill-color: #333
|
||||
}
|
||||
|
||||
ul li {
|
||||
width: 20%;
|
||||
height: 65px;
|
||||
display: block;
|
||||
float: left;
|
||||
margin-right: -1px;
|
||||
margin-left: -2px;
|
||||
margin-bottom: -2px;
|
||||
cursor: pointer;
|
||||
font-size: 14px;
|
||||
overflow: hidden;
|
||||
text-align: center;
|
||||
padding: 15px 0 10px 0;
|
||||
border: 1px solid #e2e2e2;
|
||||
background-color: #efefef;
|
||||
user-select: none;
|
||||
-ms-user-select: none;
|
||||
-moz-user-select: none;
|
||||
-webkit-user-select: none;
|
||||
transition: all .2s linear;
|
||||
-o-transition: all .2s linear;
|
||||
-moz-transition: all .2s linear;
|
||||
-webkit-transition: all .2s linear
|
||||
}
|
||||
|
||||
ul li:hover {
|
||||
color: #fff;
|
||||
background-color: #563d7c
|
||||
}
|
||||
|
||||
ul li:hover i, ul li:hover div {
|
||||
color: #fff;
|
||||
}
|
||||
|
||||
ul li i {
|
||||
color: #333;
|
||||
display: inline-block;
|
||||
font-size: 30px !important
|
||||
}
|
||||
|
||||
ul li div {
|
||||
color: #333;
|
||||
height: 35px;
|
||||
font-size: 13px;
|
||||
line-height: 35px;
|
||||
white-space: nowrap
|
||||
}
|
||||
</style>
|
||||
</head>
|
||||
<body>
|
||||
|
||||
<ul>
|
||||
{foreach $extraIcons??[] as $icon}
|
||||
<li>
|
||||
<i class="iconfont {$icon}"></i>
|
||||
<div class="icon-title">{$icon}</div>
|
||||
</li>
|
||||
{/foreach}
|
||||
{foreach $layuiIcons??[] as $icon}
|
||||
<li>
|
||||
<i class="layui-icon {$icon}"></i>
|
||||
<div class="icon-title">{$icon}</div>
|
||||
</li>
|
||||
{/foreach}
|
||||
{foreach $thinkIcons??[] as $icon}
|
||||
<li>
|
||||
<i class="iconfont {$icon}"></i>
|
||||
<div class="icon-title">{$icon}</div>
|
||||
</li>
|
||||
{/foreach}
|
||||
</ul>
|
||||
|
||||
<script src="__ROOT__/static/plugs/jquery/jquery.min.js" type="text/javascript"></script>
|
||||
<script>
|
||||
$(function () {
|
||||
$('li').on('click', function (className) {
|
||||
if ((className = $(this).find('i').get(0).className)) {
|
||||
top.$('[name="{$field}"]').val(className).trigger('change');
|
||||
top.layer.close(top.layer.getFrameIndex(window.name));
|
||||
}
|
||||
});
|
||||
});
|
||||
</script>
|
||||
</body>
|
||||
</html>
|
327
app/admin/view/api/upload.js
Normal file
327
app/admin/view/api/upload.js
Normal file
@ -0,0 +1,327 @@
|
||||
define(['md5', 'notify'], function (SparkMD5, Notify, allowMime) {
|
||||
allowMime = JSON.parse('{$exts|raw}');
|
||||
|
||||
function UploadAdapter(elem, done) {
|
||||
return new (function (elem, done) {
|
||||
let that = this;
|
||||
|
||||
/*! 初始化变量 */
|
||||
this.option = {elem: $(elem), exts: [], mimes: []};
|
||||
this.option.size = this.option.elem.data('size') || 0;
|
||||
this.option.safe = this.option.elem.data('safe') ? 1 : 0;
|
||||
this.option.hide = this.option.elem.data('hload') ? 1 : 0;
|
||||
this.option.mult = this.option.elem.data('multiple') > 0;
|
||||
this.option.path = (this.option.elem.data('path') || '').replace(/\W/g, '');
|
||||
this.option.type = this.option.safe ? 'local' : this.option.elem.attr('data-uptype') || '';
|
||||
this.option.quality = parseFloat(this.option.elem.data('quality') || '1.0');
|
||||
this.option.maxWidth = parseInt(this.option.elem.data('max-width') || '0');
|
||||
this.option.maxHeight = parseInt(this.option.elem.data('max-height') || '0');
|
||||
this.option.cutWidth = parseInt(this.option.elem.data('cut-width') || '0');
|
||||
this.option.cutHeight = parseInt(this.option.elem.data('cut-height') || '0');
|
||||
|
||||
/*! 查找表单元素, 如果没有找到将不会自动写值 */
|
||||
if (this.option.elem.data('input')) {
|
||||
this.option.input = $(this.option.elem.data('input'))
|
||||
} else if (this.option.elem.data('field')) {
|
||||
this.option.input = $('input[name="' + this.option.elem.data('field') + '"]:not([type=file])');
|
||||
this.option.elem.data('input', this.option.input.length > 0 ? this.option.input.get(0) : null);
|
||||
}
|
||||
|
||||
/*! 文件选择筛选,使用 MIME 规则过滤文件列表 */
|
||||
$((this.option.elem.data('type') || '').split(',')).map(function (i, e) {
|
||||
if (allowMime[e]) that.option.exts.push(e), that.option.mimes.push(allowMime[e]);
|
||||
});
|
||||
|
||||
/*! 初始化上传组件 */
|
||||
this.adapter = new Adapter(this.option, layui.upload.render({
|
||||
url: '{:url("admin/api.upload/file",[],false,true)}', auto: false, elem: elem, accept: 'file', multiple: this.option.mult, exts: this.option.exts.join('|'), acceptMime: this.option.mimes.join(','), choose: function (obj) {
|
||||
obj.items = [], obj.files = obj.pushFile();
|
||||
layui.each(obj.files, function (idx, file) {
|
||||
obj.items.push(file);
|
||||
file.path = that.option.path;
|
||||
file.quality = that.option.quality;
|
||||
file.maxWidth = that.option.maxWidth;
|
||||
file.maxHeight = that.option.maxHeight;
|
||||
file.cutWidth = that.option.cutWidth;
|
||||
file.cutHeight = that.option.cutHeight;
|
||||
});
|
||||
that.adapter.event('upload.choose', obj.items);
|
||||
that.adapter.upload(obj.items, done);
|
||||
layui.each(obj.files, function (idx) {
|
||||
delete obj.files[idx];
|
||||
});
|
||||
}
|
||||
}));
|
||||
})(elem, done)
|
||||
}
|
||||
|
||||
// 创建对象
|
||||
UploadAdapter.adapter = window.AdminUploadAdapter = Adapter;
|
||||
|
||||
// 上传文件
|
||||
function Adapter(option, uploader) {
|
||||
this.uploader = uploader, this.config = function (option) {
|
||||
return (this.option = Object.assign({}, this.option || {}, option || {})), this;
|
||||
}, this.init = function (option) {
|
||||
this.uploader && this.uploader.config.elem.next().val('');
|
||||
this.files = {}, this.loader = 0, this.count = {total: 0, error: 0, success: 0};
|
||||
return this.config(option).config({safe: this.option.safe || 0, type: this.option.type || ''});
|
||||
}, this.init(option);
|
||||
}
|
||||
|
||||
// 文件推送
|
||||
Adapter.prototype.upload = function (files, done) {
|
||||
let that = this.init();
|
||||
layui.each(files, function (index, file) {
|
||||
that.count.total++, file.index = index, that.files[index] = file;
|
||||
if (!that.option.hide && !file.notify) {
|
||||
file.notify = new NotifyExtend(file);
|
||||
}
|
||||
if (that.option.size && file.size > that.option.size) {
|
||||
that.event('upload.error', {file: file}, file, '{:lang("大小超出限制!")}');
|
||||
}
|
||||
});
|
||||
layui.each(files, function (index, file) {
|
||||
// 禁传异常状态文件
|
||||
if (typeof file.xstate === 'number' && file.xstate === -1) return;
|
||||
// 图片限宽限高压缩
|
||||
let isGif = /^image\/gif/.test(file.type);
|
||||
if (!isGif && /^image\//.test(file.type) && (file.maxWidth + file.maxHeight + file.cutWidth + file.cutHeight > 0 || file.quality !== 1)) {
|
||||
require(['compressor'], function (Compressor) {
|
||||
let options = {quality: file.quality, resize: 'cover'};
|
||||
if (file.cutWidth) options.width = file.cutWidth;
|
||||
if (file.cutHeight) options.height = file.cutHeight;
|
||||
if (file.maxWidth) options.maxWidth = file.maxWidth;
|
||||
if (file.maxHeight) options.maxHeight = file.maxHeight;
|
||||
new Compressor(file, Object.assign(options, {
|
||||
success(blob) {
|
||||
blob.index = file.index, blob.notify = file.notify, blob.path = file.path, files[index] = blob;
|
||||
that.hash(files[index]).then(function (file) {
|
||||
that.event('upload.hash', file).request(file, done);
|
||||
});
|
||||
}, error: function () {
|
||||
that.event('upload.error', {file: file}, file, '{:lang("图片压缩失败!")}');
|
||||
}
|
||||
}));
|
||||
});
|
||||
} else {
|
||||
that.hash(file).then(function (file) {
|
||||
that.event('upload.hash', file).request(file, done);
|
||||
});
|
||||
}
|
||||
});
|
||||
};
|
||||
|
||||
// 文件上传
|
||||
Adapter.prototype.request = function (file, done) {
|
||||
let that = this, data = {key: file.xkey, safe: that.option.safe, uptype: that.option.type};
|
||||
data.size = file.size, data.name = file.name, data.hash = file.xmd5, data.mime = file.type, data.xext = file.xext;
|
||||
jQuery.ajax("{:url('admin/api.upload/state',[],false,true)}", {
|
||||
data: data, method: 'post', success: function (ret) {
|
||||
file.id = ret.data.id || 0, file.xurl = ret.data.url;
|
||||
file.xsafe = ret.data.safe, file.xpath = ret.data.key, file.xtype = ret.data.uptype;
|
||||
if (parseInt(ret.code) === 404) {
|
||||
let uploader = {};
|
||||
uploader.uptype = ret.data.uptype;
|
||||
uploader.url = ret.data.server;
|
||||
uploader.head = {};
|
||||
uploader.form = new FormData();
|
||||
uploader.form.append('key', ret.data.key);
|
||||
uploader.form.append('safe', ret.data.safe);
|
||||
uploader.form.append('uptype', ret.data.uptype);
|
||||
if (ret.data.uptype === 'qiniu') {
|
||||
uploader.form.append('token', ret.data.token);
|
||||
} else if (ret.data.uptype === 'alist') {
|
||||
uploader.type = 'put';
|
||||
uploader.head['file-path'] = ret.data['filepath'];
|
||||
uploader.head['authorization'] = ret.data['authorization'];
|
||||
uploader.form = new FormData();
|
||||
} else if (ret.data.uptype === 'alioss') {
|
||||
uploader.form.append('policy', ret.data['policy']);
|
||||
uploader.form.append('signature', ret.data['signature']);
|
||||
uploader.form.append('OSSAccessKeyId', ret.data['OSSAccessKeyId']);
|
||||
uploader.form.append('success_action_status', '200');
|
||||
uploader.form.append('Content-Disposition', 'inline;filename=' + encodeURIComponent(file.name));
|
||||
} else if (ret.data.uptype === 'txcos') {
|
||||
uploader.form.append('q-ak', ret.data['q-ak']);
|
||||
uploader.form.append('policy', ret.data['policy']);
|
||||
uploader.form.append('q-key-time', ret.data['q-key-time']);
|
||||
uploader.form.append('q-signature', ret.data['q-signature']);
|
||||
uploader.form.append('q-sign-algorithm', ret.data['q-sign-algorithm']);
|
||||
uploader.form.append('success_action_status', '200');
|
||||
uploader.form.append('Content-Disposition', 'inline;filename=' + encodeURIComponent(file.name));
|
||||
} else if (ret.data.uptype === 'upyun') {
|
||||
uploader.form.delete('key');
|
||||
uploader.form.delete('safe');
|
||||
uploader.form.delete('uptype');
|
||||
uploader.form.append('save-key', ret.data['key']);
|
||||
uploader.form.append('policy', ret.data['policy']);
|
||||
uploader.form.append('authorization', ret.data['authorization']);
|
||||
uploader.form.append('Content-Disposition', 'inline;filename=' + encodeURIComponent(file.name));
|
||||
}
|
||||
uploader.form.append('file', file, file.name), jQuery.ajax({
|
||||
xhrFields: {withCredentials: ret.data.uptype === 'local'}, headers: uploader.head, url: uploader.url, data: uploader.form, type: uploader.type || 'post', xhr: function (xhr) {
|
||||
xhr = new XMLHttpRequest();
|
||||
return xhr.upload.addEventListener('progress', function (event) {
|
||||
file.xtotal = event.total, file.xloaded = event.loaded || 0;
|
||||
that.progress((file.xloaded / file.xtotal * 100).toFixed(2), file)
|
||||
}), xhr;
|
||||
}, contentType: false, error: function () {
|
||||
that.event('upload.error', {file: file}, file, '{:lang("上传接口异常!")}');
|
||||
}, processData: false, success: function (ret) {
|
||||
// 兼容数据格式
|
||||
if (typeof ret === 'string' && ret.length > 0) try {
|
||||
ret = JSON.parse(ret) || ret;
|
||||
} catch (e) {
|
||||
console.log(e)
|
||||
}
|
||||
if (typeof ret !== 'object') {
|
||||
ret = {code: 1, url: file.xurl, info: '{:lang("文件上传成功!")}'};
|
||||
}
|
||||
/*! 检查单个文件上传返回的结果 */
|
||||
if (typeof ret === 'object' && ret.code < 1) {
|
||||
that.event('upload.error', {file: file}, file, ret.info || '{:lang("文件上传失败!")}');
|
||||
} else if (uploader.uptype === 'alist' && parseInt(ret.code) !== 200) {
|
||||
that.event('upload.error', {file: file}, file, ret.message || '{:lang("文件上传失败!")}');
|
||||
} else {
|
||||
that.done(ret, file.index, file, done, '{:lang("文件上传成功!")}');
|
||||
}
|
||||
}
|
||||
});
|
||||
} else if (parseInt(ret.code) === 200) {
|
||||
(file.xurl = ret.data.url), that.progress('100.00', file);
|
||||
that.done({code: 1, url: file.xurl, info: file.xstats, data: {code: 200, url: file.xurl}}, file.index, file, done, '{:lang("文件秒传成功!")}');
|
||||
} else {
|
||||
that.event('upload.error', {file: file}, file, ret.info || ret.error.message || '{:lang("文件上传出错!")}');
|
||||
}
|
||||
}
|
||||
});
|
||||
};
|
||||
|
||||
// 上传进度
|
||||
Adapter.prototype.progress = function (number, file) {
|
||||
this.event('upload.progress', {number: number, file: file});
|
||||
if (file.notify) file.notify.setProgress(number);
|
||||
};
|
||||
|
||||
// 上传结果
|
||||
Adapter.prototype.done = function (ret, idx, file, done, message) {
|
||||
/*! 检查单个文件上传返回的结果 */
|
||||
if (ret.code < 1) return $.msg.tips(ret.info || '{:lang("文件上传失败!")}');
|
||||
if (typeof file.xurl !== 'string') return $.msg.tips('{:lang("无效的文件上传对象!")}');
|
||||
jQuery.post("{:url('admin/api.upload/done',[],false,true)}", {id: file.id, hash: file.xmd5});
|
||||
/*! 单个文件上传成功结果处理 */
|
||||
if (typeof done === 'function') {
|
||||
done.call(this.option.elem, file.xurl, this.files['id']);
|
||||
} else if (this.option.mult < 1 && this.option.elem.data('input')) {
|
||||
$(this.option.elem.data('input')).val(file.xurl).trigger('change', file);
|
||||
}
|
||||
// 文件上传成功事件
|
||||
this.event('push', file.xurl).event('upload.done', {file: file, data: ret}, file, message);
|
||||
/*! 所有文件上传完成后结果处理 */
|
||||
if (this.count.success + this.count.error >= this.count.total) {
|
||||
this.option.hide || $.msg.close(this.loader);
|
||||
if (this.option.mult > 0 && this.option.elem.data('input')) {
|
||||
let urls = this.option.elem.data('input').value || [];
|
||||
if (typeof urls === 'string') urls = urls.split('|');
|
||||
for (let i in this.files) urls.push(this.files[i].xurl);
|
||||
$(this.option.elem.data('input')).val(urls.join('|')).trigger('change', this.files);
|
||||
}
|
||||
this.event('upload.complete', {file: this.files}, file).init().uploader && this.uploader.reload();
|
||||
}
|
||||
};
|
||||
|
||||
/*! 触发事件过程 */
|
||||
Adapter.prototype.event = function (name, data, file, message) {
|
||||
if (name === 'upload.error') {
|
||||
this.count.error++, file.xstate = -1, file.xstats = message;
|
||||
if (file.notify) file.notify.setError(message || file.xstats || '');
|
||||
} else if (name === 'upload.done') {
|
||||
this.count.success++, file.xstate = 1, file.xstats = message;
|
||||
if (file.notify) file.notify.setSuccess(message || file.xstats || '')
|
||||
}
|
||||
if (this.option.elem) {
|
||||
this.option.elem.triggerHandler(name, data);
|
||||
if (this.option.input) this.option.input.triggerHandler(name, data);
|
||||
}
|
||||
return this;
|
||||
};
|
||||
|
||||
/**
|
||||
* 计算文件 HASH 值
|
||||
* @param {File} file 文件对象
|
||||
* @return {Promise}
|
||||
*/
|
||||
Adapter.prototype.hash = function (file) {
|
||||
let defer = jQuery.Deferred();
|
||||
file.xext = file.name.indexOf('.') > -1 ? file.name.split('.').pop() : 'tmp';
|
||||
|
||||
/*! 兼容不能计算文件 HASH 的情况 */
|
||||
let IsDate = '{$nameType|default=""}'.indexOf('date') > -1;
|
||||
if (!window.FileReader || IsDate) return jQuery.when((function (xmd5, chars) {
|
||||
while (xmd5.length < 32) xmd5 += chars.charAt(Math.floor(Math.random() * chars.length));
|
||||
return SetFileXdata(file, xmd5, 6), defer.promise();
|
||||
})(layui.util.toDateString(Date.now(), 'yyyyMMddHHmmss-'), '0123456789'));
|
||||
|
||||
/*! 读取文件并计算 HASH 值 */
|
||||
return new LoadNextChunk(file).ReadAsChunk();
|
||||
|
||||
function SetFileXdata(file, xmd5, slice) {
|
||||
file.xmd5 = xmd5, file.xstate = 0, file.xstats = '';
|
||||
file.xkey = file.xmd5.substring(0, slice || 2) + '/' + file.xmd5.substring(slice || 2) + '.' + file.xext;
|
||||
if (file.path) file.xkey = file.path + '/' + file.xkey;
|
||||
return defer.resolve(file, file.xmd5, file.xkey), file;
|
||||
}
|
||||
|
||||
function LoadNextChunk(file) {
|
||||
let that = this, reader = new FileReader(), spark = new SparkMD5.ArrayBuffer();
|
||||
let slice = File.prototype.slice || File.prototype.mozSlice || File.prototype.webkitSlice;
|
||||
this.chunkIdx = 0, this.chunkSize = 2097152, this.chunkTotal = Math.ceil(file.size / this.chunkSize);
|
||||
reader.onload = function (event) {
|
||||
spark.append(event.target.result);
|
||||
++that.chunkIdx < that.chunkTotal ? that.ReadAsChunk() : SetFileXdata(file, spark.end());
|
||||
}, reader.onerror = function () {
|
||||
defer.reject();
|
||||
}, this.ReadAsChunk = function () {
|
||||
this.start = that.chunkIdx * that.chunkSize;
|
||||
this.loaded = this.start + that.chunkSize >= file.size ? file.size : this.start + that.chunkSize;
|
||||
reader.readAsArrayBuffer(slice.call(file, this.start, this.loaded));
|
||||
defer.notify(file, (this.loaded / file.size * 100).toFixed(2));
|
||||
return defer.promise();
|
||||
};
|
||||
}
|
||||
};
|
||||
|
||||
return UploadAdapter;
|
||||
|
||||
/**
|
||||
* 上传状态提示扩展插件
|
||||
* @param {File} file 文件对象
|
||||
* @constructor
|
||||
*/
|
||||
function NotifyExtend(file) {
|
||||
let that = this, message = "{:lang('上传进度 %s', ['<span data-upload-progress>0%</span>'])}";
|
||||
this.notify = Notify.notify({width: 260, title: file.name, showProgress: true, description: message, type: 'default', position: 'top-right', closeTimeout: 0});
|
||||
this.$elem = $(this.notify.notification.nodes);
|
||||
this.$elem.find('.growl-notification__progress').addClass('is-visible');
|
||||
this.$elem.find('.growl-notification__progress-bar').addClass('transition');
|
||||
this.setProgress = function (number) {
|
||||
this.$elem.find('[data-upload-progress]').html(number + '%');
|
||||
this.$elem.find('.growl-notification__progress-bar').css({width: number + '%'});
|
||||
return this;
|
||||
}, this.setError = function (message) {
|
||||
this.$elem.find('.growl-notification__desc').html(message || '{:lang("文件上传失败!")}');
|
||||
this.$elem.removeClass('growl-notification--default').addClass('growl-notification--error')
|
||||
return this.close();
|
||||
}, this.setSuccess = function (message) {
|
||||
this.setProgress('100.00');
|
||||
this.$elem.find('.growl-notification__desc').html(message || '{:lang("文件上传成功!")}');
|
||||
this.$elem.removeClass('growl-notification--default').addClass('growl-notification--success');
|
||||
return this.close();
|
||||
}, this.close = function (timeout) {
|
||||
return setTimeout(function () {
|
||||
that.notify.close();
|
||||
}, timeout || 2000), this;
|
||||
};
|
||||
}
|
||||
});
|
161
app/admin/view/api/upload/image.html
Normal file
161
app/admin/view/api/upload/image.html
Normal file
@ -0,0 +1,161 @@
|
||||
<div class="image-dialog" id="ImageDialog">
|
||||
<div class="image-dialog-head">
|
||||
<label class="pull-left flex">
|
||||
<input class="layui-input margin-right-5" v-model="keys" style="height:30px;line-height:30px" placeholder="{:lang('请输入搜索关键词')}">
|
||||
<a class="layui-btn layui-btn-sm layui-btn-normal" @click="search">{:lang('搜 索')}</a>
|
||||
</label>
|
||||
<div class="pull-right">
|
||||
<a class="layui-btn layui-btn-sm layui-btn-normal" @click="uploadImage">{:lang('上传图片')}</a>
|
||||
</div>
|
||||
</div>
|
||||
<div class="image-dialog-body">
|
||||
<div class="image-dialog-item" v-for="x in list" @click="setItem(x)" style="display:none" v-show="show" :class="{'image-dialog-checked':x.checked}">
|
||||
<div class="uploadimage" :style="x.style"></div>
|
||||
<p class="image-dialog-item-name layui-elip" v-text="x.name"></p>
|
||||
<div class="image-dialog-item-tool">
|
||||
<span class="image-dialog-item-type">{{x.xext.toUpperCase()}}</span>
|
||||
<span class="image-dialog-item-size">{{formatSize(x.size)}}</span>
|
||||
{if auth('admin/file/remove')}
|
||||
<span class="layui-icon layui-icon-close image-dialog-item-close" @click.stop="remove(x)"></span>
|
||||
{/if}
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
<div class="image-dialog-foot">
|
||||
<div id="ImageDialogPage" class="image-dialog-page"></div>
|
||||
<div id="ImageDialogButton layui-hide" class="image-dialog-button layui-btn layui-btn-normal" v-if="data.length>0" @click="confirm">
|
||||
{php} $tag = '{{data.length}}'; {/php}
|
||||
{:lang('已选 %s 张,确认', [$tag])}
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<script>
|
||||
require(['vue'], function (vue) {
|
||||
var app = new vue({
|
||||
el: '#ImageDialog',
|
||||
data: {
|
||||
didx: 0,
|
||||
page: 1, limit: 15, show: false, mult: false,
|
||||
keys: '', list: [], data: [], idxs: {}, urls: [],
|
||||
},
|
||||
created: function () {
|
||||
this.didx = $.msg.mdx.pop();
|
||||
this.$btn = $('#{$get.id|default=""}');
|
||||
this.$ups = $('#ImageDialogUploadLayout [data-file]');
|
||||
this.mult = "{$get.file|default='image'}" === 'images';
|
||||
this.loadPage(), setTimeout(function () {
|
||||
$('#ImageDialogButton').removeClass('layui-hide');
|
||||
}, 1000);
|
||||
},
|
||||
methods: {
|
||||
// 搜索刷新数据
|
||||
search: function () {
|
||||
this.page = 1;
|
||||
this.loadPage();
|
||||
},
|
||||
// 确认选择数据
|
||||
confirm: function () {
|
||||
this.urls = [];
|
||||
this.data.forEach(function (file) {
|
||||
app.setValue(file.xurl);
|
||||
});
|
||||
this.setInput();
|
||||
},
|
||||
// 删除指定的图片
|
||||
remove: function (x) {
|
||||
$.msg.confirm('确认要移除这张图片吗?', function () {
|
||||
$.form.load('{:url("admin/file/remove")}', {id: x.id}, 'POST', function (ret) {
|
||||
ret.code > 0 ? app.loadPage() : $.msg.error(ret.info);
|
||||
return app.$forceUpdate(), false;
|
||||
})
|
||||
})
|
||||
},
|
||||
// 格式文件大小
|
||||
formatSize: function (size) {
|
||||
return $.formatFileSize(size);
|
||||
},
|
||||
// 设置单项数据
|
||||
setItem: function (item) {
|
||||
if (!this.mult) {
|
||||
this.setValue(item.xurl).setInput();
|
||||
} else if ((item.checked = !this.idxs[item.id])) {
|
||||
(this.idxs[item.id] = item) && this.data.push(item);
|
||||
} else {
|
||||
delete this.idxs[item.id];
|
||||
this.data.forEach(function (temp, idx) {
|
||||
temp.id === item.id && app.data.splice(idx, 1);
|
||||
});
|
||||
}
|
||||
},
|
||||
// 更新列表数据
|
||||
setList: function (items, count) {
|
||||
this.list = items;
|
||||
this.list.forEach(function (item) {
|
||||
item.checked = !!app.idxs[item.id]
|
||||
item.style = 'background-image:url(' + item.xurl + ')';
|
||||
});
|
||||
this.addPage(count);
|
||||
},
|
||||
// 设置选择数据
|
||||
setValue: function (xurl) {
|
||||
$.msg.close(this.didx);
|
||||
this.urls.push(xurl) && this.$btn.triggerHandler('push', xurl);
|
||||
return this;
|
||||
},
|
||||
// 设置输入表单
|
||||
setInput: function () {
|
||||
if (this.$btn.data('input')) {
|
||||
$(this.$btn.data('input')).val(this.urls.join('|')).trigger('change');
|
||||
}
|
||||
},
|
||||
// 创建分页工具条
|
||||
addPage: function (count) {
|
||||
this.show = true;
|
||||
layui.laypage.render({
|
||||
curr: this.page, count: count, limit: app.limit,
|
||||
layout: ['count', 'prev', 'page', 'next', 'refresh'],
|
||||
elem: 'ImageDialogPage', jump: function (obj, first) {
|
||||
if (!first) app.loadPage(app.page = obj.curr);
|
||||
},
|
||||
});
|
||||
},
|
||||
// 加载页面数据
|
||||
loadPage: function () {
|
||||
this.params = {page: this.page, limit: this.limit, output: 'layui.table', name: this.keys || ''};
|
||||
this.params.type = '{$get.type|default="gif,png,jpg,jpeg"}';
|
||||
$.form.load('{:url("image",[],false,true)}', this.params, 'get', function (ret) {
|
||||
return app.setList(ret.data, ret.count), false;
|
||||
});
|
||||
},
|
||||
// 上传图片文件
|
||||
uploadImage: function () {
|
||||
this.urls = [];
|
||||
this.$ups.off('push').on('push', function (e, xurl) {
|
||||
app.setValue(xurl);
|
||||
}).off('upload.complete').on('upload.complete', function () {
|
||||
app.setInput();
|
||||
}).click();
|
||||
},
|
||||
}
|
||||
});
|
||||
});
|
||||
</script>
|
||||
|
||||
<label class="layui-hide" id="ImageDialogUploadLayout">
|
||||
<!-- 图片上传组件 开始 -->
|
||||
{if isset($get.file) && $get.file eq 'image'}
|
||||
<button data-file="one" data-type="{$get.type|default='gif,png,jpg,jpeg'}"
|
||||
data-path="{$get.path|default=''}" data-size="{$get.size|default=0}"
|
||||
data-cut-width="{$get.cutWidth|default=0}" data-cut-height="{$get.cutHeight|default=0}"
|
||||
data-max-width="{$get.maxWidth|default=0}" data-max-height="{$get.maxHeight|default=0}"
|
||||
></button>
|
||||
{else}
|
||||
<button data-file="mul" data-type="{$get.type|default='gif,png,jpg,jpeg'}"
|
||||
data-path="{$get.path|default=''}" data-size="{$get.size|default=0}"
|
||||
data-cut-width="{$get.cutWidth|default=0}" data-cut-height="{$get.cutHeight|default=0}"
|
||||
data-max-width="{$get.maxWidth|default=0}" data-max-height="{$get.maxHeight|default=0}"
|
||||
></button>
|
||||
{/if}
|
||||
<!-- 图片上传组件 结束 -->
|
||||
</label>
|
143
app/admin/view/auth/form.html
Normal file
143
app/admin/view/auth/form.html
Normal file
@ -0,0 +1,143 @@
|
||||
{extend name='main'}
|
||||
|
||||
{block name="button"}
|
||||
<button data-target-submit class='layui-btn layui-btn-sm'>{:lang('保存数据')}</button>
|
||||
<button data-target-backup class="layui-btn layui-btn-sm layui-btn-danger">{:lang('取消编辑')}</button>
|
||||
{/block}
|
||||
|
||||
{block name="content"}
|
||||
<div class="think-box-shadow">
|
||||
<form method="post" id="RoleForm" class="layui-form layui-card">
|
||||
<div class="layui-card-body">
|
||||
<label class="layui-form-item relative block">
|
||||
<span class="help-label"><b>{:lang('权限名称')}</b>Auth Name</span>
|
||||
<input maxlength="100" class="layui-input" name="title" value='{$vo.title|default=""}' required vali-name="{:lang('权限名称')}" placeholder="{:lang('请输入权限名称')}">
|
||||
<span class="help-block">{:lang('访问权限名称需要保持不重复,在给用户授权时需要根据名称选择!')}</span>
|
||||
</label>
|
||||
<label class="layui-form-item relative block">
|
||||
<span class="help-label"><b>{:lang('权限描述')}</b>Auth Remark</span>
|
||||
<textarea placeholder="{:lang('请输入权限描述')}" maxlength="200" class="layui-textarea" name="desc">{$vo.desc|default=""}</textarea>
|
||||
</label>
|
||||
<div class="layui-form-item">
|
||||
<span class="help-label label-required-prev"><b>{:lang('功能节点')}</b>Auth Nodes</span>
|
||||
<ul id="zTree" class="ztree notselect"></ul>
|
||||
</div>
|
||||
<div class="hr-line-dashed"></div>
|
||||
{notempty name='vo.id'}<input name="id" value="{$vo.id}" type="hidden"/>{/notempty}
|
||||
<div class="layui-form-item text-center">
|
||||
<button data-target-submit class="layui-btn">{:lang('保存数据')}</button>
|
||||
<button data-target-backup class="layui-btn layui-btn-danger" type="button">{:lang('取消编辑')}</button>
|
||||
</div>
|
||||
</div>
|
||||
</form>
|
||||
</div>
|
||||
{/block}
|
||||
|
||||
{block name="script"}
|
||||
<script>
|
||||
require(['jquery.ztree'], function () {
|
||||
new function () {
|
||||
let that = this;
|
||||
this.data = {}, this.ztree = null, this.setting = {
|
||||
view: {showLine: false, showIcon: false, dblClickExpand: false},
|
||||
check: {enable: true, nocheck: false, chkboxType: {"Y": "ps", "N": "ps"}}, callback: {
|
||||
beforeClick: function (id, node) {
|
||||
node.children.length < 1 ? that.ztree.checkNode(node, !node.checked, true, true) : that.ztree.expandNode(node);
|
||||
return false;
|
||||
}
|
||||
}
|
||||
};
|
||||
this.renderChildren = function (list, level) {
|
||||
let childrens = [];
|
||||
for (let i in list) childrens.push({
|
||||
open: true, node: list[i].node, name: list[i].title || list[i].node,
|
||||
checked: list[i].checked || false, children: this.renderChildren(list[i]._sub_, level + 1)
|
||||
});
|
||||
return childrens;
|
||||
};
|
||||
this.syncData = function () {
|
||||
$.form.load('{:sysuri()}', {id: '{$vo.id|default=0}', action: 'json'}, 'post', function (ret) {
|
||||
return (that.data = that.renderChildren(ret.data, 1)), that.showTree(), false;
|
||||
});
|
||||
};
|
||||
this.showTree = function () {
|
||||
this.ztree = $.fn.zTree.init($("#zTree"), this.setting, this.data);
|
||||
while (true) {
|
||||
let nodes = this.ztree.getNodesByFilter(function (node) {
|
||||
return (!node.node && node.children.length < 1);
|
||||
});
|
||||
if (nodes.length < 1) break;
|
||||
for (let i in nodes) this.ztree.removeNode(nodes[i]);
|
||||
}
|
||||
};
|
||||
// 刷新数据
|
||||
this.syncData();
|
||||
// 监听表单提交
|
||||
$('#RoleForm').vali(function (form) {
|
||||
let data = that.ztree.getCheckedNodes(true);
|
||||
Object.assign(form, {nodes: [], action: 'save'})
|
||||
for (let i in data) if (data[i].node) form.nodes.push(data[i].node);
|
||||
$.form.load('{:sysuri()}', form, 'post');
|
||||
});
|
||||
};
|
||||
});
|
||||
</script>
|
||||
{/block}
|
||||
|
||||
{block name="style"}
|
||||
<style>
|
||||
ul.ztree li {
|
||||
line-height: 24px;
|
||||
white-space: normal !important;
|
||||
}
|
||||
|
||||
ul.ztree li span.button.switch {
|
||||
margin-right: 5px;
|
||||
}
|
||||
|
||||
ul.ztree ul ul li {
|
||||
display: inline-block;
|
||||
white-space: normal;
|
||||
}
|
||||
|
||||
ul.ztree > li {
|
||||
border: 1px solid rgba(0, 0, 0, 0.10);
|
||||
padding: 15px;
|
||||
background: rgba(0, 0, 0, 0.05);
|
||||
border-radius: 3px;
|
||||
margin-bottom: 10px;
|
||||
}
|
||||
|
||||
ul.ztree > li > ul {
|
||||
padding: 10px;
|
||||
margin-top: 10px;
|
||||
background: rgba(0, 0, 0, 0.05);
|
||||
border-radius: 3px;
|
||||
}
|
||||
|
||||
ul.ztree > li > ul > li {
|
||||
padding: 5px 0;
|
||||
}
|
||||
|
||||
ul.ztree > li > a > span {
|
||||
font-size: 15px;
|
||||
font-weight: 700;
|
||||
}
|
||||
|
||||
ul.ztree .level2 .button.level2 {
|
||||
width: 0;
|
||||
}
|
||||
|
||||
ul.ztree li span.button.noline_open {
|
||||
background-position-y: -73px;
|
||||
}
|
||||
|
||||
ul.ztree li span.button.noline_close {
|
||||
background-position-y: -73px;
|
||||
}
|
||||
|
||||
ul.ztree .level1 > .node_name {
|
||||
font-weight: bold;
|
||||
}
|
||||
</style>
|
||||
{/block}
|
76
app/admin/view/auth/index.html
Normal file
76
app/admin/view/auth/index.html
Normal file
@ -0,0 +1,76 @@
|
||||
{extend name='table'}
|
||||
|
||||
{block name="button"}
|
||||
<!--{if auth("add")}-->
|
||||
<button data-open='{:url("add")}' data-table-id="RoleTable" class='layui-btn layui-btn-sm layui-btn-primary'>{:lang('添加权限')}</button>
|
||||
<!--{/if}-->
|
||||
|
||||
<!--{if auth("remove")}-->
|
||||
<button data-action='{:url("remove")}' data-rule="id#{id}" data-table-id="RoleTable" data-confirm="{:lang('确定要批量删除权限吗?')}" class='layui-btn layui-btn-sm layui-btn-primary'>{:lang('批量删除')}</button>
|
||||
<!--{/if}-->
|
||||
{/block}
|
||||
|
||||
{block name="content"}
|
||||
<div class="think-box-shadow">
|
||||
{include file='auth/index_search'}
|
||||
<table id="RoleTable" data-url="{:request()->url()}" data-target-search="form.form-search"></table>
|
||||
</div>
|
||||
{/block}
|
||||
|
||||
{block name='script'}
|
||||
<script>
|
||||
$(function () {
|
||||
// 初始化表格组件
|
||||
$('#RoleTable').layTable({
|
||||
even: true, height: 'full',
|
||||
sort: {field: 'sort desc,id', type: 'desc'},
|
||||
cols: [[
|
||||
{checkbox: true, fixed: true},
|
||||
{field: 'sort', title: '{:lang("排序权重")}', align: 'center', width: 100, sort: true, templet: '#SortInputRoleTableTpl'},
|
||||
{field: 'title', title: '{:lang("权限名称")}', align: 'center', minWidth: 140},
|
||||
{field: 'desc', title: '{:lang("权限描述")}', align: 'center', minWidth: 110, templet: '<div>{{d.desc||"-"}}</div>'},
|
||||
{field: 'status', title: '{:lang("使用状态")}', align: 'center', minWidth: 110, templet: '#StatusSwitchRoleTableTpl'},
|
||||
{field: 'create_at', title: '{:lang("创建时间")}', align: 'center', minWidth: 170, sort: true},
|
||||
{toolbar: '#ToolbarRoleTableTpl', title: '{:lang("操作面板")}', align: 'center', minWidth: 210, fixed: 'right'},
|
||||
]]
|
||||
});
|
||||
|
||||
// 数据状态切换操作
|
||||
layui.form.on('switch(StatusSwitchRoleTable)', function (obj) {
|
||||
let data = {id: obj.value, status: obj.elem.checked > 0 ? 1 : 0};
|
||||
$.form.load("{:url('state')}", data, 'post', function (ret) {
|
||||
if (ret.code < 1) $.msg.error(ret.info, 3, function () {
|
||||
$('#RoleTable').trigger('reload');
|
||||
});
|
||||
return false;
|
||||
}, false);
|
||||
});
|
||||
});
|
||||
|
||||
</script>
|
||||
|
||||
<!-- 列表排序权重模板 -->
|
||||
<script type="text/html" id="SortInputRoleTableTpl">
|
||||
<input type="number" min="0" data-blur-number="0" data-action-blur="{:request()->url()}" data-value="id#{{d.id}};action#sort;sort#{value}" data-loading="false" value="{{d.sort}}" class="layui-input text-center">
|
||||
</script>
|
||||
|
||||
<!-- 数据状态切换模板 -->
|
||||
<script type="text/html" id="StatusSwitchRoleTableTpl">
|
||||
<!--{if auth("state")}-->
|
||||
<input type="checkbox" value="{{d.id}}" lay-skin="switch" lay-text="{:lang('已激活')}|{:lang('已禁用')}" lay-filter="StatusSwitchRoleTable" {{-d.status>0?'checked':''}}>
|
||||
<!--{else}-->
|
||||
{{-d.status ? '<b class="color-green">{:lang("已启用")}</b>' : '<b class="color-red">{:lang("已禁用")}</b>'}}
|
||||
<!--{/if}-->
|
||||
</script>
|
||||
|
||||
<!-- 数据操作工具条模板 -->
|
||||
<script type="text/html" id="ToolbarRoleTableTpl">
|
||||
<!--{if auth('edit')}-->
|
||||
<a class="layui-btn layui-btn-primary layui-btn-sm" data-open='{:url("edit")}?id={{d.id}}'>{:lang("编 辑")}</a>
|
||||
<!--{/if}-->
|
||||
|
||||
<!--{if auth("remove")}-->
|
||||
<a class="layui-btn layui-btn-danger layui-btn-sm" data-action="{:url('remove')}" data-value="id#{{d.id}}" data-confirm="{:lang('确定要删除权限吗?')}">{:lang("删 除")}</a>
|
||||
<!--{/if}-->
|
||||
</script>
|
||||
{/block}
|
45
app/admin/view/auth/index_search.html
Normal file
45
app/admin/view/auth/index_search.html
Normal file
@ -0,0 +1,45 @@
|
||||
<fieldset>
|
||||
<legend>{:lang('条件搜索')}</legend>
|
||||
<form class="layui-form layui-form-pane form-search" action="{:sysuri()}" onsubmit="return false" method="get" autocomplete="off">
|
||||
|
||||
<div class="layui-form-item layui-inline">
|
||||
<label class="layui-form-label">{:lang('权限名称')}</label>
|
||||
<div class="layui-input-inline">
|
||||
<input name="title" value="{$get.title|default=''}" placeholder="{:lang('请输入权限名称')}" class="layui-input">
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="layui-form-item layui-inline">
|
||||
<label class="layui-form-label">{:lang('权限描述')}</label>
|
||||
<div class="layui-input-inline">
|
||||
<input name="desc" value="{$get.desc|default=''}" placeholder="{:lang('请输入权限描述')}" class="layui-input">
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="layui-form-item layui-inline">
|
||||
<label class="layui-form-label">{:lang('使用状态')}</label>
|
||||
<div class="layui-input-inline">
|
||||
<select class="layui-select" name="status">
|
||||
<option value=''>-- {:lang('全部')} --</option>
|
||||
{foreach [lang('已禁用记录'),lang('已激活记录')] as $k=>$v}
|
||||
{if isset($get.status) and $get.status eq $k.""}
|
||||
<option selected value="{$k}">{$v}</option>
|
||||
{else}
|
||||
<option value="{$k}">{$v}</option>
|
||||
{/if}{/foreach}
|
||||
</select>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="layui-form-item layui-inline">
|
||||
<label class="layui-form-label">{:lang('创建时间')}</label>
|
||||
<div class="layui-input-inline">
|
||||
<input data-date-range name="create_at" value="{$get.create_at|default=''}" placeholder="{:lang('请选择创建时间')}" class="layui-input">
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="layui-form-item layui-inline">
|
||||
<button class="layui-btn layui-btn-primary"><i class="layui-icon"></i> {:lang('搜 索')}</button>
|
||||
</div>
|
||||
</form>
|
||||
</fieldset>
|
68
app/admin/view/base/form.html
Normal file
68
app/admin/view/base/form.html
Normal file
@ -0,0 +1,68 @@
|
||||
<form action="{:sysuri()}" method="post" data-auto="true" class="layui-form layui-card" data-table-id="BaseTable">
|
||||
|
||||
<div class="layui-card-body padding-left-40">
|
||||
|
||||
<div class="layui-form-item label-required-prev">
|
||||
<div class="help-label"><b>数据类型</b>Data Type</div>
|
||||
{if isset($vo.type)}
|
||||
<label><input readonly value="{$vo.type|default=''}" class="layui-input think-bg-gray"></label>
|
||||
{else}
|
||||
<select class="layui-select" lay-filter="DataType">
|
||||
{foreach $types as $type}{if (isset($vo.type) and $type eq $vo.type) or ($type eq input('get.type'))}
|
||||
<option selected value="{$type}">{$type}</option>
|
||||
{else}
|
||||
<option value="{$type}">{$type}</option>
|
||||
{/if}{/foreach}
|
||||
</select>
|
||||
<script>
|
||||
(function (callable) {
|
||||
layui.form.on('select(DataType)', callable);
|
||||
callable({value: "{$vo.type|default=''}" || $('[lay-filter=DataType]').val()});
|
||||
})(function (data) {
|
||||
if (data.value === '--- 新增类型 ---') {
|
||||
$('#DataTypeInput').removeClass('layui-hide').find('input').val('').focus();
|
||||
} else {
|
||||
$('#DataTypeInput').addClass('layui-hide').find('input').val(data.value);
|
||||
}
|
||||
});
|
||||
</script>
|
||||
{/if}
|
||||
<p class="help-block">请选择数据类型,数据创建后不能再次修改哦 ~</p>
|
||||
<div id="DataTypeInput" class="layui-hide relative">
|
||||
<input class="layui-input" maxlength="20" name="type" required vali-name="数据类型" placeholder="请输入数据类型" value="{$vo.type|default=''}">
|
||||
<p class="help-block">请输入新的数据类型,数据创建后不能再次修改哦 ~</p>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<label class="layui-form-item relative block">
|
||||
<span class="help-label"><b>数据编码</b>Data Code</span>
|
||||
{if isset($vo.code)}
|
||||
<input readonly maxlength="100" class="layui-input think-bg-gray" name="code" value='{$vo.code|default=""}' required placeholder="请输入数据编码">
|
||||
{else}
|
||||
<input maxlength="100" class="layui-input" name="code" value='{$vo.code|default=""}' required vali-name="数据编码" placeholder="请输入数据编码">
|
||||
{/if}
|
||||
<span class="help-block">请输入新的数据编码,数据创建后不能再次修改,同种数据类型的数据编码不能出现重复 ~</span>
|
||||
</label>
|
||||
|
||||
<label class="layui-form-item relative block">
|
||||
<span class="help-label"><b>数据名称</b>Data Name</span>
|
||||
<input maxlength="500" class="layui-input" name="name" value='{$vo.name|default=""}' required vali-name="数据名称" placeholder="请输入数据名称">
|
||||
<span class="help-block">请输入当前数据名称,请尽量保持名称的唯一性,数据名称尽量不要出现重复 ~</span>
|
||||
</label>
|
||||
|
||||
<label class="layui-form-item relative block">
|
||||
<span class="help-label"><b>数据内容</b>Data Content</span>
|
||||
<textarea name="content" class="layui-textarea" placeholder="请输入数据内容">{$vo.content|default=''}</textarea>
|
||||
</label>
|
||||
|
||||
</div>
|
||||
|
||||
<div class="hr-line-dashed"></div>
|
||||
{notempty name='vo.id'}<input type='hidden' value='{$vo.id}' name='id'>{/notempty}
|
||||
|
||||
<div class="layui-form-item text-center">
|
||||
<button class="layui-btn" type='submit'>保存数据</button>
|
||||
<button class="layui-btn layui-btn-danger" type='button' data-confirm="确定要取消编辑吗?" data-close>取消编辑</button>
|
||||
</div>
|
||||
|
||||
</form>
|
86
app/admin/view/base/index.html
Normal file
86
app/admin/view/base/index.html
Normal file
@ -0,0 +1,86 @@
|
||||
{extend name='table'}
|
||||
|
||||
{block name="button"}
|
||||
<!--{if auth("add")}-->
|
||||
<button data-table-id="BaseTable" data-modal='{:url("add")}?type={$type|default=""}' class='layui-btn layui-btn-sm layui-btn-primary'>添加数据</button>
|
||||
<!--{/if}-->
|
||||
|
||||
<!--{if auth("remove")}-->
|
||||
<button data-table-id="BaseTable" data-action='{:url("remove")}' data-rule="id#{id}" data-confirm="确定要批量删除数据吗?" class='layui-btn layui-btn-sm layui-btn-primary'>批量删除</button>
|
||||
<!--{/if}-->
|
||||
{/block}
|
||||
|
||||
{block name="content"}
|
||||
<div class="layui-tab layui-tab-card">
|
||||
<ul class="layui-tab-title">
|
||||
{foreach $types as $t}{if isset($type) and $type eq $t}
|
||||
<li class="layui-this" data-open="{:sysuri()}?type={$t}">{$t}</li>
|
||||
{else}
|
||||
<li data-open="{:sysuri()}?type={$t}">{$t}</li>
|
||||
{/if}{/foreach}
|
||||
</ul>
|
||||
<div class="layui-tab-content">
|
||||
{include file='base/index_search'}
|
||||
<table id="BaseTable" data-url="{:request()->url()}" data-target-search="form.form-search"></table>
|
||||
</div>
|
||||
</div>
|
||||
{/block}
|
||||
|
||||
{block name='script'}
|
||||
<script>
|
||||
$(function () {
|
||||
// 初始化表格组件
|
||||
$('#BaseTable').layTable({
|
||||
even: true, height: 'full',
|
||||
sort: {field: 'sort desc,id', type: 'asc'},
|
||||
where: {type: '{$type|default=""}'},
|
||||
cols: [[
|
||||
{checkbox: true, fixed: true},
|
||||
{field: 'sort', title: '{:lang("排序权重")}', width: 100, align: 'center', sort: true, templet: '#SortInputTpl'},
|
||||
// {field: 'type', title: '数据类型', minWidth: 140, align: 'center'},
|
||||
{field: 'code', title: '数据编码', width: '20%', align: 'left'},
|
||||
{field: 'name', title: '数据名称', width: '30%', align: 'left'},
|
||||
{field: 'status', title: '数据状态', minWidth: 110, align: 'center', templet: '#StatusSwitchTpl'},
|
||||
{field: 'create_at', title: '创建时间', minWidth: 170, align: 'center', sort: true},
|
||||
{toolbar: '#toolbar', align: 'center', minWidth: 150, title: '数据操作', fixed: 'right'},
|
||||
]]
|
||||
});
|
||||
|
||||
// 数据状态切换操作
|
||||
layui.form.on('switch(StatusSwitch)', function (obj) {
|
||||
var data = {id: obj.value, status: obj.elem.checked > 0 ? 1 : 0};
|
||||
$.form.load("{:url('state')}", data, 'post', function (ret) {
|
||||
if (ret.code < 1) $.msg.error(ret.info, 3, function () {
|
||||
$('#BaseTable').trigger('reload');
|
||||
});
|
||||
return false;
|
||||
}, false);
|
||||
});
|
||||
});
|
||||
</script>
|
||||
|
||||
<!-- 列表排序权重模板 -->
|
||||
<script type="text/html" id="SortInputTpl">
|
||||
<input type="number" min="0" data-blur-number="0" data-action-blur="{:sysuri()}" data-value="id#{{d.id}};action#sort;sort#{value}" data-loading="false" value="{{d.sort}}" class="layui-input text-center">
|
||||
</script>
|
||||
|
||||
<!-- 数据状态切换模板 -->
|
||||
<script type="text/html" id="StatusSwitchTpl">
|
||||
<!--{if auth("state")}-->
|
||||
<input type="checkbox" value="{{d.id}}" lay-skin="switch" lay-text="已激活|已禁用" lay-filter="StatusSwitch" {{-d.status>0?'checked':''}}>
|
||||
<!--{else}-->
|
||||
{{-d.status ? '<b class="color-green">已启用</b>' : '<b class="color-red">已禁用</b>'}}
|
||||
<!--{/if}-->
|
||||
</script>
|
||||
|
||||
<!-- 数据操作工具条模板 -->
|
||||
<script type="text/html" id="toolbar">
|
||||
<!--{if auth('edit')}-->
|
||||
<a class="layui-btn layui-btn-primary layui-btn-sm" data-event-dbclick data-title="编辑数据" data-modal='{:url("edit")}?id={{d.id}}'>编 辑</a>
|
||||
<!--{/if}-->
|
||||
|
||||
<!--{if auth("remove")}-->
|
||||
<a class="layui-btn layui-btn-danger layui-btn-sm" data-confirm="确定要删除数据吗?" data-action="{:url('remove')}" data-value="id#{{d.id}}">删 除</a>
|
||||
<!--{/if}-->
|
||||
</script>
|
||||
{/block}
|
42
app/admin/view/base/index_search.html
Normal file
42
app/admin/view/base/index_search.html
Normal file
@ -0,0 +1,42 @@
|
||||
<form class="layui-form layui-form-pane form-search" action="{:sysuri()}" onsubmit="return false" method="get" autocomplete="off">
|
||||
|
||||
<div class="layui-form-item layui-inline">
|
||||
<label class="layui-form-label">{:lang('数据编码')}</label>
|
||||
<div class="layui-input-inline">
|
||||
<input name="code" value="{$get.code|default=''}" placeholder="{:lang('请输入数据编码')}" class="layui-input">
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="layui-form-item layui-inline">
|
||||
<label class="layui-form-label">{:lang('数据名称')}</label>
|
||||
<div class="layui-input-inline">
|
||||
<input name="name" value="{$get.name|default=''}" placeholder="{:lang('请输入数据名称')}" class="layui-input">
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="layui-form-item layui-inline">
|
||||
<label class="layui-form-label">{:lang('使用状态')}</label>
|
||||
<div class="layui-input-inline">
|
||||
<select class="layui-select" name="status">
|
||||
<option value=''>-- {:lang('全部')} --</option>
|
||||
{foreach [lang('已禁用记录'),lang('已激活记录')] as $k=>$v}
|
||||
{if isset($get.status) and $get.status eq $k.""}
|
||||
<option selected value="{$k}">{$v}</option>
|
||||
{else}
|
||||
<option value="{$k}">{$v}</option>
|
||||
{/if}{/foreach}
|
||||
</select>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="layui-form-item layui-inline">
|
||||
<label class="layui-form-label">{:lang('创建时间')}</label>
|
||||
<div class="layui-input-inline">
|
||||
<input data-date-range name="create_at" value="{$get.create_at|default=''}" placeholder="{:lang('请选择创建时间')}" class="layui-input">
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="layui-form-item layui-inline">
|
||||
<button class="layui-btn layui-btn-primary"><i class="layui-icon"></i> {:lang('搜 索')}</button>
|
||||
</div>
|
||||
</form>
|
234
app/admin/view/config/index.html
Normal file
234
app/admin/view/config/index.html
Normal file
@ -0,0 +1,234 @@
|
||||
{extend name="main"}
|
||||
|
||||
{block name="button"}
|
||||
<!--{if isset($issuper) and $issuper}-->
|
||||
<a class="layui-btn layui-btn-sm layui-btn-primary" data-load="{:url('admin/api.system/config')}">{:lang('清理无效配置')}</a>
|
||||
<!--{/if}-->
|
||||
|
||||
<!--{if auth('system')}-->
|
||||
<a class="layui-btn layui-btn-sm layui-btn-primary" data-modal="{:url('system')}">{:lang('修改系统参数')}</a>
|
||||
<!--{/if}-->
|
||||
{/block}
|
||||
|
||||
{block name="content"}
|
||||
<!--{notempty name='issuper'}-->
|
||||
<div class="layui-card padding-20 shadow">
|
||||
<div class="layui-card-header notselect">
|
||||
<span class="help-label">
|
||||
<b style="color:#333!important;">{:lang('运行模式')}</b>( {:lang('仅超级管理员可配置')} )
|
||||
</span>
|
||||
</div>
|
||||
<div class="layui-card-body">
|
||||
<div class="layui-btn-group shadow-mini nowrap">
|
||||
<!--{if $app->isDebug()}-->
|
||||
<a class="layui-btn layui-btn-sm layui-btn-active">{:lang('以开发模式运行')}</a>
|
||||
<a class="layui-btn layui-btn-sm layui-btn-primary" data-confirm="{:lang('确定要切换到生产模式运行吗?')}" data-load="{:url('admin/api.system/debug')}?state=1">{:lang('以生产模式运行')}</a>
|
||||
<!--{else}-->
|
||||
<a class="layui-btn layui-btn-sm layui-btn-primary" data-confirm="{:lang('确定要切换到开发模式运行吗?')}" data-load="{:url('admin/api.system/debug')}?state=0">{:lang('以开发模式运行')}</a>
|
||||
<a class="layui-btn layui-btn-sm layui-btn-active">{:lang('以生产模式运行')}</a>
|
||||
<!--{/if}-->
|
||||
</div>
|
||||
<div class="margin-top-20">
|
||||
<p><b>{:lang('开发模式')}</b>:{:lang('开发人员或在功能调试时使用,系统异常时会显示详细的错误信息,同时还会记录操作日志及数据库 SQL 语句信息。')}</p>
|
||||
<p><b>{:lang('生产模式')}</b>:{:lang('项目正式部署上线后使用,系统异常时统一显示 “%s”,只记录重要的异常日志信息,强烈推荐上线后使用此模式。',[config('app.error_message')])}</p>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="layui-card padding-20 shadow">
|
||||
<div class="layui-card-header notselect">
|
||||
<span class="help-label">
|
||||
<b style="color:#333!important;">{:lang('富编辑器')}</b>( {:lang('仅超级管理员可配置')} )
|
||||
</span>
|
||||
</div>
|
||||
<div class="layui-card-body layui-clear">
|
||||
<div class="layui-btn-group shadow-mini">
|
||||
{if !in_array(sysconf('base.editor'),['ckeditor4','ckeditor5','wangEditor','auto'])}{php}sysconf('base.editor','ckeditor4');{/php}{/if}
|
||||
{foreach ['ckeditor4'=>'CKEditor4','ckeditor5'=>'CKEditor5','wangEditor'=>'wangEditor','auto'=>lang('自适应模式')] as $k => $v}{if sysconf('base.editor') eq $k}
|
||||
{if auth('storage')}<a data-title="配置{$v}" class="layui-btn layui-btn-sm layui-btn-active">{$v}</a>{else}<a class="layui-btn layui-btn-sm layui-btn-active">{$v}</a>{/if}
|
||||
{else}
|
||||
{if auth('storage')}<a data-title="配置{$v}" data-action="{:url('admin/api.system/editor')}" data-value="editor#{$k}" class="layui-btn layui-btn-sm layui-btn-primary">{$v}</a>{else}<a class="layui-btn layui-btn-sm layui-btn-primary">{$v}</a>{/if}
|
||||
{/if}{/foreach}
|
||||
</div>
|
||||
<div class="margin-top-20 full-width pull-left">
|
||||
<p><b>CKEditor4</b>:{:lang('旧版本编辑器,对浏览器兼容较好,但内容编辑体验稍有不足。')}</p>
|
||||
<p><b>CKEditor5</b>:{:lang('新版本编辑器,只支持新特性浏览器,对内容编辑体验较好,推荐使用。')}</p>
|
||||
<p><b>wangEditor</b>:{:lang('国产优质富文本编辑器,对于小程序及App内容支持会更友好,推荐使用。')}</p>
|
||||
<p><b>{:lang('自适应模式')}</b>:{:lang('优先使用新版本编辑器,若浏览器不支持新版本时自动降级为旧版本编辑器。')}</p>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
<!--{/notempty}-->
|
||||
|
||||
<div class="layui-card padding-20 shadow">
|
||||
<div class="layui-card-header notselect">
|
||||
<span class="help-label">
|
||||
<b style="color:#333!important;">{:lang('存储引擎')}</b>( {:lang('文件默认存储方式')} )
|
||||
</span>
|
||||
</div>
|
||||
<!-- 初始化存储配置 -->
|
||||
{if !sysconf('storage.type')}{php}sysconf('storage.type','local');{/php}{/if}
|
||||
{if !sysconf('storage.link_type')}{php}sysconf('storage.link_type','none');{/php}{/if}
|
||||
{if !sysconf('storage.name_type')}{php}sysconf('storage.name_type','xmd5');{/php}{/if}
|
||||
{if !sysconf('storage.allow_exts')}{php}sysconf('storage.allow_exts','doc,gif,ico,jpg,mp3,mp4,p12,pem,png,rar,xls,xlsx');{/php}{/if}
|
||||
{if !sysconf('storage.local_http_protocol')}{php}sysconf('storage.local_http_protocol','follow');{/php}{/if}
|
||||
<div class="layui-card-body">
|
||||
<div class="layui-btn-group shadow-mini nowrap">
|
||||
{foreach $files as $k => $v}{if sysconf('storage.type') eq $k}
|
||||
{if auth('storage')}<a data-title="配置{$v}" data-modal="{:url('storage')}?type={$k}" class="layui-btn layui-btn-sm layui-btn-active">{$v}</a>{else}<a class="layui-btn layui-btn-sm layui-btn-active">{$v}</a>{/if}
|
||||
{else}
|
||||
{if auth('storage')}<a data-title="配置{$v}" data-modal="{:url('storage')}?type={$k}" class="layui-btn layui-btn-sm layui-btn-primary">{$v}</a>{else}<a class="layui-btn layui-btn-sm layui-btn-primary">{$v}</a>{/if}
|
||||
{/if}{/foreach}
|
||||
</div>
|
||||
<div class="margin-top-20 full-width">
|
||||
<p><b>{:lang('本地服务器存储')}</b>:{:lang('文件上传到本地服务器的 `static/upload` 目录,不支持大文件上传,占用服务器磁盘空间,访问时消耗服务器带宽流量。')}</p>
|
||||
<p><b>{:lang('自建Alist存储')}</b>:{:lang('文件上传到 Alist 存储的服务器或云存储空间,根据服务配置可支持大文件上传,不占用本身服务器空间及服务器带宽流量。')}</p>
|
||||
<p><b>{:lang('七牛云对象存储')}</b>:{:lang('文件上传到七牛云存储空间,支持大文件上传,不占用服务器空间及服务器带宽流量,支持 CDN 加速访问,访问量大时推荐使用。')}</p>
|
||||
<p><b>{:lang('又拍云USS存储')}</b>:{:lang('文件上传到又拍云 USS 存储空间,支持大文件上传,不占用服务器空间及服务器带宽流量,支持 CDN 加速访问,访问量大时推荐使用。')}</p>
|
||||
<p><b>{:lang('阿里云OSS存储')}</b>:{:lang('文件上传到阿里云 OSS 存储空间,支持大文件上传,不占用服务器空间及服务器带宽流量,支持 CDN 加速访问,访问量大时推荐使用。')}</p>
|
||||
<p><b>{:lang('腾讯云COS存储')}</b>:{:lang('文件上传到腾讯云 COS 存储空间,支持大文件上传,不占用服务器空间及服务器带宽流量,支持 CDN 加速访问,访问量大时推荐使用。')}</p>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="layui-card padding-20 shadow">
|
||||
<div class="layui-card-header notselect">
|
||||
<span class="help-label">
|
||||
<b style="color:#333!important;">{:lang('系统参数')}</b>( {:lang('当前系统配置参数')} )
|
||||
</span>
|
||||
</div>
|
||||
<div class="layui-card-body">
|
||||
<div class="layui-form-item">
|
||||
<div class="help-label"><b>{:lang('网站名称')}</b>Website</div>
|
||||
<label class="relative block">
|
||||
<input readonly value="{:sysconf('site_name')}" class="layui-input layui-bg-gray">
|
||||
<a data-copy="{:sysconf('site_name')}" class="layui-icon layui-icon-release input-right-icon"></a>
|
||||
</label>
|
||||
<div class="help-block">{:lang('网站名称及网站图标,将显示在浏览器的标签上。')}</div>
|
||||
</div>
|
||||
<div class="layui-form-item">
|
||||
<div class="help-label"><b>{:lang('管理程序名称')}</b>Name</div>
|
||||
<label class="relative block">
|
||||
<input readonly value="{:sysconf('app_name')}" class="layui-input layui-bg-gray">
|
||||
<a data-copy="{:sysconf('app_name')}" class="layui-icon layui-icon-release input-right-icon"></a>
|
||||
</label>
|
||||
<div class="help-block">{:lang('管理程序名称,将显示在后台左上角标题。')}</div>
|
||||
</div>
|
||||
<div class="layui-form-item">
|
||||
<div class="help-label"><b>{:lang('管理程序版本')}</b>Version</div>
|
||||
<label class="relative block">
|
||||
<input readonly value="{:sysconf('app_version')}" class="layui-input layui-bg-gray">
|
||||
<a data-copy="{:sysconf('app_version')}" class="layui-icon layui-icon-release input-right-icon"></a>
|
||||
</label>
|
||||
<div class="help-block">{:lang('管理程序版本,将显示在后台左上角标题。')}</div>
|
||||
</div>
|
||||
<div class="layui-form-item">
|
||||
<div class="help-label"><b>{:lang('公安备案号')}</b>Beian</div>
|
||||
<label class="relative block">
|
||||
<input readonly value="{:sysconf('beian')?:'-'}" class="layui-input layui-bg-gray">
|
||||
<a data-copy="{:sysconf('beian')}" class="layui-icon layui-icon-release input-right-icon"></a>
|
||||
</label>
|
||||
<p class="help-block">
|
||||
{:lang('公安备案号,可以在 %s 查询获取,将在登录页面下面显示。',['<a target="_blank" href="https://www.beian.gov.cn/portal/registerSystemInfo">www.beian.gov.cn</a>'])}
|
||||
</p>
|
||||
</div>
|
||||
<div class="layui-form-item">
|
||||
<div class="help-label"><b>{:lang('网站备案号')}</b>Miitbeian</div>
|
||||
<label class="relative block">
|
||||
<input readonly value="{:sysconf('miitbeian')?:'-'}" class="layui-input layui-bg-gray">
|
||||
<a data-copy="{:sysconf('miitbeian')}" class="layui-icon layui-icon-release input-right-icon"></a>
|
||||
</label>
|
||||
<div class="help-block">
|
||||
{:lang('网站备案号,可以在 %s 查询获取,将显示在登录页面下面。',['<a target="_blank" href="https://beian.miit.gov.cn">beian.miit.gov.cn</a>'])}
|
||||
</div>
|
||||
</div>
|
||||
<div class="layui-form-item">
|
||||
<div class="help-label"><b>{:lang('网站版权信息')}</b>Copyright</div>
|
||||
<label class="relative block">
|
||||
<input readonly value="{:sysconf('site_copy')}" class="layui-input layui-bg-gray">
|
||||
<a data-copy="{:sysconf('site_copy')}" class="layui-icon layui-icon-release input-right-icon"></a>
|
||||
</label>
|
||||
<div class="help-block">{:lang('网站版权信息,在后台登录页面显示版本信息并链接到备案到信息备案管理系统。')}</div>
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!--{if $app->isDebug()}-->
|
||||
<div class="layui-card padding-20 shadow">
|
||||
<div class="layui-card-header notselect">
|
||||
<span class="help-label">
|
||||
<b style="color:#333!important;">{:lang('系统信息')}</b>( {:lang('仅开发模式可见')} )
|
||||
</span>
|
||||
</div>
|
||||
<div class="layui-card-body">
|
||||
<table class="layui-table" lay-even>
|
||||
<tbody>
|
||||
<tr>
|
||||
<th class="nowrap text-center">{:lang('核心框架')}</th>
|
||||
<td><a target="_blank" href="https://www.thinkphp.cn">ThinkPHP Version {$framework.version|default='None'}</a></td>
|
||||
</tr>
|
||||
<tr>
|
||||
<th class="nowrap text-center">{:lang('平台框架')}</th>
|
||||
<td><a target="_blank" href="https://thinkadmin.top">ThinkAdmin Version {$thinkadmin.version|default='6.0.0'}</a></td>
|
||||
</tr>
|
||||
<tr>
|
||||
<th class="nowrap text-center">{:lang('操作系统')}</th>
|
||||
<td>{:php_uname()}</td>
|
||||
</tr>
|
||||
<tr>
|
||||
<th class="nowrap text-center">{:lang('运行环境')}</th>
|
||||
<td>{:ucfirst($request->server('SERVER_SOFTWARE',php_sapi_name()))} & PHP {$Think.const.PHP_VERSION} & {:ucfirst(app()->db->connect()->getConfig('type'))}</td>
|
||||
</tr>
|
||||
<!-- {notempty name='systemid'} -->
|
||||
<tr>
|
||||
<th class="nowrap text-center">{:lang('系统序号')}</th>
|
||||
<td>{$systemid|default=''}</td>
|
||||
</tr>
|
||||
<!-- {/notempty} -->
|
||||
</tbody>
|
||||
</table>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
{notempty name='plugins'}
|
||||
<div class="layui-card padding-20 shadow">
|
||||
<div class="layui-card-header notselect">
|
||||
<span class="help-label">
|
||||
<b style="color:#333!important;">{:lang('应用插件')}</b>( {:lang('仅开发模式可见')} )
|
||||
</span>
|
||||
</div>
|
||||
<div class="layui-card-body">
|
||||
<table class="layui-table" lay-even>
|
||||
<thead>
|
||||
<tr>
|
||||
<th class="nowrap text-center">{:lang('应用名称')}</th>
|
||||
<th class="nowrap text-center">{:lang('插件名称')}</th>
|
||||
<th class="nowrap text-left">{:lang('插件包名')}</th>
|
||||
<th class="nowrap text-center">{:lang('插件版本')}</th>
|
||||
<th class="nowrap text-center">{:lang('授权协议')}</th>
|
||||
</tr>
|
||||
</thead>
|
||||
<tbody>
|
||||
{foreach $plugins as $key=>$plugin}
|
||||
<tr>
|
||||
<td class="nowrap text-center">{$key}</td>
|
||||
<td class="nowrap text-center">{$plugin.name|lang}</td>
|
||||
<td class="nowrap text-left">
|
||||
{if empty($plugin.install.document)}{$plugin.package}
|
||||
{else}<a target="_blank" href="{$plugin.install.document}">{$plugin.package}</a>{/if}
|
||||
</td>
|
||||
<td class="nowrap text-center">{$plugin.install.version|default='unknow'}</td>
|
||||
<td class="nowrap text-center">
|
||||
{if empty($plugin.install.license)} -
|
||||
{elseif is_array($plugin.install.license)}{$plugin.install.license|join='、',###}
|
||||
{else}{$plugin.install.license|default='-'}{/if}
|
||||
</td>
|
||||
</tr>
|
||||
{/foreach}
|
||||
</tbody>
|
||||
</table>
|
||||
</div>
|
||||
</div>
|
||||
{/notempty}
|
||||
<!--{/if}-->
|
||||
{/block}
|
49
app/admin/view/config/storage-0.html
Normal file
49
app/admin/view/config/storage-0.html
Normal file
@ -0,0 +1,49 @@
|
||||
<div class="layui-form-item">
|
||||
<label class="layui-form-label label-required">
|
||||
<b class="color-green">命名方式</b><br><span class="nowrap color-desc">NameType</span>
|
||||
</label>
|
||||
<div class="layui-input-block">
|
||||
<div class="layui-input help-checks">
|
||||
{foreach ['xmd5'=>'文件哈希值 ( 支持秒传 )','date'=>'日期+随机 ( 普通上传 )'] as $k=>$v}
|
||||
<label class="think-radio notselect">
|
||||
{if sysconf('storage.name_type') eq $k}
|
||||
<input checked type="radio" name="storage.name_type" value="{$k}" lay-ignore> {$v}
|
||||
{else}
|
||||
<input type="radio" name="storage.name_type" value="{$k}" lay-ignore> {$v}
|
||||
{/if}
|
||||
</label>
|
||||
{/foreach}
|
||||
</div>
|
||||
<p class="help-block">类型为“文件哈希”时可以实现文件秒传功能,同一个文件只需上传一次节省存储空间,推荐使用。</p>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="layui-form-item">
|
||||
<label class="layui-form-label label-required">
|
||||
<b class="color-green">链接类型</b><br><span class="nowrap color-desc">LinkType</span>
|
||||
</label>
|
||||
<div class="layui-input-block">
|
||||
<div class="layui-input help-checks">
|
||||
{foreach ['none'=>'简洁链接','full'=>'完整链接','none+compress'=>'简洁并压缩图片','full+compress'=>'完整并压缩图片'] as $k=>$v}
|
||||
<label class="think-radio notselect">
|
||||
{if sysconf('storage.link_type') eq $k}
|
||||
<input checked type="radio" name="storage.link_type" value="{$k}" lay-ignore> {$v}
|
||||
{else}
|
||||
<input type="radio" name="storage.link_type" value="{$k}" lay-ignore> {$v}
|
||||
{/if}
|
||||
</label>
|
||||
{/foreach}
|
||||
</div>
|
||||
<p class="help-block">类型为“简洁链接”时链接将只返回 hash 地址,而“完整链接”将携带参数保留文件名,图片压缩功能云平台会单独收费。</p>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="layui-form-item">
|
||||
<label class="layui-form-label" for="storage.allow_exts">
|
||||
<b class="color-green">允许类型</b><br><span class="nowrap color-desc">AllowExts</span>
|
||||
</label>
|
||||
<div class="layui-input-block">
|
||||
<input id="storage.allow_exts" type="text" name="storage.allow_exts" value="{:sysconf('storage.allow_exts')}" required vali-name="文件后缀" placeholder="请输入系统文件上传后缀" class="layui-input">
|
||||
<p class="help-block">设置系统允许上传的文件后缀,多个以英文逗号隔开如:png,jpg,rar,doc,未包含在设置内的文件后缀将不被允许上传。</p>
|
||||
</div>
|
||||
</div>
|
98
app/admin/view/config/storage-alioss.html
Normal file
98
app/admin/view/config/storage-alioss.html
Normal file
@ -0,0 +1,98 @@
|
||||
<form action="{:sysuri()}" method="post" data-auto="true" class="layui-form layui-card">
|
||||
<div class="layui-card-body padding-top-20">
|
||||
|
||||
<div class="color-text layui-code text-center layui-bg-gray" style="border-left-width:1px;margin:0 0 15px 40px">
|
||||
<p class="margin-bottom-5 font-w7">文件将上传到 <a target="_blank" href="https://www.aliyun.com/minisite/goods?taskCode=shareNew2205&recordId=3646641&userCode=ztlqlu4v">阿里云</a> OSS 存储,需要配置 OSS 公开访问及跨域策略</p>
|
||||
<p>配置跨域访问 CORS 规则,设置:来源 Origin 为 *,允许 Methods 为 POST,允许 Headers 为 *</p>
|
||||
</div>
|
||||
|
||||
{include file='config/storage-0'}
|
||||
|
||||
<div class="layui-form-item">
|
||||
<label class="layui-form-label label-required">
|
||||
<b class="color-green">访问协议</b><br><span class="nowrap color-desc">Protocol</span>
|
||||
</label>
|
||||
<div class="layui-input-block">
|
||||
{if !sysconf('storage.alioss_http_protocol')}{php}sysconf('storage.alioss_http_protocol','http');{/php}{/if}
|
||||
<div class="layui-input help-checks">
|
||||
{foreach ['http'=>'HTTP','https'=>'HTTPS','auto'=>"AUTO"] as $protocol=>$remark}
|
||||
<label class="think-radio">
|
||||
{if sysconf('storage.alioss_http_protocol') eq $protocol}
|
||||
<input checked type="radio" name="storage.alioss_http_protocol" value="{$protocol}" lay-ignore> {$remark}
|
||||
{else}
|
||||
<input type="radio" name="storage.alioss_http_protocol" value="{$protocol}" lay-ignore> {$remark}
|
||||
{/if}
|
||||
</label>
|
||||
{/foreach}
|
||||
</div>
|
||||
<p class="help-block">阿里云OSS存储访问协议,其中 HTTPS 需要配置证书才能使用(AUTO 为相对协议)</p>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="layui-form-item">
|
||||
<label class="layui-form-label">
|
||||
<b class="color-green">存储区域</b><br><span class="nowrap color-desc label-required">Region</span>
|
||||
</label>
|
||||
<div class="layui-input-block">
|
||||
<select class="layui-select" name="storage.alioss_point" lay-search>
|
||||
{foreach $points as $point => $title}
|
||||
{if sysconf('storage.alioss_point') eq $point}
|
||||
<option selected value="{$point}">{$title}( {$point} )</option>
|
||||
{else}
|
||||
<option value="{$point}">{$title}( {$point} )</option>
|
||||
{/if}{/foreach}
|
||||
</select>
|
||||
<p class="help-block">阿里云OSS存储空间所在区域,需要严格对应储存所在区域才能上传文件</p>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="layui-form-item">
|
||||
<label class="layui-form-label" for="storage.alioss_bucket">
|
||||
<b class="color-green">空间名称</b><br><span class="nowrap color-desc">Bucket</span>
|
||||
</label>
|
||||
<div class="layui-input-block">
|
||||
<input id="storage.alioss_bucket" type="text" name="storage.alioss_bucket" value="{:sysconf('storage.alioss_bucket')}" required vali-name="空间名称" placeholder="请输入阿里云OSS存储 Bucket (空间名称)" class="layui-input">
|
||||
<p class="help-block">填写阿里云OSS存储空间名称,如:think-admin-oss(需要是全区唯一的值,不存在时会自动创建)</p>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="layui-form-item">
|
||||
<label class="layui-form-label" for="storage.alioss_http_domain">
|
||||
<b class="color-green">访问域名</b><br><span class="nowrap color-desc">Domain</span>
|
||||
</label>
|
||||
<div class="layui-input-block">
|
||||
<input id="storage.alioss_http_domain" type="text" name="storage.alioss_http_domain" value="{:sysconf('storage.alioss_http_domain')}" required vali-name="访问域名" placeholder="请输入阿里云OSS存储 Domain (访问域名)" class="layui-input">
|
||||
<p class="help-block">填写阿里云OSS存储外部访问域名,不需要填写访问协议,如:static.alioss.thinkadmin.top</p>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="layui-form-item">
|
||||
<label class="layui-form-label" for="storage.alioss_access_key">
|
||||
<b class="color-green">访问密钥</b><br><span class="nowrap color-desc">AccessKey</span>
|
||||
</label>
|
||||
<div class="layui-input-block">
|
||||
<input id="storage.alioss_access_key" type="text" name="storage.alioss_access_key" value="{:sysconf('storage.alioss_access_key')}" required vali-name="访问密钥" placeholder="请输入阿里云OSS存储 AccessKey (访问密钥)" class="layui-input">
|
||||
<p class="help-block">可以在 [ 阿里云 > 个人中心 ] 设置并获取到访问密钥</p>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="layui-form-item">
|
||||
<label class="layui-form-label" for="storage.alioss_secret_key">
|
||||
<b class="color-green">安全密钥</b><br><span class="nowrap color-desc">SecretKey</span>
|
||||
</label>
|
||||
<div class="layui-input-block">
|
||||
<input id="storage.alioss_secret_key" type="text" name="storage.alioss_secret_key" value="{:sysconf('storage.alioss_secret_key')}" maxlength="43" required vali-name="安全密钥" placeholder="请输入阿里云OSS存储 SecretKey (安全密钥)" class="layui-input">
|
||||
<p class="help-block">可以在 [ 阿里云 > 个人中心 ] 设置并获取到安全密钥</p>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="hr-line-dashed margin-left-40"></div>
|
||||
<input type="hidden" name="storage.type" value="alioss">
|
||||
|
||||
<div class="layui-form-item text-center padding-left-40">
|
||||
<button class="layui-btn" type="submit">保存配置</button>
|
||||
<button class="layui-btn layui-btn-danger" type='button' data-confirm="确定要取消修改吗?" data-close>取消修改</button>
|
||||
</div>
|
||||
|
||||
</div>
|
||||
</form>
|
81
app/admin/view/config/storage-alist.html
Normal file
81
app/admin/view/config/storage-alist.html
Normal file
@ -0,0 +1,81 @@
|
||||
<form action="{:sysuri()}" method="post" data-auto="true" class="layui-form layui-card">
|
||||
<div class="layui-card-body padding-top-20">
|
||||
|
||||
<div class="color-text layui-code text-center layui-bg-gray" style="border-left-width:1px;margin:0 0 15px 40px">
|
||||
<p class="margin-bottom-5 font-w7">文件将上传到 <a target="_blank" href="https://alist.nn.ci/zh/">Alist</a> 自建存储,需要自行搭建 <a target="_blank" href="https://alist.nn.ci/zh/">Alist</a> 存储服务器。</p>
|
||||
<p>Alist 是一个支持多种存储的文件列表程序,可将各种云盘及本地磁盘资源进行整合。</p>
|
||||
<p>建议不要开放匿名用户访问,尽量使用独立账号管理,需要关闭 “签名所有” 让文件可以直接访问。</p>
|
||||
</div>
|
||||
|
||||
{include file='config/storage-0'}
|
||||
|
||||
<div class="layui-form-item">
|
||||
<label class="layui-form-label label-required">
|
||||
<b class="color-green">访问协议</b><br><span class="nowrap color-desc">Protocol</span>
|
||||
</label>
|
||||
<div class="layui-input-block">
|
||||
{if !sysconf('storage.alist_http_protocol')}{php}sysconf('storage.alist_http_protocol','http');{/php}{/if}
|
||||
<div class="layui-input help-checks">
|
||||
{foreach ['http'=>'HTTP','https'=>'HTTPS','auto'=>"AUTO"] as $protocol=>$remark}
|
||||
<label class="think-radio">
|
||||
{if sysconf('storage.alist_http_protocol') eq $protocol}
|
||||
<input checked type="radio" name="storage.alist_http_protocol" value="{$protocol}" lay-ignore> {$remark}
|
||||
{else}
|
||||
<input type="radio" name="storage.alist_http_protocol" value="{$protocol}" lay-ignore> {$remark}
|
||||
{/if}
|
||||
</label>
|
||||
{/foreach}
|
||||
</div>
|
||||
<p class="help-block">请选择 Alist 存储访问协议,其中 HTTPS 需要配置证书才能使用( AUTO 为相对协议 )</p>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="layui-form-item">
|
||||
<label class="layui-form-label" for="storage.alist_http_domain">
|
||||
<b class="color-green">访问域名</b><br><span class="nowrap color-desc">Domain</span>
|
||||
</label>
|
||||
<div class="layui-input-block">
|
||||
<input id="storage.alist_http_domain" type="text" name="storage.alist_http_domain" value="{:sysconf('storage.alist_http_domain')}" required vali-name="访问域名" placeholder="请输入 Alist 存储的访问域名" class="layui-input">
|
||||
<p class="help-block">请填写 Alist 存储访问域名,不需要填写访问协议,如:storage.thinkadmin.top</p>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="layui-form-item">
|
||||
<label class="layui-form-label" for="storage.alist_savepath">
|
||||
<b class="color-green">存储目录</b><br><span class="nowrap color-desc">Directory</span>
|
||||
</label>
|
||||
<div class="layui-input-block">
|
||||
<input id="storage.alist_savepath" type="text" name="storage.alist_savepath" value="{:sysconf('storage.alist_savepath')}" required vali-name="存储目录" placeholder="请输入 Alist 存储目录" class="layui-input">
|
||||
<p class="help-block">请填写 Alist 用户基本目录的相对存储位置,填写 / 表示用户基本目录( 需要拥有读写权限 )</p>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="layui-form-item">
|
||||
<label class="layui-form-label" for="storage.alist_username">
|
||||
<b class="color-green">用户账号</b><br><span class="nowrap color-desc">Username</span>
|
||||
</label>
|
||||
<div class="layui-input-block">
|
||||
<input id="storage.alist_username" name="storage.alist_username" value="{:sysconf('storage.alist_username')}" maxlength="100" required vali-name="用户账号" placeholder="请输入 Alist 存储的用户账号" class="layui-input">
|
||||
<p class="help-block">请填写 Alist 用户账号,注意此账号需要拥有上面存储目录的访问权限。</p>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="layui-form-item">
|
||||
<label class="layui-form-label" for="storage.alist_password">
|
||||
<b class="color-green">用户密码</b><br><span class="nowrap color-desc">Password</span>
|
||||
</label>
|
||||
<div class="layui-input-block">
|
||||
<input id="storage.alist_password" name="storage.alist_password" value="{:sysconf('storage.alist_password')}" maxlength="100" required vali-name="用户密码" placeholder="请输入 Alist 存储的用户密码" class="layui-input">
|
||||
<p class="help-block">请填写 Alist 用户登录密码,用于生成文件上传的接口认证令牌,如果填写错误将无法上传文件。</p>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="hr-line-dashed margin-left-40"></div>
|
||||
<input type="hidden" name="storage.type" value="alist">
|
||||
|
||||
<div class="layui-form-item text-center padding-left-40">
|
||||
<button class="layui-btn" type="submit">保存配置</button>
|
||||
<button class="layui-btn layui-btn-danger" type='button' data-confirm="确定要取消修改吗?" data-close>取消修改</button>
|
||||
</div>
|
||||
</div>
|
||||
</form>
|
51
app/admin/view/config/storage-local.html
Normal file
51
app/admin/view/config/storage-local.html
Normal file
@ -0,0 +1,51 @@
|
||||
<form action="{:sysuri()}" method="post" data-auto="true" class="layui-form layui-card">
|
||||
<div class="layui-card-body padding-top-20">
|
||||
|
||||
<div class="color-text layui-code text-center layui-bg-gray" style="border-left-width:1px;margin:0 0 15px 40px">
|
||||
<p class="margin-bottom-5 font-w7">文件将存储在本地服务器,默认保存在 public/upload 目录,文件以 HASH 命名。</p>
|
||||
<p>文件存储的目录需要有读写权限,有足够的存储空间。<span class="color-red">特别注意,本地存储暂不支持图片压缩!</span></p>
|
||||
</div>
|
||||
|
||||
{include file='config/storage-0'}
|
||||
|
||||
<div class="layui-form-item">
|
||||
<label class="layui-form-label label-required">
|
||||
<b class="color-green">访问协议</b><br><span class="nowrap color-desc">Protocol</span>
|
||||
</label>
|
||||
<div class="layui-input-block">
|
||||
{if !sysconf('storage.local_http_protocol')}{php}sysconf('storage.local_http_protocol','follow');{/php}{/if}
|
||||
<div class="layui-input help-checks">
|
||||
{foreach ['follow'=>'FOLLOW','http'=>'HTTP','https'=>'HTTPS','path'=>'PATH','auto'=>'AUTO'] as $protocol=>$remark}
|
||||
<label class="think-radio">
|
||||
{if sysconf('storage.local_http_protocol') eq $protocol}
|
||||
<input checked type="radio" name="storage.local_http_protocol" value="{$protocol}" lay-ignore> {$remark}
|
||||
{else}
|
||||
<input type="radio" name="storage.local_http_protocol" value="{$protocol}" lay-ignore> {$remark}
|
||||
{/if}
|
||||
</label>
|
||||
{/foreach}
|
||||
</div>
|
||||
<p class="help-block">本地存储访问协议,其中 HTTPS 需要配置证书才能使用( FOLLOW 跟随系统,PATH 文件路径,AUTO 相对协议 )</p>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="layui-form-item">
|
||||
<label class="layui-form-label" for="storage.local_http_domain">
|
||||
<b class="color-green">访问域名</b><br><span class="nowrap color-desc">Domain</span>
|
||||
</label>
|
||||
<div class="layui-input-block">
|
||||
<input id="storage.local_http_domain" type="text" name="storage.local_http_domain" value="{:sysconf('storage.local_http_domain')}" placeholder="请输入上传后的访问域名 (非必填项)" class="layui-input">
|
||||
<p class="help-block">填写上传后的访问域名(不指定时根据当前访问地址自动计算),不需要填写访问协议,如:static.thinkadmin.top</p>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="hr-line-dashed margin-left-40"></div>
|
||||
<input type="hidden" name="storage.type" value="local">
|
||||
|
||||
<div class="layui-form-item text-center padding-left-40">
|
||||
<button class="layui-btn" type="submit">保存配置</button>
|
||||
<button class="layui-btn layui-btn-danger" type='button' data-confirm="确定要取消修改吗?" data-close>取消修改</button>
|
||||
</div>
|
||||
|
||||
</div>
|
||||
</form>
|
97
app/admin/view/config/storage-qiniu.html
Normal file
97
app/admin/view/config/storage-qiniu.html
Normal file
@ -0,0 +1,97 @@
|
||||
<form action="{:sysuri()}" method="post" data-auto="true" class="layui-form layui-card">
|
||||
<div class="layui-card-body padding-top-20">
|
||||
|
||||
<div class="color-text layui-code text-center layui-bg-gray" style="border-left-width:1px;margin:0 0 15px 40px">
|
||||
<p class="margin-bottom-5 font-w7">文件将上传到 <a target="_blank" href="https://s.qiniu.com/rYr26v">七牛云</a> 存储,对象存储需要配置为公开访问的 Bucket 空间</p>
|
||||
完成实名认证后可获得 10G 免费存储空间哦!<a target="_blank" href="https://s.qiniu.com/rYr26v">我要免费申请</a>
|
||||
</div>
|
||||
|
||||
{include file='config/storage-0'}
|
||||
|
||||
<div class="layui-form-item">
|
||||
<label class="layui-form-label label-required">
|
||||
<b class="color-green">访问协议</b><br><span class="nowrap color-desc">Protocol</span>
|
||||
</label>
|
||||
<div class="layui-input-block">
|
||||
{if !sysconf('storage.qiniu_http_protocol')}{php}sysconf('storage.qiniu_http_protocol','http');{/php}{/if}
|
||||
<div class="layui-input help-checks">
|
||||
{foreach ['http'=>'HTTP','https'=>'HTTPS','auto'=>"AUTO"] as $protocol=>$remark}
|
||||
<label class="think-radio">
|
||||
{if sysconf('storage.qiniu_http_protocol') eq $protocol}
|
||||
<input checked type="radio" name="storage.qiniu_http_protocol" value="{$protocol}" lay-ignore> {$remark}
|
||||
{else}
|
||||
<input type="radio" name="storage.qiniu_http_protocol" value="{$protocol}" lay-ignore> {$remark}
|
||||
{/if}
|
||||
</label>
|
||||
{/foreach}
|
||||
</div>
|
||||
<p class="help-block">七牛云存储访问协议,其中 HTTPS 需要配置证书才能使用( AUTO 为相对协议 )</p>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="layui-form-item">
|
||||
<label class="layui-form-label">
|
||||
<b class="color-green">存储区域</b><br><span class="nowrap color-desc label-required">Region</span>
|
||||
</label>
|
||||
<div class="layui-input-block">
|
||||
<select class="layui-select" name="storage.qiniu_region" lay-search>
|
||||
{foreach $points as $point => $title}
|
||||
{if sysconf('storage.qiniu_region') eq $point}
|
||||
<option selected value="{$point}">{$title}( {$point} )</option>
|
||||
{else}
|
||||
<option value="{$point}">{$title}( {$point} )</option>
|
||||
{/if}{/foreach}
|
||||
</select>
|
||||
<p class="help-block">七牛云存储空间所在区域,需要严格对应储存所在区域才能上传文件</p>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="layui-form-item">
|
||||
<label class="layui-form-label" for="storage.qiniu_bucket">
|
||||
<b class="color-green">空间名称</b><br><span class="nowrap color-desc">Bucket</span>
|
||||
</label>
|
||||
<div class="layui-input-block">
|
||||
<input id="storage.qiniu_bucket" type="text" name="storage.qiniu_bucket" value="{:sysconf('storage.qiniu_bucket')}" required vali-name="空间名称" placeholder="请输入七牛云存储 Bucket (空间名称)" class="layui-input">
|
||||
<p class="help-block">填写七牛云存储空间名称,如:static</p>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="layui-form-item">
|
||||
<label class="layui-form-label" for="storage.qiniu_http_domain">
|
||||
<b class="color-green">访问域名</b><br><span class="nowrap color-desc">Domain</span>
|
||||
</label>
|
||||
<div class="layui-input-block">
|
||||
<input id="storage.qiniu_http_domain" type="text" name="storage.qiniu_http_domain" value="{:sysconf('storage.qiniu_http_domain')}" required vali-name="访问域名" placeholder="请输入七牛云存储 Domain (访问域名)" class="layui-input">
|
||||
<p class="help-block">填写七牛云存储访问域名,不需要填写访问协议,如:static.qiniu.thinkadmin.top</p>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="layui-form-item">
|
||||
<label class="layui-form-label" for="storage.qiniu_access_key">
|
||||
<b class="color-green">访问密钥</b><br><span class="nowrap color-desc">AccessKey</span>
|
||||
</label>
|
||||
<div class="layui-input-block">
|
||||
<input id="storage.qiniu_access_key" type="text" name="storage.qiniu_access_key" value="{:sysconf('storage.qiniu_access_key')}" required vali-name="访问密钥" placeholder="请输入七牛云授权 AccessKey (访问密钥)" class="layui-input">
|
||||
<p class="help-block">可以在 [ 七牛云 > 个人中心 ] 设置并获取到访问密钥</p>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="layui-form-item">
|
||||
<label class="layui-form-label" for="storage.qiniu_secret_key">
|
||||
<b class="color-green">安全密钥</b><br><span class="nowrap color-desc">SecretKey</span>
|
||||
</label>
|
||||
<div class="layui-input-block">
|
||||
<input id="storage.qiniu_secret_key" type="text" name="storage.qiniu_secret_key" value="{:sysconf('storage.qiniu_secret_key')}" maxlength="43" required vali-name="安全密钥" placeholder="请输入七牛云授权 SecretKey (安全密钥)" class="layui-input">
|
||||
<p class="help-block">可以在 [ 七牛云 > 个人中心 ] 设置并获取到安全密钥</p>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="hr-line-dashed margin-left-40"></div>
|
||||
<input type="hidden" name="storage.type" value="qiniu">
|
||||
|
||||
<div class="layui-form-item text-center padding-left-40">
|
||||
<button class="layui-btn" type="submit">保存配置</button>
|
||||
<button class="layui-btn layui-btn-danger" type='button' data-confirm="确定要取消修改吗?" data-close>取消修改</button>
|
||||
</div>
|
||||
</div>
|
||||
</form>
|
96
app/admin/view/config/storage-txcos.html
Normal file
96
app/admin/view/config/storage-txcos.html
Normal file
@ -0,0 +1,96 @@
|
||||
<form action="{:sysuri()}" method="post" data-auto="true" class="layui-form layui-card">
|
||||
<div class="layui-card-body padding-top-20">
|
||||
|
||||
<div class="color-text layui-code text-center layui-bg-gray" style="border-left-width:1px;margin:0 0 15px 40px">
|
||||
<p class="margin-bottom-5 font-w7">文件将上传到 <a target="_blank" href="https://curl.qcloud.com/4t0Mbw2K">腾讯云</a> COS 存储,需要配置 COS 公有读私有写访问权限及跨域策略</p>
|
||||
<p>配置跨域访问 CORS 规则,设置来源 Origin 为 *,允许 Methods 为 POST,允许 Headers 为 *</p>
|
||||
</div>
|
||||
|
||||
{include file='config/storage-0'}
|
||||
|
||||
<div class="layui-form-item">
|
||||
<label class="layui-form-label label-required">
|
||||
<b class="color-green">访问协议</b><br><span class="nowrap color-desc">Protocol</span>
|
||||
</label>
|
||||
<div class="layui-input-block">
|
||||
{if !sysconf('storage.txcos_http_protocol')}{php}sysconf('storage.txcos_http_protocol','http');{/php}{/if}
|
||||
{foreach ['http'=>'HTTP','https'=>'HTTPS','auto'=>"AUTO"] as $protocol=>$remark}
|
||||
<label class="think-radio">
|
||||
{if sysconf('storage.txcos_http_protocol') eq $protocol}
|
||||
<input checked type="radio" name="storage.txcos_http_protocol" value="{$protocol}" lay-ignore> {$remark}
|
||||
{else}
|
||||
<input type="radio" name="storage.txcos_http_protocol" value="{$protocol}" lay-ignore> {$remark}
|
||||
{/if}
|
||||
</label>
|
||||
{/foreach}
|
||||
<p class="help-block">腾讯云COS存储访问协议,其中 HTTPS 需要配置证书才能使用( AUTO 为相对协议 )</p>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="layui-form-item">
|
||||
<label class="layui-form-label">
|
||||
<b class="color-green">存储区域</b><br><span class="nowrap color-desc label-required">Region</span>
|
||||
</label>
|
||||
<div class="layui-input-block">
|
||||
<select class="layui-select" name="storage.txcos_point" lay-search>
|
||||
{foreach $points as $point => $title}
|
||||
{if sysconf('storage.txcos_point') eq $point}
|
||||
<option selected value="{$point}">{$title}( {$point} )</option>
|
||||
{else}
|
||||
<option value="{$point}">{$title}( {$point} )</option>
|
||||
{/if}{/foreach}
|
||||
</select>
|
||||
<p class="help-block">腾讯云COS存储空间所在区域,需要严格对应储存所在区域才能上传文件</p>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="layui-form-item">
|
||||
<label class="layui-form-label" for="storage.txcos_bucket">
|
||||
<b class="color-green">空间名称</b><br><span class="nowrap color-desc">Bucket</span>
|
||||
</label>
|
||||
<div class="layui-input-block">
|
||||
<input id="storage.txcos_bucket" type="text" name="storage.txcos_bucket" value="{:sysconf('storage.txcos_bucket')}" required vali-name="空间名称" placeholder="请输入腾讯云COS存储 Bucket" class="layui-input">
|
||||
<p class="help-block">填写腾讯云COS存储空间名称,如:thinkadmin-1251143395</p>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="layui-form-item">
|
||||
<label class="layui-form-label" for="storage.txcos_http_domain">
|
||||
<b class="color-green">访问域名</b><br><span class="nowrap color-desc">Domain</span>
|
||||
</label>
|
||||
<div class="layui-input-block">
|
||||
<input id="storage.txcos_http_domain" type="text" name="storage.txcos_http_domain" value="{:sysconf('storage.txcos_http_domain')}" required vali-name="访问域名" placeholder="请输入腾讯云COS存储 Domain" class="layui-input">
|
||||
<p class="help-block">填写腾讯云COS存储外部访问域名,不需要填写访问协议,如:static.txcos.thinkadmin.top</p>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="layui-form-item">
|
||||
<label class="layui-form-label" for="storage.txcos_access_key">
|
||||
<b class="color-green">访问密钥</b><br><span class="nowrap color-desc">AccessKey</span>
|
||||
</label>
|
||||
<div class="layui-input-block">
|
||||
<input id="storage.txcos_access_key" type="text" name="storage.txcos_access_key" value="{:sysconf('storage.txcos_access_key')}" required vali-name="访问密钥" placeholder="请输入腾讯云COS存储 AccessKey" class="layui-input">
|
||||
<p class="help-block">可以在 [ 腾讯云 > 个人中心 ] 设置并获取到访问密钥</p>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="layui-form-item">
|
||||
<label class="layui-form-label" for="storage.txcos_secret_key">
|
||||
<b class="color-green">安全密钥</b><br><span class="nowrap color-desc">SecretKey</span>
|
||||
</label>
|
||||
<div class="layui-input-block">
|
||||
<input id="storage.txcos_secret_key" type="text" name="storage.txcos_secret_key" value="{:sysconf('storage.txcos_secret_key')}" maxlength="43" required vali-name="安全密钥" placeholder="请输入腾讯云COS存储 SecretKey" class="layui-input">
|
||||
<p class="help-block">可以在 [ 腾讯云 > 个人中心 ] 设置并获取到安全密钥</p>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="hr-line-dashed margin-left-40"></div>
|
||||
<input type="hidden" name="storage.type" value="txcos">
|
||||
|
||||
<div class="layui-form-item text-center padding-left-40">
|
||||
<button class="layui-btn" type="submit">保存配置</button>
|
||||
<button class="layui-btn layui-btn-danger" type='button' data-confirm="确定要取消修改吗?" data-close>取消修改</button>
|
||||
</div>
|
||||
|
||||
</div>
|
||||
</form>
|
81
app/admin/view/config/storage-upyun.html
Normal file
81
app/admin/view/config/storage-upyun.html
Normal file
@ -0,0 +1,81 @@
|
||||
<form action="{:sysuri()}" method="post" data-auto="true" class="layui-form layui-card">
|
||||
<div class="layui-card-body padding-top-20">
|
||||
|
||||
<div class="color-text layui-code text-center layui-bg-gray" style="border-left-width:1px;margin:0 0 15px 40px">
|
||||
<p class="margin-bottom-5 font-w7">文件将上传到 <a target="_blank" href="https://console.upyun.com/register/?invite=PN1cRmjRb">又拍云</a> USS 存储,需要配置 USS 公开访问及跨域策略</p>
|
||||
<p>配置跨域访问 CORS 规则,设置来源 Origin 为 *,允许 Methods 为 POST,允许 Headers 为 *</p>
|
||||
</div>
|
||||
|
||||
{include file='config/storage-0'}
|
||||
|
||||
<div class="layui-form-item">
|
||||
<label class="layui-form-label label-required">
|
||||
<b class="color-green">访问协议</b><br><span class="nowrap color-desc">Protocol</span>
|
||||
</label>
|
||||
<div class="layui-input-block">
|
||||
{if !sysconf('storage.upyun_http_protocol')}{php}sysconf('storage.upyun_http_protocol','http');{/php}{/if}
|
||||
<div class="layui-input help-checks">
|
||||
{foreach ['http'=>'HTTP','https'=>'HTTPS','auto'=>"AUTO"] as $protocol=>$remark}
|
||||
<label class="think-radio">
|
||||
{if sysconf('storage.upyun_http_protocol') eq $protocol}
|
||||
<input checked type="radio" name="storage.upyun_http_protocol" value="{$protocol}" lay-ignore> {$remark}
|
||||
{else}
|
||||
<input type="radio" name="storage.upyun_http_protocol" value="{$protocol}" lay-ignore> {$remark}
|
||||
{/if}
|
||||
</label>
|
||||
{/foreach}
|
||||
</div>
|
||||
<p class="help-block">又拍云存储访问协议,其中 HTTPS 需要配置证书才能使用(AUTO 为相对协议)</p>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="layui-form-item">
|
||||
<label class="layui-form-label" for="storage.upyun_bucket">
|
||||
<b class="color-green">空间名称</b><br><span class="nowrap color-desc">Bucket</span>
|
||||
</label>
|
||||
<div class="layui-input-block">
|
||||
<input id="storage.upyun_bucket" name="storage.upyun_bucket" value="{:sysconf('storage.upyun_bucket')}" required vali-name="空间名称" placeholder="请输入又拍云存储 Bucket (空间名称)" class="layui-input">
|
||||
<p class="help-block">填写又拍云存储空间名称,如:think-admin-uss(需要是全区唯一的值,不存在时会自动创建)</p>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="layui-form-item">
|
||||
<label class="layui-form-label" for="storage.upyun_http_domain">
|
||||
<b class="color-green">访问域名</b><br><span class="nowrap color-desc">Domain</span>
|
||||
</label>
|
||||
<div class="layui-input-block">
|
||||
<input id="storage.upyun_http_domain" name="storage.upyun_http_domain" value="{:sysconf('storage.upyun_http_domain')}" required vali-name="访问域名" placeholder="请输入又拍云存储 Domain (访问域名)" class="layui-input">
|
||||
<p class="help-block">填写又拍云存储外部访问域名,不需要填写访问协议,如:static.uss.thinkadmin.top</p>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="layui-form-item">
|
||||
<label class="layui-form-label" for="storage.upyun_access_key">
|
||||
<b class="color-green">操作账号</b><br><span class="nowrap color-desc">Username</span>
|
||||
</label>
|
||||
<div class="layui-input-block">
|
||||
<input id="storage.upyun_access_key" name="storage.upyun_access_key" value="{:sysconf('storage.upyun_access_key')}" maxlength="100" required vali-name="操作员账号" placeholder="请输入又拍云存储 Username (操作员账号)" class="layui-input">
|
||||
<p class="help-block">可以在 [ 账户管理 > 操作员 ] 设置操作员账号并将空间给予授权。</p>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="layui-form-item">
|
||||
<label class="layui-form-label" for="storage.upyun_secret_key">
|
||||
<b class="color-green">操作密码</b><br><span class="nowrap color-desc">Password</span>
|
||||
</label>
|
||||
<div class="layui-input-block">
|
||||
<input id="storage.upyun_secret_key" name="storage.upyun_secret_key" value="{:sysconf('storage.upyun_secret_key')}" maxlength="100" required vali-name="操作员密码" placeholder="请输入又拍云存储 Password (操作员密码)" class="layui-input">
|
||||
<p class="help-block">可以在 [ 账户管理 > 操作员 ] 设置操作员密码并将空间给予授权</p>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="hr-line-dashed margin-left-40"></div>
|
||||
<input type="hidden" name="storage.type" value="upyun">
|
||||
|
||||
<div class="layui-form-item text-center padding-left-40">
|
||||
<button class="layui-btn" type="submit">保存配置</button>
|
||||
<button class="layui-btn layui-btn-danger" type='button' data-confirm="确定要取消修改吗?" data-close>取消修改</button>
|
||||
</div>
|
||||
|
||||
</div>
|
||||
</form>
|
129
app/admin/view/config/system.html
Normal file
129
app/admin/view/config/system.html
Normal file
@ -0,0 +1,129 @@
|
||||
<form action="{:sysuri()}" method="post" data-auto="true" class="layui-form layui-card">
|
||||
<div class="layui-card-body padding-left-40">
|
||||
|
||||
<div class="layui-row layui-col-space15 margin-bottom-5">
|
||||
<div class="layui-col-xs4 padding-bottom-0">
|
||||
<label class="relative block">
|
||||
<span class="help-label"><b>登录表单标题</b>Login Name</span>
|
||||
<input name="login_name" required placeholder="请输入登录页面的表单标题" vali-name="登录标题" value="{:sysconf('login_name')?:'系统管理'}" class="layui-input">
|
||||
</label>
|
||||
</div>
|
||||
<div class="layui-col-xs4 padding-bottom-0">
|
||||
<div class="help-label label-required-prev"><b>后台登录入口</b>Login Entry</div>
|
||||
<label class="layui-input relative block nowrap label-required-null">
|
||||
<span>{:sysuri('@',[],false,true)}</span>
|
||||
<input autofocus required pattern="[a-zA-Z_][a-zA-Z0-9_]*" vali-name="登录入口" placeholder="请输入后台登录入口" class="layui-input inline-block padding-0 border-0" style="width:100px;background:none" value="{:substr(sysuri('admin/index/index',[],false), strlen(sysuri('@')))}" name="xpath">
|
||||
</label>
|
||||
</div>
|
||||
<div class="layui-col-xs4 padding-bottom-0">
|
||||
<div class="help-label label-required-prev"><b>后台默认配色</b>Theme Style</div>
|
||||
<select class="layui-select" name="site_theme" lay-filter="SiteTheme">
|
||||
{foreach $themes as $k=>$v}{if sysconf('base.site_theme') eq $k}
|
||||
<option selected value="{$k}">{$v}</option>
|
||||
{else}
|
||||
<option value="{$k}">{$v}</option>
|
||||
{/if}{/foreach}
|
||||
</select>
|
||||
</div>
|
||||
<div class="layui-col-xs12 padding-top-0 padding-bottom-0">
|
||||
<span class="help-block">后台登录入口是由英文字母开头,且不能有相同名称的模块,设置之后原地址不能继续访问,请谨慎配置 ~</span>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="layui-form-item margin-bottom-5">
|
||||
<div class="help-label"><b>登录背景图片</b>Background Image</div>
|
||||
<div class="layui-textarea help-images">
|
||||
<input type="hidden" value="{:sysconf('login_image')}" name="login_image">
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="layui-form-item margin-bottom-5">
|
||||
<div class="help-label label-required-prev"><b>JWT 接口密钥</b>Jwt Key</div>
|
||||
<label class="relative block label-required-null">
|
||||
<input class="layui-input" pattern=".{32}" maxlength="32" required vali-name="接口密钥" placeholder="请输入32位JWT接口密钥" name="data.jwtkey" value="{:sysconf('data.jwtkey')?:md5(uniqid(strval(rand(1000,9999)),true))}">
|
||||
<a class="input-right-icon layui-icon layui-icon-refresh" id="RefreshJwtKey"></a>
|
||||
</label>
|
||||
<div class="help-block sub-span-blue">
|
||||
请输入 <span>32</span> 位 <span>JWT</span> 接口密钥,在使用 <span>JWT</span> 接口时需要使用此密钥进行加密及签名!
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="layui-form-item margin-bottom-5">
|
||||
<div class="help-label label-required-prev"><b>浏览器小图标</b>Browser Icon</div>
|
||||
<div class="relative block label-required-null">
|
||||
<input class="layui-input" required pattern="url" vali-name="图标文件" data-tips-image data-tips-hover placeholder="请上传浏览器图标" value="{:sysconf('site_icon')}" name="site_icon">
|
||||
<a class="input-right-icon layui-icon layui-icon-upload-drag" data-file="btn" data-type="png,jpg,jpeg" data-field="site_icon"></a>
|
||||
</div>
|
||||
<div class="help-block sub-span-blue">
|
||||
建议上传 <span>128x128</span> 或 <span>256x256</span> 的 <span>JPG</span>,<span>PNG</span>,<span>JPEG</span> 图片,保存后会自动生成 <span>48x48</span> 的 <span>ICO</span> 文件 ~
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="layui-row layui-col-space15 margin-bottom-5">
|
||||
<div class="layui-col-xs4 padding-bottom-0">
|
||||
<label class="layui-form-item margin-bottom-5 relative block">
|
||||
<span class="help-label"><b>网站名称</b>Site Name</span>
|
||||
<input name="site_name" required placeholder="请输入网站名称" vali-name="网站名称" value="{:sysconf('site_name')}" class="layui-input">
|
||||
<span class="help-block">网站名称将显示在浏览器的标签上 ~</span>
|
||||
</label>
|
||||
</div>
|
||||
<div class="layui-col-xs4 padding-bottom-0">
|
||||
<label class="layui-form-item margin-bottom-5 relative block">
|
||||
<span class="help-label"><b>后台程序名称</b>App Name</span>
|
||||
<input name="app_name" required placeholder="请输入程序名称" vali-name="程序名称" value="{:sysconf('app_name')}" class="layui-input">
|
||||
<span class="help-block">管理程序名称显示在后台左上标题处 ~</span>
|
||||
</label>
|
||||
</div>
|
||||
<div class="layui-col-xs4 padding-bottom-0">
|
||||
<label class="layui-form-item margin-bottom-5 relative block">
|
||||
<span class="help-label"><b>后台程序版本</b>App Version</span>
|
||||
<input name="app_version" placeholder="请输入程序版本" value="{:sysconf('app_version')}" class="layui-input">
|
||||
<span class="help-block">管理程序版本显示在后台左上标题处 ~</span>
|
||||
</label>
|
||||
</div>
|
||||
<div class="layui-col-xs4 padding-top-0 padding-bottom-0">
|
||||
<label class="relative block">
|
||||
<span class="help-label"><b>公安安备号</b>Beian</span>
|
||||
<input name="beian" placeholder="请输入公安安备号" value="{:sysconf('beian')}" class="layui-input">
|
||||
</label>
|
||||
</div>
|
||||
<div class="layui-col-xs4 padding-top-0 padding-bottom-0">
|
||||
<label class="relative block">
|
||||
<span class="help-label"><b>网站备案号</b>Miitbeian</span>
|
||||
<input name="miitbeian" placeholder="请输入网站备案号" value="{:sysconf('miitbeian')}" class="layui-input">
|
||||
</label>
|
||||
</div>
|
||||
<div class="layui-col-xs4 padding-top-0 padding-bottom-0">
|
||||
<label class="relative block">
|
||||
<span class="help-label"><b>网站版权信息</b>Copyright</span>
|
||||
<input name="site_copy" required placeholder="请输入版权信息" vali-name="版权信息" value="{:sysconf('site_copy')}" class="layui-input">
|
||||
</label>
|
||||
</div>
|
||||
<div class="layui-col-xs12 help-block padding-top-0">
|
||||
网站备案号和公安备案号可以在<a target="_blank" href="https://beian.miit.gov.cn">备案管理中心</a>查询并获取,网站上线时必需配置备案号,备案号会链接到信息备案管理系统 ~
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="hr-line-dashed"></div>
|
||||
<div class="layui-form-item text-center">
|
||||
<button class="layui-btn" type="submit">保存配置</button>
|
||||
<button class="layui-btn layui-btn-danger" type='button' data-confirm="确定要取消修改吗?" data-close>取消修改</button>
|
||||
</div>
|
||||
</form>
|
||||
|
||||
<script>
|
||||
$('[name=login_image]').uploadMultipleImage();
|
||||
require(['md5'], function (md5) {
|
||||
$('body').off('click', '#RefreshJwtKey').on('click', '#RefreshJwtKey', function () {
|
||||
$(this).parent().find('input').val(md5.hash(Date.toString() + Math.random()));
|
||||
});
|
||||
});
|
||||
layui.form.on('select(SiteTheme)', function (data) {
|
||||
var alls = '', prox = 'layui-layout-theme-', curt = prox + data.value;
|
||||
$(data.elem.options).map(function () {
|
||||
if (this.value !== data.value) alls += ' ' + prox + this.value;
|
||||
});
|
||||
$('.layui-layout-body').removeClass(alls).addClass(curt)
|
||||
});
|
||||
</script>
|
568
app/admin/view/error.php
Normal file
568
app/admin/view/error.php
Normal file
@ -0,0 +1,568 @@
|
||||
<?php
|
||||
|
||||
if (!function_exists('parse_padding')) {
|
||||
function parse_padding($source)
|
||||
{
|
||||
$length = strlen(strval(count($source['source']) + $source['first']));
|
||||
return 40 + ($length - 1) * 8;
|
||||
}
|
||||
}
|
||||
|
||||
if (!function_exists('parse_class')) {
|
||||
function parse_class($name): string
|
||||
{
|
||||
$names = explode('\\', $name);
|
||||
return '<abbr title="' . $name . '">' . end($names) . '</abbr>';
|
||||
}
|
||||
}
|
||||
|
||||
if (!function_exists('parse_file')) {
|
||||
function parse_file($file, $line): string
|
||||
{
|
||||
return '<a class="toggle" title="' . "{$file} line {$line}" . '">' . basename($file) . " line {$line}" . '</a>';
|
||||
}
|
||||
}
|
||||
|
||||
if (!function_exists('parse_args')) {
|
||||
function parse_args($args): string
|
||||
{
|
||||
$result = [];
|
||||
foreach ($args as $key => $item) {
|
||||
switch (true) {
|
||||
case is_object($item):
|
||||
$value = sprintf('<em>object</em>(%s)', parse_class(get_class($item)));
|
||||
break;
|
||||
case is_array($item):
|
||||
if (count($item) > 3) {
|
||||
$value = sprintf('[%s, ...]', parse_args(array_slice($item, 0, 3)));
|
||||
} else {
|
||||
$value = sprintf('[%s]', parse_args($item));
|
||||
}
|
||||
break;
|
||||
case is_string($item):
|
||||
if (strlen($item) > 20) {
|
||||
$value = sprintf(
|
||||
'\'<a class="toggle" title="%s">%s...</a>\'',
|
||||
htmlentities($item),
|
||||
htmlentities(substr($item, 0, 20))
|
||||
);
|
||||
} else {
|
||||
$value = sprintf("'%s'", htmlentities($item));
|
||||
}
|
||||
break;
|
||||
case is_int($item):
|
||||
case is_float($item):
|
||||
$value = $item;
|
||||
break;
|
||||
case is_null($item):
|
||||
$value = '<em>null</em>';
|
||||
break;
|
||||
case is_bool($item):
|
||||
$value = '<em>' . ($item ? 'true' : 'false') . '</em>';
|
||||
break;
|
||||
case is_resource($item):
|
||||
$value = '<em>resource</em>';
|
||||
break;
|
||||
default:
|
||||
$value = htmlentities(str_replace("\n", '', var_export(strval($item), true)));
|
||||
break;
|
||||
}
|
||||
|
||||
$result[] = is_int($key) ? $value : "'{$key}' => {$value}";
|
||||
}
|
||||
|
||||
return implode(', ', $result);
|
||||
}
|
||||
}
|
||||
if (!function_exists('echo_value')) {
|
||||
function echo_value($val)
|
||||
{
|
||||
if (is_array($val) || is_object($val)) {
|
||||
echo htmlentities(json_encode($val, JSON_PRETTY_PRINT));
|
||||
} elseif (is_bool($val)) {
|
||||
echo $val ? 'true' : 'false';
|
||||
} elseif (is_scalar($val)) {
|
||||
echo htmlentities($val);
|
||||
} else {
|
||||
echo 'Resource';
|
||||
}
|
||||
}
|
||||
}
|
||||
?>
|
||||
<!DOCTYPE html>
|
||||
<html lang="zh">
|
||||
<head>
|
||||
<meta charset="UTF-8">
|
||||
<title>系统发生错误</title>
|
||||
<meta name="robots" content="noindex,nofollow"/>
|
||||
<style>
|
||||
/* Base */
|
||||
body {
|
||||
color: #333;
|
||||
font: 16px Verdana, "Helvetica Neue", helvetica, Arial, 'Microsoft YaHei', sans-serif;
|
||||
margin: 0;
|
||||
padding: 0 20px 20px;
|
||||
}
|
||||
|
||||
h1 {
|
||||
margin: 10px 0 0;
|
||||
font-size: 28px;
|
||||
font-weight: 500;
|
||||
line-height: 32px;
|
||||
}
|
||||
|
||||
h2 {
|
||||
color: #4288ce;
|
||||
font-weight: 400;
|
||||
padding: 6px 0;
|
||||
margin: 6px 0 0;
|
||||
font-size: 18px;
|
||||
border-bottom: 1px solid #eee;
|
||||
}
|
||||
|
||||
h3 {
|
||||
margin: 12px;
|
||||
font-size: 16px;
|
||||
font-weight: bold;
|
||||
}
|
||||
|
||||
abbr {
|
||||
cursor: help;
|
||||
text-decoration: underline;
|
||||
text-decoration-style: dotted;
|
||||
}
|
||||
|
||||
a {
|
||||
color: #868686;
|
||||
cursor: pointer;
|
||||
}
|
||||
|
||||
a:hover {
|
||||
text-decoration: underline;
|
||||
}
|
||||
|
||||
.line-error {
|
||||
background: #f8cbcb;
|
||||
}
|
||||
|
||||
.echo table {
|
||||
width: 100%;
|
||||
}
|
||||
|
||||
.echo pre {
|
||||
padding: 16px;
|
||||
overflow: auto;
|
||||
font-size: 85%;
|
||||
line-height: 1.45;
|
||||
background-color: #f7f7f7;
|
||||
border: 0;
|
||||
border-radius: 3px;
|
||||
font-family: Consolas, "Liberation Mono", Menlo, Courier, monospace;
|
||||
}
|
||||
|
||||
.echo pre > pre {
|
||||
padding: 0;
|
||||
margin: 0;
|
||||
}
|
||||
|
||||
/* Exception Info */
|
||||
.exception {
|
||||
margin-top: 20px;
|
||||
}
|
||||
|
||||
.exception .message {
|
||||
padding: 12px;
|
||||
border: 1px solid #ddd;
|
||||
border-bottom: 0 none;
|
||||
line-height: 18px;
|
||||
font-size: 16px;
|
||||
border-top-left-radius: 4px;
|
||||
border-top-right-radius: 4px;
|
||||
font-family: Consolas, "Liberation Mono", Courier, Verdana, "微软雅黑", serif;
|
||||
}
|
||||
|
||||
.exception .code {
|
||||
float: left;
|
||||
text-align: center;
|
||||
color: #fff;
|
||||
margin-right: 12px;
|
||||
padding: 16px;
|
||||
border-radius: 4px;
|
||||
background: #999;
|
||||
}
|
||||
|
||||
.exception .source-code {
|
||||
padding: 6px;
|
||||
border: 1px solid #ddd;
|
||||
|
||||
background: #f9f9f9;
|
||||
overflow-x: auto;
|
||||
|
||||
}
|
||||
|
||||
.exception .source-code pre {
|
||||
margin: 0;
|
||||
}
|
||||
|
||||
.exception .source-code pre ol {
|
||||
margin: 0;
|
||||
color: #4288ce;
|
||||
display: inline-block;
|
||||
min-width: 100%;
|
||||
box-sizing: border-box;
|
||||
font-size: 14px;
|
||||
font-family: "Century Gothic", Consolas, "Liberation Mono", Courier, Verdana, serif;
|
||||
padding-left: <?php echo (isset($source) && ! empty($source)) ? parse_padding($source): 40;?> px;
|
||||
}
|
||||
|
||||
.exception .source-code pre li {
|
||||
border-left: 1px solid #ddd;
|
||||
height: 18px;
|
||||
line-height: 18px;
|
||||
}
|
||||
|
||||
.exception .source-code pre code {
|
||||
color: #333;
|
||||
height: 100%;
|
||||
display: inline-block;
|
||||
border-left: 1px solid #fff;
|
||||
font-size: 14px;
|
||||
font-family: Consolas, "Liberation Mono", Courier, Verdana, "微软雅黑", serif;
|
||||
}
|
||||
|
||||
.exception .trace {
|
||||
padding: 6px;
|
||||
border: 1px solid #ddd;
|
||||
border-top: 0 none;
|
||||
line-height: 16px;
|
||||
font-size: 14px;
|
||||
font-family: Consolas, "Liberation Mono", Courier, Verdana, "微软雅黑", serif;
|
||||
}
|
||||
|
||||
.exception .trace h2:hover {
|
||||
text-decoration: underline;
|
||||
cursor: pointer;
|
||||
}
|
||||
|
||||
.exception .trace ol {
|
||||
margin: 12px;
|
||||
}
|
||||
|
||||
.exception .trace ol li {
|
||||
padding: 2px 4px;
|
||||
}
|
||||
|
||||
.exception div:last-child {
|
||||
border-bottom-left-radius: 4px;
|
||||
border-bottom-right-radius: 4px;
|
||||
}
|
||||
|
||||
/* Exception Variables */
|
||||
.exception-var table {
|
||||
width: 100%;
|
||||
margin: 12px 0;
|
||||
box-sizing: border-box;
|
||||
table-layout: fixed;
|
||||
word-wrap: break-word;
|
||||
}
|
||||
|
||||
.exception-var table caption {
|
||||
text-align: left;
|
||||
font-size: 16px;
|
||||
font-weight: bold;
|
||||
padding: 6px 0;
|
||||
}
|
||||
|
||||
.exception-var table caption small {
|
||||
font-weight: 300;
|
||||
display: inline-block;
|
||||
margin-left: 10px;
|
||||
color: #ccc;
|
||||
}
|
||||
|
||||
.exception-var table tbody {
|
||||
font-size: 13px;
|
||||
font-family: Consolas, "Liberation Mono", Courier, "微软雅黑", serif;
|
||||
}
|
||||
|
||||
.exception-var table td {
|
||||
padding: 0 6px;
|
||||
vertical-align: top;
|
||||
word-break: break-all;
|
||||
}
|
||||
|
||||
.exception-var table td:first-child {
|
||||
width: 28%;
|
||||
font-weight: bold;
|
||||
white-space: nowrap;
|
||||
}
|
||||
|
||||
.exception-var table td pre {
|
||||
margin: 0;
|
||||
}
|
||||
|
||||
/* Copyright Info */
|
||||
.copyright {
|
||||
margin-top: 24px;
|
||||
padding: 12px 0;
|
||||
border-top: 1px solid #eee;
|
||||
}
|
||||
|
||||
/* SPAN elements with the classes below are added by prettyprint. */
|
||||
pre.prettyprint .pln {
|
||||
color: #000
|
||||
}
|
||||
|
||||
/* plain text */
|
||||
pre.prettyprint .str {
|
||||
color: #080
|
||||
}
|
||||
|
||||
/* string content */
|
||||
pre.prettyprint .kwd {
|
||||
color: #008
|
||||
}
|
||||
|
||||
/* a keyword */
|
||||
pre.prettyprint .com {
|
||||
color: #800
|
||||
}
|
||||
|
||||
/* a comment */
|
||||
pre.prettyprint .typ {
|
||||
color: #606
|
||||
}
|
||||
|
||||
/* a type name */
|
||||
pre.prettyprint .lit {
|
||||
color: #066
|
||||
}
|
||||
|
||||
/* a literal value */
|
||||
/* punctuation, lisp open bracket, lisp close bracket */
|
||||
pre.prettyprint .pun, pre.prettyprint .opn, pre.prettyprint .clo {
|
||||
color: #660
|
||||
}
|
||||
|
||||
pre.prettyprint .tag {
|
||||
color: #008
|
||||
}
|
||||
|
||||
/* a markup tag name */
|
||||
pre.prettyprint .atn {
|
||||
color: #606
|
||||
}
|
||||
|
||||
/* a markup attribute name */
|
||||
pre.prettyprint .atv {
|
||||
color: #080
|
||||
}
|
||||
|
||||
/* a markup attribute value */
|
||||
pre.prettyprint .dec, pre.prettyprint .var {
|
||||
color: #606
|
||||
}
|
||||
|
||||
/* a declaration; a variable name */
|
||||
pre.prettyprint .fun {
|
||||
color: red
|
||||
}
|
||||
|
||||
/* a function name */
|
||||
</style>
|
||||
</head>
|
||||
<body>
|
||||
<?php if (\think\facade\App::isDebug()) { ?>
|
||||
<?php foreach ($traces as $index => $trace) { ?>
|
||||
<div class="exception">
|
||||
<div class="message">
|
||||
<div class="info">
|
||||
<div>
|
||||
<h2><?php echo "#{$index} [{$trace['code']}]" . sprintf('%s in %s', parse_class($trace['name']), parse_file($trace['file'], $trace['line'])); ?></h2>
|
||||
</div>
|
||||
<div><h1><?php echo nl2br(htmlentities($trace['message'])); ?></h1></div>
|
||||
</div>
|
||||
</div>
|
||||
<?php if (!empty($trace['source'])) { ?>
|
||||
<div class="source-code">
|
||||
<pre class="prettyprint lang-php">
|
||||
<ol start="<?php echo $trace['source']['first']; ?>"><!--<?php foreach ((array)$trace['source']['source'] as $key => $value) { ?>--><li class="line-<?php echo " {$index}-" . ($key + $trace['source']['first']) . ($trace['line'] === $key + $trace['source']['first'] ? ' line-error' : ''); ?>"><code><?php echo htmlentities($value); ?></code></li><!--<?php } ?>--></ol>
|
||||
</pre>
|
||||
</div>
|
||||
<?php } ?>
|
||||
<div class="trace">
|
||||
<h2 data-expand="<?php echo 0 === $index ? '1' : '0'; ?>">Call Stack</h2>
|
||||
<ol>
|
||||
<li><?php echo sprintf('in %s', parse_file($trace['file'], $trace['line'])); ?></li>
|
||||
<?php foreach ((array)$trace['trace'] as $value) { ?>
|
||||
<li>
|
||||
<?php
|
||||
// Show Function
|
||||
if ($value['function']) {
|
||||
echo sprintf('at %s%s%s(%s)', isset($value['class']) ? parse_class($value['class']) : '', $value['type'] ?? '', $value['function'], isset($value['args']) ? parse_args($value['args']) : '');
|
||||
}
|
||||
|
||||
// Show line
|
||||
if (isset($value['file']) && isset($value['line'])) {
|
||||
echo sprintf(' in %s', parse_file($value['file'], $value['line']));
|
||||
}
|
||||
?>
|
||||
</li>
|
||||
<?php } ?>
|
||||
</ol>
|
||||
</div>
|
||||
</div>
|
||||
<?php } ?>
|
||||
<?php } else { ?>
|
||||
<div class="exception">
|
||||
<div class="info"><h1><?php echo htmlentities(isset($message) ? $message : ''); ?></h1></div>
|
||||
</div>
|
||||
<?php } ?>
|
||||
|
||||
<?php if (!empty($datas)) { ?>
|
||||
<div class="exception-var">
|
||||
<h2>Exception Datas</h2>
|
||||
<?php foreach ((array)$datas as $label => $value) { ?>
|
||||
<table>
|
||||
<?php if (empty($value)) { ?>
|
||||
<caption><?php echo $label; ?><small>empty</small></caption>
|
||||
<?php } else { ?>
|
||||
<caption><?php echo $label; ?></caption>
|
||||
<tbody>
|
||||
<?php foreach ((array)$value as $key => $val) { ?>
|
||||
<tr>
|
||||
<td><?php echo htmlentities($key); ?></td>
|
||||
<td><?php echo_value($val); ?></td>
|
||||
</tr>
|
||||
<?php } ?>
|
||||
</tbody>
|
||||
<?php } ?>
|
||||
</table>
|
||||
<?php } ?>
|
||||
</div>
|
||||
<?php } ?>
|
||||
|
||||
<?php if (!empty($tables)) { ?>
|
||||
<div class="exception-var">
|
||||
<h2>Environment Variables</h2>
|
||||
<?php foreach ((array)$tables as $label => $value) { ?>
|
||||
<table>
|
||||
<?php if (empty($value)) { ?>
|
||||
<caption><?php echo $label; ?><small>empty</small></caption>
|
||||
<?php } else { ?>
|
||||
<caption><?php echo $label; ?></caption>
|
||||
<tbody>
|
||||
<?php foreach ((array)$value as $key => $val) { ?>
|
||||
<tr>
|
||||
<td><?php echo htmlentities($key); ?></td>
|
||||
<td><?php echo_value($val); ?></td>
|
||||
</tr>
|
||||
<?php } ?>
|
||||
</tbody>
|
||||
<?php } ?>
|
||||
</table>
|
||||
<?php } ?>
|
||||
</div>
|
||||
<?php } ?>
|
||||
|
||||
<?php if (\think\facade\App::isDebug()) { ?>
|
||||
<script>
|
||||
function $(selector, node) {
|
||||
var elements;
|
||||
node = node || document;
|
||||
if (document.querySelectorAll) {
|
||||
elements = node.querySelectorAll(selector);
|
||||
} else {
|
||||
switch (selector.substr(0, 1)) {
|
||||
case '#':
|
||||
elements = [node.getElementById(selector.substr(1))];
|
||||
break;
|
||||
case '.':
|
||||
if (document.getElementsByClassName) {
|
||||
elements = node.getElementsByClassName(selector.substr(1));
|
||||
} else {
|
||||
elements = get_elements_by_class(selector.substr(1), node);
|
||||
}
|
||||
break;
|
||||
default:
|
||||
elements = node.getElementsByTagName();
|
||||
}
|
||||
}
|
||||
return elements;
|
||||
|
||||
function get_elements_by_class(search_class, node, tag) {
|
||||
var elements = [], eles,
|
||||
pattern = new RegExp('(^|\\s)' + search_class + '(\\s|$)');
|
||||
|
||||
node = node || document;
|
||||
tag = tag || '*';
|
||||
|
||||
eles = node.getElementsByTagName(tag);
|
||||
for (var i = 0; i < eles.length; i++) {
|
||||
if (pattern.test(eles[i].className)) {
|
||||
elements.push(eles[i])
|
||||
}
|
||||
}
|
||||
|
||||
return elements;
|
||||
}
|
||||
}
|
||||
|
||||
$.getScript = function (src, func) {
|
||||
var script = document.createElement('script');
|
||||
|
||||
script.async = 'async';
|
||||
script.src = src;
|
||||
script.onload = func || function () {
|
||||
};
|
||||
|
||||
$('head')[0].appendChild(script);
|
||||
}
|
||||
|
||||
;(function () {
|
||||
var files = $('.toggle');
|
||||
var ol = $('ol', $('.prettyprint')[0]);
|
||||
var li = $('li', ol[0]);
|
||||
|
||||
// 短路径和长路径变换
|
||||
for (var i = 0; i < files.length; i++) {
|
||||
files[i].ondblclick = function () {
|
||||
var title = this.title;
|
||||
|
||||
this.title = this.innerHTML;
|
||||
this.innerHTML = title;
|
||||
}
|
||||
}
|
||||
|
||||
(function () {
|
||||
var expand = function (dom, expand) {
|
||||
var ol = $('ol', dom.parentNode)[0];
|
||||
expand = undefined === expand ? dom.attributes['data-expand'].value === '0' : undefined;
|
||||
if (expand) {
|
||||
dom.attributes['data-expand'].value = '1';
|
||||
ol.style.display = 'none';
|
||||
dom.innerText = 'Call Stack (展开)';
|
||||
} else {
|
||||
dom.attributes['data-expand'].value = '0';
|
||||
ol.style.display = 'block';
|
||||
dom.innerText = 'Call Stack (折叠)';
|
||||
}
|
||||
};
|
||||
var traces = $('.trace');
|
||||
for (var i = 0; i < traces.length; i++) {
|
||||
var h2 = $('h2', traces[i])[0];
|
||||
expand(h2);
|
||||
h2.onclick = function () {
|
||||
expand(this);
|
||||
};
|
||||
}
|
||||
})();
|
||||
|
||||
$.getScript('//cdn.bootcss.com/prettify/r298/prettify.min.js', function () {
|
||||
prettyPrint();
|
||||
});
|
||||
})();
|
||||
</script>
|
||||
<?php } ?>
|
||||
</body>
|
||||
</html>
|
40
app/admin/view/file/form.html
Normal file
40
app/admin/view/file/form.html
Normal file
@ -0,0 +1,40 @@
|
||||
<form action="{:sysuri()}" method="post" data-auto="true" class="layui-form layui-card" data-table-id="FileTable">
|
||||
|
||||
<div class="layui-card-body padding-left-40">
|
||||
|
||||
<label class="layui-form-item relative block">
|
||||
<span class="help-label"><b>{:lang('文件名称')}</b>Name</span>
|
||||
<input maxlength="100" class="layui-input" name="name" value='{$vo.name|default=""}' required vali-name="文件名称" placeholder="请输入文件名称">
|
||||
</label>
|
||||
|
||||
<label class="layui-form-item relative block">
|
||||
<span class="help-label"><b>{:lang('文件大小')}</b>Size</span>
|
||||
<input maxlength="100" class="layui-input layui-bg-gray" value='{$vo.size|default=0|format_bytes}' readonly>
|
||||
</label>
|
||||
|
||||
<label class="layui-form-item relative block">
|
||||
<span class="help-label"><b>{:lang('存储方式')}</b>Type</span>
|
||||
<input maxlength="100" class="layui-input layui-bg-gray" value='{$types[$vo.type]??""}' readonly>
|
||||
</label>
|
||||
|
||||
<label class="layui-form-item relative block">
|
||||
<span class="help-label"><b>{:lang('文件哈希')}</b>Hash</span>
|
||||
<input maxlength="100" class="layui-input layui-bg-gray" value='{$vo.hash|default=""}' readonly>
|
||||
</label>
|
||||
|
||||
<label class="layui-form-item relative block">
|
||||
<span class="help-label"><b>{:lang('文件链接')}</b>Link</span>
|
||||
<input maxlength="100" class="layui-input layui-bg-gray" value='{$vo.xurl|default=""}' readonly>
|
||||
</label>
|
||||
|
||||
|
||||
</div>
|
||||
|
||||
<div class="hr-line-dashed"></div>
|
||||
{notempty name='vo.id'}<input type='hidden' value='{$vo.id}' name='id'>{/notempty}
|
||||
|
||||
<div class="layui-form-item text-center">
|
||||
<button class="layui-btn" type='submit'>{:lang('保存数据')}</button>
|
||||
<button class="layui-btn layui-btn-danger" type='button' data-confirm="{:lang('确定要取消编辑吗?')}" data-close>{:lang('取消编辑')}</button>
|
||||
</div>
|
||||
</form>
|
64
app/admin/view/file/index.html
Normal file
64
app/admin/view/file/index.html
Normal file
@ -0,0 +1,64 @@
|
||||
{extend name='table'}
|
||||
|
||||
{block name="button"}
|
||||
<!--{if auth("distinct")}-->
|
||||
<a data-table-id="FileTable" data-load='{:url("distinct")}' class='layui-btn layui-btn-sm layui-btn-primary'>{:lang('清理重复')}</a>
|
||||
<!--{/if}-->
|
||||
<!--{if auth("remove")}-->
|
||||
<a data-confirm="{:lang('确定删除这些记录吗?')}" data-table-id="FileTable" data-action='{:url("remove")}' data-rule="id#{id}" class='layui-btn layui-btn-sm layui-btn-primary'>{:lang('批量删除')}</a>
|
||||
<!--{/if}-->
|
||||
{/block}
|
||||
|
||||
{block name="content"}
|
||||
<div class="think-box-shadow">
|
||||
{include file='file/index_search'}
|
||||
<table id="FileTable" data-url="{:sysuri('index')}" data-target-search="form.form-search"></table>
|
||||
</div>
|
||||
<script>
|
||||
$(function () {
|
||||
$('#FileTable').layTable({
|
||||
even: true, height: 'full',
|
||||
sort: {field: 'id', type: 'desc'},
|
||||
cols: [[
|
||||
{checkbox: true, fixed: true},
|
||||
{field: 'id', title: 'ID', width: 80, align: 'center', sort: true},
|
||||
{field: 'name', title: '{:lang("文件名称")}', width: '12%', align: 'center'},
|
||||
{field: 'hash', title: '{:lang("文件哈希")}', width: '15%', align: 'center', templet: '<div><code>{{d.hash}}</code></div>'},
|
||||
{field: 'size', title: '{:lang("文件大小")}', align: 'center', width: '7%', sort: true, templet: '<div>{{-$.formatFileSize(d.size)}}</div>'},
|
||||
{field: 'xext', title: '{:lang("文件后缀")}', align: 'center', width: '7%', sort: true},
|
||||
{
|
||||
field: 'xurl', title: '{:lang("查看文件")}', width: '7%', align: 'center', templet: function (d) {
|
||||
if (typeof d.mime === 'string' && /^image\//.test(d.mime)) {
|
||||
return laytpl('<div><a target="_blank" data-tips-hover data-tips-image="{{d.xurl}}"><i class="layui-icon layui-icon-picture"></i></a></div>').render(d)
|
||||
}
|
||||
if (typeof d.mime === 'string' && /^video\//.test(d.mime)) {
|
||||
return laytpl('<div><a target="_blank" data-video-player="{{d.xurl}}" data-tips-text="{:lang(\'播放视频\')}"><i class="layui-icon layui-icon-video"></i></a></div>').render(d);
|
||||
}
|
||||
if (typeof d.mime === 'string' && /^audio\//.test(d.mime)) {
|
||||
return laytpl('<div><a target="_blank" data-video-player="{{d.xurl}}" data-tips-text="{:lang(\'播放音频\')}"><i class="layui-icon layui-icon-headset"></i></a></div>').render(d);
|
||||
}
|
||||
return laytpl('<div><a target="_blank" href="{{d.xurl}}" data-tips-text="{:lang(\'查看下载\')}"><i class="layui-icon layui-icon-file"></i></a></div>').render(d);
|
||||
}
|
||||
},
|
||||
{
|
||||
field: 'isfast', title: '{:lang("上传方式")}', align: 'center', width: '8%', templet: function (d) {
|
||||
return d.isfast ? '<b class="color-green">{:lang("秒传")}</b>' : '<b class="color-blue">{:lang("普通")}</b>';
|
||||
}
|
||||
},
|
||||
{field: 'ctype', title: '{:lang("存储方式")}', align: 'center', width: '10%'},
|
||||
{field: 'create_at', title: '{:lang("创建时间")}', align: 'center', width: '15%', sort: true},
|
||||
{toolbar: '#toolbar', title: '{:lang("操作面板")}', align: 'center', minWidth: 150, fixed: 'right'}
|
||||
]]
|
||||
});
|
||||
});
|
||||
</script>
|
||||
|
||||
<script type="text/html" id="toolbar">
|
||||
<!--{if auth("edit")}-->
|
||||
<a class="layui-btn layui-btn-sm" data-modal="{:url('edit')}?id={{d.id}}" data-title="编辑文件信息">{:lang("编 辑")}</a>
|
||||
<!--{/if}-->
|
||||
<!--{if auth("remove")}-->
|
||||
<a class="layui-btn layui-btn-sm layui-btn-danger" data-action="{:url('remove')}" data-value="id#{{d.id}}">{:lang("删 除")}</a>
|
||||
<!--{/if}-->
|
||||
</script>
|
||||
{/block}
|
58
app/admin/view/file/index_search.html
Normal file
58
app/admin/view/file/index_search.html
Normal file
@ -0,0 +1,58 @@
|
||||
<fieldset>
|
||||
<legend>{:lang('条件搜索')}</legend>
|
||||
<form class="layui-form layui-form-pane form-search" action="{:sysuri()}" onsubmit="return false" method="get" autocomplete="off">
|
||||
|
||||
<div class="layui-form-item layui-inline">
|
||||
<label class="layui-form-label">{:lang('文件名称')}</label>
|
||||
<label class="layui-input-inline">
|
||||
<input name="name" value="{$get.name|default=''}" placeholder="{:lang('请输入文件名称')}" class="layui-input">
|
||||
</label>
|
||||
</div>
|
||||
|
||||
<div class="layui-form-item layui-inline">
|
||||
<label class="layui-form-label">{:lang('文件哈希')}</label>
|
||||
<label class="layui-input-inline">
|
||||
<input name="hash" value="{$get.hash|default=''}" placeholder="{:lang('请输入文件哈希')}" class="layui-input">
|
||||
</label>
|
||||
</div>
|
||||
|
||||
<div class="layui-form-item layui-inline">
|
||||
<label class="layui-form-label">{:lang('文件后缀')}</label>
|
||||
<div class="layui-input-inline">
|
||||
<select name="xext" lay-search class="layui-select">
|
||||
<option value=''>-- {:lang('全部')} --</option>
|
||||
{foreach $xexts as $v}{if isset($get.xext) and $k eq $get.xext}
|
||||
<option selected value="{$v}">{$v}</option>
|
||||
{else}
|
||||
<option value="{$v}">{$v}</option>
|
||||
{/if}{/foreach}
|
||||
</select>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="layui-form-item layui-inline">
|
||||
<label class="layui-form-label">{:lang('存储方式')}</label>
|
||||
<div class="layui-input-inline">
|
||||
<select name="type" lay-search class="layui-select">
|
||||
<option value=''>-- {:lang('全部')} --</option>
|
||||
{foreach $types as $k=>$v}{if isset($get.type) and $k eq $get.type}
|
||||
<option selected value="{$k}">{$v}</option>
|
||||
{else}
|
||||
<option value="{$k}">{$v}</option>
|
||||
{/if}{/foreach}
|
||||
</select>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="layui-form-item layui-inline">
|
||||
<label class="layui-form-label">{:lang('创建时间')}</label>
|
||||
<div class="layui-input-inline">
|
||||
<input data-date-range name="create_at" value="{$get.create_at|default=''}" placeholder="{:lang('请选择创建时间')}" class="layui-input">
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="layui-form-item layui-inline">
|
||||
<button class="layui-btn layui-btn-primary"><i class="layui-icon"></i> {:lang('搜 索')}</button>
|
||||
</div>
|
||||
</form>
|
||||
</fieldset>
|
35
app/admin/view/full.html
Normal file
35
app/admin/view/full.html
Normal file
@ -0,0 +1,35 @@
|
||||
<!DOCTYPE html>
|
||||
<html lang="zh">
|
||||
<head>
|
||||
<title>{block name="title"}{$title|default=''}{if !empty($title)} · {/if}{:sysconf('site_name')}{/block}</title>
|
||||
<meta charset="utf-8">
|
||||
<meta name="renderer" content="webkit">
|
||||
<meta name="format-detection" content="telephone=no">
|
||||
<meta name="mobile-web-app-capable" content="yes">
|
||||
<meta http-equiv="X-UA-Compatible" content="IE=edge,chrome=1">
|
||||
<meta name="apple-mobile-web-app-status-bar-style" content="black">
|
||||
<meta name="viewport" content="width=device-width,initial-scale=1,maximum-scale=0.4">
|
||||
<link rel="stylesheet" href="__ROOT__/static/plugs/layui/css/layui.css?at={:date('md')}">
|
||||
<link rel="stylesheet" href="__ROOT__/static/theme/css/iconfont.css?at={:date('md')}">
|
||||
<link rel="stylesheet" href="__ROOT__/static/theme/css/console.css?at={:date('md')}">
|
||||
<link rel="stylesheet" href="__ROOT__/static/extra/style.css?at={:date('md')}">
|
||||
{if file_exists(syspath("public/static/extra/icon/iconfont.css"))}
|
||||
<link rel="stylesheet" href="__ROOT__/static/extra/icon/iconfont.css?at={:date('md')}">
|
||||
{/if}
|
||||
{block name="style"}{/block}
|
||||
<script src="__ROOT__/static/plugs/jquery/pace.min.js"></script>
|
||||
<script src="{:url('admin/api.plugs/script',[],false,false)}"></script>
|
||||
</head>
|
||||
<body class="layui-layout-body">
|
||||
{block name='body'}
|
||||
<div class="layui-layout layui-layout-admin layui-layout-left-hide">
|
||||
<div class="layui-body think-bg-white margin-0 padding-0" style="top:0">{block name='content'}{/block}</div>
|
||||
</div>
|
||||
{/block}
|
||||
<script src="__ROOT__/static/plugs/layui/layui.js"></script>
|
||||
<script src="__ROOT__/static/plugs/require/require.js"></script>
|
||||
<script src="__ROOT__/static/admin.js"></script>
|
||||
<script src="__ROOT__/static/extra/script.js"></script>
|
||||
{block name='script'}{/block}
|
||||
</body>
|
||||
</html>
|
50
app/admin/view/index/index-left.html
Normal file
50
app/admin/view/index/index-left.html
Normal file
@ -0,0 +1,50 @@
|
||||
<div class="layui-side">
|
||||
<a class="layui-side-target" data-target-menu-type></a>
|
||||
<a class="layui-logo layui-elip" href="{:sysuri('@')}" title="{:sysconf('app_name')}">
|
||||
<span class="headimg headimg-no headimg-xs" data-lazy-src="{:sysconf('site_icon')}"></span>
|
||||
<span class="headtxt">{:sysconf('app_name')} {if sysconf('app_version')}<sup>{:sysconf('app_version')}</sup>{/if}</span>
|
||||
</a>
|
||||
<div class="layui-side-scroll">
|
||||
<div class="layui-side-icon">
|
||||
{foreach $menus as $one}
|
||||
<div>
|
||||
<a data-menu-node="m-{$one.id}" data-open="{$one.url}" data-target-tips="{$one.title|default=''}">
|
||||
{notempty name='one.icon'}<i class="{$one.icon|default=''}"></i>{/notempty}
|
||||
<span>{$one.title|default=''}</span>
|
||||
</a>
|
||||
</div>
|
||||
{/foreach}
|
||||
</div>
|
||||
<div class="layui-side-tree">
|
||||
{foreach $menus as $one}{notempty name='one.sub'}
|
||||
<ul class="layui-nav layui-nav-tree layui-hide" data-menu-layout="m-{$one.id}">
|
||||
{foreach $one.sub as $two}{empty name='two.sub'}
|
||||
<li class="layui-nav-item">
|
||||
<a data-target-tips="{$two.title}" data-menu-node="m-{$one.id}-{$two.id}" data-open="{$two.url}">
|
||||
<span class='nav-icon {$two.icon|default="layui-icon layui-icon-senior"}'></span>
|
||||
<span class="nav-text">{$two.title|default=''}</span>
|
||||
</a>
|
||||
</li>
|
||||
{else}
|
||||
<li class="layui-nav-item" data-submenu-layout='m-{$one.id}-{$two.id}'>
|
||||
<a data-target-tips="{$two.title}">
|
||||
<span class='nav-icon layui-hide {$two.icon|default="layui-icon layui-icon-triangle-d"}'></span>
|
||||
<span class="nav-text">{$two.title|default=''}</span>
|
||||
</a>
|
||||
<dl class="layui-nav-child">
|
||||
{foreach $two.sub as $thr}
|
||||
<dd>
|
||||
<a data-target-tips="{$thr.title}" data-open="{$thr.url}" data-menu-node="m-{$one.id}-{$two.id}-{$thr.id}">
|
||||
<span class='nav-icon {$thr.icon|default="layui-icon layui-icon-senior"}'></span>
|
||||
<span class="nav-text">{$thr.title|default=''}</span>
|
||||
</a>
|
||||
</dd>
|
||||
{/foreach}
|
||||
</dl>
|
||||
</li>
|
||||
{/empty}{/foreach}
|
||||
</ul>
|
||||
{/notempty}{/foreach}
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
44
app/admin/view/index/index-top.html
Normal file
44
app/admin/view/index/index-top.html
Normal file
@ -0,0 +1,44 @@
|
||||
<div class="layui-header">
|
||||
<ul class="layui-nav layui-layout-left">
|
||||
<li class="layui-nav-item" lay-unselect>
|
||||
<a class="text-center" data-target-menu-type>
|
||||
<i class="layui-icon layui-icon-spread-left"></i>
|
||||
</a>
|
||||
</li>
|
||||
<li class="layui-nav-item" lay-unselect>
|
||||
<a class="layui-logo-hide layui-elip" href="{:sysuri('@')}" title="{:sysconf('app_name')}">
|
||||
<span class="headimg headimg-no headimg-xs" data-lazy-src="{:sysconf('site_icon')}"></span>
|
||||
</a>
|
||||
</li>
|
||||
{foreach $menus as $one}
|
||||
<li class="layui-nav-item">
|
||||
<a data-menu-node="m-{$one.id}" data-open="{$one.url}"><span>{$one.title|default=''}</span></a>
|
||||
</li>
|
||||
{/foreach}
|
||||
</ul>
|
||||
<ul class="layui-nav layui-layout-right">
|
||||
<li lay-unselect class="layui-nav-item"><a data-reload><i class="layui-icon layui-icon-refresh-3"></i></a></li>
|
||||
{if session('user.username')}
|
||||
<li class="layui-nav-item">
|
||||
<dl class="layui-nav-child">
|
||||
<dd lay-unselect><a data-modal="{:sysuri('admin/index/info',['id'=>session('user.id')])}"><i class="layui-icon layui-icon-set-fill"></i> {:lang('基本资料')}</a></dd>
|
||||
<dd lay-unselect><a data-modal="{:sysuri('admin/index/pass',['id'=>session('user.id')])}"><i class="layui-icon layui-icon-component"></i> {:lang('安全设置')}</a></dd>
|
||||
{if isset($super) and $super}
|
||||
<dd lay-unselect><a data-load="{:sysuri('admin/api.system/push')}"><i class="layui-icon layui-icon-template-1"></i> {:lang('缓存加速')}</a></dd>
|
||||
<dd lay-unselect><a data-load="{:sysuri('admin/api.system/clear')}"><i class="layui-icon layui-icon-fonts-clear"></i> {:lang('清理缓存')}</a></dd>
|
||||
{/if}
|
||||
<dd lay-unselect><a data-width="520px" data-modal="{:sysuri('admin/index/theme')}"><i class="layui-icon layui-icon-theme"></i> {:lang('配色方案')}</a></dd>
|
||||
<dd lay-unselect><a data-load="{:sysuri('admin/login/out')}" data-confirm="{:lang('确定要退出登录吗?')}"><i class="layui-icon layui-icon-release"></i> {:lang('退出登录')}</a></dd>
|
||||
</dl>
|
||||
<a class="layui-elip">
|
||||
<span class="headimg" data-lazy-src="{:htmlentities(session('user.headimg'))}"></span>
|
||||
<span>{:htmlentities(lang(session('user.nickname')?:session('user.username')))}</span>
|
||||
</a>
|
||||
</li>
|
||||
{else}
|
||||
<li class="layui-nav-item">
|
||||
<a data-href="{:sysuri('admin/login/index')}"><i class="layui-icon layui-icon-username"></i> {:lang('立即登录')}</a>
|
||||
</li>
|
||||
{/if}
|
||||
</ul>
|
||||
</div>
|
65
app/admin/view/index/index.html
Normal file
65
app/admin/view/index/index.html
Normal file
@ -0,0 +1,65 @@
|
||||
<!DOCTYPE html>
|
||||
<html lang="zh">
|
||||
|
||||
<head>
|
||||
<title>{block name="title"}{$title|default=''}{if !empty($title)} · {/if}{:sysconf('site_name')}{/block}</title>
|
||||
<meta charset="utf-8">
|
||||
<meta name="renderer" content="webkit">
|
||||
<meta name="format-detection" content="telephone=no">
|
||||
<meta name="mobile-web-app-capable" content="yes">
|
||||
<meta http-equiv="X-UA-Compatible" content="IE=edge,chrome=1">
|
||||
<meta name="apple-mobile-web-app-status-bar-style" content="black">
|
||||
<meta name="viewport" content="width=device-width,initial-scale=1,maximum-scale=0.4">
|
||||
<link rel="stylesheet" href="__ROOT__/static/plugs/layui/css/layui.css?at={:date('md')}">
|
||||
<link rel="stylesheet" href="__ROOT__/static/theme/css/iconfont.css?at={:date('md')}">
|
||||
{if file_exists(syspath("public/static/extra/icon/iconfont.css"))}
|
||||
<link rel="stylesheet" href="__ROOT__/static/extra/icon/iconfont.css?at={:date('md')}">
|
||||
{/if}
|
||||
<link rel="stylesheet" href="__ROOT__/static/theme/css/console.css?at={:date('md')}">
|
||||
<link rel="stylesheet" href="__ROOT__/static/extra/style.css?at={:date('md')}">
|
||||
{block name="style"}{/block}
|
||||
<script src="__ROOT__/static/plugs/jquery/pace.min.js"></script>
|
||||
<script src="{:url('admin/api.plugs/script',[],false,false)}"></script>
|
||||
</head>
|
||||
|
||||
<body class="layui-layout-body layui-layout-theme-{$theme|default='default'}">
|
||||
|
||||
{block name='body'}
|
||||
<div class="layui-layout layui-layout-admin layui-layout-left-hide">
|
||||
|
||||
<!-- 左则菜单 开始 -->
|
||||
{include file="index/index-left"}
|
||||
<!-- 左则菜单 结束 -->
|
||||
|
||||
<!-- 顶部菜单 开始 -->
|
||||
{include file='index/index-top'}
|
||||
<!-- 顶部菜单 结束 -->
|
||||
|
||||
<!-- 主体内容 开始 -->
|
||||
<div class="layui-body">
|
||||
<div class="think-page-body">
|
||||
{block name='content'}{/block}
|
||||
</div>
|
||||
<!-- 页面加载动画 -->
|
||||
<div class="think-page-loader layui-hide">
|
||||
<div class="loader"></div>
|
||||
</div>
|
||||
</div>
|
||||
<!-- 主体内容 结束 -->
|
||||
</div>
|
||||
|
||||
<!-- 加载动画 开始 -->
|
||||
<div class="think-page-loader">
|
||||
<div class="loader"></div>
|
||||
</div>
|
||||
<!-- 加载动画 结束 -->
|
||||
|
||||
{/block}
|
||||
<script src="__ROOT__/static/plugs/layui/layui.js"></script>
|
||||
<script src="__ROOT__/static/plugs/require/require.js"></script>
|
||||
<script src="__ROOT__/static/admin.js"></script>
|
||||
<script src="__ROOT__/static/extra/script.js"></script>
|
||||
{block name='script'}{/block}
|
||||
</body>
|
||||
|
||||
</html>
|
36
app/admin/view/index/theme.html
Normal file
36
app/admin/view/index/theme.html
Normal file
@ -0,0 +1,36 @@
|
||||
<form action="{:sysuri()}" method="post" data-auto="true" class="layui-form layui-card" id="theme">
|
||||
<div class="layui-card-body padding-left-40">
|
||||
|
||||
<div class="layui-form-item margin-bottom-5 label-required-prev">
|
||||
<div class="help-label"><b>后台配色方案</b>Theme Style</div>
|
||||
<div class="layui-textarea think-bg-gray" style="min-height:unset">
|
||||
{foreach $themes as $k=>$v}
|
||||
<label class="think-radio">
|
||||
{if isset($theme) and $theme eq $k}
|
||||
<input name="site_theme" type="radio" value="{$k}" lay-ignore checked> {$v}
|
||||
{else}
|
||||
<input name="site_theme" type="radio" value="{$k}" lay-ignore> {$v}
|
||||
{/if}
|
||||
</label>
|
||||
{/foreach}
|
||||
</div>
|
||||
<p class="help-block">切换配色方案,需要保存成功后配色方案才会永久生效,下次登录也会有效哦 ~</p>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="hr-line-dashed"></div>
|
||||
<div class="layui-form-item text-center">
|
||||
<button class="layui-btn" type="submit">保存配置</button>
|
||||
<button class="layui-btn layui-btn-danger" type='button' data-close>取消修改</button>
|
||||
</div>
|
||||
</form>
|
||||
|
||||
<script>
|
||||
$('form#theme input[name=site_theme]').on('click', function () {
|
||||
var alls = '', that = this, prox = 'layui-layout-theme-', curt = prox + that.value;
|
||||
$('form#theme input[name=site_theme]').map(function () {
|
||||
if (this.value !== that.value) alls += ' ' + prox + this.value;
|
||||
});
|
||||
$('.layui-layout-body').removeClass(alls).addClass(curt)
|
||||
});
|
||||
</script>
|
57
app/admin/view/login/index.html
Normal file
57
app/admin/view/login/index.html
Normal file
@ -0,0 +1,57 @@
|
||||
{extend name="index/index"}
|
||||
|
||||
{block name='style'}
|
||||
<meta name="viewport" content="width=device-width,initial-scale=1,maximum-scale=1">
|
||||
<script>if (location.href.indexOf('#') > -1) location.replace(location.href.split('#')[0])</script>
|
||||
<link rel="stylesheet" href="__ROOT__/static/theme/css/login.css">
|
||||
{/block}
|
||||
|
||||
{block name="body"}
|
||||
<div class="login-container" {$loginStyle|raw}>
|
||||
<div class="header notselect layui-hide-xs">
|
||||
<a href="{:url('@')}" class="title">{:sysconf('app_name')}<span>{:sysconf('app_version')}</span></a>
|
||||
{notempty name='runtimeMode'}
|
||||
<a class="pull-right layui-anim layui-anim-fadein" target="_blank" href='https://gitee.com/zoujingli/ThinkAdmin'>
|
||||
<img src='https://gitee.com/zoujingli/ThinkAdmin/widgets/widget_1.svg' alt='Fork me on Gitee'>
|
||||
</a>
|
||||
{/notempty}
|
||||
</div>
|
||||
<form data-login-form onsubmit="return false" method="post" class="layui-anim layui-anim-upbit" autocomplete="off">
|
||||
<h2 class="notselect">{:sysconf('login_name')?:'系统管理'}</h2>
|
||||
<ul>
|
||||
<li class="username">
|
||||
<label class="label-required-null">
|
||||
<i class="layui-icon layui-icon-username"></i>
|
||||
<input class="layui-input" required pattern="^\S{4,}$" vali-name="登录账号" name="username" autofocus autocomplete="off" placeholder="登录账号">
|
||||
</label>
|
||||
</li>
|
||||
<li class="password">
|
||||
<label class="label-required-null">
|
||||
<i class="layui-icon layui-icon-password"></i>
|
||||
<input class="layui-input" required pattern="^\S{4,}$" vali-name="登录密码" name="password" maxlength="32" type="password" autocomplete="off" placeholder="登录密码" lay-affix="eye">
|
||||
</label>
|
||||
</li>
|
||||
<li class="verify layui-hide">
|
||||
<label class="inline-block relative label-required-null">
|
||||
<i class="layui-icon layui-icon-picture-fine"></i>
|
||||
<input class="layui-input" required pattern="^\S{4,}$" name="verify" maxlength="4" autocomplete="off" vali-name="验证码" placeholder="验证码">
|
||||
</label>
|
||||
<label data-captcha="{:url('admin/login/captcha',[],false)}" data-field-verify="verify" data-field-uniqid="uniqid" data-captcha-type="{$captchaType}" data-captcha-token="{$captchaToken}"></label>
|
||||
</li>
|
||||
<li class="text-center padding-top-20">
|
||||
<button type="submit" class="layui-btn layui-disabled full-width" data-form-loaded="立即登入">正在载入</button>
|
||||
</li>
|
||||
</ul>
|
||||
</form>
|
||||
<div class="footer notselect">
|
||||
<p class="layui-hide-xs">推荐使用 <a target="_blank" href="https://www.google.cn/chrome">Google Chrome</a> 或 <a target="_blank" href="https://www.microsoft.com/zh-cn/edge#platform">Microsoft Edge</a> 浏览器访问</p>
|
||||
{:sysconf('site_copy')}
|
||||
{if sysconf('beian')}<span class="padding-5">|</span><a target="_blank" href="https://www.beian.gov.cn/portal/registerSystemInfo">{:sysconf('beian')}</a>{/if}
|
||||
{if sysconf('miitbeian')}<span class="padding-5">|</span><a target="_blank" href="https://beian.miit.gov.cn/">{:sysconf('miitbeian')}</a>{/if}
|
||||
</div>
|
||||
</div>
|
||||
{/block}
|
||||
|
||||
{block name='script'}
|
||||
<script src="__ROOT__/static/login.js"></script>
|
||||
{/block}
|
23
app/admin/view/main.html
Normal file
23
app/admin/view/main.html
Normal file
@ -0,0 +1,23 @@
|
||||
<div class="layui-card">
|
||||
{block name='style'}{/block}
|
||||
{block name='header'}
|
||||
{notempty name='title'}
|
||||
<div class="layui-card-header">
|
||||
<span class="layui-icon font-s10 color-desc margin-right-5"></span>{$title|lang}
|
||||
<div class="pull-right">{block name='button'}{/block}</div>
|
||||
</div>
|
||||
{/notempty}
|
||||
{/block}
|
||||
<div class="layui-card-line"></div>
|
||||
<div class="layui-card-body">
|
||||
<div class="layui-card-html">
|
||||
{notempty name='showErrorMessage'}
|
||||
<div class="think-box-notify" type="error">
|
||||
<b>{:lang('系统提示:')}</b><span>{$showErrorMessage|raw}</span>
|
||||
</div>
|
||||
{/notempty}
|
||||
{block name='content'}{/block}
|
||||
</div>
|
||||
</div>
|
||||
{block name='script'}{/block}
|
||||
</div>
|
97
app/admin/view/menu/form.html
Normal file
97
app/admin/view/menu/form.html
Normal file
@ -0,0 +1,97 @@
|
||||
<form action="{:sysuri()}" method="post" data-auto="true" class="layui-form layui-card" data-table-id="MenuTable">
|
||||
|
||||
<div class="layui-card-body">
|
||||
|
||||
<div class="layui-form-item">
|
||||
<label class="layui-form-label label-required-next">{:lang('上级菜单')}</label>
|
||||
<div class="layui-input-block">
|
||||
<select name='pid' class='layui-select' lay-search>
|
||||
{foreach $menus as $menu}{eq name='menu.id' value='$vo.pid|default=0'}
|
||||
<option selected value='{$menu.id}'>{$menu.spl|raw}{$menu.title}</option>
|
||||
{else}
|
||||
<option value='{$menu.id}'>{$menu.spl|raw}{$menu.title}</option>
|
||||
{/eq}{/foreach}
|
||||
</select>
|
||||
<p class="help-block"><b>必选</b>,请选择上级菜单或顶级菜单 ( 目前最多支持三级菜单 )</p>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="layui-form-item">
|
||||
<label class="layui-form-label">{:lang('菜单名称')}</label>
|
||||
<div class="layui-input-block">
|
||||
<input name="title" value='{$vo.title|default=""}' required vali-name="菜单名称" placeholder="请输入菜单名称" class="layui-input">
|
||||
<p class="help-block"><b>必选</b>,请填写菜单名称 ( 如:系统管理 ),建议字符不要太长,一般 4-6 个汉字</p>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="layui-form-item">
|
||||
<label class="layui-form-label">{:lang('菜单链接')}</label>
|
||||
<div class="layui-input-block">
|
||||
<input onblur="this.value=this.value === ''?'#':this.value" name="url" required vali-name="菜单链接" placeholder="请输入菜单链接" value="{$vo.url|default='#'}" class="layui-input">
|
||||
<p class="help-block">
|
||||
<b>必选</b>,请填写链接地址或选择系统节点 ( 如:https://domain.com/admin/user/index.html 或 admin/user/index )
|
||||
<br>当填写链接地址时,以下面的 “权限节点” 来判断菜单自动隐藏或显示,注意未填写 “权限节点” 时将不会隐藏该菜单哦
|
||||
</p>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="layui-form-item">
|
||||
<label class="layui-form-label">{:lang('链接参数')}</label>
|
||||
<div class="layui-input-block">
|
||||
<input name="params" placeholder="请输入链接参数" value="{$vo.params|default=''}" class="layui-input">
|
||||
<p class="help-block"><b>可选</b>,设置菜单链接的 GET 访问参数 ( 如:name=1&age=3 )</p>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="layui-form-item">
|
||||
<label class="layui-form-label">{:lang('权限节点')}</label>
|
||||
<div class="layui-input-block">
|
||||
<input name="node" placeholder="请输入权限节点" value="{$vo.node|default=''}" class="layui-input">
|
||||
<p class="help-block"><b>可选</b>,请填写系统权限节点 ( 如:admin/user/index ),未填写时默认解释"菜单链接"判断是否拥有访问权限;</p>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="layui-form-item">
|
||||
<label class="layui-form-label">{:lang('菜单图标')}</label>
|
||||
<div class="layui-input-block">
|
||||
<div class="layui-input-inline">
|
||||
<input placeholder="请输入或选择图标" name="icon" value='{$vo.icon|default=""}' class="layui-input">
|
||||
</div>
|
||||
<span style="padding:0 12px;min-width:45px" class='layui-btn layui-btn-primary'>
|
||||
<i style="font-size:1.2em;margin:0;float:none" class='{$vo.icon|default=""}'></i>
|
||||
</span>
|
||||
<button data-icon='icon' type='button' class='layui-btn layui-btn-primary'>{:lang('选择图标')}</button>
|
||||
<p class="help-block"><b>可选</b>,设置菜单选项前置图标,目前支持 layui 字体图标及 iconfont 定制字体图标。</p>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
</div>
|
||||
|
||||
<div class="hr-line-dashed"></div>
|
||||
{notempty name='vo.id'}<input type='hidden' value='{$vo.id}' name='id'>{/notempty}
|
||||
|
||||
<div class="layui-form-item text-center">
|
||||
<button class="layui-btn" type='submit'>{:lang('保存数据')}</button>
|
||||
<button class="layui-btn layui-btn-danger" type='button' data-confirm="{:lang('确定要取消编辑吗?')}" data-close>{:lang('取消编辑')}</button>
|
||||
</div>
|
||||
</form>
|
||||
|
||||
<script>
|
||||
require(['jquery.autocompleter'], function () {
|
||||
$('[name="icon"]').on('change', function () {
|
||||
$(this).parent().next().find('i').get(0).className = this.value
|
||||
}), $('input[name=url]').autocompleter({
|
||||
limit: 6, highlightMatches: true, template: '{{ label }} <span> {{ title }} </span>', callback: function (node) {
|
||||
$('input[name=node]').val(node);
|
||||
}, source: (function (subjects, data) {
|
||||
for (var i in subjects) data.push({value: subjects[i].node, label: subjects[i].node, title: subjects[i].title});
|
||||
return data;
|
||||
})(JSON.parse('{$nodes|raw|json_encode}'), [])
|
||||
}), $('input[name=node]').autocompleter({
|
||||
limit: 5, highlightMatches: true, template: '{{ label }} <span> {{ title }} </span>', source: (function (subjects, data) {
|
||||
for (var i in subjects) data.push({value: subjects[i].node, label: subjects[i].node, title: subjects[i].title});
|
||||
return data;
|
||||
})(JSON.parse('{$auths|raw|json_encode}'), [])
|
||||
});
|
||||
});
|
||||
</script>
|
119
app/admin/view/menu/index.html
Normal file
119
app/admin/view/menu/index.html
Normal file
@ -0,0 +1,119 @@
|
||||
{extend name='table'}
|
||||
|
||||
{block name="button"}
|
||||
<!--{if $type eq 'index' and auth("add")}-->
|
||||
<button data-modal='{:url("add")}' data-table-id="MenuTable" class='layui-btn layui-btn-sm layui-btn-primary'>{:lang('添加菜单')}</button>
|
||||
<!--{/if}-->
|
||||
|
||||
<!--{if $type eq 'index' and auth("state")}-->
|
||||
<button data-action='{:url("state")}' data-table-id="MenuTable" data-rule="id#{sps};status#0" class='layui-btn layui-btn-sm layui-btn-primary'>{:lang('禁用菜单')}</button>
|
||||
<!--{/if}-->
|
||||
|
||||
<!--{if $type eq 'recycle' and auth("state")}-->
|
||||
<button data-action='{:url("state")}' data-table-id="MenuTable" data-rule="id#{spp};status#1" class='layui-btn layui-btn-sm layui-btn-primary'>{:lang('激活菜单')}</button>
|
||||
<!--{/if}-->
|
||||
{/block}
|
||||
|
||||
{block name="content"}
|
||||
<div class="layui-tab layui-tab-card">
|
||||
<ul class="layui-tab-title">
|
||||
{foreach ['index'=>lang('全部菜单'),'recycle'=>lang('回 收 站')] as $k=>$v}
|
||||
{if isset($type) and $type eq $k and isset($pid) and $pid == ''}
|
||||
<li class="layui-this" data-open="{:url('index')}?type={$k}">{$v}</li>
|
||||
{else}
|
||||
<li data-open="{:url('index')}?type={$k}">{$v}</li>
|
||||
{/if}{/foreach}
|
||||
|
||||
<!--顶级菜单-->
|
||||
<!--{foreach $menupList as $k=>$v}-->
|
||||
<li data-open="{:url('index')}?type=index&pid={$v.id}" {if isset($pid) and $pid eq $v.id}class="layui-this"{/if} >{$v.title}</li>
|
||||
<!--{/foreach}-->
|
||||
</ul>
|
||||
<div class="layui-tab-content">
|
||||
<table id="MenuTable" data-url="{:request()->url()}" data-target-search="form.form-search"></table>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<script>
|
||||
$(function () {
|
||||
$('#MenuTable').layTable({
|
||||
even: true, height: 'full', page: false,
|
||||
sort: {field: 'sort desc,id', type: 'asc'},
|
||||
where: {type: '{$type|default="index"}', pid: '{$pid|default=""}'},
|
||||
filter: function (items) {
|
||||
var type = this.where.type;
|
||||
return items.filter(function (item) {
|
||||
return !(type === 'index' && parseInt(item.status) === 0);
|
||||
});
|
||||
},
|
||||
cols: [[
|
||||
{checkbox: true, field: 'sps'},
|
||||
{field: 'sort', title: '{:lang("排序权重")}', width: 100, align: 'center', templet: '#SortInputTpl'},
|
||||
{field: 'icon', title: '{:lang("图 标")}', width: 80, align: 'center', templet: '<div><i class="{{d.icon}} font-s18"></i></div>'},
|
||||
{field: 'title', title: '{:lang("菜单名称")}', minWidth: 220, templet: '<div><span class="color-desc">{{d.spl}}</span>{{d.title}}</div>'},
|
||||
{field: 'url', title: '{:lang("跳转链接")}', minWidth: 200},
|
||||
{field: 'status', title: '{:lang("使用状态")}', minWidth: 120, align: 'center', templet: '#StatusSwitchTpl'},
|
||||
{toolbar: '#toolbar', title: '{:lang("操作面板")}', minWidth: 150, align: 'center', fixed: 'right'},
|
||||
]]
|
||||
});
|
||||
|
||||
// 数据状态切换操作
|
||||
layui.form.on('switch(StatusSwitch)', function (object) {
|
||||
object.data = {status: object.elem.checked > 0 ? 1 : 0};
|
||||
object.data.id = object.value.split('|')[object.data.status] || object.value;
|
||||
$.form.load("{:url('state')}", object.data, 'post', function (ret) {
|
||||
if (ret.code < 1) $.msg.error(ret.info, 3, function () {
|
||||
$('#MenuTable').trigger('reload');
|
||||
}); else {
|
||||
$('#MenuTable').trigger('reload');
|
||||
}
|
||||
return false;
|
||||
}, false);
|
||||
});
|
||||
});
|
||||
</script>
|
||||
|
||||
<!-- 数据状态切换模板 -->
|
||||
<script type="text/html" id="StatusSwitchTpl">
|
||||
<!--{if auth("state")}-->
|
||||
{{# if( "{$type|default='index'}"==='index' || (d.spc<1 || d.status<1)){ }}
|
||||
<input type="checkbox" value="{{d.sps}}|{{d.spp}}" lay-text="{:lang('已激活')}|{:lang('已禁用')}" lay-filter="StatusSwitch" lay-skin="switch" {{-d.status>0?'checked':''}}>
|
||||
{{# }else{ }}
|
||||
{{-d.status ? '<b class="color-green">{:lang('已激活')}</b>' : '<b class="color-red">{:lang('已禁用')}</b>'}}
|
||||
{{# } }}
|
||||
<!--{else}-->
|
||||
{{-d.status ? '<b class="color-green">{:lang('已激活')}</b>' : '<b class="color-red">{:lang('已禁用')}</b>'}}
|
||||
<!--{/if}-->
|
||||
</script>
|
||||
|
||||
<!-- 列表排序权重模板 -->
|
||||
<script type="text/html" id="SortInputTpl">
|
||||
<input type="number" min="0" data-blur-number="0" data-action-blur="{:sysuri()}" data-value="id#{{d.id}};action#sort;sort#{value}" data-loading="false" value="{{d.sort}}" class="layui-input text-center">
|
||||
</script>
|
||||
|
||||
<!-- 操控面板的模板 -->
|
||||
<script type="text/html" id="toolbar">
|
||||
|
||||
<!-- {if isset($type) and $type eq 'index'} -->
|
||||
<!-- {if auth('add')} -->
|
||||
{{# if(d.spt<2){ }}
|
||||
<a class="layui-btn layui-btn-sm layui-btn-primary" data-title="添加系统菜单" data-modal='{:url("add")}?pid={{d.id}}'>{:lang('添 加')}</a>
|
||||
{{# }else{ }}
|
||||
<a class="layui-btn layui-btn-sm layui-btn-disabled">{:lang('添 加')}</a>
|
||||
{{# } }}
|
||||
<!-- {/if} -->
|
||||
{if auth('edit')}
|
||||
<a class="layui-btn layui-btn-sm" data-event-dbclick data-title="编辑系统菜单" data-modal='{:url("edit")}?id={{d.id}}'>{:lang('编 辑')}</a>
|
||||
{/if}
|
||||
<!-- {else} -->
|
||||
{if auth('remove')}
|
||||
{{# if( (d.spc<1 || d.status<1)){ }}
|
||||
<a class="layui-btn layui-btn-sm layui-btn-danger" data-confirm="确定要删除菜单吗?" data-action="{:url('remove')}" data-value="id#{{d.sps}}">删 除</a>
|
||||
{{# }else{ }}
|
||||
<a class="layui-btn layui-btn-disabled layui-btn-sm">删 除</a>
|
||||
{{# } }}
|
||||
{/if}
|
||||
<!-- {/if} -->
|
||||
|
||||
</script>
|
||||
{/block}
|
47
app/admin/view/oplog/index.html
Normal file
47
app/admin/view/oplog/index.html
Normal file
@ -0,0 +1,47 @@
|
||||
{extend name='table'}
|
||||
|
||||
{block name="button"}
|
||||
<!--{if auth("remove")}-->
|
||||
<button data-table-id="OplogTable" data-action='{:url("remove")}' data-rule="id#{id}" data-confirm="确定要删除选中的日志吗?" class='layui-btn layui-btn-sm layui-btn-primary'>{:lang('批量删除')}</button>
|
||||
<!--{/if}-->
|
||||
|
||||
<!--{if auth("clear")}-->
|
||||
<button data-table-id="OplogTable" data-load='{:url("clear")}' data-confirm="确定要清空所有日志吗?" class='layui-btn layui-btn-sm layui-btn-primary'>{:lang('清空数据')}</button>
|
||||
<!--{/if}-->
|
||||
{/block}
|
||||
|
||||
{block name="content"}
|
||||
<div class="think-box-shadow">
|
||||
{include file='oplog/index_search'}
|
||||
<table id="OplogTable" data-url="{:request()->url()}" data-target-search="form.form-search"></table>
|
||||
</div>
|
||||
{/block}
|
||||
|
||||
{block name='script'}
|
||||
<script>
|
||||
$(function () {
|
||||
$('#OplogTable').layTable({
|
||||
even: true, height: 'full',
|
||||
sort: {field: 'id', type: 'desc'},
|
||||
cols: [[
|
||||
{checkbox: true},
|
||||
{field: 'id', title: 'ID', width: 80, sort: true, align: 'center'},
|
||||
{field: 'username', title: '{:lang("操作账号")}', minWidth: 100, width: '8%', sort: true, align: 'center'},
|
||||
{field: 'node', title: '{:lang("操作节点")}', minWidth: 120},
|
||||
{field: 'action', title: '{:lang("操作行为")}', minWidth: 120},
|
||||
{field: 'content', title: '{:lang("操作内容")}', minWidth: 150},
|
||||
{field: 'geoip', title: '{:lang("访问地址")}', minWidth: 100, width: '10%'},
|
||||
{field: 'geoisp', title: '{:lang("网络服务商")}', minWidth: 100},
|
||||
{field: 'create_at', title: '{:lang("创建时间")}', minWidth: 170, align: 'center', sort: true},
|
||||
{toolbar: '#toolbar', title: '{:lang("操作面板")}', align: 'center', minWidth: 80, width: '8%', fixed: 'right'}
|
||||
]]
|
||||
});
|
||||
});
|
||||
</script>
|
||||
|
||||
<script type="text/html" id="toolbar">
|
||||
<!--{if auth('remove')}-->
|
||||
<a data-action='{:url("remove")}' data-value="id#{{d.id}}" data-confirm="确认要删除这条记录吗?" class="layui-btn layui-btn-sm layui-btn-danger">{:lang('删 除')}</a>
|
||||
<!--{/if}-->
|
||||
</script>
|
||||
{/block}
|
87
app/admin/view/oplog/index_search.html
Normal file
87
app/admin/view/oplog/index_search.html
Normal file
@ -0,0 +1,87 @@
|
||||
<fieldset>
|
||||
<legend>{:lang('条件搜索')}</legend>
|
||||
<form class="layui-form layui-form-pane form-search" action="{:sysuri()}" onsubmit="return false" method="get" autocomplete="off">
|
||||
|
||||
<div class="layui-form-item layui-inline">
|
||||
<label class="layui-form-label">{:lang('操作账号')}</label>
|
||||
<div class="layui-input-inline">
|
||||
<select name='username' lay-search class="layui-select">
|
||||
<option value=''>-- {:lang('全部')} --</option>
|
||||
{foreach $users as $user}{if $user eq input('get.username')}
|
||||
<option selected value="{$user}">{$user}</option>
|
||||
{else}
|
||||
<option value="{$user}">{$user}</option>
|
||||
{/if}{/foreach}
|
||||
</select>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="layui-form-item layui-inline">
|
||||
<label class="layui-form-label">{:lang('操作行为')}</label>
|
||||
<div class="layui-input-inline">
|
||||
<select name="action" lay-search class="layui-select">
|
||||
<option value=''>-- {:lang('全部')} --</option>
|
||||
{foreach $actions as $action}{if $action eq input('get.action')}
|
||||
<option selected value="{$action}">{$action}</option>
|
||||
{else}
|
||||
<option value="{$action}">{$action}</option>
|
||||
{/if}{/foreach}
|
||||
</select>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="layui-form-item layui-inline">
|
||||
<label class="layui-form-label">{:lang('操作节点')}</label>
|
||||
<label class="layui-input-inline">
|
||||
<input name="node" value="{$get.node|default=''}" placeholder="{:lang('请输入操作节点')}" class="layui-input">
|
||||
</label>
|
||||
</div>
|
||||
|
||||
<div class="layui-form-item layui-inline">
|
||||
<label class="layui-form-label">{:lang('操作内容')}</label>
|
||||
<label class="layui-input-inline">
|
||||
<input name="content" value="{$get.content|default=''}" placeholder="{:lang('请输入操作内容')}" class="layui-input">
|
||||
</label>
|
||||
</div>
|
||||
|
||||
<div class="layui-form-item layui-inline">
|
||||
<label class="layui-form-label">{:lang('访问地址')}</label>
|
||||
<label class="layui-input-inline">
|
||||
<input name="geoip" value="{$get.geoip|default=''}" placeholder="{:lang('请输入访问地址')}" class="layui-input">
|
||||
</label>
|
||||
</div>
|
||||
|
||||
<div class="layui-form-item layui-inline">
|
||||
<label class="layui-form-label">{:lang('创建时间')}</label>
|
||||
<label class="layui-input-inline">
|
||||
<input data-date-range name="create_at" value="{$get.create_at|default=''}" placeholder="{:lang('请选择创建时间')}" class="layui-input">
|
||||
</label>
|
||||
</div>
|
||||
|
||||
<div class="layui-form-item layui-inline">
|
||||
<button type="submit" class="layui-btn layui-btn-primary"><i class="layui-icon"></i> {:lang('搜 索')}</button>
|
||||
<button type="button" data-form-export="{:url('index')}?type={$type|default=''}" class="layui-btn layui-btn-primary">
|
||||
<i class="layui-icon layui-icon-export"></i> {:lang('导 出')}
|
||||
</button>
|
||||
</div>
|
||||
</form>
|
||||
</fieldset>
|
||||
|
||||
<script>
|
||||
require(['excel'], function (excel) {
|
||||
excel.bind(function (data) {
|
||||
|
||||
// 设置表格内容
|
||||
data.forEach(function (item, index) {
|
||||
data[index] = [item.id, item.username, item.node, item.geoip, item.geoisp, item.action, item.content, item.create_at];
|
||||
});
|
||||
|
||||
// 设置表头内容
|
||||
data.unshift(['ID', '{:lang("操作账号")}', '{:lang("操作节点")}', '{:lang("访问地址")}', '{:lang("网络服务商")}', '{:lang("操作行为")}', '{:lang("操作内容")}', '{:lang("创建时间")}']);
|
||||
|
||||
// 应用表格样式
|
||||
return this.withStyle(data, {A: 60, B: 80, C: 99, E: 120, G: 120});
|
||||
|
||||
}, '{:lang("操作日志")}' + layui.util.toDateString(Date.now(), '_yyyyMMdd_HHmmss'));
|
||||
});
|
||||
</script>
|
131
app/admin/view/queue/index.html
Normal file
131
app/admin/view/queue/index.html
Normal file
@ -0,0 +1,131 @@
|
||||
{extend name='table'}
|
||||
|
||||
{block name="button"}
|
||||
|
||||
{if isset($super) and $super}
|
||||
|
||||
<a data-table-id="QueueTable" class="layui-btn layui-btn-sm layui-btn-primary" data-queue="{:url('admin/api.plugs/optimize')}">{:lang('优化数据库')}</a>
|
||||
|
||||
{if isset($iswin) and ($iswin or php_sapi_name() eq 'cli')}
|
||||
<button data-load='{:url("admin/api.queue/start")}' class='layui-btn layui-btn-sm layui-btn-primary'>{:lang('开启服务')}</button>
|
||||
<button data-load='{:url("admin/api.queue/stop")}' class='layui-btn layui-btn-sm layui-btn-primary'>{:lang('关闭服务')}</button>
|
||||
{/if}
|
||||
|
||||
{if auth("clean")}
|
||||
<button data-table-id="QueueTable" data-queue='{:url("clean")}' class='layui-btn layui-btn-sm layui-btn-primary'>{:lang('定时清理')}</button>
|
||||
{/if}
|
||||
|
||||
{/if}
|
||||
|
||||
{if auth("remove")}
|
||||
<button data-table-id="QueueTable" data-action='{:url("remove")}' data-rule="id#{id}" data-confirm="{:lang('确定批量删除记录吗?')}" class='layui-btn layui-btn-sm layui-btn-primary'>{:lang('批量删除')}</button>
|
||||
{/if}
|
||||
|
||||
{/block}
|
||||
|
||||
{block name="content"}
|
||||
<div class="think-box-notify" type="info">
|
||||
|
||||
<!--{if isset($super) and $super}-->
|
||||
<b>{:lang('服务状态')}:</b><b class="margin-right-5" data-queue-message><span class="color-desc">{:lang('检查中')}</span></b>
|
||||
<b data-tips-text="{:lang('点击可复制【服务启动指令】')}" class="layui-icon pointer margin-right-20" data-copy="{$command|default=''}"></b>
|
||||
<script>$('[data-queue-message]').load('{:sysuri("admin/api.queue/status")}');</script>
|
||||
<!--{/if}-->
|
||||
|
||||
<b>{:lang('任务统计')}:</b>{:lang('待处理 %s 个任务,处理中 %s 个任务,已完成 %s 个任务,已失败 %s 个任务。', [
|
||||
'<b class="color-text" data-extra="pre">..</b>',
|
||||
'<b class="color-blue" data-extra="dos">..</b>',
|
||||
'<b class="color-green" data-extra="oks">..</b>',
|
||||
'<b class="color-red" data-extra="ers">..</b>'
|
||||
])}
|
||||
|
||||
</div>
|
||||
|
||||
<div class="think-box-shadow">
|
||||
{include file='queue/index_search'}
|
||||
<table id="QueueTable" data-line="2" data-url="{:request()->url()}" data-target-search="form.form-search"></table>
|
||||
</div>
|
||||
{/block}
|
||||
|
||||
{block name='script'}
|
||||
<script>
|
||||
$(function () {
|
||||
$('#QueueTable').layTable({
|
||||
even: true, height: 'full',
|
||||
sort: {field: 'loops_time desc,id', type: 'desc'},
|
||||
// 扩展数据处理,需要返回原 items 数据
|
||||
filter: function (items, result) {
|
||||
return result && result.extra && $('[data-extra]').map(function () {
|
||||
this.innerHTML = result.extra[this.dataset.extra] || 0;
|
||||
}), items;
|
||||
},
|
||||
cols: [[
|
||||
{checkbox: true, fixed: 'left'},
|
||||
{
|
||||
field: 'id', title: '{:lang("任务名称")}', width: '25%', sort: true, templet: function (d) {
|
||||
if (d.loops_time > 0) {
|
||||
d.one = '<span class="layui-badge think-bg-blue">循</span>';
|
||||
} else {
|
||||
d.one = '<span class="layui-badge think-bg-red">次</span>';
|
||||
}
|
||||
if (parseInt(d.rscript) === 1) {
|
||||
d.two = '<span class="layui-badge layui-bg-green">复</span>';
|
||||
} else {
|
||||
d.two = '<span class="layui-badge think-bg-violet">单</span>';
|
||||
}
|
||||
return laytpl('{{-d.one}}任务编号:<b>{{d.code}}</b><br>{{-d.two}}任务名称:{{d.title}}').render(d);
|
||||
}
|
||||
},
|
||||
{
|
||||
field: 'exec_time', title: '{:lang("任务计划")}', width: '25%', templet: function (d) {
|
||||
d.html = '执行指令:' + d.command + '<br>计划执行:' + d.exec_time;
|
||||
if (d.loops_time > 0) {
|
||||
return d.html + ' ( 每 <b class="color-blue">' + d.loops_time + '</b> 秒 ) ';
|
||||
} else {
|
||||
return d.html + ' <span class="color-desc">( 单次任务 )</span> ';
|
||||
}
|
||||
}
|
||||
},
|
||||
{
|
||||
field: 'loops_time', title: '{:lang("任务状态")}', width: '30%', templet: function (d) {
|
||||
d.html = ([
|
||||
'<span class="pull-left layui-badge layui-badge-middle layui-bg-gray">未知</span>',
|
||||
'<span class="pull-left layui-badge layui-badge-middle layui-bg-black">等待</span>',
|
||||
'<span class="pull-left layui-badge layui-badge-middle layui-bg-blue">执行</span>',
|
||||
'<span class="pull-left layui-badge layui-badge-middle layui-bg-green">完成</span>',
|
||||
'<span class="pull-left layui-badge layui-badge-middle layui-bg-red">失败</span>',
|
||||
][d.status] || '') + '执行时间:';
|
||||
d.enter_time = d.enter_time || '';
|
||||
d.outer_time = d.outer_time || '0.0000';
|
||||
if (d.enter_time.length > 12) {
|
||||
d.html += d.enter_time.substring(12) + '<span class="color-desc"> ( ' + d.outer_time + ' ) </span>';
|
||||
d.html += ' 已执行 <b class="color-blue">' + (d.attempts || 0) + '</b> 次';
|
||||
} else {
|
||||
d.html += '<span class="color-desc">任务未执行</span>'
|
||||
}
|
||||
return d.html + '<br>执行结果:<span class="color-blue">' + (d.exec_desc || '<span class="color-desc">未获取到执行结果</span>') + '</span>';
|
||||
}
|
||||
},
|
||||
{toolbar: '#toolbar', title: '{:lang("操作面板")}', align: 'center', minWidth: 210, fixed: 'right'}
|
||||
]]
|
||||
});
|
||||
});
|
||||
</script>
|
||||
|
||||
<script type="text/html" id="toolbar">
|
||||
|
||||
<!--{if auth('redo')}-->
|
||||
{{# if(d.status===4||d.status===3){ }}
|
||||
<a class="layui-btn layui-btn-sm" data-confirm="确定要重置该任务吗?" data-queue="{:url('redo')}?code={{d.code}}">{:lang('重 置')}</a>
|
||||
{{# }else{ }}
|
||||
<a class="layui-btn layui-btn-sm layui-btn-disabled">{:lang('重 置')}</a>
|
||||
{{# } }}
|
||||
<!--{/if}-->
|
||||
|
||||
<!--{if auth('remove')}-->
|
||||
<a class='layui-btn layui-btn-sm layui-btn-danger' data-confirm="{:lang('确定要删除该记录吗?')}" data-action='{:url("remove")}' data-value="id#{{d.id}}">{:lang('删 除')}</a>
|
||||
<!--{/if}-->
|
||||
|
||||
<a class='layui-btn layui-btn-sm layui-btn-normal' onclick="$.loadQueue('{{d.code}}',false,this)">{:lang('日 志')}</a>
|
||||
</script>
|
||||
{/block}
|
45
app/admin/view/queue/index_search.html
Normal file
45
app/admin/view/queue/index_search.html
Normal file
@ -0,0 +1,45 @@
|
||||
<fieldset>
|
||||
<legend>{:lang('条件搜索')}</legend>
|
||||
<form class="layui-form layui-form-pane form-search" action="{:sysuri()}" onsubmit="return false" method="get" autocomplete="off">
|
||||
|
||||
<div class="layui-form-item layui-inline">
|
||||
<label class="layui-form-label">{:lang('编号名称')}</label>
|
||||
<label class="layui-input-inline">
|
||||
<input name="title" value="{$get.title|default=''}" placeholder="{:lang('请输入名称或编号')}" class="layui-input">
|
||||
</label>
|
||||
</div>
|
||||
|
||||
<div class="layui-form-item layui-inline">
|
||||
<label class="layui-form-label">{:lang('任务指令')}</label>
|
||||
<label class="layui-input-inline">
|
||||
<input name="command" value="{$get.command|default=''}" placeholder="{:lang('请输入任务指令')}" class="layui-input">
|
||||
</label>
|
||||
</div>
|
||||
|
||||
<div class="layui-form-item layui-inline">
|
||||
<label class="layui-form-label">{:lang('任务状态')}</label>
|
||||
<label class="layui-input-inline">
|
||||
<select name="status" class="layui-select">
|
||||
<option value=''>-- {:lang('全部')} --</option>
|
||||
{foreach ['1'=>lang('等待处理'),'2'=>lang('正在处理'),'3'=>lang('处理完成'),'4'=>lang('处理失败')] as $k=>$v}
|
||||
{if isset($get.status) and $get.status eq $k}
|
||||
<option selected value="{$k}">{$v}</option>
|
||||
{else}
|
||||
<option value="{$k}">{$v}</option>
|
||||
{/if}{/foreach}
|
||||
</select>
|
||||
</label>
|
||||
</div>
|
||||
|
||||
<div class="layui-form-item layui-inline">
|
||||
<label class="layui-form-label">{:lang('计划时间')}</label>
|
||||
<label class="layui-input-inline">
|
||||
<input data-date-range name="exec_time" value="{$get.exec_time|default=''}" placeholder="{:lang('请选择计划时间')}" class="layui-input">
|
||||
</label>
|
||||
</div>
|
||||
|
||||
<div class="layui-form-item layui-inline">
|
||||
<button class="layui-btn layui-btn-primary"><i class="layui-icon"></i> {:lang('搜 索')}</button>
|
||||
</div>
|
||||
</form>
|
||||
</fieldset>
|
23
app/admin/view/table.html
Normal file
23
app/admin/view/table.html
Normal file
@ -0,0 +1,23 @@
|
||||
<div class="layui-card">
|
||||
{block name='style'}{/block}
|
||||
{block name='header'}
|
||||
{notempty name='title'}
|
||||
<div class="layui-card-header">
|
||||
<span class="layui-icon font-s10 color-desc margin-right-5"></span>{$title|lang}
|
||||
<div class="pull-right">{block name='button'}{/block}</div>
|
||||
</div>
|
||||
{/notempty}
|
||||
{/block}
|
||||
<div class="layui-card-line"></div>
|
||||
<div class="layui-card-body">
|
||||
<div class="layui-card-table">
|
||||
{notempty name='showErrorMessage'}
|
||||
<div class="think-box-notify" type="error">
|
||||
<b>{:lang('系统提示:')}</b><span>{$showErrorMessage|raw}</span>
|
||||
</div>
|
||||
{/notempty}
|
||||
{block name='content'}{/block}
|
||||
</div>
|
||||
</div>
|
||||
{block name='script'}{/block}
|
||||
</div>
|
114
app/admin/view/user/form.html
Normal file
114
app/admin/view/user/form.html
Normal file
@ -0,0 +1,114 @@
|
||||
<form action="{:sysuri()}" method="post" data-auto="true" class="layui-form layui-card" data-table-id="UserTable">
|
||||
<div class="layui-card-body padding-left-40">
|
||||
|
||||
<fieldset class="layui-bg-gray">
|
||||
<legend><b class="layui-badge think-bg-violet">用户账号</b></legend>
|
||||
|
||||
<div class="layui-row layui-col-space15">
|
||||
<div class="layui-col-xs2 text-center padding-top-15">
|
||||
<input type="hidden" data-cut-width="500" data-cut-height="500" data-max-width="500" data-max-height="500" name="headimg" value="{$vo.headimg|default=''}">
|
||||
<script>$('[name=headimg]').uploadOneImage();</script>
|
||||
</div>
|
||||
<div class="layui-col-xs5">
|
||||
<label class="block relative">
|
||||
<span class="help-label"><b>登录账号</b>User Name</span>
|
||||
{if isset($vo) and isset($vo.username)}
|
||||
<input name="username" value='{$vo.username|default=""}' required readonly class="layui-input think-bg-gray">
|
||||
{else}
|
||||
<input name="username" value='{$vo.username|default=""}' required pattern="^.{4,}$" vali-name="登录账号" placeholder="请输入登录账号" class="layui-input">
|
||||
{/if}
|
||||
<span class="help-block">登录账号不能少于4位字符,创建后不能再次修改.</span>
|
||||
</label>
|
||||
</div>
|
||||
<div class="layui-col-xs5">
|
||||
<label class="block relative">
|
||||
<span class="help-label"><b>用户名称</b>Nick Name</span>
|
||||
<input name="nickname" value='{$vo.nickname|default=""}' required vali-name="用户名称" placeholder="请输入用户名称" class="layui-input">
|
||||
<span class="help-block">用于区分用户数据的用户名称,请尽量不要重复.</span>
|
||||
</label>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
</fieldset>
|
||||
|
||||
{if !empty($bases) || !empty($auths)}
|
||||
<fieldset class="layui-bg-gray">
|
||||
<legend><b class="layui-badge think-bg-violet">用户权限</b></legend>
|
||||
{if !empty($bases)}
|
||||
<div class="layui-form-item">
|
||||
<div class="help-label"><b>角色身份</b>Role Identity</div>
|
||||
<div class="layui-textarea help-checks">
|
||||
{foreach $bases as $base}
|
||||
<label class="think-checkbox">
|
||||
{if isset($vo.usertype) and $vo.usertype eq $base.code}
|
||||
<input type="radio" checked name="usertype" value="{$base.code}" lay-ignore>{$base.name}
|
||||
{else}
|
||||
<input type="radio" name="usertype" value="{$base.code}" lay-ignore>{$base.name}
|
||||
{/if}
|
||||
</label>
|
||||
{/foreach}
|
||||
</div>
|
||||
</div>
|
||||
{/if}
|
||||
{if !empty($auths)}
|
||||
<div class="layui-form-item">
|
||||
<div class="help-label"><b>访问权限</b>Role Permission</div>
|
||||
<div class="layui-textarea help-checks">
|
||||
{if isset($vo.username) and $vo.username eq $super}
|
||||
<span class="color-desc padding-left-5">超级用户拥所有访问权限,不需要配置权限。</span>
|
||||
{else}{foreach $auths as $authorize}
|
||||
<label class="think-checkbox">
|
||||
{if in_array($authorize.id, $vo.authorize)}
|
||||
<input type="checkbox" checked name="authorize[]" value="{$authorize.id}" lay-ignore>{$authorize.title}
|
||||
{else}
|
||||
<input type="checkbox" name="authorize[]" value="{$authorize.id}" lay-ignore>{$authorize.title}
|
||||
{/if}
|
||||
</label>
|
||||
{/foreach}{/if}
|
||||
</div>
|
||||
</div>
|
||||
{/if}
|
||||
</fieldset>
|
||||
{/if}
|
||||
|
||||
<fieldset class="layui-bg-gray">
|
||||
<legend><b class="layui-badge think-bg-violet">用户资料</b></legend>
|
||||
<div class="layui-row layui-col-space15">
|
||||
<div class="layui-col-xs4">
|
||||
<label class="relative block">
|
||||
<span class="help-label"><b>联系邮箱</b>Contact Email</span>
|
||||
<input name="contact_mail" value='{$vo.contact_mail|default=""}' pattern="email" vali-name="联系邮箱" placeholder="请输入联系电子邮箱" class="layui-input">
|
||||
<span class="color-desc">可选,请填写用户常用的电子邮箱</span>
|
||||
</label>
|
||||
</div>
|
||||
<div class="layui-col-xs4">
|
||||
<label class="relative block">
|
||||
<span class="help-label"><b>联系手机</b>Contact Mobile</span>
|
||||
<input type="tel" maxlength="11" name="contact_phone" value='{$vo.contact_phone|default=""}' pattern="phone" vali-name="联系手机" placeholder="请输入用户联系手机" class="layui-input">
|
||||
<span class="color-desc">可选,请填写用户常用的联系手机号</span>
|
||||
</label>
|
||||
</div>
|
||||
<div class="layui-col-xs4">
|
||||
<label class="relative block">
|
||||
<span class="help-label"><b>联系QQ</b>Contact QQ</span>
|
||||
<input name="contact_qq" maxlength="11" value='{$vo.contact_qq|default=""}' pattern="qq" vali-name="联系QQ" placeholder="请输入常用的联系QQ" class="layui-input">
|
||||
<span class="color-desc">可选,请填写用户常用的联系QQ号</span>
|
||||
</label>
|
||||
</div>
|
||||
</div>
|
||||
<label class="layui-form-item block relative margin-top-10">
|
||||
<span class="help-label"><b>用户描述</b>User Remark</span>
|
||||
<textarea placeholder="请输入用户描述" class="layui-textarea" name="describe">{$vo.describe|default=""}</textarea>
|
||||
</label>
|
||||
</fieldset>
|
||||
|
||||
</div>
|
||||
|
||||
<div class="hr-line-dashed"></div>
|
||||
{notempty name='vo.id'}<input type='hidden' value='{$vo.id}' name='id'>{/notempty}
|
||||
|
||||
<div class="layui-form-item text-center">
|
||||
<button class="layui-btn" type='submit'>保存数据</button>
|
||||
<button class="layui-btn layui-btn-danger" type='button' data-confirm="确定要取消编辑吗?" data-close>取消编辑</button>
|
||||
</div>
|
||||
</form>
|
116
app/admin/view/user/index.html
Normal file
116
app/admin/view/user/index.html
Normal file
@ -0,0 +1,116 @@
|
||||
{extend name='table'}
|
||||
|
||||
{block name="button"}
|
||||
{if isset($type) and $type eq 'index'}
|
||||
<!--{if auth("add")}-->
|
||||
<button data-modal='{:url("add")}' data-title="{:lang('添加用户')}" class='layui-btn layui-btn-sm layui-btn-primary'>{:lang('添加用户')}</button>
|
||||
<!--{/if}-->
|
||||
<!--{if auth("state")}-->
|
||||
<a data-confirm="确定要禁用这些用户吗?" data-table-id="UserTable" data-action="{:url('state')}" data-rule="id#{id};status#0" class='layui-btn layui-btn-sm layui-btn-primary'>{:lang('批量禁用')}</a>
|
||||
<!--{/if}-->
|
||||
{else}
|
||||
<!--{if auth("state")}-->
|
||||
<a data-confirm="确定要恢复这些账号吗?" data-table-id="UserTable" data-action="{:url('state')}" data-rule="id#{id};status#1" class='layui-btn layui-btn-sm layui-btn-primary'>{:lang('批量恢复')}</a>
|
||||
<!--{/if}-->
|
||||
<!--{if auth("remove")}-->
|
||||
<a data-confirm="确定永久删除这些账号吗?" data-table-id="UserTable" data-action='{:url("remove")}' data-rule="id#{id}" class='layui-btn layui-btn-sm layui-btn-primary'>{:lang('批量删除')}</a>
|
||||
<!--{/if}-->
|
||||
{/if}
|
||||
{/block}
|
||||
|
||||
{block name="content"}
|
||||
<div class="layui-tab layui-tab-card">
|
||||
<ul class="layui-tab-title">
|
||||
{foreach ['index'=>lang('系统用户'),'recycle'=>lang('回 收 站')] as $k=>$v}{if isset($type) and $type eq $k}
|
||||
<li data-open="{:url('index')}?type={$k}" class="layui-this">{$v}</li>
|
||||
{else}
|
||||
<li data-open="{:url('index')}?type={$k}">{$v}</li>
|
||||
{/if}{/foreach}
|
||||
</ul>
|
||||
<div class="layui-tab-content">
|
||||
{include file='user/index_search'}
|
||||
<table id="UserTable" data-url="{:sysuri('index')}" data-target-search="form.form-search"></table>
|
||||
</div>
|
||||
</div>
|
||||
<script>
|
||||
$(function () {
|
||||
$('#UserTable').layTable({
|
||||
even: true, height: 'full',
|
||||
sort: {field: 'sort desc,id', type: 'desc'},
|
||||
where: {type: '{$type|default="index"}'},
|
||||
cols: [[
|
||||
{checkbox: true, fixed: true},
|
||||
{field: 'sort', title: '{:lang("排序权重")}', width: 100, align: 'center', sort: true, templet: '#SortInputTpl'},
|
||||
{
|
||||
field: 'headimg', title: '{:lang("头像")}', width: 60, align: 'center', templet: function (d) {
|
||||
if (!d.headimg) return '-';
|
||||
return layui.laytpl('<div class="headimg headimg-ss shadow-inset margin-0" data-tips-image data-tips-hover data-lazy-src="{{d.headimg}}"></div>').render(d);
|
||||
}
|
||||
},
|
||||
{field: 'username', title: '{:lang("登录账号")}', minWidth: 100, align: 'center', templet: '<div>{{d.username||"-"}}</div>'},
|
||||
{field: 'nickname', title: '{:lang("用户名称")}', minWidth: 100, align: 'center', templet: '<div>{{d.nickname||"-"}}</div>'},
|
||||
/* {notempty name='bases'} */
|
||||
{
|
||||
field: 'usertype', title: '{:lang("角色身份")}', minWidth: 100, align: 'center', templet: function (d) {
|
||||
d.userinfo = d.userinfo || {};
|
||||
return d.userinfo.code ? (d.userinfo.name + ' ( ' + d.userinfo.code + ' ) ') : '-';
|
||||
}
|
||||
},
|
||||
/* {/notempty} */
|
||||
// {field: 'contact_mail', title: '联系邮箱', minWidth: 80, templet: '<div>{{d.contact_mail||"-"}}</div>'},
|
||||
// {field: 'contact_phone', title: '联系电话', minWidth: 80, templet: '<div>{{d.contact_phone||"-"}}</div>'},
|
||||
{field: 'status', title: '{:lang("使用状态")}', align: 'center', minWidth: 110, templet: '#StatusSwitchTpl'},
|
||||
{field: 'login_num', title: '{:lang("登录次数")}', align: 'center', minWidth: 100, sort: true},
|
||||
{field: 'login_at', title: '{:lang("最后登录")}', align: 'center', minWidth: 170, sort: true},
|
||||
{field: 'create_at', title: '{:lang("创建时间")}', align: 'center', minWidth: 170, sort: true},
|
||||
{toolbar: '#toolbar', title: '{:lang("操作面板")}', align: 'center', minWidth: 180, fixed: 'right'}
|
||||
]]
|
||||
});
|
||||
|
||||
// 数据状态切换操作
|
||||
layui.form.on('switch(StatusSwitch)', function (obj) {
|
||||
var data = {id: obj.value, status: obj.elem.checked > 0 ? 1 : 0};
|
||||
$.form.load("{:url('state')}", data, 'post', function (ret) {
|
||||
if (ret.code < 1) $.msg.error(ret.info, 3, function () {
|
||||
$('#UserTable').trigger('reload');
|
||||
}); else {
|
||||
$('#UserTable').trigger('reload')
|
||||
}
|
||||
return false;
|
||||
}, false);
|
||||
});
|
||||
});
|
||||
</script>
|
||||
|
||||
<!-- 数据状态切换模板 -->
|
||||
<script type="text/html" id="StatusSwitchTpl">
|
||||
<!--{if auth("state")}-->
|
||||
<input type="checkbox" value="{{d.id}}" lay-skin="switch" lay-text="{:lang('已激活')}|{:lang('已禁用')}" lay-filter="StatusSwitch" {{-d.status>0?'checked':''}}>
|
||||
<!--{else}-->
|
||||
{{-d.status ? '<b class="color-green">{:lang("已激活")}</b>' : '<b class="color-red">{:lang("已禁用")}</b>'}}
|
||||
<!--{/if}-->
|
||||
</script>
|
||||
|
||||
<!-- 列表排序权重模板 -->
|
||||
<script type="text/html" id="SortInputTpl">
|
||||
<input type="number" min="0" data-blur-number="0" data-action-blur="{:sysuri()}" data-value="id#{{d.id}};action#sort;sort#{value}" data-loading="false" value="{{d.sort}}" class="layui-input text-center">
|
||||
</script>
|
||||
|
||||
<script type="text/html" id="toolbar">
|
||||
{if isset($type) and $type eq 'index'}
|
||||
<!--{if auth("edit")}-->
|
||||
<a class="layui-btn layui-btn-sm" data-event-dbclick data-title="{:lang('编辑用户')}" data-modal='{:url("edit")}?id={{d.id}}'>{:lang('编 辑')}</a>
|
||||
<!--{/if}-->
|
||||
<!--{if auth("pass")}-->
|
||||
<a class="layui-btn layui-btn-sm layui-btn-normal" data-title="{:lang('设置密码')}" data-modal='{:url("pass")}?id={{d.id}}'>{:lang('密 码')}</a>
|
||||
<!--{/if}-->
|
||||
{else}
|
||||
<!--{if auth("edit")}-->
|
||||
<a class="layui-btn layui-btn-sm" data-event-dbclick data-title="{:lang('编辑用户')}" data-modal='{:url("edit")}?id={{d.id}}'>{:lang('编 辑')}</a>
|
||||
<!--{/if}-->
|
||||
<!--{if auth("remove")}-->
|
||||
<a class="layui-btn layui-btn-sm layui-btn-danger" data-confirm="{:lang('确定要永久删除吗?')}" data-action="{:url('remove')}" data-value="id#{{d.id}}">{:lang('删 除')}</a>
|
||||
<!--{/if}-->
|
||||
{/if}
|
||||
</script>
|
||||
{/block}
|
44
app/admin/view/user/index_search.html
Normal file
44
app/admin/view/user/index_search.html
Normal file
@ -0,0 +1,44 @@
|
||||
<form class="layui-form layui-form-pane form-search" action="{:sysuri()}" onsubmit="return false" method="get" autocomplete="off">
|
||||
|
||||
<div class="layui-form-item layui-inline">
|
||||
<label class="layui-form-label">{:lang('账号名称')}</label>
|
||||
<label class="layui-input-inline">
|
||||
<input name="username" value="{$get.username|default=''}" placeholder="{:lang('请输入账号或名称')}" class="layui-input">
|
||||
</label>
|
||||
</div>
|
||||
|
||||
<!--{notempty name='bases'}-->
|
||||
<div class="layui-form-item layui-inline">
|
||||
<label class="layui-form-label">{:lang('角色身份')}</label>
|
||||
<div class="layui-input-inline">
|
||||
<select name="usertype" lay-search class="layui-select">
|
||||
<option value=''>-- {:lang('全部')} --</option>
|
||||
{foreach $bases as $base}{if $base.code eq input('get.usertype')}
|
||||
<option selected value="{$base.code|default=''}">{$base.name|default=''} ( {$base.code|default=''} )</option>
|
||||
{else}
|
||||
<option value="{$base.code|default=''}">{$base.name|default=''} ( {$base.code|default=''} )</option>
|
||||
{/if}{/foreach}
|
||||
</select>
|
||||
</div>
|
||||
</div>
|
||||
<!--{/notempty}-->
|
||||
|
||||
<div class="layui-form-item layui-inline">
|
||||
<label class="layui-form-label">{:lang('最后登录')}</label>
|
||||
<div class="layui-input-inline">
|
||||
<input data-date-range name="login_at" value="{$get.login_at|default=''}" placeholder="{:lang('请选择登录时间')}" class="layui-input">
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="layui-form-item layui-inline">
|
||||
<label class="layui-form-label">{:lang('创建时间')}</label>
|
||||
<div class="layui-input-inline">
|
||||
<input data-date-range name="create_at" value="{$get.create_at|default=''}" placeholder="{:lang('请选择创建时间')}" class="layui-input">
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<div class="layui-form-item layui-inline">
|
||||
<input type="hidden" name="type" value="{$type|default='index'}">
|
||||
<button class="layui-btn layui-btn-primary"><i class="layui-icon"></i> {:lang('搜 索')}</button>
|
||||
</div>
|
||||
</form>
|
40
app/admin/view/user/pass.html
Normal file
40
app/admin/view/user/pass.html
Normal file
@ -0,0 +1,40 @@
|
||||
<form action="{:sysuri()}" method="post" data-auto="true" class="layui-form layui-card" data-table-id="UserTable">
|
||||
<div class="layui-card-body padding-left-40">
|
||||
|
||||
|
||||
<label class="layui-form-item relative block">
|
||||
<span class="help-label"><b>登录用户账号</b>Username</span>
|
||||
<input disabled value='{$vo.username|default=""}' class="layui-input think-bg-gray">
|
||||
<span class="help-block">登录用户账号创建后,不允许再次修改。</span>
|
||||
</label>
|
||||
|
||||
<!--{if $verify}-->
|
||||
<label class="layui-form-item relative block">
|
||||
<span class="help-label"><b>旧的登录密码</b>Old Password</span>
|
||||
<input type="password" autofocus name="oldpassword" value='' pattern="^\S{1,}$" required vali-name="验证密码" placeholder="请输入旧的登录密码" class="layui-input">
|
||||
<span class="color-desc">请输入旧密码来验证修改权限,旧密码不限制格式。</span>
|
||||
</label>
|
||||
<!--{/if}-->
|
||||
|
||||
<label class="layui-form-item relative block">
|
||||
<span class="help-label"><b>新的登录密码</b>New Password</span>
|
||||
<input type="password" name="password" maxlength="32" pattern="^(?![\d]+$)(?![a-zA-Z]+$)(?![^\da-zA-Z]+$).{6,32}$" required vali-name="登录密码" placeholder="请输入新的登录密码" class="layui-input">
|
||||
<span class="color-desc">密码必须包含大小写字母、数字、符号的任意两者组合。</span>
|
||||
</label>
|
||||
|
||||
<label class="layui-form-item relative block">
|
||||
<span class="help-label"><b>重复登录密码</b>Repeat Password</span>
|
||||
<input type="password" name="repassword" maxlength="32" pattern="^(?![\d]+$)(?![a-zA-Z]+$)(?![^\da-zA-Z]+$).{6,32}$" required vali-name="重复密码" placeholder="请重复输入登录密码" class="layui-input">
|
||||
<span class="color-desc">密码必须包含大小写字母、数字、符号的任意两者组合。</span>
|
||||
</label>
|
||||
|
||||
</div>
|
||||
|
||||
<div class="hr-line-dashed"></div>
|
||||
{notempty name='vo.id'}<input type='hidden' value='{$vo.id}' name='id'>{/notempty}
|
||||
|
||||
<div class="layui-form-item text-center">
|
||||
<button class="layui-btn" type='submit'>保存数据</button>
|
||||
<button class="layui-btn layui-btn-danger" type='button' data-confirm="确定要取消编辑吗?" data-close>取消编辑</button>
|
||||
</div>
|
||||
</form>
|
27
app/index/controller/Index.php
Normal file
27
app/index/controller/Index.php
Normal file
@ -0,0 +1,27 @@
|
||||
<?php
|
||||
|
||||
// +----------------------------------------------------------------------
|
||||
// | Static Plugin for ThinkAdmin
|
||||
// +----------------------------------------------------------------------
|
||||
// | 版权所有 2014~2024 ThinkAdmin [ thinkadmin.top ]
|
||||
// +----------------------------------------------------------------------
|
||||
// | 官方网站: https://thinkadmin.top
|
||||
// +----------------------------------------------------------------------
|
||||
// | 开源协议 ( https://mit-license.org )
|
||||
// | 免责声明 ( https://thinkadmin.top/disclaimer )
|
||||
// +----------------------------------------------------------------------
|
||||
// | gitee 代码仓库:https://gitee.com/zoujingli/think-plugs-static
|
||||
// | github 代码仓库:https://github.com/zoujingli/think-plugs-static
|
||||
// +----------------------------------------------------------------------
|
||||
|
||||
namespace app\index\controller;
|
||||
|
||||
use think\admin\Controller;
|
||||
|
||||
class Index extends Controller
|
||||
{
|
||||
public function index()
|
||||
{
|
||||
$this->redirect(sysuri('admin/login/index'));
|
||||
}
|
||||
}
|
109
app/wechat/Service.php
Normal file
109
app/wechat/Service.php
Normal file
@ -0,0 +1,109 @@
|
||||
<?php
|
||||
|
||||
// +----------------------------------------------------------------------
|
||||
// | Wechat Plugin for ThinkAdmin
|
||||
// +----------------------------------------------------------------------
|
||||
// | 版权所有 2014~2025 Anyon <zoujingli@qq.com>
|
||||
// +----------------------------------------------------------------------
|
||||
// | 官方网站: https://thinkadmin.top
|
||||
// +----------------------------------------------------------------------
|
||||
// | 开源协议 ( https://mit-license.org )
|
||||
// | 免责声明 ( https://thinkadmin.top/disclaimer )
|
||||
// +----------------------------------------------------------------------
|
||||
// | gitee 代码仓库:https://gitee.com/zoujingli/think-plugs-wechat
|
||||
// | github 代码仓库:https://github.com/zoujingli/think-plugs-wechat
|
||||
// +----------------------------------------------------------------------
|
||||
|
||||
declare (strict_types=1);
|
||||
|
||||
namespace app\wechat;
|
||||
|
||||
use app\wechat\command\Auto;
|
||||
use app\wechat\command\Clear;
|
||||
use app\wechat\command\Fans;
|
||||
use app\wechat\service\AutoService;
|
||||
use app\wechat\service\PaymentService;
|
||||
use think\admin\extend\CodeExtend;
|
||||
use think\admin\Plugin;
|
||||
use think\Request;
|
||||
|
||||
/**
|
||||
* 组件注册服务
|
||||
* @class Service
|
||||
* @package app\wechat
|
||||
*/
|
||||
class Service extends Plugin
|
||||
{
|
||||
/**
|
||||
* 定义插件名称
|
||||
* @var string
|
||||
*/
|
||||
protected $appName = '微信管理';
|
||||
|
||||
/**
|
||||
* 定义安装包名
|
||||
* @var string
|
||||
*/
|
||||
protected $package = 'zoujingli/think-plugs-wechat';
|
||||
|
||||
/**
|
||||
* 注册组件服务
|
||||
* @return void
|
||||
*/
|
||||
public function register(): void
|
||||
{
|
||||
// 注册模块指令
|
||||
$this->commands([Fans::class, Auto::class, Clear::class]);
|
||||
|
||||
// 注册粉丝关注事件
|
||||
$this->app->event->listen('WechatFansSubscribe', static function ($openid) {
|
||||
AutoService::register($openid);
|
||||
});
|
||||
|
||||
// 注册支付通知路由
|
||||
$this->app->route->any('/plugin-wxpay-notify/:vars', static function (Request $request) {
|
||||
try {
|
||||
$data = json_decode(CodeExtend::deSafe64($request->param('vars')), true);
|
||||
return PaymentService::notify($data);
|
||||
} catch (\Exception|\Error $exception) {
|
||||
return "Error: {$exception->getMessage()}";
|
||||
}
|
||||
});
|
||||
}
|
||||
|
||||
/**
|
||||
* 增加微信配置
|
||||
* @return array[]
|
||||
*/
|
||||
public static function menu(): array
|
||||
{
|
||||
$code = app(static::class)->appCode;
|
||||
// 设置插件菜单
|
||||
return [
|
||||
[
|
||||
'name' => '微信管理',
|
||||
'subs' => [
|
||||
['name' => '微信接口配置', 'icon' => 'layui-icon layui-icon-set', 'node' => "{$code}/config/options"],
|
||||
['name' => '微信支付配置', 'icon' => 'layui-icon layui-icon-rmb', 'node' => "{$code}/config/payment"],
|
||||
],
|
||||
],
|
||||
[
|
||||
'name' => '微信定制',
|
||||
'subs' => [
|
||||
['name' => '微信粉丝管理', 'icon' => 'layui-icon layui-icon-username', 'node' => "{$code}/fans/index"],
|
||||
['name' => '微信图文管理', 'icon' => 'layui-icon layui-icon-template-1', 'node' => "{$code}/news/index"],
|
||||
['name' => '微信菜单配置', 'icon' => 'layui-icon layui-icon-cellphone', 'node' => "{$code}/menu/index"],
|
||||
['name' => '回复规则管理', 'icon' => 'layui-icon layui-icon-engine', 'node' => "{$code}/keys/index"],
|
||||
['name' => '关注自动回复', 'icon' => 'layui-icon layui-icon-release', 'node' => "{$code}/auto/index"],
|
||||
],
|
||||
],
|
||||
[
|
||||
'name' => '微信支付',
|
||||
'subs' => [
|
||||
['name' => '微信支付行为', 'icon' => 'layui-icon layui-icon-rmb', 'node' => "{$code}/payment.record/index"],
|
||||
['name' => '微信退款管理', 'icon' => 'layui-icon layui-icon-engine', 'node' => "{$code}/payment.refund/index"],
|
||||
]
|
||||
]
|
||||
];
|
||||
}
|
||||
}
|
151
app/wechat/command/Auto.php
Normal file
151
app/wechat/command/Auto.php
Normal file
@ -0,0 +1,151 @@
|
||||
<?php
|
||||
|
||||
// +----------------------------------------------------------------------
|
||||
// | Wechat Plugin for ThinkAdmin
|
||||
// +----------------------------------------------------------------------
|
||||
// | 版权所有 2014~2025 Anyon <zoujingli@qq.com>
|
||||
// +----------------------------------------------------------------------
|
||||
// | 官方网站: https://thinkadmin.top
|
||||
// +----------------------------------------------------------------------
|
||||
// | 开源协议 ( https://mit-license.org )
|
||||
// | 免责声明 ( https://thinkadmin.top/disclaimer )
|
||||
// +----------------------------------------------------------------------
|
||||
// | gitee 代码仓库:https://gitee.com/zoujingli/think-plugs-wechat
|
||||
// | github 代码仓库:https://github.com/zoujingli/think-plugs-wechat
|
||||
// +----------------------------------------------------------------------
|
||||
|
||||
declare (strict_types=1);
|
||||
|
||||
namespace app\wechat\command;
|
||||
|
||||
use app\wechat\model\WechatAuto;
|
||||
use app\wechat\service\MediaService;
|
||||
use app\wechat\service\WechatService;
|
||||
use think\admin\Command;
|
||||
use think\console\Input;
|
||||
use think\console\input\Argument;
|
||||
use think\console\Output;
|
||||
|
||||
/**
|
||||
* 向指定用户推送消息
|
||||
* @class Auto
|
||||
* @package app\wechat\command
|
||||
*/
|
||||
class Auto extends Command
|
||||
{
|
||||
/** @var string */
|
||||
private $openid;
|
||||
|
||||
/**
|
||||
* 配置消息指令
|
||||
*/
|
||||
protected function configure()
|
||||
{
|
||||
$this->setName('xadmin:fansmsg');
|
||||
$this->addArgument('openid', Argument::OPTIONAL, 'wechat user openid', '');
|
||||
$this->addArgument('autocode', Argument::OPTIONAL, 'wechat auto message', '');
|
||||
$this->setDescription('Wechat Users Push AutoMessage for ThinkAdmin');
|
||||
}
|
||||
|
||||
/**
|
||||
* @param Input $input
|
||||
* @param Output $output
|
||||
* @throws \WeChat\Exceptions\InvalidResponseException
|
||||
* @throws \WeChat\Exceptions\LocalCacheException
|
||||
* @throws \think\admin\Exception
|
||||
* @throws \think\db\exception\DataNotFoundException
|
||||
* @throws \think\db\exception\DbException
|
||||
* @throws \think\db\exception\ModelNotFoundException
|
||||
*/
|
||||
protected function execute(Input $input, Output $output)
|
||||
{
|
||||
$code = $input->getArgument('autocode');
|
||||
$this->openid = $input->getArgument('openid');
|
||||
if (empty($code)) $this->setQueueError('Message Code cannot be empty');
|
||||
if (empty($this->openid)) $this->setQueueError('Wechat Openid cannot be empty');
|
||||
|
||||
// 查询微信消息对象
|
||||
$map = ['code' => $code, 'status' => 1];
|
||||
$data = WechatAuto::mk()->where($map)->find();
|
||||
if (empty($data)) $this->setQueueError('Message Data Query failed');
|
||||
|
||||
// 发送微信客服消息
|
||||
$this->buildMessage($data->toArray());
|
||||
}
|
||||
|
||||
/**
|
||||
* 关键字处理
|
||||
* @param array $data
|
||||
* @throws \WeChat\Exceptions\InvalidResponseException
|
||||
* @throws \WeChat\Exceptions\LocalCacheException
|
||||
* @throws \think\admin\Exception
|
||||
* @throws \think\db\exception\DataNotFoundException
|
||||
* @throws \think\db\exception\DbException
|
||||
* @throws \think\db\exception\ModelNotFoundException
|
||||
*/
|
||||
private function buildMessage(array $data)
|
||||
{
|
||||
$type = strtolower($data['type']);
|
||||
$result = [0, '待发送的消息不符合规则'];
|
||||
if ($type === 'text' && !empty($data['content'])) {
|
||||
$result = $this->sendMessage('text', ['content' => $data['content']]);
|
||||
}
|
||||
if ($type === 'voice' && !empty($data['voice_url'])) {
|
||||
if ($mediaId = MediaService::upload($data['voice_url'], 'voice')) {
|
||||
$result = $this->sendMessage('voice', ['media_id' => $mediaId]);
|
||||
}
|
||||
}
|
||||
if ($type === 'image' && !empty($data['image_url'])) {
|
||||
if ($mediaId = MediaService::upload($data['image_url'])) {
|
||||
$result = $this->sendMessage('image', ['media_id' => $mediaId]);
|
||||
}
|
||||
}
|
||||
if ($type === 'news') {
|
||||
[$item, $news] = [MediaService::news($data['news_id']), []];
|
||||
if (isset($item['articles']) && is_array($item['articles'])) {
|
||||
$host = sysconf('base.site_host') ?: true;
|
||||
foreach ($item['articles'] as $vo) if (empty($news)) $news[] = [
|
||||
'url' => url("@wechat/api.view/item/id/{$vo['id']}", [], false, $host)->build(),
|
||||
'title' => $vo['title'], 'picurl' => $vo['local_url'], 'description' => $vo['digest'],
|
||||
];
|
||||
$result = $this->sendMessage('news', ['articles' => $news]);
|
||||
}
|
||||
}
|
||||
if ($type === 'music' && !empty($data['music_url']) && !empty($data['music_title']) && !empty($data['music_desc'])) {
|
||||
$mediaId = $data['music_image'] ? MediaService::upload($data['music_image']) : '';
|
||||
$result = $this->sendMessage('music', [
|
||||
'hqmusicurl' => $data['music_url'], 'musicurl' => $data['music_url'],
|
||||
'description' => $data['music_desc'], 'title' => $data['music_title'], 'thumb_media_id' => $mediaId,
|
||||
]);
|
||||
}
|
||||
if ($type === 'video' && !empty($data['video_url']) && !empty($data['video_desc']) && !empty($data['video_title'])) {
|
||||
$video = ['title' => $data['video_title'], 'introduction' => $data['video_desc']];
|
||||
if ($mediaId = MediaService::upload($data['video_url'], 'video', $video)) {
|
||||
$result = $this->sendMessage('video', ['media_id' => $mediaId, 'title' => $data['video_title'], 'description' => $data['video_desc']]);
|
||||
}
|
||||
}
|
||||
if (empty($result[0])) {
|
||||
$this->setQueueError($result[1]);
|
||||
} else {
|
||||
$this->setQueueSuccess($result[1]);
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* 推送客服消息
|
||||
* @param string $type 消息类型
|
||||
* @param array $data 消息对象
|
||||
* @return array
|
||||
*/
|
||||
private function sendMessage(string $type, array $data): array
|
||||
{
|
||||
try {
|
||||
WechatService::WeChatCustom()->send([
|
||||
$type => $data, 'touser' => $this->openid, 'msgtype' => $type,
|
||||
]);
|
||||
return [1, '向微信用户推送消息成功'];
|
||||
} catch (\Exception $exception) {
|
||||
return [0, $exception->getMessage()];
|
||||
}
|
||||
}
|
||||
}
|
60
app/wechat/command/Clear.php
Normal file
60
app/wechat/command/Clear.php
Normal file
@ -0,0 +1,60 @@
|
||||
<?php
|
||||
|
||||
// +----------------------------------------------------------------------
|
||||
// | Wechat Plugin for ThinkAdmin
|
||||
// +----------------------------------------------------------------------
|
||||
// | 版权所有 2014~2025 Anyon <zoujingli@qq.com>
|
||||
// +----------------------------------------------------------------------
|
||||
// | 官方网站: https://thinkadmin.top
|
||||
// +----------------------------------------------------------------------
|
||||
// | 开源协议 ( https://mit-license.org )
|
||||
// | 免责声明 ( https://thinkadmin.top/disclaimer )
|
||||
// +----------------------------------------------------------------------
|
||||
// | gitee 代码仓库:https://gitee.com/zoujingli/think-plugs-wechat
|
||||
// | github 代码仓库:https://github.com/zoujingli/think-plugs-wechat
|
||||
// +----------------------------------------------------------------------
|
||||
|
||||
declare (strict_types=1);
|
||||
|
||||
namespace app\wechat\command;
|
||||
|
||||
use app\wechat\model\WechatPaymentRecord;
|
||||
use think\admin\Command;
|
||||
use think\console\Input;
|
||||
use think\console\Output;
|
||||
|
||||
/**
|
||||
* 微信支付单清理任务
|
||||
* @class Clear
|
||||
* @package app\wechat\command
|
||||
*/
|
||||
class Clear extends Command
|
||||
{
|
||||
protected function configure()
|
||||
{
|
||||
$this->setName('xadmin:fanspay');
|
||||
$this->setDescription('Wechat Users Payment auto clear for ThinkAdmin');
|
||||
}
|
||||
|
||||
/**
|
||||
* 执行支付单清理任务
|
||||
* @param \think\console\Input $input
|
||||
* @param \think\console\Output $output
|
||||
* @throws \think\admin\Exception
|
||||
* @throws \think\db\exception\DbException
|
||||
*/
|
||||
protected function execute(Input $input, Output $output)
|
||||
{
|
||||
$query = WechatPaymentRecord::mq();
|
||||
$query->where(['payment_status' => 0]);
|
||||
$query->whereTime('create_time', '<', strtotime('-24 hours'));
|
||||
[$total, $count] = [(clone $query)->count(), 0];
|
||||
if (empty($total)) $this->setQueueSuccess('无需清理24小时未支付!');
|
||||
/** @var \think\Model $item */
|
||||
foreach ($query->cursor() as $item) {
|
||||
$this->setQueueMessage($total, ++$count, sprintf('开始清理 %s 支付单...', $item->getAttr('code')));
|
||||
$item->delete();
|
||||
$this->setQueueMessage($total, $count, sprintf('完成清理 %s 支付单!', $item->getAttr('code')), 1);
|
||||
}
|
||||
}
|
||||
}
|
149
app/wechat/command/Fans.php
Normal file
149
app/wechat/command/Fans.php
Normal file
@ -0,0 +1,149 @@
|
||||
<?php
|
||||
|
||||
// +----------------------------------------------------------------------
|
||||
// | Wechat Plugin for ThinkAdmin
|
||||
// +----------------------------------------------------------------------
|
||||
// | 版权所有 2014~2025 Anyon <zoujingli@qq.com>
|
||||
// +----------------------------------------------------------------------
|
||||
// | 官方网站: https://thinkadmin.top
|
||||
// +----------------------------------------------------------------------
|
||||
// | 开源协议 ( https://mit-license.org )
|
||||
// | 免责声明 ( https://thinkadmin.top/disclaimer )
|
||||
// +----------------------------------------------------------------------
|
||||
// | gitee 代码仓库:https://gitee.com/zoujingli/think-plugs-wechat
|
||||
// | github 代码仓库:https://github.com/zoujingli/think-plugs-wechat
|
||||
// +----------------------------------------------------------------------
|
||||
|
||||
declare (strict_types=1);
|
||||
|
||||
namespace app\wechat\command;
|
||||
|
||||
use app\wechat\model\WechatFans;
|
||||
use app\wechat\model\WechatFansTags;
|
||||
use app\wechat\service\FansService;
|
||||
use app\wechat\service\WechatService;
|
||||
use think\admin\Command;
|
||||
|
||||
/**
|
||||
* 微信粉丝管理指令
|
||||
* @class Fans
|
||||
* @package app\wechat\command
|
||||
*/
|
||||
class Fans extends Command
|
||||
{
|
||||
/**
|
||||
* 配置指令
|
||||
*/
|
||||
protected function configure()
|
||||
{
|
||||
$this->setName('xadmin:fansall');
|
||||
$this->setDescription('Wechat Users Data Synchronize for ThinkAdmin');
|
||||
}
|
||||
|
||||
/**
|
||||
* 任务执行处理
|
||||
* @throws \WeChat\Exceptions\InvalidResponseException
|
||||
* @throws \WeChat\Exceptions\LocalCacheException
|
||||
* @throws \think\admin\Exception
|
||||
*/
|
||||
public function handle()
|
||||
{
|
||||
$this->setQueueSuccess($this->_list() . $this->_black());
|
||||
}
|
||||
|
||||
/**
|
||||
* 同步微信粉丝列表
|
||||
* @param string $next
|
||||
* @param integer $done
|
||||
* @return string
|
||||
* @throws \WeChat\Exceptions\InvalidResponseException
|
||||
* @throws \WeChat\Exceptions\LocalCacheException
|
||||
* @throws \think\admin\Exception
|
||||
*/
|
||||
protected function _list(string $next = '', int $done = 0): string
|
||||
{
|
||||
$appid = WechatService::getAppid();
|
||||
$this->process->message('开始获取微信用户数据');
|
||||
while (is_string($next)) {
|
||||
$result = WechatService::WeChatUser()->getUserList($next);
|
||||
if (is_array($result) && !empty($result['data']['openid'])) {
|
||||
foreach (array_chunk($result['data']['openid'], 100) as $openids) {
|
||||
$info = WechatService::WeChatUser()->getBatchUserInfo($openids);
|
||||
if (is_array($info) && !empty($info['user_info_list'])) {
|
||||
foreach ($info['user_info_list'] as $user) if (isset($user['nickname'])) {
|
||||
$this->queue->message($result['total'], ++$done, "-> 开始获取 {$user['openid']} {$user['nickname']}");
|
||||
FansService::set($user, $appid);
|
||||
$this->queue->message($result['total'], $done, "-> 完成更新 {$user['openid']} {$user['nickname']}", 1);
|
||||
}
|
||||
}
|
||||
}
|
||||
$next = $result['total'] > $done ? $result['next_openid'] : null;
|
||||
} else {
|
||||
$next = null;
|
||||
}
|
||||
}
|
||||
$this->process->message($done > 0 ? '微信用户数据获取完成' : '未获取到微信用户数据');
|
||||
$this->process->message('');
|
||||
return sprintf('共获取 %d 个用户数据', $done);
|
||||
}
|
||||
|
||||
/**
|
||||
* 同步粉丝黑名单列表
|
||||
* @param string $next
|
||||
* @param integer $done
|
||||
* @return string
|
||||
* @throws \WeChat\Exceptions\InvalidResponseException
|
||||
* @throws \WeChat\Exceptions\LocalCacheException
|
||||
*/
|
||||
public function _black(string $next = '', int $done = 0): string
|
||||
{
|
||||
$wechat = WechatService::WeChatUser();
|
||||
$this->setQueueProgress('开始更新黑名单列表');
|
||||
|
||||
// 清理原来的黑名单,重新批量更新黑名单列表
|
||||
WechatFans::mk()->where(['is_black' => 1])->update(['is_black' => 0]);
|
||||
|
||||
$result = ['total' => 0];
|
||||
while (!is_null($next) && is_array($result = $wechat->getBlackList($next))) {
|
||||
if (empty($result['data']['openid'])) break;
|
||||
foreach (array_chunk($result['data']['openid'], 100) as $chunk) {
|
||||
$done += count($chunk);
|
||||
WechatFans::mk()->whereIn('openid', $chunk)->update(['is_black' => 1]);
|
||||
}
|
||||
$next = $result['total'] > $done ? $result['next_openid'] : null;
|
||||
}
|
||||
$this->setQueueProgress(sprintf('完成更新 %s 个黑名单', $result['total']), null, 1);
|
||||
$this->output->newLine();
|
||||
if (empty($result['total'])) {
|
||||
return ', 其中黑名单 0 人';
|
||||
} else {
|
||||
return sprintf(', 其中黑名单 %s 人', $result['total']);
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* 同步粉丝标签列表
|
||||
* @param integer $done
|
||||
* @return string
|
||||
* @throws \WeChat\Exceptions\InvalidResponseException
|
||||
* @throws \WeChat\Exceptions\LocalCacheException
|
||||
* @throws \think\admin\Exception
|
||||
*/
|
||||
public function _tags(int $done = 0): string
|
||||
{
|
||||
$appid = WechatService::getAppid();
|
||||
$this->output->comment('开始获取微信用户标签数据');
|
||||
if (is_array($list = WechatService::WeChatTags()->getTags()) && !empty($list['tags'])) {
|
||||
$count = count($list['tags']);
|
||||
foreach ($list['tags'] as &$tag) {
|
||||
$tag['appid'] = $appid;
|
||||
$this->queue->message($count, ++$done, "-> {$tag['name']}");
|
||||
}
|
||||
WechatFansTags::mk()->where(['appid' => $appid])->delete();
|
||||
WechatFansTags::mk()->insertAll($list['tags']);
|
||||
}
|
||||
$this->output->comment($done > 0 ? '微信用户标签数据获取完成' : '未获取到微信用户标签数据');
|
||||
$this->output->newLine();
|
||||
return sprintf(', 获取到 %s 个标签', $done);
|
||||
}
|
||||
}
|
130
app/wechat/controller/Auto.php
Normal file
130
app/wechat/controller/Auto.php
Normal file
@ -0,0 +1,130 @@
|
||||
<?php
|
||||
|
||||
// +----------------------------------------------------------------------
|
||||
// | Wechat Plugin for ThinkAdmin
|
||||
// +----------------------------------------------------------------------
|
||||
// | 版权所有 2014~2025 Anyon <zoujingli@qq.com>
|
||||
// +----------------------------------------------------------------------
|
||||
// | 官方网站: https://thinkadmin.top
|
||||
// +----------------------------------------------------------------------
|
||||
// | 开源协议 ( https://mit-license.org )
|
||||
// | 免责声明 ( https://thinkadmin.top/disclaimer )
|
||||
// +----------------------------------------------------------------------
|
||||
// | gitee 代码仓库:https://gitee.com/zoujingli/think-plugs-wechat
|
||||
// | github 代码仓库:https://github.com/zoujingli/think-plugs-wechat
|
||||
// +----------------------------------------------------------------------
|
||||
|
||||
declare (strict_types=1);
|
||||
|
||||
namespace app\wechat\controller;
|
||||
|
||||
use app\wechat\model\WechatAuto;
|
||||
use think\admin\Controller;
|
||||
use think\admin\extend\CodeExtend;
|
||||
use think\admin\helper\QueryHelper;
|
||||
use think\admin\service\SystemService;
|
||||
|
||||
/**
|
||||
* 关注自动回复
|
||||
* @class Auto
|
||||
* @package app\wechat\controller
|
||||
*/
|
||||
class Auto extends Controller
|
||||
{
|
||||
/**
|
||||
* 消息类型
|
||||
* @var array
|
||||
*/
|
||||
public $types = [
|
||||
'text' => '文字', 'news' => '图文',
|
||||
'image' => '图片', 'music' => '音乐',
|
||||
'video' => '视频', 'voice' => '语音',
|
||||
];
|
||||
|
||||
/**
|
||||
* 关注自动回复
|
||||
* @auth true
|
||||
* @menu true
|
||||
* @throws \think\db\exception\DataNotFoundException
|
||||
* @throws \think\db\exception\DbException
|
||||
* @throws \think\db\exception\ModelNotFoundException
|
||||
*/
|
||||
public function index()
|
||||
{
|
||||
$this->type = $this->get['type'] ?? 'index';
|
||||
WechatAuto::mQuery()->layTable(function () {
|
||||
$this->title = '关注自动回复';
|
||||
}, function (QueryHelper $query) {
|
||||
$query->like('code,type#mtype')->dateBetween('create_at');
|
||||
$query->where(['status' => intval($this->type === 'index')]);
|
||||
});
|
||||
}
|
||||
|
||||
/**
|
||||
* 列表数据处理
|
||||
* @param array $data
|
||||
*/
|
||||
protected function _index_page_filter(array &$data)
|
||||
{
|
||||
foreach ($data as &$vo) {
|
||||
$vo['type'] = $this->types[$vo['type']] ?? $vo['type'];
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* 添加自动回复
|
||||
* @auth true
|
||||
*/
|
||||
public function add()
|
||||
{
|
||||
$this->title = '添加自动回复';
|
||||
WechatAuto::mForm('form');
|
||||
}
|
||||
|
||||
/**
|
||||
* 编辑自动回复
|
||||
* @auth true
|
||||
*/
|
||||
public function edit()
|
||||
{
|
||||
$this->title = '编辑自动回复';
|
||||
WechatAuto::mForm('form');
|
||||
}
|
||||
|
||||
/**
|
||||
* 添加数据处理
|
||||
* @param array $data
|
||||
*/
|
||||
protected function _form_filter(array &$data)
|
||||
{
|
||||
if (empty($data['code'])) {
|
||||
$data['code'] = CodeExtend::uniqidNumber(18, 'AM');
|
||||
}
|
||||
if ($this->request->isGet()) {
|
||||
$this->defaultImage = SystemService::uri('/static/theme/img/image.png', '__FULL__');
|
||||
} else {
|
||||
$data['content'] = strip_tags($data['content'] ?? '', '<a>');
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* 修改规则状态
|
||||
* @auth true
|
||||
*/
|
||||
public function state()
|
||||
{
|
||||
WechatAuto::mSave($this->_vali([
|
||||
'status.in:0,1' => '状态值范围异常!',
|
||||
'status.require' => '状态值不能为空!',
|
||||
]));
|
||||
}
|
||||
|
||||
/**
|
||||
* 删除自动回复
|
||||
* @auth true
|
||||
*/
|
||||
public function remove()
|
||||
{
|
||||
WechatAuto::mDelete();
|
||||
}
|
||||
}
|
235
app/wechat/controller/Config.php
Normal file
235
app/wechat/controller/Config.php
Normal file
@ -0,0 +1,235 @@
|
||||
<?php
|
||||
|
||||
// +----------------------------------------------------------------------
|
||||
// | Wechat Plugin for ThinkAdmin
|
||||
// +----------------------------------------------------------------------
|
||||
// | 版权所有 2014~2025 Anyon <zoujingli@qq.com>
|
||||
// +----------------------------------------------------------------------
|
||||
// | 官方网站: https://thinkadmin.top
|
||||
// +----------------------------------------------------------------------
|
||||
// | 开源协议 ( https://mit-license.org )
|
||||
// | 免责声明 ( https://thinkadmin.top/disclaimer )
|
||||
// +----------------------------------------------------------------------
|
||||
// | gitee 代码仓库:https://gitee.com/zoujingli/think-plugs-wechat
|
||||
// | github 代码仓库:https://github.com/zoujingli/think-plugs-wechat
|
||||
// +----------------------------------------------------------------------
|
||||
|
||||
declare (strict_types=1);
|
||||
|
||||
namespace app\wechat\controller;
|
||||
|
||||
use app\wechat\service\WechatService;
|
||||
use think\admin\Builder;
|
||||
use think\admin\Controller;
|
||||
use think\admin\storage\LocalStorage;
|
||||
|
||||
/**
|
||||
* 微信授权绑定
|
||||
* @class Config
|
||||
* @package app\wechat\controller
|
||||
*/
|
||||
class Config extends Controller
|
||||
{
|
||||
/**
|
||||
* 微信授权配置
|
||||
* @auth true
|
||||
* @menu true
|
||||
* @throws \think\admin\Exception
|
||||
*/
|
||||
public function options()
|
||||
{
|
||||
$this->thrNotify = sysuri('wechat/api.push/index', [], false, true);
|
||||
if ($this->request->isGet()) {
|
||||
try {
|
||||
// 生成微信授权链接
|
||||
$source = enbase64url(sysuri('admin/index/index', [], false, true) . '#' . $this->request->url());
|
||||
$authurl = sysconf('wechat.service_authurl|raw') ?: "https://open.cuci.cc/plugin-wechat-service/api.push/auth?source=SOURCE";
|
||||
$this->authurl = str_replace('source=SOURCE', "source={$source}", $authurl);
|
||||
// 授权成功后的参数保存
|
||||
if (input('?appid') && input('?appkey')) {
|
||||
sysconf('wechat.type', 'thr');
|
||||
sysconf('wechat.thr_appid', input('appid'));
|
||||
sysconf('wechat.thr_appkey', input('appkey'));
|
||||
WechatService::ThinkServiceConfig()->setApiNotifyUri($this->thrNotify);
|
||||
}
|
||||
// 读取授权的微信参数
|
||||
$this->wechat = WechatService::ThinkServiceConfig()->getConfig();
|
||||
} catch (\Exception $exception) {
|
||||
$this->wechat = [];
|
||||
$this->message = $exception->getMessage();
|
||||
}
|
||||
$this->geoip = $this->app->cache->get('mygeoip', '');
|
||||
if (empty($this->geoip)) {
|
||||
$this->geoip = gethostbyname($this->request->host());
|
||||
$this->app->cache->set('mygeoip', $this->geoip, 360);
|
||||
}
|
||||
$this->title = '微信授权配置';
|
||||
$this->fetch();
|
||||
} else {
|
||||
foreach ($this->request->post() as $k => $v) sysconf($k, $v);
|
||||
if ($this->request->post('wechat.type') === 'thr') try {
|
||||
WechatService::ThinkServiceConfig()->setApiNotifyUri($this->thrNotify);
|
||||
} catch (\Exception $exception) {
|
||||
$this->error($exception->getMessage());
|
||||
}
|
||||
sysoplog('微信授权配置', '修改微信授权配置成功');
|
||||
$this->success('微信授权修改成功!', admuri('', ['uniqid' => uniqid()]));
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* 微信授权测试
|
||||
* @auth true
|
||||
*/
|
||||
public function options_test()
|
||||
{
|
||||
$this->fetch();
|
||||
}
|
||||
|
||||
/**
|
||||
* 微信第三方平台接口配置
|
||||
* @auth true
|
||||
* @throws \think\admin\Exception
|
||||
*/
|
||||
public function options_jsonrpc()
|
||||
{
|
||||
if ($this->request->isGet()) {
|
||||
$authUrl = sysconf('wechat.service_authurl|raw') ?: "https://open.cuci.cc/plugin-wechat-service/api.push/auth?source=SOURCE";
|
||||
$jsonRpc = sysconf('wechat.service_jsonrpc|raw') ?: 'https://open.cuci.cc/plugin-wechat-service/api.client/jsonrpc?token=TOKEN';
|
||||
Builder::mk()
|
||||
->addTextInput('auth_url', '公众号授权跳转入口', 'Getway', true, '进行微信授权时会跳转到这个页面,由微信管理员扫二维码进行授权。', '^https?://.*?auth.*?source=SOURCE')
|
||||
->addTextInput('json_rpc', '第三方服务平台接口', 'JsonRpc', true, '由应用插件 <a target="_blank" href="https://thinkadmin.top/plugin/think-plugs-wechat-service.html">ThinkPlugsWechatService</a> 提供的第三方服务平台 JSON-RPC 接口地址。', '^https?://.*?jsonrpc.*?token=TOKEN')
|
||||
->addSubmitButton('保存参数')->addCancelButton()->fetch(['vo' => ['auth_url' => $authUrl, 'json_rpc' => $jsonRpc]]);
|
||||
} else {
|
||||
$data = $this->_vali([
|
||||
'auth_url.require' => '授权跳转不能为空!',
|
||||
'json_rpc.require' => '接口地址不能为空!'
|
||||
]);
|
||||
sysconf('wechat.service_authurl', $data['auth_url']);
|
||||
sysconf('wechat.service_jsonrpc', $data['json_rpc']);
|
||||
$this->success('接口地址保存成功!');
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* 绑定小程序
|
||||
* @auth true
|
||||
* @throws \think\admin\Exception
|
||||
*/
|
||||
public function options_wxapp()
|
||||
{
|
||||
if ($this->request->isGet()) {
|
||||
$data = sysdata('plugin.wechat.wxapp') ?: [];
|
||||
Builder::mk()
|
||||
->addTextInput('appid', '小程序', 'AppId', true, '<b>必选</b>,微信小程序 AppID 需要微信公众号平台获取!', '^wx[0-9a-z]{16}$', ['maxlength' => 18])
|
||||
->addTextInput('appkey', '小程序密钥', 'AppSecret', true, '<b>必选</b>,微信小程序 AppSecret 需要微信公众号平台获取!', '^[0-9a-z]{32}$', ['maxlength' => 32])
|
||||
->addSubmitButton('保存参数')->addCancelButton()
|
||||
->fetch(['vo' => ['appid' => $data['appid'] ?? '', 'appkey' => $data['appkey'] ?? '']]);
|
||||
} else {
|
||||
sysdata('plugin.wechat.wxapp', $this->_vali([
|
||||
'appid.require' => '小程序ID不能为空!',
|
||||
'appkey.require' => '小程序密钥不能为空!'
|
||||
]));
|
||||
$this->success('参数保存成功!');
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* 微信支付配置
|
||||
* @auth true
|
||||
* @menu true
|
||||
* @throws \think\admin\Exception
|
||||
*/
|
||||
public function payment()
|
||||
{
|
||||
if ($this->request->isGet()) {
|
||||
$this->title = '微信支付配置';
|
||||
$local = LocalStorage::instance();
|
||||
$this->data = sysdata('plugin.wechat.payment');
|
||||
$this->mch_ssl_cer = $this->data['mch_ssl_cer'] ?? '';
|
||||
$this->mch_ssl_key = $this->data['mch_ssl_key'] ?? '';
|
||||
$this->mch_ssl_p12 = $this->data['mch_ssl_p12'] ?? '';
|
||||
$this->mch_ssl_pay = $this->data['mch_ssl_pay'] ?? '';
|
||||
if (!$local->has($this->mch_ssl_cer, true)) $this->mch_ssl_cer = '';
|
||||
if (!$local->has($this->mch_ssl_key, true)) $this->mch_ssl_key = '';
|
||||
if (!$local->has($this->mch_ssl_p12, true)) $this->mch_ssl_p12 = '';
|
||||
if (!$local->has($this->mch_ssl_pay, true)) $this->mch_ssl_pay = '';
|
||||
$this->fetch();
|
||||
} else {
|
||||
$this->error('抱歉,数据提交地址错误!');
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* 微信支付修改
|
||||
* @auth true
|
||||
* @throws \think\admin\Exception
|
||||
*/
|
||||
public function payment_save()
|
||||
{
|
||||
if ($this->request->isPost()) {
|
||||
$local = LocalStorage::instance();
|
||||
$wechat = $this->request->post('wechat');
|
||||
// if (empty($wechat['mch_pay_sid'])) {
|
||||
// $this->error('微信支付公钥序号为空!');
|
||||
// }
|
||||
// if (empty($wechat['mch_ssl_pay']) || !$local->has($wechat['mch_ssl_pay'], true)) {
|
||||
// $this->error('微信支付公钥不能为空!');
|
||||
// }
|
||||
// PEM 证书模式处理
|
||||
if ($wechat['mch_ssl_type'] === 'pem') {
|
||||
if (empty($wechat['mch_ssl_key']) || !$local->has($wechat['mch_ssl_key'], true)) {
|
||||
$this->error('商户证书密钥不能为空!');
|
||||
}
|
||||
if (empty($wechat['mch_ssl_cer']) || !$local->has($wechat['mch_ssl_cer'], true)) {
|
||||
$this->error('商户证书公钥不能为空!');
|
||||
}
|
||||
}
|
||||
// P12 证书模式转 PEM 模式
|
||||
if ($wechat['mch_ssl_type'] === 'p12') {
|
||||
if (empty($wechat['mch_ssl_p12']) || !$local->has($wechat['mch_ssl_p12'], true)) {
|
||||
$this->error('商户证书 P12 不能为空!');
|
||||
}
|
||||
if (openssl_pkcs12_read($local->get($wechat['mch_ssl_p12'], true), $certs, $wechat['mch_id'])) {
|
||||
$name1 = sprintf("wxpay/%s_%s_cer.pem", $wechat['mch_id'], md5($certs['cert']));
|
||||
$name2 = sprintf("wxpay/%s_%s_key.pem", $wechat['mch_id'], md5($certs['pkey']));
|
||||
$wechat['mch_ssl_cer'] = $local->set($name1, $certs['cert'], true)['url'];
|
||||
$wechat['mch_ssl_key'] = $local->set($name2, $certs['pkey'], true)['url'];
|
||||
$wechat['mch_ssl_type'] = 'pem';
|
||||
} else {
|
||||
$this->error('商户账号与 P12 证书不匹配!');
|
||||
}
|
||||
}
|
||||
// 记录文本格式参数,兼容分布式部署
|
||||
sysdata('plugin.wechat.payment', [
|
||||
'appid' => WechatService::getAppid(),
|
||||
'mch_id' => $wechat['mch_id'],
|
||||
'mch_key' => $wechat['mch_key'],
|
||||
'mch_v3_key' => $wechat['mch_v3_key'],
|
||||
'mch_ssl_cer' => $wechat['mch_ssl_cer'],
|
||||
'mch_ssl_key' => $wechat['mch_ssl_key'],
|
||||
'mch_ssl_pay' => $wechat['mch_ssl_pay'],
|
||||
'mch_pay_sid' => $wechat['mch_pay_sid'] ?? '',
|
||||
'ssl_pay_text' => $local->get($wechat['mch_ssl_pay'], true),
|
||||
'ssl_cer_text' => $local->get($wechat['mch_ssl_cer'], true),
|
||||
'ssl_key_text' => $local->get($wechat['mch_ssl_key'], true),
|
||||
]);
|
||||
// 同步更新证书内容
|
||||
WechatService::withWxpayCert(['mch_id' => $wechat['mch_id']]);
|
||||
// 记录操作历史并返回保存结果
|
||||
sysoplog('微信支付配置', '修改微信支付配置成功');
|
||||
$this->success('微信支付配置成功!');
|
||||
} else {
|
||||
$this->error('抱歉,访问方式错误!');
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* 微信支付测试
|
||||
* @auth true
|
||||
*/
|
||||
public function payment_test()
|
||||
{
|
||||
$this->fetch();
|
||||
}
|
||||
}
|
129
app/wechat/controller/Fans.php
Normal file
129
app/wechat/controller/Fans.php
Normal file
@ -0,0 +1,129 @@
|
||||
<?php
|
||||
|
||||
// +----------------------------------------------------------------------
|
||||
// | Wechat Plugin for ThinkAdmin
|
||||
// +----------------------------------------------------------------------
|
||||
// | 版权所有 2014~2025 Anyon <zoujingli@qq.com>
|
||||
// +----------------------------------------------------------------------
|
||||
// | 官方网站: https://thinkadmin.top
|
||||
// +----------------------------------------------------------------------
|
||||
// | 开源协议 ( https://mit-license.org )
|
||||
// | 免责声明 ( https://thinkadmin.top/disclaimer )
|
||||
// +----------------------------------------------------------------------
|
||||
// | gitee 代码仓库:https://gitee.com/zoujingli/think-plugs-wechat
|
||||
// | github 代码仓库:https://github.com/zoujingli/think-plugs-wechat
|
||||
// +----------------------------------------------------------------------
|
||||
|
||||
declare (strict_types=1);
|
||||
|
||||
namespace app\wechat\controller;
|
||||
|
||||
use app\wechat\model\WechatFans;
|
||||
use app\wechat\model\WechatFansTags;
|
||||
use app\wechat\service\WechatService;
|
||||
use think\admin\Controller;
|
||||
use think\admin\helper\QueryHelper;
|
||||
use think\exception\HttpResponseException;
|
||||
|
||||
/**
|
||||
* 微信用户管理
|
||||
* @class Fans
|
||||
* @package app\wechat\controller
|
||||
*/
|
||||
class Fans extends Controller
|
||||
{
|
||||
/**
|
||||
* 微信用户管理
|
||||
* @auth true
|
||||
* @menu true
|
||||
* @throws \think\db\exception\DataNotFoundException
|
||||
* @throws \think\db\exception\DbException
|
||||
* @throws \think\db\exception\ModelNotFoundException
|
||||
*/
|
||||
public function index()
|
||||
{
|
||||
WechatFans::mQuery()->layTable(function () {
|
||||
$this->title = '微信用户管理';
|
||||
}, static function (QueryHelper $query) {
|
||||
$query->where(['appid' => WechatService::getAppid()]);
|
||||
$query->like('nickname')->equal('subscribe,is_black')->dateBetween('subscribe_at');
|
||||
});
|
||||
}
|
||||
|
||||
/**
|
||||
* 列表数据处理
|
||||
* @param array $data
|
||||
*/
|
||||
protected function _index_page_filter(array &$data)
|
||||
{
|
||||
foreach ($data as &$vo) $vo['subscribe_at'] = format_datetime($vo['subscribe_at']);
|
||||
}
|
||||
|
||||
/**
|
||||
* 同步用户数据
|
||||
* @auth true
|
||||
*/
|
||||
public function sync()
|
||||
{
|
||||
sysoplog('微信授权配置', '创建粉丝用户同步任务');
|
||||
$this->_queue('同步微信用户数据', "xadmin:fansall");
|
||||
}
|
||||
|
||||
/**
|
||||
* 黑名单列表操作
|
||||
* @auth true
|
||||
*/
|
||||
public function black()
|
||||
{
|
||||
try {
|
||||
$data = $this->_vali([
|
||||
'black.require' => '操作类型不能为空!',
|
||||
'openid.require' => '操作用户不能为空!',
|
||||
]);
|
||||
foreach (array_chunk(str2arr($data['openid']), 20) as $openids) {
|
||||
if ($data['black']) {
|
||||
WechatService::WeChatUser()->batchBlackList($openids);
|
||||
WechatFans::mk()->whereIn('openid', $openids)->update(['is_black' => 1]);
|
||||
} else {
|
||||
WechatService::WeChatUser()->batchUnblackList($openids);
|
||||
WechatFans::mk()->whereIn('openid', $openids)->update(['is_black' => 0]);
|
||||
}
|
||||
}
|
||||
if (empty($data['black'])) {
|
||||
$this->success('移出黑名单成功!');
|
||||
} else {
|
||||
$this->success('拉入黑名单成功!');
|
||||
}
|
||||
} catch (HttpResponseException $exception) {
|
||||
throw $exception;
|
||||
} catch (\Exception $exception) {
|
||||
$this->error("黑名单操作失败,请稍候再试!<br>{$exception->getMessage()}");
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* 删除用户信息
|
||||
* @auth true
|
||||
*/
|
||||
public function remove()
|
||||
{
|
||||
WechatFans::mDelete();
|
||||
}
|
||||
|
||||
/**
|
||||
* 清空用户数据
|
||||
* @auth true
|
||||
*/
|
||||
public function truncate()
|
||||
{
|
||||
try {
|
||||
WechatFans::mQuery()->empty();
|
||||
WechatFansTags::mQuery()->empty();
|
||||
$this->success('清空用户数据成功!');
|
||||
} catch (HttpResponseException $exception) {
|
||||
throw $exception;
|
||||
} catch (\Exception $exception) {
|
||||
$this->error("清空用户数据失败,{$exception->getMessage()}");
|
||||
}
|
||||
}
|
||||
}
|
163
app/wechat/controller/Keys.php
Normal file
163
app/wechat/controller/Keys.php
Normal file
@ -0,0 +1,163 @@
|
||||
<?php
|
||||
|
||||
// +----------------------------------------------------------------------
|
||||
// | Wechat Plugin for ThinkAdmin
|
||||
// +----------------------------------------------------------------------
|
||||
// | 版权所有 2014~2025 Anyon <zoujingli@qq.com>
|
||||
// +----------------------------------------------------------------------
|
||||
// | 官方网站: https://thinkadmin.top
|
||||
// +----------------------------------------------------------------------
|
||||
// | 开源协议 ( https://mit-license.org )
|
||||
// | 免责声明 ( https://thinkadmin.top/disclaimer )
|
||||
// +----------------------------------------------------------------------
|
||||
// | gitee 代码仓库:https://gitee.com/zoujingli/think-plugs-wechat
|
||||
// | github 代码仓库:https://github.com/zoujingli/think-plugs-wechat
|
||||
// +----------------------------------------------------------------------
|
||||
|
||||
declare (strict_types=1);
|
||||
|
||||
namespace app\wechat\controller;
|
||||
|
||||
use app\wechat\model\WechatKeys;
|
||||
use app\wechat\service\WechatService;
|
||||
use think\admin\Controller;
|
||||
use think\admin\helper\QueryHelper;
|
||||
use think\admin\service\SystemService;
|
||||
use think\exception\HttpResponseException;
|
||||
|
||||
/**
|
||||
* 回复规则管理
|
||||
* @class Keys
|
||||
* @package app\wechat\controller
|
||||
*/
|
||||
class Keys extends Controller
|
||||
{
|
||||
/**
|
||||
* 消息类型
|
||||
* @var array
|
||||
*/
|
||||
public $types = [
|
||||
'text' => '文字', 'news' => '图文', 'image' => '图片', 'music' => '音乐',
|
||||
'video' => '视频', 'voice' => '语音', 'customservice' => '转客服',
|
||||
];
|
||||
|
||||
/**
|
||||
* 回复规则管理
|
||||
* @auth true
|
||||
* @menu true
|
||||
* @throws \think\db\exception\DataNotFoundException
|
||||
* @throws \think\db\exception\DbException
|
||||
* @throws \think\db\exception\ModelNotFoundException
|
||||
*/
|
||||
public function index()
|
||||
{
|
||||
// 关键字二维码生成
|
||||
if ($this->request->get('action') === 'qrc') try {
|
||||
$wechat = WechatService::WeChatQrcode();
|
||||
$result = $wechat->create($this->request->get('keys', ''));
|
||||
$this->success('生成二维码成功!', "javascript:$.previewImage('{$wechat->url($result['ticket'])}')");
|
||||
} catch (HttpResponseException $exception) {
|
||||
throw $exception;
|
||||
} catch (\Exception $exception) {
|
||||
$this->error("生成二维码失败,请稍候再试!<br> {$exception->getMessage()}");
|
||||
}
|
||||
// 数据列表分页处理
|
||||
$this->type = $this->get['type'] ?? 'index';
|
||||
WechatKeys::mQuery()->layTable(function () {
|
||||
$this->title = '回复规则管理';
|
||||
}, function (QueryHelper $query) {
|
||||
$query->whereNotIn('keys', ['subscribe', 'default']);
|
||||
$query->like('keys,type#mtype')->dateBetween('create_at');
|
||||
$query->where(['status' => intval($this->type === 'index')]);
|
||||
});
|
||||
}
|
||||
|
||||
/**
|
||||
* 列表数据处理
|
||||
* @param array $data
|
||||
*/
|
||||
protected function _index_page_filter(array &$data)
|
||||
{
|
||||
foreach ($data as &$vo) {
|
||||
$vo['type'] = $this->types[$vo['type']] ?? $vo['type'];
|
||||
$vo['qrc'] = sysuri('wechat/keys/index') . "?action=qrc&keys={$vo['keys']}";
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* 添加回复规则
|
||||
* @auth true
|
||||
*/
|
||||
public function add()
|
||||
{
|
||||
$this->title = '添加回复规则';
|
||||
WechatKeys::mForm('form');
|
||||
}
|
||||
|
||||
/**
|
||||
* 编辑回复规则
|
||||
* @auth true
|
||||
*/
|
||||
public function edit()
|
||||
{
|
||||
$this->title = '编辑回复规则';
|
||||
WechatKeys::mForm('form');
|
||||
}
|
||||
|
||||
/**
|
||||
* 修改规则状态
|
||||
* @auth true
|
||||
*/
|
||||
public function state()
|
||||
{
|
||||
WechatKeys::mSave($this->_vali([
|
||||
'status.in:0,1' => '状态值范围异常!',
|
||||
'status.require' => '状态值不能为空!',
|
||||
]));
|
||||
}
|
||||
|
||||
/**
|
||||
* 删除回复规则
|
||||
* @auth true
|
||||
*/
|
||||
public function remove()
|
||||
{
|
||||
WechatKeys::mDelete();
|
||||
}
|
||||
|
||||
/**
|
||||
* 配置订阅回复
|
||||
* @auth true
|
||||
*/
|
||||
public function subscribe()
|
||||
{
|
||||
$this->title = '编辑订阅回复规则';
|
||||
WechatKeys::mForm('form', 'keys', [], ['keys' => 'subscribe']);
|
||||
}
|
||||
|
||||
/**
|
||||
* 配置默认回复
|
||||
* @auth true
|
||||
*/
|
||||
public function defaults()
|
||||
{
|
||||
$this->title = '编辑默认回复规则';
|
||||
WechatKeys::mForm('form', 'keys', [], ['keys' => 'default']);
|
||||
}
|
||||
|
||||
/**
|
||||
* 添加数据处理
|
||||
* @param array $data
|
||||
* @throws \think\db\exception\DbException
|
||||
*/
|
||||
protected function _form_filter(array &$data)
|
||||
{
|
||||
if ($this->request->isPost()) {
|
||||
$map = [['keys', '=', $data['keys']], ['id', '<>', $data['id'] ?? 0]];
|
||||
if (WechatKeys::mk()->where($map)->count() > 0) $this->error('关键字已经存在!');
|
||||
$data['content'] = strip_tags($data['content'] ?? '', '<a>');
|
||||
} elseif ($this->request->isGet()) {
|
||||
$this->defaultImage = SystemService::uri('/static/theme/img/image.png', '__FULL__');
|
||||
}
|
||||
}
|
||||
}
|
171
app/wechat/controller/Menu.php
Normal file
171
app/wechat/controller/Menu.php
Normal file
@ -0,0 +1,171 @@
|
||||
<?php
|
||||
|
||||
// +----------------------------------------------------------------------
|
||||
// | Wechat Plugin for ThinkAdmin
|
||||
// +----------------------------------------------------------------------
|
||||
// | 版权所有 2014~2025 Anyon <zoujingli@qq.com>
|
||||
// +----------------------------------------------------------------------
|
||||
// | 官方网站: https://thinkadmin.top
|
||||
// +----------------------------------------------------------------------
|
||||
// | 开源协议 ( https://mit-license.org )
|
||||
// | 免责声明 ( https://thinkadmin.top/disclaimer )
|
||||
// +----------------------------------------------------------------------
|
||||
// | gitee 代码仓库:https://gitee.com/zoujingli/think-plugs-wechat
|
||||
// | github 代码仓库:https://github.com/zoujingli/think-plugs-wechat
|
||||
// +----------------------------------------------------------------------
|
||||
|
||||
declare (strict_types=1);
|
||||
|
||||
namespace app\wechat\controller;
|
||||
|
||||
use app\wechat\service\WechatService;
|
||||
use think\admin\Controller;
|
||||
use think\exception\HttpResponseException;
|
||||
|
||||
/**
|
||||
* 微信菜单管理
|
||||
* @class Menu
|
||||
* @package app\wechat\controller
|
||||
*/
|
||||
class Menu extends Controller
|
||||
{
|
||||
/**
|
||||
* 存储数据名称
|
||||
* @var string
|
||||
*/
|
||||
protected $ckey = 'wechat_menu_data';
|
||||
|
||||
/**
|
||||
* 微信菜单的类型
|
||||
* @var array
|
||||
*/
|
||||
protected $menuTypes = [
|
||||
'click' => '匹配规则',
|
||||
'view' => '跳转网页',
|
||||
'miniprogram' => '打开小程序',
|
||||
'customservice' => '转多客服',
|
||||
'scancode_push' => '扫码推事件',
|
||||
'scancode_waitmsg' => '扫码推事件且弹出“消息接收中”提示框',
|
||||
'pic_sysphoto' => '弹出系统拍照发图',
|
||||
'pic_photo_or_album' => '弹出拍照或者相册发图',
|
||||
'pic_weixin' => '弹出微信相册发图器',
|
||||
'location_select' => '弹出地理位置选择器',
|
||||
];
|
||||
|
||||
/**
|
||||
* 微信菜单管理
|
||||
* @auth true
|
||||
* @menu true
|
||||
* @throws \think\admin\Exception
|
||||
* @throws \think\db\exception\DataNotFoundException
|
||||
* @throws \think\db\exception\DbException
|
||||
* @throws \think\db\exception\ModelNotFoundException
|
||||
*/
|
||||
public function index()
|
||||
{
|
||||
if ($this->request->get('output') === 'json') {
|
||||
$map = [['keys', 'notin', ['subscribe', 'default']], ['status', '=', 1]];
|
||||
$result = $this->app->db->name('WechatKeys')->where($map)->order('sort desc,id desc')->select();
|
||||
$this->success('获取数据成功!', ['menudata' => sysdata($this->ckey), 'keysdata' => $result->toArray()]);
|
||||
} else {
|
||||
$this->title = '微信菜单定制';
|
||||
$this->fetch();
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* 取消微信菜单
|
||||
* @auth true
|
||||
*/
|
||||
public function cancel()
|
||||
{
|
||||
try {
|
||||
WechatService::WeChatMenu()->delete();
|
||||
$this->success('公众号菜单取消成功!');
|
||||
} catch (HttpResponseException $exception) {
|
||||
sysoplog('微信管理', '取消微信菜单成功');
|
||||
throw $exception;
|
||||
} catch (\Exception $exception) {
|
||||
$this->error("菜单取消失败,请稍候再试!<br> {$exception->getMessage()}");
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* 编辑微信菜单
|
||||
* @auth true
|
||||
*/
|
||||
public function push()
|
||||
{
|
||||
if ($this->request->isPost()) {
|
||||
$data = $this->request->post('data');
|
||||
if (empty($data)) try {
|
||||
WechatService::WeChatMenu()->delete();
|
||||
sysoplog('微信菜单管理', '删除微信菜单成功');
|
||||
$this->success('删除微信菜单成功!', '');
|
||||
} catch (HttpResponseException $exception) {
|
||||
throw $exception;
|
||||
} catch (\Exception $exception) {
|
||||
sysoplog('微信菜单管理', "删除微信菜单失败:{$exception->getMessage()}");
|
||||
$this->error("删除微信菜单失败,请稍候再试!<br>{$exception->getMessage()}");
|
||||
} else try {
|
||||
sysdata($this->ckey, $this->_buildMenuData(json_decode($data, true)));
|
||||
WechatService::WeChatMenu()->create(['button' => sysdata($this->ckey)]);
|
||||
sysoplog('微信菜单管理', '发布微信菜单成功');
|
||||
$this->success('保存发布菜单成功!', '');
|
||||
} catch (HttpResponseException $exception) {
|
||||
throw $exception;
|
||||
} catch (\Exception $exception) {
|
||||
sysoplog('微信菜单管理', "发布微信菜单失败:{$exception->getMessage()}");
|
||||
$this->error("微信菜单发布失败,请稍候再试!<br> {$exception->getMessage()}");
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* 菜单数据处理
|
||||
* @param array $list
|
||||
* @return array
|
||||
*/
|
||||
private function _buildMenuData(array $list): array
|
||||
{
|
||||
foreach ($list as &$item) {
|
||||
if (empty($item['sub_button'])) {
|
||||
$item = $this->_buildMenuDataItem($item);
|
||||
} else {
|
||||
$button = ['name' => $item['name'], 'sub_button' => []];
|
||||
foreach ($item['sub_button'] as $sub) {
|
||||
$button['sub_button'][] = $this->_buildMenuDataItem($sub);
|
||||
}
|
||||
$item = $button;
|
||||
}
|
||||
}
|
||||
return $list;
|
||||
}
|
||||
|
||||
/**
|
||||
* 单个微信菜单数据处理
|
||||
* @param array $item
|
||||
* @return array
|
||||
*/
|
||||
private function _buildMenuDataItem(array $item): array
|
||||
{
|
||||
switch (strtolower($item['type'])) {
|
||||
case 'pic_weixin':
|
||||
case 'pic_sysphoto':
|
||||
case 'scancode_push':
|
||||
case 'location_select':
|
||||
case 'scancode_waitmsg':
|
||||
case 'pic_photo_or_album':
|
||||
return ['name' => $item['name'], 'type' => $item['type'], 'key' => $item['key'] ?? $item['type']];
|
||||
case 'click':
|
||||
if (empty($item['key'])) $this->error('匹配规则存在空的选项');
|
||||
return ['name' => $item['name'], 'type' => $item['type'], 'key' => $item['key']];
|
||||
case 'view':
|
||||
return ['name' => $item['name'], 'type' => $item['type'], 'url' => $item['url']];
|
||||
case 'miniprogram':
|
||||
return ['name' => $item['name'], 'type' => $item['type'], 'url' => $item['url'], 'appid' => $item['appid'], 'pagepath' => $item['pagepath']];
|
||||
default:
|
||||
return [];
|
||||
}
|
||||
}
|
||||
}
|
154
app/wechat/controller/News.php
Normal file
154
app/wechat/controller/News.php
Normal file
@ -0,0 +1,154 @@
|
||||
<?php
|
||||
|
||||
// +----------------------------------------------------------------------
|
||||
// | Wechat Plugin for ThinkAdmin
|
||||
// +----------------------------------------------------------------------
|
||||
// | 版权所有 2014~2025 Anyon <zoujingli@qq.com>
|
||||
// +----------------------------------------------------------------------
|
||||
// | 官方网站: https://thinkadmin.top
|
||||
// +----------------------------------------------------------------------
|
||||
// | 开源协议 ( https://mit-license.org )
|
||||
// | 免责声明 ( https://thinkadmin.top/disclaimer )
|
||||
// +----------------------------------------------------------------------
|
||||
// | gitee 代码仓库:https://gitee.com/zoujingli/think-plugs-wechat
|
||||
// | github 代码仓库:https://github.com/zoujingli/think-plugs-wechat
|
||||
// +----------------------------------------------------------------------
|
||||
|
||||
declare (strict_types=1);
|
||||
|
||||
namespace app\wechat\controller;
|
||||
|
||||
use app\wechat\model\WechatNews;
|
||||
use app\wechat\model\WechatNewsArticle;
|
||||
use app\wechat\service\MediaService;
|
||||
use think\admin\Controller;
|
||||
use think\admin\helper\QueryHelper;
|
||||
use think\admin\service\AdminService;
|
||||
|
||||
/**
|
||||
* 微信图文管理
|
||||
* @class News
|
||||
* @package app\wechat\controller
|
||||
*/
|
||||
class News extends Controller
|
||||
{
|
||||
/**
|
||||
* 微信图文管理
|
||||
* @auth true
|
||||
* @menu true
|
||||
*/
|
||||
public function index()
|
||||
{
|
||||
$this->title = '微信图文列表';
|
||||
WechatNews::mQuery(null, static function (QueryHelper $query) {
|
||||
$query->where(['is_deleted' => 0])->order('id desc')->page();
|
||||
});
|
||||
}
|
||||
|
||||
/**
|
||||
* 图文列表数据处理
|
||||
* @param array $data
|
||||
* @throws \think\db\exception\DataNotFoundException
|
||||
* @throws \think\db\exception\DbException
|
||||
* @throws \think\db\exception\ModelNotFoundException
|
||||
*/
|
||||
protected function _page_filter(array &$data)
|
||||
{
|
||||
foreach ($data as &$vo) {
|
||||
$vo = MediaService::news($vo['id']);
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* 图文选择器
|
||||
* @auth true
|
||||
*/
|
||||
public function select()
|
||||
{
|
||||
$this->index();
|
||||
}
|
||||
|
||||
/**
|
||||
* 添加微信图文
|
||||
* @auth true
|
||||
*/
|
||||
public function add()
|
||||
{
|
||||
if ($this->request->isGet()) {
|
||||
$this->title = '新建微信图文';
|
||||
$this->fetch('form');
|
||||
} else {
|
||||
$update = [
|
||||
'create_by' => AdminService::getUserId(),
|
||||
'article_id' => $this->_buildArticle($this->request->post('data', [])),
|
||||
];
|
||||
if (WechatNews::mk()->save($update) !== false) {
|
||||
$this->success('图文添加成功!', 'javascript:history.back()');
|
||||
} else {
|
||||
$this->error('图文添加失败,请稍候再试!');
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* 编辑微信图文
|
||||
* @auth true
|
||||
* @throws \think\db\exception\DataNotFoundException
|
||||
* @throws \think\db\exception\DbException
|
||||
* @throws \think\db\exception\ModelNotFoundException
|
||||
*/
|
||||
public function edit()
|
||||
{
|
||||
$this->id = $this->request->get('id');
|
||||
if (empty($this->id)) $this->error('参数错误,请稍候再试!');
|
||||
if ($this->request->isGet()) {
|
||||
if ($this->request->get('output') === 'json') {
|
||||
$this->success('获取数据成功!', MediaService::news($this->id));
|
||||
} else {
|
||||
$this->title = '编辑微信图文';
|
||||
$this->fetch('form');
|
||||
}
|
||||
} else {
|
||||
$ids = $this->_buildArticle($this->request->post('data', []));
|
||||
[$map, $data] = [['id' => $this->id], ['article_id' => $ids]];
|
||||
if (WechatNews::mk()->where($map)->update($data) !== false) {
|
||||
$this->success('图文更新成功!', 'javascript:history.back()');
|
||||
} else {
|
||||
$this->error('更新失败,请稍候再试!');
|
||||
}
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* 删除微信图文
|
||||
* auth true
|
||||
*/
|
||||
public function remove()
|
||||
{
|
||||
WechatNews::mDelete();
|
||||
}
|
||||
|
||||
/**
|
||||
* 图文更新操作
|
||||
* @param array $data
|
||||
* @return string
|
||||
*/
|
||||
private function _buildArticle(array $data): string
|
||||
{
|
||||
$ids = [];
|
||||
foreach ($data as $vo) {
|
||||
if (empty($vo['digest'])) {
|
||||
$vo['digest'] = mb_substr(strip_tags(preg_replace('#(\s+| )#', '', $vo['content'])), 0, 120);
|
||||
}
|
||||
$vo['create_at'] = date('Y-m-d H:i:s');
|
||||
if (empty($vo['id'])) {
|
||||
$result = $id = WechatNewsArticle::mk()->insertGetId($vo);
|
||||
} else {
|
||||
$id = intval($vo['id']);
|
||||
$result = WechatNewsArticle::mk()->where('id', $id)->update($vo);
|
||||
}
|
||||
if ($result) $ids[] = $id;
|
||||
}
|
||||
return join(',', $ids);
|
||||
}
|
||||
}
|
99
app/wechat/controller/api/Js.php
Normal file
99
app/wechat/controller/api/Js.php
Normal file
@ -0,0 +1,99 @@
|
||||
<?php
|
||||
|
||||
// +----------------------------------------------------------------------
|
||||
// | Wechat Plugin for ThinkAdmin
|
||||
// +----------------------------------------------------------------------
|
||||
// | 版权所有 2014~2025 Anyon <zoujingli@qq.com>
|
||||
// +----------------------------------------------------------------------
|
||||
// | 官方网站: https://thinkadmin.top
|
||||
// +----------------------------------------------------------------------
|
||||
// | 开源协议 ( https://mit-license.org )
|
||||
// | 免责声明 ( https://thinkadmin.top/disclaimer )
|
||||
// +----------------------------------------------------------------------
|
||||
// | gitee 代码仓库:https://gitee.com/zoujingli/think-plugs-wechat
|
||||
// | github 代码仓库:https://github.com/zoujingli/think-plugs-wechat
|
||||
// +----------------------------------------------------------------------
|
||||
|
||||
declare (strict_types=1);
|
||||
|
||||
namespace app\wechat\controller\api;
|
||||
|
||||
use app\wechat\service\WechatService;
|
||||
use think\admin\Controller;
|
||||
use think\Response;
|
||||
|
||||
/**
|
||||
* 前端JS获取控制器
|
||||
* @class Js
|
||||
* @package app\wechat\controller\api
|
||||
*/
|
||||
class Js extends Controller
|
||||
{
|
||||
/** @var string */
|
||||
protected $params;
|
||||
|
||||
/** @var string */
|
||||
protected $openid;
|
||||
|
||||
/** @var string */
|
||||
protected $fansinfo;
|
||||
|
||||
/**
|
||||
* 生成网页授权的JS内容
|
||||
* @return \think\Response
|
||||
* @throws \WeChat\Exceptions\InvalidResponseException
|
||||
* @throws \WeChat\Exceptions\LocalCacheException
|
||||
* @throws \think\admin\Exception
|
||||
*/
|
||||
public function index(): Response
|
||||
{
|
||||
$mode = $this->request->get('mode', 1);
|
||||
$source = $this->request->server('http_referer') ?: $this->request->url(true);
|
||||
$userinfo = WechatService::getWebOauthInfo($source, $mode, false);
|
||||
if (empty($userinfo['openid'])) {
|
||||
$content = 'alert("Wechat webOauth failed.")';
|
||||
} else {
|
||||
$this->openid = $userinfo['openid'];
|
||||
$this->params = json_encode(WechatService::getWebJssdkSign($source));
|
||||
$this->fansinfo = json_encode($userinfo['fansinfo'] ?? [], JSON_UNESCAPED_UNICODE);
|
||||
// 生成数据授权令牌
|
||||
$this->token = uniqid('oauth') . rand(10000, 99999);
|
||||
$this->app->cache->set($this->openid, $this->token, 3600);
|
||||
// 生成前端JS变量代码
|
||||
$content = $this->_buildContent();
|
||||
}
|
||||
return Response::create($content)->contentType('application/x-javascript');
|
||||
}
|
||||
|
||||
/**
|
||||
* 给指定地址创建签名参数
|
||||
* @throws \WeChat\Exceptions\InvalidResponseException
|
||||
* @throws \WeChat\Exceptions\LocalCacheException
|
||||
* @throws \think\admin\Exception
|
||||
*/
|
||||
public function sdk()
|
||||
{
|
||||
$data = $this->_vali(['url.require' => '签名地址不能为空!']);
|
||||
$this->success('获取签名参数', WechatService::getWebJssdkSign($data['url']));
|
||||
}
|
||||
|
||||
/**
|
||||
* 生成授权内容
|
||||
* @return string
|
||||
*/
|
||||
private function _buildContent(): string
|
||||
{
|
||||
return <<<EOF
|
||||
if(typeof wx === 'object'){
|
||||
wx.token="{$this->token}";
|
||||
wx.openid="{$this->openid}";
|
||||
wx.fansinfo={$this->fansinfo};
|
||||
wx.config({$this->params});
|
||||
wx.ready(function(){
|
||||
wx.hideOptionMenu();
|
||||
wx.hideAllNonBaseMenuItem();
|
||||
});
|
||||
}
|
||||
EOF;
|
||||
}
|
||||
}
|
71
app/wechat/controller/api/Login.php
Normal file
71
app/wechat/controller/api/Login.php
Normal file
@ -0,0 +1,71 @@
|
||||
<?php
|
||||
|
||||
// +----------------------------------------------------------------------
|
||||
// | Wechat Plugin for ThinkAdmin
|
||||
// +----------------------------------------------------------------------
|
||||
// | 版权所有 2014~2025 Anyon <zoujingli@qq.com>
|
||||
// +----------------------------------------------------------------------
|
||||
// | 官方网站: https://thinkadmin.top
|
||||
// +----------------------------------------------------------------------
|
||||
// | 开源协议 ( https://mit-license.org )
|
||||
// | 免责声明 ( https://thinkadmin.top/disclaimer )
|
||||
// +----------------------------------------------------------------------
|
||||
// | gitee 代码仓库:https://gitee.com/zoujingli/think-plugs-wechat
|
||||
// | github 代码仓库:https://github.com/zoujingli/think-plugs-wechat
|
||||
// +----------------------------------------------------------------------
|
||||
|
||||
declare (strict_types=1);
|
||||
|
||||
namespace app\wechat\controller\api;
|
||||
|
||||
use app\wechat\service\LoginService;
|
||||
use think\admin\Controller;
|
||||
|
||||
/**
|
||||
* 微信扫码登录
|
||||
* @class Login
|
||||
* @package app\wechat\controller\api
|
||||
*/
|
||||
class Login extends Controller
|
||||
{
|
||||
/**
|
||||
* 显示二维码
|
||||
* @return void
|
||||
*/
|
||||
public function qrc()
|
||||
{
|
||||
$mode = intval(input('mode', '0'));
|
||||
$data = LoginService::qrcode(LoginService::gcode(), $mode);
|
||||
$this->success('登录二维码', $data);
|
||||
}
|
||||
|
||||
/**
|
||||
* 微信授权处理
|
||||
* @throws \WeChat\Exceptions\InvalidResponseException
|
||||
* @throws \WeChat\Exceptions\LocalCacheException
|
||||
* @throws \think\admin\Exception
|
||||
*/
|
||||
public function oauth()
|
||||
{
|
||||
$data = $this->_vali(['auth.default' => '', 'mode.default' => '0']);
|
||||
if (LoginService::oauth($data['auth'], intval($data['mode']))) {
|
||||
$this->fetch('success', ['message' => '授权成功']);
|
||||
} else {
|
||||
$this->fetch('failed', ['message' => '授权失败']);
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* 获取授权信息
|
||||
* 用定时器请求这个接口
|
||||
*/
|
||||
public function query()
|
||||
{
|
||||
$data = $this->_vali(['code.require' => '编号不能为空!']);
|
||||
if ($fans = LoginService::query($data['code'])) {
|
||||
$this->success('获取授权信息', $fans);
|
||||
} else {
|
||||
$this->error('未获取到授权!');
|
||||
}
|
||||
}
|
||||
}
|
333
app/wechat/controller/api/Push.php
Normal file
333
app/wechat/controller/api/Push.php
Normal file
@ -0,0 +1,333 @@
|
||||
<?php
|
||||
|
||||
// +----------------------------------------------------------------------
|
||||
// | Wechat Plugin for ThinkAdmin
|
||||
// +----------------------------------------------------------------------
|
||||
// | 版权所有 2014~2025 Anyon <zoujingli@qq.com>
|
||||
// +----------------------------------------------------------------------
|
||||
// | 官方网站: https://thinkadmin.top
|
||||
// +----------------------------------------------------------------------
|
||||
// | 开源协议 ( https://mit-license.org )
|
||||
// | 免责声明 ( https://thinkadmin.top/disclaimer )
|
||||
// +----------------------------------------------------------------------
|
||||
// | gitee 代码仓库:https://gitee.com/zoujingli/think-plugs-wechat
|
||||
// +----------------------------------------------------------------------
|
||||
|
||||
declare (strict_types=1);
|
||||
|
||||
namespace app\wechat\controller\api;
|
||||
|
||||
use app\wechat\service\FansService;
|
||||
use app\wechat\service\MediaService;
|
||||
use app\wechat\service\WechatService;
|
||||
use think\admin\Controller;
|
||||
|
||||
/**
|
||||
* 微信消息推送处理
|
||||
* @class Push
|
||||
* @package app\wechat\controller\api
|
||||
*/
|
||||
class Push extends Controller
|
||||
{
|
||||
|
||||
/**
|
||||
* 公众号 APPID
|
||||
* @var string
|
||||
*/
|
||||
protected $appid;
|
||||
|
||||
/**
|
||||
* 微信用户 OPENID
|
||||
* @var string
|
||||
*/
|
||||
protected $openid;
|
||||
|
||||
/**
|
||||
* 消息是否加密码
|
||||
* @var boolean
|
||||
*/
|
||||
protected $encrypt;
|
||||
|
||||
/**
|
||||
* 请求微信 OPENID
|
||||
* @var string
|
||||
*/
|
||||
protected $fromOpenid;
|
||||
|
||||
/**
|
||||
* 微信消息对象
|
||||
* @var array
|
||||
*/
|
||||
protected $receive;
|
||||
|
||||
/**
|
||||
* 微信实例对象
|
||||
* @var \WeChat\Receive
|
||||
*/
|
||||
protected $wechat;
|
||||
|
||||
/**
|
||||
* 强制返回JSON消息
|
||||
* @var boolean
|
||||
*/
|
||||
protected $forceJson = false;
|
||||
|
||||
/**
|
||||
* 强制客服消息回复
|
||||
* @var boolean
|
||||
*/
|
||||
protected $forceCustom = false;
|
||||
|
||||
/**
|
||||
* 获取网络出口IP
|
||||
* @return string
|
||||
*/
|
||||
public function geoip(): string
|
||||
{
|
||||
return $this->request->ip();
|
||||
}
|
||||
|
||||
/**
|
||||
* 消息推送处理接口
|
||||
* @return string
|
||||
*/
|
||||
public function index(): string
|
||||
{
|
||||
try {
|
||||
if (WechatService::getType() === 'thr') {
|
||||
$this->forceJson = true; // 直接返回JSON数据到SERVICE
|
||||
$this->forceCustom = false; // 直接使用客服消息模式推送
|
||||
$this->appid = $this->request->post('appid', '', null);
|
||||
$this->openid = $this->request->post('openid', '', null);
|
||||
$this->encrypt = boolval($this->request->post('encrypt', 0));
|
||||
$this->receive = $this->_arrayChangeKeyCase(json_decode(input('params', '[]'), true));
|
||||
if (empty($this->appid) || empty($this->openid) || empty($this->receive)) {
|
||||
throw new \think\admin\Exception('微信API实例缺失必要参数[appid,openid,receive]');
|
||||
}
|
||||
} else {
|
||||
$this->forceJson = false; // 直接返回JSON对象数据
|
||||
$this->forceCustom = false; // 直接使用客服消息推送
|
||||
$this->appid = WechatService::getAppid();
|
||||
$this->wechat = WechatService::WeChatReceive();
|
||||
$this->openid = $this->wechat->getOpenid();
|
||||
$this->encrypt = $this->wechat->isEncrypt();
|
||||
$this->receive = $this->_arrayChangeKeyCase($this->wechat->getReceive());
|
||||
}
|
||||
$this->fromOpenid = $this->receive['tousername'] ?? '';
|
||||
// 消息类型:text, event, image, voice, shortvideo, location, link
|
||||
if (method_exists($this, ($method = $this->receive['msgtype'] ?? ''))) {
|
||||
if (is_string($result = $this->$method())) return $result;
|
||||
} else {
|
||||
$this->app->log->notice("The {$method} event pushed by wechat was not handled. from {$this->openid}");
|
||||
}
|
||||
} catch (\Exception $exception) {
|
||||
$this->app->log->error("{$exception->getFile()}:{$exception->getLine()} [{$exception->getCode()}] {$exception->getMessage()}");
|
||||
}
|
||||
return $this->fromOpenid ? 'success' : '';
|
||||
}
|
||||
|
||||
/**
|
||||
* 文件消息处理
|
||||
* @return boolean|string
|
||||
* @throws \WeChat\Exceptions\InvalidDecryptException
|
||||
* @throws \WeChat\Exceptions\InvalidResponseException
|
||||
* @throws \WeChat\Exceptions\LocalCacheException
|
||||
* @throws \think\admin\Exception
|
||||
* @throws \think\db\exception\DataNotFoundException
|
||||
* @throws \think\db\exception\DbException
|
||||
* @throws \think\db\exception\ModelNotFoundException
|
||||
*/
|
||||
protected function text()
|
||||
{
|
||||
return $this->_keys("WechatKeys#keys#{$this->receive['content']}", false, $this->forceCustom);
|
||||
}
|
||||
|
||||
/**
|
||||
* 事件消息处理
|
||||
* @return boolean|string
|
||||
* @throws \WeChat\Exceptions\InvalidDecryptException
|
||||
* @throws \WeChat\Exceptions\InvalidResponseException
|
||||
* @throws \WeChat\Exceptions\LocalCacheException
|
||||
* @throws \think\admin\Exception
|
||||
* @throws \think\db\exception\DataNotFoundException
|
||||
* @throws \think\db\exception\DbException
|
||||
* @throws \think\db\exception\ModelNotFoundException
|
||||
*/
|
||||
protected function event()
|
||||
{
|
||||
switch (strtolower($this->receive['event'])) {
|
||||
case 'unsubscribe':
|
||||
$this->app->event->trigger('WechatFansUnSubscribe', $this->openid);
|
||||
return $this->_setUserInfo(false);
|
||||
case 'subscribe':
|
||||
[$this->app->event->trigger('WechatFansSubscribe', $this->openid), $this->_setUserInfo(true)];
|
||||
if (isset($this->receive['eventkey']) && is_string($this->receive['eventkey'])) {
|
||||
if (($key = preg_replace('/^qrscene_/i', '', $this->receive['eventkey']))) {
|
||||
return $this->_keys("WechatKeys#keys#{$key}", false, true);
|
||||
}
|
||||
}
|
||||
return $this->_keys('WechatKeys#keys#subscribe', true, $this->forceCustom);
|
||||
case 'scan':
|
||||
case 'click':
|
||||
if (empty($this->receive['eventkey'])) return false;
|
||||
return $this->_keys("WechatKeys#keys#{$this->receive['eventkey']}", false, $this->forceCustom);
|
||||
case 'scancode_push':
|
||||
case 'scancode_waitmsg':
|
||||
if (empty($this->receive['scancodeinfo']['scanresult'])) return false;
|
||||
return $this->_keys("WechatKeys#keys#{$this->receive['scancodeinfo']['scanresult']}", false, $this->forceCustom);
|
||||
case 'view':
|
||||
case 'location':
|
||||
default:
|
||||
return false;
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* 关键字处理
|
||||
* @param string $rule 关键字规则
|
||||
* @param boolean $last 重复回复消息处理
|
||||
* @param boolean $custom 是否使用客服消息发送
|
||||
* @return boolean|string
|
||||
* @throws \WeChat\Exceptions\InvalidDecryptException
|
||||
* @throws \WeChat\Exceptions\InvalidResponseException
|
||||
* @throws \WeChat\Exceptions\LocalCacheException
|
||||
* @throws \think\admin\Exception
|
||||
* @throws \think\db\exception\DataNotFoundException
|
||||
* @throws \think\db\exception\DbException
|
||||
* @throws \think\db\exception\ModelNotFoundException
|
||||
*/
|
||||
private function _keys(string $rule, bool $last = false, bool $custom = false)
|
||||
{
|
||||
if (is_numeric(stripos($rule, '#reply#text:'))) {
|
||||
[, $content] = explode('#reply#text:', $rule);
|
||||
return $this->_buildMessage('text', ['Content' => $content]);
|
||||
}
|
||||
[$table, $field, $value] = explode('#', $rule . '##');
|
||||
$data = $this->app->db->name($table)->where([$field => $value])->find();
|
||||
if (empty($data['type']) || (array_key_exists('status', $data) && empty($data['status']))) {
|
||||
return $last ? false : $this->_keys('WechatKeys#keys#default', true, $custom);
|
||||
}
|
||||
switch (strtolower($data['type'])) {
|
||||
case 'keys':
|
||||
$content = empty($data['content']) ? $data['name'] : $data['content'];
|
||||
return $this->_keys("WechatKeys#keys#{$content}", $last, $custom);
|
||||
case 'text':
|
||||
return $this->_sendMessage('text', ['content' => $data['content']], $custom);
|
||||
case 'customservice':
|
||||
return $this->_sendMessage('customservice', ['content' => $data['content']]);
|
||||
case 'voice':
|
||||
if (empty($data['voice_url']) || !($mediaId = MediaService::upload($data['voice_url'], 'voice'))) return false;
|
||||
return $this->_sendMessage('voice', ['media_id' => $mediaId], $custom);
|
||||
case 'image':
|
||||
if (empty($data['image_url']) || !($mediaId = MediaService::upload($data['image_url']))) return false;
|
||||
return $this->_sendMessage('image', ['media_id' => $mediaId], $custom);
|
||||
case 'news':
|
||||
[$news, $articles] = [MediaService::news($data['news_id']), []];
|
||||
if (empty($news['articles'])) return false;
|
||||
foreach ($news['articles'] as $vo) $articles[] = [
|
||||
'url' => url("@wechat/api.view/item/id/{$vo['id']}", [], false, true)->build(),
|
||||
'title' => $vo['title'], 'picurl' => $vo['local_url'], 'description' => $vo['digest'],
|
||||
];
|
||||
return $this->_sendMessage('news', ['articles' => $articles], $custom);
|
||||
case 'music':
|
||||
if (empty($data['music_url']) || empty($data['music_title']) || empty($data['music_desc'])) return false;
|
||||
$mediaId = $data['music_image'] ? MediaService::upload($data['music_image']) : '';
|
||||
return $this->_sendMessage('music', [
|
||||
'hqmusicurl' => $data['music_url'], 'musicurl' => $data['music_url'],
|
||||
'description' => $data['music_desc'], 'title' => $data['music_title'], 'thumb_media_id' => $mediaId,
|
||||
], $custom);
|
||||
case 'video':
|
||||
if (empty($data['video_url']) || empty($data['video_desc']) || empty($data['video_title'])) return false;
|
||||
$video = ['title' => $data['video_title'], 'introduction' => $data['video_desc']];
|
||||
if (!($mediaId = MediaService::upload($data['video_url'], 'video', $video))) return false;
|
||||
return $this->_sendMessage('video', ['media_id' => $mediaId, 'title' => $data['video_title'], 'description' => $data['video_desc']], $custom);
|
||||
default:
|
||||
return false;
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* 发送消息到微信
|
||||
* @param string $type 消息类型(text|image|voice|video|music|news|mpnews|wxcard)
|
||||
* @param array $data 消息内容数据对象
|
||||
* @param boolean $custom 是否使用客服消息发送
|
||||
* @return string|void
|
||||
* @throws \WeChat\Exceptions\InvalidDecryptException
|
||||
* @throws \WeChat\Exceptions\InvalidResponseException
|
||||
* @throws \WeChat\Exceptions\LocalCacheException
|
||||
*/
|
||||
private function _sendMessage(string $type, array $data, bool $custom = false)
|
||||
{
|
||||
if ($custom) {
|
||||
WechatService::WeChatCustom()->send(['touser' => $this->openid, 'msgtype' => $type, $type => $data]);
|
||||
} else switch (strtolower($type)) {
|
||||
case 'text': // 发送文本消息
|
||||
return $this->_buildMessage($type, ['Content' => $data['content']]);
|
||||
case 'news': // 发送图文消息
|
||||
foreach ($data['articles'] as &$v) {
|
||||
$v = ['PicUrl' => $v['picurl'], 'Title' => $v['title'], 'Description' => $v['description'], 'Url' => $v['url']];
|
||||
}
|
||||
return $this->_buildMessage($type, ['Articles' => $data['articles'], 'ArticleCount' => count($data['articles'])]);
|
||||
case 'image': // 发送图片消息
|
||||
return $this->_buildMessage($type, ['Image' => ['MediaId' => $data['media_id']]]);
|
||||
case 'voice': // 发送语言消息
|
||||
return $this->_buildMessage($type, ['Voice' => ['MediaId' => $data['media_id']]]);
|
||||
case 'video': // 发送视频消息
|
||||
return $this->_buildMessage($type, ['Video' => ['Title' => $data['title'], 'Description' => $data['description'], 'MediaId' => $data['media_id']]]);
|
||||
case 'music': // 发送音乐消息
|
||||
return $this->_buildMessage($type, ['Music' => ['Title' => $data['title'], 'Description' => $data['description'], 'MusicUrl' => $data['musicurl'], 'HQMusicUrl' => $data['musicurl'], 'ThumbMediaId' => $data['thumb_media_id']]]);
|
||||
case 'customservice': // 转交客服消息
|
||||
if ($data['content']) $this->_sendMessage('text', $data, true);
|
||||
return $this->_buildMessage('transfer_customer_service');
|
||||
default:
|
||||
return 'success';
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* 消息数据生成
|
||||
* @param mixed $type 消息类型
|
||||
* @param array $data 消息内容
|
||||
* @return string
|
||||
* @throws \WeChat\Exceptions\InvalidDecryptException
|
||||
*/
|
||||
private function _buildMessage(string $type, array $data = []): string
|
||||
{
|
||||
$data = array_merge($data, ['ToUserName' => $this->openid, 'FromUserName' => $this->fromOpenid, 'CreateTime' => time(), 'MsgType' => $type]);
|
||||
return $this->forceJson ? json_encode($data, JSON_UNESCAPED_UNICODE) : WechatService::WeChatReceive()->reply($data, true, $this->encrypt);
|
||||
}
|
||||
|
||||
/**
|
||||
* 同步粉丝状态
|
||||
* @param boolean $state 订阅状态
|
||||
* @return boolean
|
||||
*/
|
||||
private function _setUserInfo(bool $state): bool
|
||||
{
|
||||
if ($state) {
|
||||
try {
|
||||
$user = WechatService::WeChatUser()->getUserInfo($this->openid);
|
||||
return FansService::set(array_merge($user, ['subscribe' => 1, 'appid' => $this->appid]));
|
||||
} catch (\Exception $exception) {
|
||||
$this->app->log->error(__METHOD__ . " {$this->openid} get userinfo faild. {$exception->getMessage()}");
|
||||
return false;
|
||||
}
|
||||
} else {
|
||||
return FansService::set(['subscribe' => 0, 'openid' => $this->openid, 'appid' => $this->appid]);
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* 数组健值全部转小写
|
||||
* @param array $data
|
||||
* @return array
|
||||
*/
|
||||
private function _arrayChangeKeyCase(array $data): array
|
||||
{
|
||||
$data = array_change_key_case($data);
|
||||
foreach ($data as $key => $vo) if (is_array($vo)) {
|
||||
$data[$key] = $this->_arrayChangeKeyCase($vo);
|
||||
}
|
||||
return $data;
|
||||
}
|
||||
}
|
209
app/wechat/controller/api/Test.php
Normal file
209
app/wechat/controller/api/Test.php
Normal file
@ -0,0 +1,209 @@
|
||||
<?php
|
||||
|
||||
// +----------------------------------------------------------------------
|
||||
// | Wechat Plugin for ThinkAdmin
|
||||
// +----------------------------------------------------------------------
|
||||
// | 版权所有 2014~2025 Anyon <zoujingli@qq.com>
|
||||
// +----------------------------------------------------------------------
|
||||
// | 官方网站: https://thinkadmin.top
|
||||
// +----------------------------------------------------------------------
|
||||
// | 开源协议 ( https://mit-license.org )
|
||||
// | 免责声明 ( https://thinkadmin.top/disclaimer )
|
||||
// +----------------------------------------------------------------------
|
||||
// | gitee 代码仓库:https://gitee.com/zoujingli/think-plugs-wechat
|
||||
// | github 代码仓库:https://github.com/zoujingli/think-plugs-wechat
|
||||
// +----------------------------------------------------------------------
|
||||
|
||||
declare (strict_types=1);
|
||||
|
||||
namespace app\wechat\controller\api;
|
||||
|
||||
use app\wechat\service\MediaService;
|
||||
use app\wechat\service\PaymentService;
|
||||
use app\wechat\service\WechatService;
|
||||
use think\admin\Controller;
|
||||
use think\admin\extend\CodeExtend;
|
||||
use think\Response;
|
||||
use WeChat\Contracts\Tools;
|
||||
|
||||
/**
|
||||
* 微信测试工具
|
||||
* @class Test
|
||||
* @package app\wechat\controller\api
|
||||
*/
|
||||
class Test extends Controller
|
||||
{
|
||||
/**
|
||||
* 微信JSAPI支付二维码
|
||||
* @login true
|
||||
* @return \think\Response
|
||||
*/
|
||||
public function jsapiQrc(): Response
|
||||
{
|
||||
$this->url = sysuri('wechat/api.test/jsapi', [], false, true);
|
||||
return $this->_buildQrcResponse($this->url);
|
||||
}
|
||||
|
||||
/**
|
||||
* 显示网页授权二维码
|
||||
* @login true
|
||||
* @return \think\Response
|
||||
*/
|
||||
public function oauthQrc(): Response
|
||||
{
|
||||
$this->url = sysuri('wechat/api.test/oauth', [], false, true);
|
||||
return $this->_buildQrcResponse($this->url);
|
||||
}
|
||||
|
||||
/**
|
||||
* 显示网页授权二维码
|
||||
* @login true
|
||||
* @return \think\Response
|
||||
*/
|
||||
public function jssdkQrc(): Response
|
||||
{
|
||||
$this->url = sysuri('wechat/api.test/jssdk', [], false, true);
|
||||
return $this->_buildQrcResponse($this->url);
|
||||
}
|
||||
|
||||
/**
|
||||
* 微信扫码支付模式一二维码显示
|
||||
* @login true
|
||||
* @return \think\Response
|
||||
*/
|
||||
public function scanOneQrc(): Response
|
||||
{
|
||||
$pay = WechatService::WePayOrder();
|
||||
return $this->_buildQrcResponse($pay->qrcParams('8888888'));
|
||||
}
|
||||
|
||||
/**
|
||||
* 扫码支付模式二测试二维码
|
||||
* @login true
|
||||
* @return \think\Response
|
||||
* @throws \think\admin\Exception
|
||||
*/
|
||||
public function scanTwoQrc(): Response
|
||||
{
|
||||
$code = CodeExtend::uniqidDate(18, 'TX');
|
||||
$result = PaymentService::create('', $code, "扫码支付测试 {$code}", '0.01', PaymentService::WECHAT_QRC, '0.01');
|
||||
return $this->_buildQrcResponse($result['params']['code_url']);
|
||||
// $result = WechatService::WePayOrder()->create([
|
||||
// 'body' => '测试商品',
|
||||
// 'total_fee' => '1',
|
||||
// 'trade_type' => 'NATIVE',
|
||||
// 'notify_url' => sysuri('wechat/api.test/notify', [], false, true),
|
||||
// 'out_trade_no' => CodeExtend::uniqidNumber(18),
|
||||
// 'spbill_create_ip' => $this->request->ip(),
|
||||
// ]);
|
||||
// return $this->_buildQrcResponse($result['code_url']);
|
||||
}
|
||||
|
||||
/**
|
||||
* 网页授权测试
|
||||
* @throws \WeChat\Exceptions\InvalidResponseException
|
||||
* @throws \WeChat\Exceptions\LocalCacheException
|
||||
* @throws \think\admin\Exception
|
||||
*/
|
||||
public function oauth()
|
||||
{
|
||||
$this->url = $this->request->url(true);
|
||||
$this->fans = WechatService::getWebOauthInfo($this->url, 1);
|
||||
$this->fetch();
|
||||
}
|
||||
|
||||
/**
|
||||
* JSSDK测试
|
||||
* @throws \WeChat\Exceptions\InvalidResponseException
|
||||
* @throws \WeChat\Exceptions\LocalCacheException
|
||||
* @throws \think\admin\Exception
|
||||
*/
|
||||
public function jssdk()
|
||||
{
|
||||
$this->options = WechatService::getWebJssdkSign();
|
||||
$this->fetch();
|
||||
}
|
||||
|
||||
/**
|
||||
* 微信扫码支付模式一通知处理
|
||||
* -- 注意,需要在微信商户配置支付通知地址
|
||||
* @return string
|
||||
* @throws \WeChat\Exceptions\InvalidResponseException
|
||||
* @throws \WeChat\Exceptions\LocalCacheException
|
||||
*/
|
||||
public function scanOneNotify(): string
|
||||
{
|
||||
$pay = WechatService::WePayOrder();
|
||||
$notify = $pay->getNotify();
|
||||
p('======= 来自扫码支付1的数据 ======');
|
||||
p($notify);
|
||||
// 微信统一下单处理
|
||||
$options = [
|
||||
'body' => "测试商品,商品ID:{$notify['product_id']}",
|
||||
'total_fee' => '1',
|
||||
'trade_type' => 'NATIVE',
|
||||
'notify_url' => sysuri('wechat/api.test/notify', [], false, true),
|
||||
'out_trade_no' => CodeExtend::uniqidDate(18),
|
||||
'spbill_create_ip' => $this->request->ip(),
|
||||
];
|
||||
p('======= 来自扫码支付1统一下单结果 ======');
|
||||
p($order = $pay->create($options));
|
||||
// 回复XML文本
|
||||
$result = [
|
||||
'return_code' => 'SUCCESS',
|
||||
'return_msg' => '处理成功',
|
||||
'appid' => $notify['appid'],
|
||||
'mch_id' => $notify['mch_id'],
|
||||
'nonce_str' => Tools::createNoncestr(),
|
||||
'prepay_id' => $order['prepay_id'],
|
||||
'result_code' => 'SUCCESS',
|
||||
];
|
||||
$result['sign'] = $pay->getPaySign($result);
|
||||
p('======= 来自扫码支付1返回的结果 ======');
|
||||
p($result);
|
||||
return Tools::arr2xml($result);
|
||||
}
|
||||
|
||||
/**
|
||||
* 微信JSAPI支付测试
|
||||
* @return void|string
|
||||
* @throws \WeChat\Exceptions\InvalidResponseException
|
||||
* @throws \WeChat\Exceptions\LocalCacheException
|
||||
* @throws \think\admin\Exception
|
||||
*/
|
||||
public function jsapi()
|
||||
{
|
||||
// 微信用户信息
|
||||
$this->user = WechatService::getWebOauthInfo($this->request->url(true));
|
||||
if (empty($this->user['openid'])) return '<h3>网页授权获取OPENID失败!</h3>';
|
||||
// 生成支付参数
|
||||
$oCode = CodeExtend::uniqidDate(18, 'TX');
|
||||
$this->result = PaymentService::create($this->user['openid'], $oCode, "JSAPI 支付测试 {$oCode}", '0.01', PaymentService::WECHAT_GZH);
|
||||
$this->optionJson = json_encode($this->result['params'], JSON_UNESCAPED_UNICODE | JSON_UNESCAPED_SLASHES);
|
||||
$this->configJson = json_encode(WechatService::getWebJssdkSign(), JSON_UNESCAPED_UNICODE | JSON_UNESCAPED_SLASHES);
|
||||
$this->fetch();
|
||||
}
|
||||
|
||||
/**
|
||||
* 支付通知接收处理
|
||||
* @return string
|
||||
* @throws \WeChat\Exceptions\InvalidResponseException
|
||||
*/
|
||||
public function notify(): string
|
||||
{
|
||||
$wechat = WechatService::WePayOrder();
|
||||
p($wechat->getNotify());
|
||||
return 'SUCCESS';
|
||||
}
|
||||
|
||||
/**
|
||||
* 创建二维码响应对应
|
||||
* @param string $url 二维码内容
|
||||
* @return \think\Response
|
||||
*/
|
||||
private function _buildQrcResponse(string $url): Response
|
||||
{
|
||||
$result = MediaService::getQrcode($url);
|
||||
return response($result->getString(), 200, ['Content-Type' => $result->getMimeType()]);
|
||||
}
|
||||
}
|
107
app/wechat/controller/api/View.php
Normal file
107
app/wechat/controller/api/View.php
Normal file
@ -0,0 +1,107 @@
|
||||
<?php
|
||||
|
||||
// +----------------------------------------------------------------------
|
||||
// | Wechat Plugin for ThinkAdmin
|
||||
// +----------------------------------------------------------------------
|
||||
// | 版权所有 2014~2025 Anyon <zoujingli@qq.com>
|
||||
// +----------------------------------------------------------------------
|
||||
// | 官方网站: https://thinkadmin.top
|
||||
// +----------------------------------------------------------------------
|
||||
// | 开源协议 ( https://mit-license.org )
|
||||
// | 免责声明 ( https://thinkadmin.top/disclaimer )
|
||||
// +----------------------------------------------------------------------
|
||||
// | gitee 代码仓库:https://gitee.com/zoujingli/think-plugs-wechat
|
||||
// | github 代码仓库:https://github.com/zoujingli/think-plugs-wechat
|
||||
// +----------------------------------------------------------------------
|
||||
|
||||
declare (strict_types=1);
|
||||
|
||||
namespace app\wechat\controller\api;
|
||||
|
||||
use app\wechat\model\WechatNewsArticle;
|
||||
use app\wechat\service\MediaService;
|
||||
use think\admin\Controller;
|
||||
|
||||
/**
|
||||
* 微信图文显示
|
||||
* @class View
|
||||
* @package app\wechat\controller\api
|
||||
*/
|
||||
class View extends Controller
|
||||
{
|
||||
|
||||
/**
|
||||
* 图文列表展示
|
||||
* @param string|integer $id 图文ID编号
|
||||
* @throws \think\db\exception\DataNotFoundException
|
||||
* @throws \think\db\exception\DbException
|
||||
* @throws \think\db\exception\ModelNotFoundException
|
||||
*/
|
||||
public function news($id = 0)
|
||||
{
|
||||
$this->id = $id ?: input('id', 0);
|
||||
$this->news = MediaService::news($this->id);
|
||||
$this->fetch();
|
||||
}
|
||||
|
||||
/**
|
||||
* 文章内容展示
|
||||
* @param string|integer $id 文章ID编号
|
||||
* @throws \think\db\exception\DbException
|
||||
*/
|
||||
public function item($id = 0)
|
||||
{
|
||||
$map = ['id' => $id ?: input('id', 0)];
|
||||
$modal = WechatNewsArticle::mk()->where($map)->findOrEmpty();
|
||||
$modal->isExists() && $modal->newQuery()->where($map)->setInc('read_num');
|
||||
$this->fetch('item', ['info' => $modal->toArray()]);
|
||||
}
|
||||
|
||||
/**
|
||||
* 文本展示
|
||||
*/
|
||||
public function text()
|
||||
{
|
||||
$text = strip_tags(input('content', ''), '<a><img>');
|
||||
$this->fetch('text', ['content' => $text]);
|
||||
}
|
||||
|
||||
/**
|
||||
* 图片展示
|
||||
*/
|
||||
public function image()
|
||||
{
|
||||
$text = strip_tags(input('content', ''), '<a><img>');
|
||||
$this->fetch('image', ['content' => $text]);
|
||||
}
|
||||
|
||||
/**
|
||||
* 视频展示
|
||||
*/
|
||||
public function video()
|
||||
{
|
||||
$this->url = strip_tags(input('url', ''), '<a><img>');
|
||||
$this->title = strip_tags(input('title', ''), '<a><img>');
|
||||
$this->fetch();
|
||||
}
|
||||
|
||||
/**
|
||||
* 语音展示
|
||||
*/
|
||||
public function voice()
|
||||
{
|
||||
$this->url = strip_tags(input('url', ''), '<a><img>');
|
||||
$this->fetch();
|
||||
}
|
||||
|
||||
/**
|
||||
* 音乐展示
|
||||
*/
|
||||
public function music()
|
||||
{
|
||||
$this->url = strip_tags(input('url', ''), '<a><img>');
|
||||
$this->desc = strip_tags(input('desc', ''), '<a><img>');
|
||||
$this->title = strip_tags(input('title', ''), '<a><img>');
|
||||
$this->fetch();
|
||||
}
|
||||
}
|
89
app/wechat/controller/payment/Record.php
Normal file
89
app/wechat/controller/payment/Record.php
Normal file
@ -0,0 +1,89 @@
|
||||
<?php
|
||||
|
||||
// +----------------------------------------------------------------------
|
||||
// | Wechat Plugin for ThinkAdmin
|
||||
// +----------------------------------------------------------------------
|
||||
// | 版权所有 2014~2025 Anyon <zoujingli@qq.com>
|
||||
// +----------------------------------------------------------------------
|
||||
// | 官方网站: https://thinkadmin.top
|
||||
// +----------------------------------------------------------------------
|
||||
// | 开源协议 ( https://mit-license.org )
|
||||
// | 免责声明 ( https://thinkadmin.top/disclaimer )
|
||||
// +----------------------------------------------------------------------
|
||||
// | gitee 代码仓库:https://gitee.com/zoujingli/think-plugs-wechat
|
||||
// | github 代码仓库:https://github.com/zoujingli/think-plugs-wechat
|
||||
// +----------------------------------------------------------------------
|
||||
|
||||
declare (strict_types=1);
|
||||
|
||||
namespace app\wechat\controller\payment;
|
||||
|
||||
use app\wechat\model\WechatFans;
|
||||
use app\wechat\model\WechatPaymentRecord;
|
||||
use app\wechat\service\PaymentService;
|
||||
use think\admin\Controller;
|
||||
use think\admin\helper\QueryHelper;
|
||||
use think\exception\HttpResponseException;
|
||||
|
||||
/**
|
||||
* 微信支付行为管理
|
||||
* @class Record
|
||||
* @package app\wechat\controller
|
||||
*/
|
||||
class Record extends Controller
|
||||
{
|
||||
/**
|
||||
* 微信支付行为管理
|
||||
* @auth true
|
||||
* @menu true
|
||||
* @return void
|
||||
* @throws \think\db\exception\DataNotFoundException
|
||||
* @throws \think\db\exception\DbException
|
||||
* @throws \think\db\exception\ModelNotFoundException
|
||||
*/
|
||||
public function index()
|
||||
{
|
||||
WechatPaymentRecord::mQuery()->layTable(function () {
|
||||
$this->title = '支付行为管理';
|
||||
}, static function (QueryHelper $query) {
|
||||
$db = WechatFans::mQuery()->like('openid|nickname#nickname')->db();
|
||||
if ($db->getOptions('where')) $query->whereRaw("openid in {$db->field('openid')->buildSql()}");
|
||||
$query->like('order_code|order_name#order')->dateBetween('create_time');
|
||||
$query->with(['bindFans'])->equal('payment_status');
|
||||
});
|
||||
}
|
||||
|
||||
/**
|
||||
* 创建退款申请
|
||||
* @auth true
|
||||
* @return void
|
||||
*/
|
||||
public function refund()
|
||||
{
|
||||
try {
|
||||
$data = $this->_vali(['code.require' => '支付号不能为空!']);
|
||||
$recode = WechatPaymentRecord::mk()->where($data)->findOrEmpty();
|
||||
if ($recode->isEmpty()) $this->error('支付单不存在!');
|
||||
if ($recode->getAttr('payment_status') < 1) $this->error('支付单未完成支付!');
|
||||
$reason = "来自订单 {$recode['order_code']} 的退款!";
|
||||
sysoplog('微信支付退款', "支付单 {$data['code']} 发起退款!");
|
||||
[$state, $message] = PaymentService::refund($data['code'], strval($recode->getAttr('payment_amount')), $reason);
|
||||
$state ? $this->success($message) : $this->error($message);
|
||||
} catch (HttpResponseException $exception) {
|
||||
throw $exception;
|
||||
} catch (\Exception $exception) {
|
||||
$this->error($exception->getMessage());
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* 清理未支付数据
|
||||
* @auth true
|
||||
* @return void
|
||||
*/
|
||||
public function clear()
|
||||
{
|
||||
sysoplog('微信支付清理', '创建粉丝未支付数据清理任务');
|
||||
$this->_queue('清理微信未支付数据', "xadmin:fanspay", 0, [], 0, 600);
|
||||
}
|
||||
}
|
61
app/wechat/controller/payment/Refund.php
Normal file
61
app/wechat/controller/payment/Refund.php
Normal file
@ -0,0 +1,61 @@
|
||||
<?php
|
||||
|
||||
// +----------------------------------------------------------------------
|
||||
// | Wechat Plugin for ThinkAdmin
|
||||
// +----------------------------------------------------------------------
|
||||
// | 版权所有 2014~2025 Anyon <zoujingli@qq.com>
|
||||
// +----------------------------------------------------------------------
|
||||
// | 官方网站: https://thinkadmin.top
|
||||
// +----------------------------------------------------------------------
|
||||
// | 开源协议 ( https://mit-license.org )
|
||||
// | 免责声明 ( https://thinkadmin.top/disclaimer )
|
||||
// +----------------------------------------------------------------------
|
||||
// | gitee 代码仓库:https://gitee.com/zoujingli/think-plugs-wechat
|
||||
// | github 代码仓库:https://github.com/zoujingli/think-plugs-wechat
|
||||
// +----------------------------------------------------------------------
|
||||
|
||||
declare (strict_types=1);
|
||||
|
||||
namespace app\wechat\controller\payment;
|
||||
|
||||
use app\wechat\model\WechatFans;
|
||||
use app\wechat\model\WechatPaymentRecord;
|
||||
use app\wechat\model\WechatPaymentRefund;
|
||||
use think\admin\Controller;
|
||||
use think\admin\helper\QueryHelper;
|
||||
use think\db\Query;
|
||||
|
||||
/**
|
||||
* 支付退款管理
|
||||
* @class Refund
|
||||
* @package app\wechat\controller
|
||||
*/
|
||||
class Refund extends Controller
|
||||
{
|
||||
/**
|
||||
* 支付退款管理
|
||||
* @auth true
|
||||
* @menu true
|
||||
* @return void
|
||||
* @throws \think\db\exception\DataNotFoundException
|
||||
* @throws \think\db\exception\DbException
|
||||
* @throws \think\db\exception\ModelNotFoundException
|
||||
*/
|
||||
public function index()
|
||||
{
|
||||
WechatPaymentRefund::mQuery()->layTable(function () {
|
||||
$this->title = '支付退款管理';
|
||||
}, function (QueryHelper $query) {
|
||||
$query->like('code|refund_trade#refund')->withoutField('refund_notify');
|
||||
$query->with(['record' => function (Query $query) {
|
||||
$query->withoutField('payment_notify');
|
||||
}]);
|
||||
if (($this->get['order'] ?? '') . ($this->get['nickname'] ?? '') . ($this->get['payment'] ?? '') . ($this->get['refund'] ?? '') !== '') {
|
||||
$db1 = WechatFans::mQuery()->field('openid')->like('openid|nickname#nickname')->db();
|
||||
$db2 = WechatPaymentRecord::mQuery()->like('order_code|order_name#order,code|payment_trade#payment');
|
||||
$db2->whereRaw("openid in {$db1->buildSql()}");
|
||||
$query->whereRaw("record_code in {$db2->field('code')->db()->buildSql()}");
|
||||
}
|
||||
});
|
||||
}
|
||||
}
|
27
app/wechat/lang/en-us.php
Normal file
27
app/wechat/lang/en-us.php
Normal file
@ -0,0 +1,27 @@
|
||||
<?php
|
||||
|
||||
// +----------------------------------------------------------------------
|
||||
// | Wechat Plugin for ThinkAdmin
|
||||
// +----------------------------------------------------------------------
|
||||
// | 版权所有 2014~2025 Anyon <zoujingli@qq.com>
|
||||
// +----------------------------------------------------------------------
|
||||
// | 官方网站: https://thinkadmin.top
|
||||
// +----------------------------------------------------------------------
|
||||
// | 开源协议 ( https://mit-license.org )
|
||||
// | 免责声明 ( https://thinkadmin.top/disclaimer )
|
||||
// +----------------------------------------------------------------------
|
||||
// | gitee 代码仓库:https://gitee.com/zoujingli/think-plugs-wechat
|
||||
// | github 代码仓库:https://github.com/zoujingli/think-plugs-wechat
|
||||
// +----------------------------------------------------------------------
|
||||
|
||||
declare (strict_types=1);
|
||||
|
||||
$extra = [];
|
||||
$extra['请选择微信对接方式,其中微信开放平台授权模式需要微信开放平台支持,还需要搭建第三方服务平台托管系统!'] = 'Please select WeChat docking mode, of which WeChat Open platform authorization mode needs WeChat Open platform support, and a third-party service platform hosting system needs to be built!';
|
||||
$extra['使用微信开放平台授权模式时,微信将授权给第三方服务平台托管系统,消息数据使用 %s 通信协议转发。'] = 'When using WeChat Open platform authorization mode, WeChat will authorize the third-party service platform hosting system, and the message data will be forwarded using %s communication protocol.';
|
||||
$extra['使用微信公众平台直接模式时,需要在微信公众号平台配置授权IP及网页授权域名,将公众号平台获取到的参数填写到下面。'] = 'When using the direct mode of the WeChat public platform, you need to configure the authorized IP and web page authorized domain name on the WeChat official account platform, and fill in the parameters obtained by the official account platform below.';
|
||||
|
||||
return array_merge($extra, [
|
||||
'微信公众平台直接模式' => 'WeChat public platform direct mode',
|
||||
'微信开放平台授权模式' => 'WeChat Open platform authorization mode'
|
||||
]);
|
59
app/wechat/model/WechatAuto.php
Normal file
59
app/wechat/model/WechatAuto.php
Normal file
@ -0,0 +1,59 @@
|
||||
<?php
|
||||
|
||||
// +----------------------------------------------------------------------
|
||||
// | Wechat Plugin for ThinkAdmin
|
||||
// +----------------------------------------------------------------------
|
||||
// | 版权所有 2014~2025 Anyon <zoujingli@qq.com>
|
||||
// +----------------------------------------------------------------------
|
||||
// | 官方网站: https://thinkadmin.top
|
||||
// +----------------------------------------------------------------------
|
||||
// | 开源协议 ( https://mit-license.org )
|
||||
// | 免责声明 ( https://thinkadmin.top/disclaimer )
|
||||
// +----------------------------------------------------------------------
|
||||
// | gitee 代码仓库:https://gitee.com/zoujingli/think-plugs-wechat
|
||||
// | github 代码仓库:https://github.com/zoujingli/think-plugs-wechat
|
||||
// +----------------------------------------------------------------------
|
||||
|
||||
declare (strict_types=1);
|
||||
|
||||
namespace app\wechat\model;
|
||||
|
||||
use think\admin\Model;
|
||||
|
||||
/**
|
||||
* 微信自动回复模型
|
||||
*
|
||||
* @property int $create_by 创建人
|
||||
* @property int $id
|
||||
* @property int $news_id 图文ID
|
||||
* @property int $status 状态(0禁用,1启用)
|
||||
* @property string $appid 公众号APPID
|
||||
* @property string $code 消息编号
|
||||
* @property string $content 文本内容
|
||||
* @property string $create_at 创建时间
|
||||
* @property string $image_url 图片链接
|
||||
* @property string $music_desc 音乐描述
|
||||
* @property string $music_image 缩略图片
|
||||
* @property string $music_title 音乐标题
|
||||
* @property string $music_url 音乐链接
|
||||
* @property string $time 延迟时间
|
||||
* @property string $type 类型(text,image,news)
|
||||
* @property string $video_desc 视频描述
|
||||
* @property string $video_title 视频标题
|
||||
* @property string $video_url 视频URL
|
||||
* @property string $voice_url 语音链接
|
||||
* @class WechatAuto
|
||||
* @package app\wechat\model
|
||||
*/
|
||||
class WechatAuto extends Model
|
||||
{
|
||||
/**
|
||||
* 格式化创建时间
|
||||
* @param mixed $value
|
||||
* @return string
|
||||
*/
|
||||
public function getCreateAtAttr($value): string
|
||||
{
|
||||
return format_datetime($value);
|
||||
}
|
||||
}
|
52
app/wechat/model/WechatFans.php
Normal file
52
app/wechat/model/WechatFans.php
Normal file
@ -0,0 +1,52 @@
|
||||
<?php
|
||||
|
||||
// +----------------------------------------------------------------------
|
||||
// | Wechat Plugin for ThinkAdmin
|
||||
// +----------------------------------------------------------------------
|
||||
// | 版权所有 2014~2025 Anyon <zoujingli@qq.com>
|
||||
// +----------------------------------------------------------------------
|
||||
// | 官方网站: https://thinkadmin.top
|
||||
// +----------------------------------------------------------------------
|
||||
// | 开源协议 ( https://mit-license.org )
|
||||
// | 免责声明 ( https://thinkadmin.top/disclaimer )
|
||||
// +----------------------------------------------------------------------
|
||||
// | gitee 代码仓库:https://gitee.com/zoujingli/think-plugs-wechat
|
||||
// | github 代码仓库:https://github.com/zoujingli/think-plugs-wechat
|
||||
// +----------------------------------------------------------------------
|
||||
|
||||
declare (strict_types=1);
|
||||
|
||||
namespace app\wechat\model;
|
||||
|
||||
use think\admin\Model;
|
||||
|
||||
/**
|
||||
* 微信粉丝用户模型
|
||||
*
|
||||
* @property int $id
|
||||
* @property int $is_black 是否为黑名单状态
|
||||
* @property int $sex 用户性别(1男性,2女性,0未知)
|
||||
* @property int $subscribe 关注状态(0未关注,1已关注)
|
||||
* @property int $subscribe_time 关注时间
|
||||
* @property string $appid 公众号APPID
|
||||
* @property string $city 用户所在城市
|
||||
* @property string $country 用户所在国家
|
||||
* @property string $create_at 创建时间
|
||||
* @property string $headimgurl 用户头像
|
||||
* @property string $language 用户的语言(zh_CN)
|
||||
* @property string $nickname 用户昵称
|
||||
* @property string $openid 粉丝openid
|
||||
* @property string $province 用户所在省份
|
||||
* @property string $qr_scene 二维码场景值
|
||||
* @property string $qr_scene_str 二维码场景内容
|
||||
* @property string $remark 备注
|
||||
* @property string $subscribe_at 关注时间
|
||||
* @property string $subscribe_scene 扫码关注场景
|
||||
* @property string $tagid_list 粉丝标签id
|
||||
* @property string $unionid 粉丝unionid
|
||||
* @class WechatFans
|
||||
* @package app\wechat\model
|
||||
*/
|
||||
class WechatFans extends Model
|
||||
{
|
||||
}
|
36
app/wechat/model/WechatFansTags.php
Normal file
36
app/wechat/model/WechatFansTags.php
Normal file
@ -0,0 +1,36 @@
|
||||
<?php
|
||||
|
||||
// +----------------------------------------------------------------------
|
||||
// | Wechat Plugin for ThinkAdmin
|
||||
// +----------------------------------------------------------------------
|
||||
// | 版权所有 2014~2025 Anyon <zoujingli@qq.com>
|
||||
// +----------------------------------------------------------------------
|
||||
// | 官方网站: https://thinkadmin.top
|
||||
// +----------------------------------------------------------------------
|
||||
// | 开源协议 ( https://mit-license.org )
|
||||
// | 免责声明 ( https://thinkadmin.top/disclaimer )
|
||||
// +----------------------------------------------------------------------
|
||||
// | gitee 代码仓库:https://gitee.com/zoujingli/think-plugs-wechat
|
||||
// | github 代码仓库:https://github.com/zoujingli/think-plugs-wechat
|
||||
// +----------------------------------------------------------------------
|
||||
|
||||
declare (strict_types=1);
|
||||
|
||||
namespace app\wechat\model;
|
||||
|
||||
use think\admin\Model;
|
||||
|
||||
/**
|
||||
* 微信粉丝标签模型
|
||||
*
|
||||
* @property int $count 粉丝总数
|
||||
* @property int $id
|
||||
* @property string $appid 公众号APPID
|
||||
* @property string $create_at 创建日期
|
||||
* @property string $name 标签名称
|
||||
* @class WechatFansTags
|
||||
* @package app\wechat\model
|
||||
*/
|
||||
class WechatFansTags extends Model
|
||||
{
|
||||
}
|
59
app/wechat/model/WechatKeys.php
Normal file
59
app/wechat/model/WechatKeys.php
Normal file
@ -0,0 +1,59 @@
|
||||
<?php
|
||||
|
||||
// +----------------------------------------------------------------------
|
||||
// | Wechat Plugin for ThinkAdmin
|
||||
// +----------------------------------------------------------------------
|
||||
// | 版权所有 2014~2025 Anyon <zoujingli@qq.com>
|
||||
// +----------------------------------------------------------------------
|
||||
// | 官方网站: https://thinkadmin.top
|
||||
// +----------------------------------------------------------------------
|
||||
// | 开源协议 ( https://mit-license.org )
|
||||
// | 免责声明 ( https://thinkadmin.top/disclaimer )
|
||||
// +----------------------------------------------------------------------
|
||||
// | gitee 代码仓库:https://gitee.com/zoujingli/think-plugs-wechat
|
||||
// | github 代码仓库:https://github.com/zoujingli/think-plugs-wechat
|
||||
// +----------------------------------------------------------------------
|
||||
|
||||
declare (strict_types=1);
|
||||
|
||||
namespace app\wechat\model;
|
||||
|
||||
use think\admin\Model;
|
||||
|
||||
/**
|
||||
* 微信回复关键词模型
|
||||
*
|
||||
* @property int $create_by 创建人
|
||||
* @property int $id
|
||||
* @property int $news_id 图文ID
|
||||
* @property int $sort 排序字段
|
||||
* @property int $status 状态(0禁用,1启用)
|
||||
* @property string $appid 公众号APPID
|
||||
* @property string $content 文本内容
|
||||
* @property string $create_at 创建时间
|
||||
* @property string $image_url 图片链接
|
||||
* @property string $keys 关键字
|
||||
* @property string $music_desc 音乐描述
|
||||
* @property string $music_image 缩略图片
|
||||
* @property string $music_title 音乐标题
|
||||
* @property string $music_url 音乐链接
|
||||
* @property string $type 类型(text,image,news)
|
||||
* @property string $video_desc 视频描述
|
||||
* @property string $video_title 视频标题
|
||||
* @property string $video_url 视频URL
|
||||
* @property string $voice_url 语音链接
|
||||
* @class WechatKeys
|
||||
* @package app\wechat\model
|
||||
*/
|
||||
class WechatKeys extends Model
|
||||
{
|
||||
/**
|
||||
* 格式化创建时间
|
||||
* @param mixed $value
|
||||
* @return string
|
||||
*/
|
||||
public function getCreateAtAttr($value): string
|
||||
{
|
||||
return format_datetime($value);
|
||||
}
|
||||
}
|
39
app/wechat/model/WechatMedia.php
Normal file
39
app/wechat/model/WechatMedia.php
Normal file
@ -0,0 +1,39 @@
|
||||
<?php
|
||||
|
||||
// +----------------------------------------------------------------------
|
||||
// | Wechat Plugin for ThinkAdmin
|
||||
// +----------------------------------------------------------------------
|
||||
// | 版权所有 2014~2025 Anyon <zoujingli@qq.com>
|
||||
// +----------------------------------------------------------------------
|
||||
// | 官方网站: https://thinkadmin.top
|
||||
// +----------------------------------------------------------------------
|
||||
// | 开源协议 ( https://mit-license.org )
|
||||
// | 免责声明 ( https://thinkadmin.top/disclaimer )
|
||||
// +----------------------------------------------------------------------
|
||||
// | gitee 代码仓库:https://gitee.com/zoujingli/think-plugs-wechat
|
||||
// | github 代码仓库:https://github.com/zoujingli/think-plugs-wechat
|
||||
// +----------------------------------------------------------------------
|
||||
|
||||
declare (strict_types=1);
|
||||
|
||||
namespace app\wechat\model;
|
||||
|
||||
use think\admin\Model;
|
||||
|
||||
/**
|
||||
* 微信媒体文件模型
|
||||
*
|
||||
* @property int $id
|
||||
* @property string $appid 公众号ID
|
||||
* @property string $create_at 创建时间
|
||||
* @property string $local_url 本地文件链接
|
||||
* @property string $md5 文件哈希
|
||||
* @property string $media_id 永久素材MediaID
|
||||
* @property string $media_url 远程图片链接
|
||||
* @property string $type 媒体类型
|
||||
* @class WechatMedia
|
||||
* @package app\wechat\model
|
||||
*/
|
||||
class WechatMedia extends Model
|
||||
{
|
||||
}
|
38
app/wechat/model/WechatNews.php
Normal file
38
app/wechat/model/WechatNews.php
Normal file
@ -0,0 +1,38 @@
|
||||
<?php
|
||||
|
||||
// +----------------------------------------------------------------------
|
||||
// | Wechat Plugin for ThinkAdmin
|
||||
// +----------------------------------------------------------------------
|
||||
// | 版权所有 2014~2025 Anyon <zoujingli@qq.com>
|
||||
// +----------------------------------------------------------------------
|
||||
// | 官方网站: https://thinkadmin.top
|
||||
// +----------------------------------------------------------------------
|
||||
// | 开源协议 ( https://mit-license.org )
|
||||
// | 免责声明 ( https://thinkadmin.top/disclaimer )
|
||||
// +----------------------------------------------------------------------
|
||||
// | gitee 代码仓库:https://gitee.com/zoujingli/think-plugs-wechat
|
||||
// | github 代码仓库:https://github.com/zoujingli/think-plugs-wechat
|
||||
// +----------------------------------------------------------------------
|
||||
|
||||
declare (strict_types=1);
|
||||
|
||||
namespace app\wechat\model;
|
||||
|
||||
use think\admin\Model;
|
||||
|
||||
/**
|
||||
* 微信图文主模型
|
||||
*
|
||||
* @property int $create_by 创建人
|
||||
* @property int $id
|
||||
* @property int $is_deleted 删除状态(0未删除,1已删除)
|
||||
* @property string $article_id 关联图文ID(用英文逗号做分割)
|
||||
* @property string $create_at 创建时间
|
||||
* @property string $local_url 永久素材外网URL
|
||||
* @property string $media_id 永久素材MediaID
|
||||
* @class WechatNews
|
||||
* @package app\wechat\model
|
||||
*/
|
||||
class WechatNews extends Model
|
||||
{
|
||||
}
|
41
app/wechat/model/WechatNewsArticle.php
Normal file
41
app/wechat/model/WechatNewsArticle.php
Normal file
@ -0,0 +1,41 @@
|
||||
<?php
|
||||
|
||||
// +----------------------------------------------------------------------
|
||||
// | Wechat Plugin for ThinkAdmin
|
||||
// +----------------------------------------------------------------------
|
||||
// | 版权所有 2014~2025 Anyon <zoujingli@qq.com>
|
||||
// +----------------------------------------------------------------------
|
||||
// | 官方网站: https://thinkadmin.top
|
||||
// +----------------------------------------------------------------------
|
||||
// | 开源协议 ( https://mit-license.org )
|
||||
// | 免责声明 ( https://thinkadmin.top/disclaimer )
|
||||
// +----------------------------------------------------------------------
|
||||
// | gitee 代码仓库:https://gitee.com/zoujingli/think-plugs-wechat
|
||||
// | github 代码仓库:https://github.com/zoujingli/think-plugs-wechat
|
||||
// +----------------------------------------------------------------------
|
||||
|
||||
declare (strict_types=1);
|
||||
|
||||
namespace app\wechat\model;
|
||||
|
||||
use think\admin\Model;
|
||||
|
||||
/**
|
||||
* 微信图文详细模型
|
||||
*
|
||||
* @property int $id
|
||||
* @property int $read_num 阅读数量
|
||||
* @property int $show_cover_pic 显示封面(0不显示,1显示)
|
||||
* @property string $author 文章作者
|
||||
* @property string $content 图文内容
|
||||
* @property string $content_source_url 原文地址
|
||||
* @property string $create_at 创建时间
|
||||
* @property string $digest 摘要内容
|
||||
* @property string $local_url 永久素材URL
|
||||
* @property string $title 素材标题
|
||||
* @class WechatNewsArticle
|
||||
* @package app\wechat\model
|
||||
*/
|
||||
class WechatNewsArticle extends Model
|
||||
{
|
||||
}
|
115
app/wechat/model/WechatPaymentRecord.php
Normal file
115
app/wechat/model/WechatPaymentRecord.php
Normal file
@ -0,0 +1,115 @@
|
||||
<?php
|
||||
|
||||
// +----------------------------------------------------------------------
|
||||
// | Wechat Plugin for ThinkAdmin
|
||||
// +----------------------------------------------------------------------
|
||||
// | 版权所有 2014~2025 Anyon <zoujingli@qq.com>
|
||||
// +----------------------------------------------------------------------
|
||||
// | 官方网站: https://thinkadmin.top
|
||||
// +----------------------------------------------------------------------
|
||||
// | 开源协议 ( https://mit-license.org )
|
||||
// | 免责声明 ( https://thinkadmin.top/disclaimer )
|
||||
// +----------------------------------------------------------------------
|
||||
// | gitee 代码仓库:https://gitee.com/zoujingli/think-plugs-wechat
|
||||
// | github 代码仓库:https://github.com/zoujingli/think-plugs-wechat
|
||||
// +----------------------------------------------------------------------
|
||||
|
||||
declare (strict_types=1);
|
||||
|
||||
namespace app\wechat\model;
|
||||
|
||||
use app\wechat\service\PaymentService;
|
||||
use think\admin\Model;
|
||||
use think\model\relation\HasOne;
|
||||
|
||||
/**
|
||||
* 微信支付行为模型
|
||||
*
|
||||
* @property float $order_amount 原订单金额
|
||||
* @property float $payment_amount 实际到账金额
|
||||
* @property float $refund_amount 退款金额
|
||||
* @property int $id
|
||||
* @property int $payment_status 支付状态(0未付,1已付,2取消)
|
||||
* @property int $refund_status 退款状态(0未退,1已退)
|
||||
* @property string $appid 发起APPID
|
||||
* @property string $code 发起支付号
|
||||
* @property string $create_time 创建时间
|
||||
* @property string $openid 用户OPENID
|
||||
* @property string $order_code 原订单编号
|
||||
* @property string $order_name 原订单标题
|
||||
* @property string $payment_bank 支付银行类型
|
||||
* @property string $payment_notify 支付结果通知
|
||||
* @property string $payment_remark 支付状态备注
|
||||
* @property string $payment_time 支付完成时间
|
||||
* @property string $payment_trade 平台交易编号
|
||||
* @property string $type 交易方式
|
||||
* @property string $update_time 更新时间
|
||||
* @property-read \app\wechat\model\WechatFans $bind_fans
|
||||
* @property-read \app\wechat\model\WechatFans $fans
|
||||
* @class WechatPaymentRecord
|
||||
* @package app\wechat\model
|
||||
*/
|
||||
class WechatPaymentRecord extends Model
|
||||
{
|
||||
/**
|
||||
* 关联用户粉丝数据
|
||||
* @return \think\model\relation\HasOne
|
||||
*/
|
||||
public function fans(): HasOne
|
||||
{
|
||||
return $this->hasOne(WechatFans::class, 'openid', 'openid');
|
||||
}
|
||||
|
||||
/**
|
||||
* 绑定用户粉丝数据
|
||||
* @return \think\model\relation\HasOne
|
||||
*/
|
||||
public function bindFans(): HasOne
|
||||
{
|
||||
return $this->fans()->bind([
|
||||
'fans_headimg' => 'headimgurl',
|
||||
'fans_nickname' => 'nickname',
|
||||
]);
|
||||
}
|
||||
|
||||
/**
|
||||
* 格式化输出时间格式
|
||||
* @param mixed $value
|
||||
* @return string
|
||||
*/
|
||||
public function getCreateTimeAttr($value): string
|
||||
{
|
||||
return $value ? format_datetime($value) : '';
|
||||
}
|
||||
|
||||
/**
|
||||
* 格式化输出时间格式
|
||||
* @param mixed $value
|
||||
* @return string
|
||||
*/
|
||||
public function getUpdateTimeAttr($value): string
|
||||
{
|
||||
return $value ? format_datetime($value) : '';
|
||||
}
|
||||
|
||||
/**
|
||||
* 格式化输出时间格式
|
||||
* @param mixed $value
|
||||
* @return string
|
||||
*/
|
||||
public function getPaymentTimeAttr($value): string
|
||||
{
|
||||
return $value ? format_datetime($value) : '';
|
||||
}
|
||||
|
||||
/**
|
||||
* 转换数据类型
|
||||
* @return array
|
||||
*/
|
||||
public function toArray(): array
|
||||
{
|
||||
$data = parent::toArray();
|
||||
$data['type_name'] = PaymentService::tradeTypeNames[$data['type']] ?? $data['type'];
|
||||
return $data;
|
||||
}
|
||||
}
|
84
app/wechat/model/WechatPaymentRefund.php
Normal file
84
app/wechat/model/WechatPaymentRefund.php
Normal file
@ -0,0 +1,84 @@
|
||||
<?php
|
||||
|
||||
// +----------------------------------------------------------------------
|
||||
// | Wechat Plugin for ThinkAdmin
|
||||
// +----------------------------------------------------------------------
|
||||
// | 版权所有 2014~2025 Anyon <zoujingli@qq.com>
|
||||
// +----------------------------------------------------------------------
|
||||
// | 官方网站: https://thinkadmin.top
|
||||
// +----------------------------------------------------------------------
|
||||
// | 开源协议 ( https://mit-license.org )
|
||||
// | 免责声明 ( https://thinkadmin.top/disclaimer )
|
||||
// +----------------------------------------------------------------------
|
||||
// | gitee 代码仓库:https://gitee.com/zoujingli/think-plugs-wechat
|
||||
// | github 代码仓库:https://github.com/zoujingli/think-plugs-wechat
|
||||
// +----------------------------------------------------------------------
|
||||
|
||||
declare (strict_types=1);
|
||||
|
||||
namespace app\wechat\model;
|
||||
|
||||
use think\admin\Model;
|
||||
use think\model\relation\HasOne;
|
||||
|
||||
/**
|
||||
* 微信支付退款模型
|
||||
*
|
||||
* @property float $refund_amount 实际到账金额
|
||||
* @property int $id
|
||||
* @property int $refund_status 支付状态(0未付,1已付,2取消)
|
||||
* @property string $code 发起支付号
|
||||
* @property string $create_time 创建时间
|
||||
* @property string $record_code 子支付编号
|
||||
* @property string $refund_account 退款目标账号
|
||||
* @property string $refund_notify 退款交易通知
|
||||
* @property string $refund_remark 支付状态备注
|
||||
* @property string $refund_scode 退款状态码
|
||||
* @property string $refund_time 支付完成时间
|
||||
* @property string $refund_trade 平台交易编号
|
||||
* @property string $update_time 更新时间
|
||||
* @property-read \app\wechat\model\WechatPaymentRecord $record
|
||||
* @class WechatPaymentRefund
|
||||
* @package app\wechat\model
|
||||
*/
|
||||
class WechatPaymentRefund extends Model
|
||||
{
|
||||
/**
|
||||
* 关联支付订单
|
||||
* @return \think\model\relation\HasOne
|
||||
*/
|
||||
public function record(): HasOne
|
||||
{
|
||||
return $this->hasOne(WechatPaymentRecord::class, 'code', 'record_code')->with('bindfans');
|
||||
}
|
||||
|
||||
/**
|
||||
* 格式化输出时间格式
|
||||
* @param mixed $value
|
||||
* @return string
|
||||
*/
|
||||
public function getCreateTimeAttr($value): string
|
||||
{
|
||||
return $value ? format_datetime($value) : '';
|
||||
}
|
||||
|
||||
/**
|
||||
* 格式化输出时间格式
|
||||
* @param mixed $value
|
||||
* @return string
|
||||
*/
|
||||
public function getUpdateTimeAttr($value): string
|
||||
{
|
||||
return $value ? format_datetime($value) : '';
|
||||
}
|
||||
|
||||
/**
|
||||
* 格式化输出时间格式
|
||||
* @param mixed $value
|
||||
* @return string
|
||||
*/
|
||||
public function getRefundTimeAttr($value): string
|
||||
{
|
||||
return $value ? format_datetime($value) : '';
|
||||
}
|
||||
}
|
58
app/wechat/service/AutoService.php
Normal file
58
app/wechat/service/AutoService.php
Normal file
@ -0,0 +1,58 @@
|
||||
<?php
|
||||
|
||||
// +----------------------------------------------------------------------
|
||||
// | Wechat Plugin for ThinkAdmin
|
||||
// +----------------------------------------------------------------------
|
||||
// | 版权所有 2014~2025 Anyon <zoujingli@qq.com>
|
||||
// +----------------------------------------------------------------------
|
||||
// | 官方网站: https://thinkadmin.top
|
||||
// +----------------------------------------------------------------------
|
||||
// | 开源协议 ( https://mit-license.org )
|
||||
// | 免责声明 ( https://thinkadmin.top/disclaimer )
|
||||
// +----------------------------------------------------------------------
|
||||
// | gitee 代码仓库:https://gitee.com/zoujingli/think-plugs-wechat
|
||||
// | github 代码仓库:https://github.com/zoujingli/think-plugs-wechat
|
||||
// +----------------------------------------------------------------------
|
||||
|
||||
declare (strict_types=1);
|
||||
|
||||
namespace app\wechat\service;
|
||||
|
||||
use app\wechat\model\WechatAuto;
|
||||
use think\admin\Service;
|
||||
use think\admin\service\QueueService;
|
||||
|
||||
/**
|
||||
* 关注自动回复服务
|
||||
* @class AutoService
|
||||
* @package app\wechat\service
|
||||
*/
|
||||
class AutoService extends Service
|
||||
{
|
||||
/**
|
||||
* 注册微信用户推送任务
|
||||
* @param string $openid
|
||||
* @throws \think\admin\Exception
|
||||
*/
|
||||
public static function register(string $openid)
|
||||
{
|
||||
foreach (WechatAuto::mk()->where(['status' => 1])->order('time asc')->cursor() as $vo) {
|
||||
[$name, $time] = ["推送客服消息 {$vo['code']}#{$openid}", static::parseTimeString($vo['time'])];
|
||||
QueueService::register($name, "xadmin:fansmsg {$openid} {$vo['code']}", $time);
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* 解析配置时间格式
|
||||
* @param string $time
|
||||
* @return int
|
||||
*/
|
||||
private static function parseTimeString(string $time): int
|
||||
{
|
||||
if (preg_match('|^.*?(\d{2}).*?(\d{2}).*?(\d{2}).*?$|', $time, $vars)) {
|
||||
return intval($vars[1]) * 3600 * intval($vars[2]) * 60 + intval($vars[3]);
|
||||
} else {
|
||||
return 0;
|
||||
}
|
||||
}
|
||||
}
|
63
app/wechat/service/FansService.php
Normal file
63
app/wechat/service/FansService.php
Normal file
@ -0,0 +1,63 @@
|
||||
<?php
|
||||
|
||||
// +----------------------------------------------------------------------
|
||||
// | Wechat Plugin for ThinkAdmin
|
||||
// +----------------------------------------------------------------------
|
||||
// | 版权所有 2014~2025 Anyon <zoujingli@qq.com>
|
||||
// +----------------------------------------------------------------------
|
||||
// | 官方网站: https://thinkadmin.top
|
||||
// +----------------------------------------------------------------------
|
||||
// | 开源协议 ( https://mit-license.org )
|
||||
// | 免责声明 ( https://thinkadmin.top/disclaimer )
|
||||
// +----------------------------------------------------------------------
|
||||
// | gitee 代码仓库:https://gitee.com/zoujingli/think-plugs-wechat
|
||||
// | github 代码仓库:https://github.com/zoujingli/think-plugs-wechat
|
||||
// +----------------------------------------------------------------------
|
||||
|
||||
declare (strict_types=1);
|
||||
|
||||
namespace app\wechat\service;
|
||||
|
||||
use app\wechat\model\WechatFans;
|
||||
use think\admin\Library;
|
||||
use think\admin\Service;
|
||||
|
||||
/**
|
||||
* 微信粉丝信息
|
||||
* @class FansService
|
||||
* @package app\wechat\service
|
||||
*/
|
||||
class FansService extends Service
|
||||
{
|
||||
|
||||
/**
|
||||
* 增加或更新粉丝信息
|
||||
* @param array $user 粉丝信息
|
||||
* @param string $appid 微信APPID
|
||||
* @return boolean
|
||||
*/
|
||||
public static function set(array $user, string $appid = ''): bool
|
||||
{
|
||||
if (isset($user['subscribe_time'])) {
|
||||
$user['subscribe_at'] = date('Y-m-d H:i:s', $user['subscribe_time']);
|
||||
}
|
||||
if (isset($user['tagid_list']) && is_array($user['tagid_list'])) {
|
||||
$user['tagid_list'] = arr2str($user['tagid_list']);
|
||||
}
|
||||
if ($appid !== '') $user['appid'] = $appid;
|
||||
unset($user['privilege'], $user['groupid']);
|
||||
foreach ($user as $k => $v) if ($v === '') unset($user[$k]);
|
||||
Library::$sapp->event->trigger('WechatFansUpdate', $user);
|
||||
return !!WechatFans::mUpdate($user, 'openid');
|
||||
}
|
||||
|
||||
/**
|
||||
* 获取粉丝信息
|
||||
* @param string $openid
|
||||
* @return array
|
||||
*/
|
||||
public static function get(string $openid): array
|
||||
{
|
||||
return WechatFans::mk()->where(['openid' => $openid])->findOrEmpty()->toArray();
|
||||
}
|
||||
}
|
107
app/wechat/service/LoginService.php
Normal file
107
app/wechat/service/LoginService.php
Normal file
@ -0,0 +1,107 @@
|
||||
<?php
|
||||
|
||||
// +----------------------------------------------------------------------
|
||||
// | Wechat Plugin for ThinkAdmin
|
||||
// +----------------------------------------------------------------------
|
||||
// | 版权所有 2014~2025 Anyon <zoujingli@qq.com>
|
||||
// +----------------------------------------------------------------------
|
||||
// | 官方网站: https://thinkadmin.top
|
||||
// +----------------------------------------------------------------------
|
||||
// | 开源协议 ( https://mit-license.org )
|
||||
// | 免责声明 ( https://thinkadmin.top/disclaimer )
|
||||
// +----------------------------------------------------------------------
|
||||
// | gitee 代码仓库:https://gitee.com/zoujingli/think-plugs-wechat
|
||||
// | github 代码仓库:https://github.com/zoujingli/think-plugs-wechat
|
||||
// +----------------------------------------------------------------------
|
||||
|
||||
declare (strict_types=1);
|
||||
|
||||
namespace app\wechat\service;
|
||||
|
||||
use think\admin\Library;
|
||||
|
||||
/**
|
||||
* 微信扫码登录服务
|
||||
* @class LoginService
|
||||
* @package app\wechat\service
|
||||
*/
|
||||
class LoginService
|
||||
{
|
||||
private const expire = 3600;
|
||||
private const prefix = 'wxlogin';
|
||||
|
||||
/**
|
||||
* 生成请求编号
|
||||
* @return string
|
||||
*/
|
||||
public static function gcode(): string
|
||||
{
|
||||
return md5(uniqid(strval(rand(0, 10000)), true));
|
||||
}
|
||||
|
||||
/**
|
||||
* 生成授权码
|
||||
* @param string $code 请求编号
|
||||
* @return string
|
||||
*/
|
||||
public static function gauth(string $code): string
|
||||
{
|
||||
return self::prefix . md5($code);
|
||||
}
|
||||
|
||||
/**
|
||||
* 生成授权二维码
|
||||
* @param string $code 请求编号
|
||||
* @param integer $mode 授权模式
|
||||
* @param boolean|string $domain
|
||||
* @return array
|
||||
*/
|
||||
public static function qrcode(string $code, int $mode = 0, $domain = true): array
|
||||
{
|
||||
$data = ['auth' => self::gauth($code), 'mode' => $mode];
|
||||
$image = MediaService::getQrcode(sysuri('wechat/api.login/oauth', $data, false, $domain));
|
||||
return ['code' => $code, 'auth' => $data['auth'], 'image' => $image->getDataUri()];
|
||||
}
|
||||
|
||||
/**
|
||||
* 发起网页授权处理
|
||||
* @param string $auth 授权编号
|
||||
* @param integer $mode 授权模式
|
||||
* @return boolean
|
||||
* @throws \WeChat\Exceptions\InvalidResponseException
|
||||
* @throws \WeChat\Exceptions\LocalCacheException
|
||||
* @throws \think\admin\Exception
|
||||
*/
|
||||
public static function oauth(string $auth = '', int $mode = 0): bool
|
||||
{
|
||||
if (stripos($auth, self::prefix) === 0) {
|
||||
$url = Library::$sapp->request->url(true);
|
||||
$fans = WechatService::getWebOauthInfo($url, $mode);
|
||||
if (isset($fans['openid'])) {
|
||||
Library::$sapp->cache->set($auth, $fans, self::expire);
|
||||
return true;
|
||||
}
|
||||
}
|
||||
return false;
|
||||
}
|
||||
|
||||
/**
|
||||
* 检查是否授权
|
||||
* @param string $code 请求编号
|
||||
* @return ?array
|
||||
*/
|
||||
public static function query(string $code): ?array
|
||||
{
|
||||
return Library::$sapp->cache->get(self::gauth($code));
|
||||
}
|
||||
|
||||
/**
|
||||
* 删除授权缓存
|
||||
* @param string $code
|
||||
* @return bool
|
||||
*/
|
||||
public static function remove(string $code): bool
|
||||
{
|
||||
return Library::$sapp->cache->delete(self::gauth($code));
|
||||
}
|
||||
}
|
124
app/wechat/service/MediaService.php
Normal file
124
app/wechat/service/MediaService.php
Normal file
@ -0,0 +1,124 @@
|
||||
<?php
|
||||
|
||||
// +----------------------------------------------------------------------
|
||||
// | Wechat Plugin for ThinkAdmin
|
||||
// +----------------------------------------------------------------------
|
||||
// | 版权所有 2014~2025 Anyon <zoujingli@qq.com>
|
||||
// +----------------------------------------------------------------------
|
||||
// | 官方网站: https://thinkadmin.top
|
||||
// +----------------------------------------------------------------------
|
||||
// | 开源协议 ( https://mit-license.org )
|
||||
// | 免责声明 ( https://thinkadmin.top/disclaimer )
|
||||
// +----------------------------------------------------------------------
|
||||
// | gitee 代码仓库:https://gitee.com/zoujingli/think-plugs-wechat
|
||||
// | github 代码仓库:https://github.com/zoujingli/think-plugs-wechat
|
||||
// +----------------------------------------------------------------------
|
||||
|
||||
declare (strict_types=1);
|
||||
|
||||
namespace app\wechat\service;
|
||||
|
||||
use app\wechat\model\WechatMedia;
|
||||
use app\wechat\model\WechatNews;
|
||||
use app\wechat\model\WechatNewsArticle;
|
||||
use Endroid\QrCode\Builder\Builder;
|
||||
use Endroid\QrCode\Encoding\Encoding;
|
||||
use Endroid\QrCode\ErrorCorrectionLevel\ErrorCorrectionLevelHigh;
|
||||
use Endroid\QrCode\RoundBlockSizeMode\RoundBlockSizeModeMargin;
|
||||
use Endroid\QrCode\Writer\PngWriter;
|
||||
use Endroid\QrCode\Writer\Result\ResultInterface;
|
||||
use think\admin\Service;
|
||||
use think\admin\Storage;
|
||||
use WeChat\Contracts\MyCurlFile;
|
||||
|
||||
/**
|
||||
* 微信素材管理
|
||||
* @class MediaService
|
||||
* @package app\wechat\service
|
||||
*/
|
||||
class MediaService extends Service
|
||||
{
|
||||
/**
|
||||
* 通过图文ID读取图文信息
|
||||
* @param mixed $id 本地图文ID
|
||||
* @param mixed $map 额外的查询条件
|
||||
* @return array
|
||||
* @throws \think\db\exception\DataNotFoundException
|
||||
* @throws \think\db\exception\DbException
|
||||
* @throws \think\db\exception\ModelNotFoundException
|
||||
*/
|
||||
public static function news($id, $map = []): array
|
||||
{
|
||||
// 文章主体数据
|
||||
$data = WechatNews::mk()->where(['id' => $id, 'is_deleted' => 0])->where($map)->findOrEmpty()->toArray();
|
||||
if (empty($data)) return [];
|
||||
|
||||
// 文章内容编号
|
||||
$data['articles'] = [];
|
||||
$aids = $data['articleids'] = str2arr($data['article_id']);
|
||||
if (empty($aids)) return $data;
|
||||
|
||||
// 文章内容列表
|
||||
$items = WechatNewsArticle::mk()->whereIn('id', $aids)->withoutField('create_by,create_at')->select()->toArray();
|
||||
foreach ($aids as $aid) foreach ($items as $item) if (intval($item['id']) === intval($aid)) $data['articles'][] = $item;
|
||||
|
||||
// 返回文章内容
|
||||
return $data;
|
||||
}
|
||||
|
||||
/**
|
||||
* 上传图片永久素材
|
||||
* @param string $url 文件地址
|
||||
* @param string $type 文件类型
|
||||
* @param array $video 视频信息
|
||||
* @return string media_id
|
||||
* @throws \WeChat\Exceptions\InvalidResponseException
|
||||
* @throws \WeChat\Exceptions\LocalCacheException
|
||||
* @throws \think\admin\Exception
|
||||
*/
|
||||
public static function upload(string $url, string $type = 'image', array $video = []): string
|
||||
{
|
||||
$map = ['md5' => md5($url), 'appid' => WechatService::getAppid()];
|
||||
if (($mediaId = WechatMedia::mk()->where($map)->value('media_id'))) return $mediaId;
|
||||
$result = WechatService::WeChatMedia()->addMaterial(static::buildCurlFile($url), $type, $video);
|
||||
WechatMedia::mUpdate([
|
||||
'md5' => $map['md5'],
|
||||
'type' => $type,
|
||||
'appid' => $map['appid'],
|
||||
'media_id' => $result['media_id'],
|
||||
'media_url' => $result['url'] ?? '',
|
||||
'local_url' => $url,
|
||||
], 'type', $map);
|
||||
return $result['media_id'];
|
||||
}
|
||||
|
||||
/**
|
||||
* 创建 CURL 文件对象
|
||||
* @param string $local 文件路径或网络地址
|
||||
* @return MyCurlFile
|
||||
* @throws \WeChat\Exceptions\LocalCacheException
|
||||
*/
|
||||
private static function buildCurlFile(string $local): MyCurlFile
|
||||
{
|
||||
if (file_exists($local)) {
|
||||
return new MyCurlFile($local);
|
||||
} else {
|
||||
return new MyCurlFile(Storage::down($local)['file']);
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* 获取二维码内容接口
|
||||
* @param string $text 二维码文本内容
|
||||
* @return \Endroid\QrCode\Writer\Result\ResultInterface
|
||||
*/
|
||||
public static function getQrcode(string $text): ResultInterface
|
||||
{
|
||||
return Builder::create()->data($text)->size(300)->margin(15)
|
||||
->writer(new PngWriter())->encoding(new Encoding('UTF-8'))
|
||||
->writerOptions([])->validateResult(false)
|
||||
->roundBlockSizeMode(new RoundBlockSizeModeMargin())
|
||||
->errorCorrectionLevel(new ErrorCorrectionLevelHigh())
|
||||
->build();
|
||||
}
|
||||
}
|
406
app/wechat/service/PaymentService.php
Normal file
406
app/wechat/service/PaymentService.php
Normal file
@ -0,0 +1,406 @@
|
||||
<?php
|
||||
|
||||
// +----------------------------------------------------------------------
|
||||
// | Wechat Plugin for ThinkAdmin
|
||||
// +----------------------------------------------------------------------
|
||||
// | 版权所有 2014~2025 Anyon <zoujingli@qq.com>
|
||||
// +----------------------------------------------------------------------
|
||||
// | 官方网站: https://thinkadmin.top
|
||||
// +----------------------------------------------------------------------
|
||||
// | 开源协议 ( https://mit-license.org )
|
||||
// | 免责声明 ( https://thinkadmin.top/disclaimer )
|
||||
// +----------------------------------------------------------------------
|
||||
// | gitee 代码仓库:https://gitee.com/zoujingli/think-plugs-wechat
|
||||
// | github 代码仓库:https://github.com/zoujingli/think-plugs-wechat
|
||||
// +----------------------------------------------------------------------
|
||||
|
||||
declare (strict_types=1);
|
||||
|
||||
namespace app\wechat\service;
|
||||
|
||||
use app\wechat\model\WechatPaymentRecord;
|
||||
use app\wechat\model\WechatPaymentRefund;
|
||||
use think\admin\Exception;
|
||||
use think\admin\extend\CodeExtend;
|
||||
use think\admin\Library;
|
||||
use think\Response;
|
||||
use WePayV3\Order;
|
||||
|
||||
/**
|
||||
* 微信V3支付服务
|
||||
* @class PaymentService
|
||||
* @package app\wechat\service
|
||||
*/
|
||||
class PaymentService
|
||||
{
|
||||
// 微信支付类型
|
||||
public const WECHAT_APP = 'wechat_app';
|
||||
public const WECHAT_GZH = 'wechat_gzh';
|
||||
public const WECHAT_XCX = 'wechat_xcx';
|
||||
public const WECHAT_WAP = 'wechat_wap';
|
||||
public const WECHAT_QRC = 'wechat_qrc';
|
||||
|
||||
// 微信支付类型转换
|
||||
private const tradeTypes = [
|
||||
self::WECHAT_APP => 'APP',
|
||||
self::WECHAT_WAP => 'MWEB',
|
||||
self::WECHAT_GZH => 'JSAPI',
|
||||
self::WECHAT_XCX => 'JSAPI',
|
||||
self::WECHAT_QRC => 'NATIVE',
|
||||
];
|
||||
|
||||
// 微信支付类型名称
|
||||
public const tradeTypeNames = [
|
||||
self::WECHAT_APP => '微信APP支付',
|
||||
self::WECHAT_WAP => '微信H5支付',
|
||||
self::WECHAT_GZH => '服务号支付',
|
||||
self::WECHAT_XCX => '小程序支付',
|
||||
self::WECHAT_QRC => '二维码支付',
|
||||
];
|
||||
|
||||
/**
|
||||
* 创建微信支付订单
|
||||
* @param string $openid 用户标识
|
||||
* @param string $oCode 订单单号
|
||||
* @param string $oName 订单标题
|
||||
* @param string $oAmount 订单金额(元)
|
||||
* @param string $pType 支付类型
|
||||
* @param ?string $pAmount 支付金额(元)
|
||||
* @param ?string $pRemark 支付描述
|
||||
* @return array
|
||||
* @throws \think\admin\Exception
|
||||
*/
|
||||
public static function create(string $openid, string $oCode, string $oName, string $oAmount, string $pType, ?string $pAmount = null, ?string $pRemark = null): array
|
||||
{
|
||||
try {
|
||||
// 检查订单是否完成
|
||||
if (self::isPayed($oCode, $oAmount, $oPayed)) {
|
||||
return ['code' => 1, 'info' => '已完成支付!', 'data' => [], 'params' => []];
|
||||
}
|
||||
// 检查剩余支付金额
|
||||
$pAmount = floatval(is_null($pAmount) ? (floatval($oAmount) - $oPayed) : $pAmount);
|
||||
if ($oPayed + $pAmount > floatval($oAmount)) {
|
||||
return ['code' => 0, 'info' => '支付总额超出!', 'data' => [], 'params' => []];
|
||||
}
|
||||
$config = WechatService::getConfig(true);
|
||||
do $pCode = CodeExtend::uniqidNumber(16, 'P');
|
||||
while (WechatPaymentRecord::mk()->master()->where(['code' => $pCode])->findOrEmpty()->isExists());
|
||||
$data = [
|
||||
'appid' => $config['appid'],
|
||||
'mchid' => $config['mch_id'],
|
||||
'payer' => ['openid' => $openid],
|
||||
'amount' => ['total' => intval($pAmount * 100), 'currency' => 'CNY'],
|
||||
'notify_url' => static::withNotifyUrl($pCode),
|
||||
'description' => empty($pRemark) ? $oName : ($oName . '-' . $pRemark),
|
||||
'out_trade_no' => $pCode,
|
||||
];
|
||||
$tradeType = static::tradeTypes[$pType] ?? '';
|
||||
if (in_array($pType, [static::WECHAT_WAP, static::WECHAT_QRC])) {
|
||||
unset($data['payer']);
|
||||
}
|
||||
if ($pType === static::WECHAT_WAP) {
|
||||
$tradeType = 'h5';
|
||||
$data['scene_info'] = ['h5_info' => ['type' => 'Wap'], 'payer_client_ip' => request()->ip()];
|
||||
}
|
||||
$params = static::withPayment($config)->create(strtolower($tradeType), $data);
|
||||
// 创建支付记录
|
||||
static::createPaymentAction($openid, $oCode, $oName, $oAmount, $pType, $pCode, strval($pAmount));
|
||||
// 返回支付参数
|
||||
return ['code' => 1, 'info' => '创建支付成功', 'data' => $data, 'params' => $params];
|
||||
} catch (Exception $exception) {
|
||||
throw $exception;
|
||||
} catch (\Exception $exception) {
|
||||
throw new Exception($exception->getMessage(), $exception->getCode());
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* 查询微信支付订单
|
||||
* @param string $pCode 订单单号
|
||||
* @return array
|
||||
*/
|
||||
public static function query(string $pCode): array
|
||||
{
|
||||
try {
|
||||
$result = static::withPayment()->query($pCode);
|
||||
if (isset($result['trade_state']) && $result['trade_state'] === 'SUCCESS') {
|
||||
$extra = [
|
||||
'openid' => $result['payer']['openid'] ?? null,
|
||||
'payment_bank' => $result['bank_type'],
|
||||
'payment_time' => date('Y-m-d H:i:s', strtotime($result['success_time'])),
|
||||
'payment_remark' => $result['trade_state_desc'] ?? null,
|
||||
'payment_notify' => json_encode($result, 64 | 256),
|
||||
];
|
||||
$pAmount = strval($result['amount']['total'] / 100);
|
||||
static::updatePaymentAction($result['out_trade_no'], $result['transaction_id'], $pAmount, $extra);
|
||||
}
|
||||
return $result;
|
||||
} catch (\Exception $exception) {
|
||||
return ['trade_state' => 'ERROR', 'trade_state_desc' => $exception->getMessage()];
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* 支付结果处理
|
||||
* @param array|null $data
|
||||
* @return \think\Response
|
||||
*/
|
||||
public static function notify(?array $data = null): Response
|
||||
{
|
||||
try {
|
||||
p(Library::$sapp->request->post(), false, 'wechat_pay_notify');
|
||||
$notify = static::withPayment()->notify(Library::$sapp->request->post());
|
||||
$result = empty($notify['result']) ? [] : json_decode($notify['result'], true);
|
||||
p($result, false, 'wechat_pay_notify');
|
||||
p('------------------', false, 'wechat_pay_notify');
|
||||
if (empty($result) || !is_array($result)) {
|
||||
empty($data['order']) || self::query($data['order']);
|
||||
return response('error', 500);
|
||||
}
|
||||
//订单支付通知处理
|
||||
if ($data['scen'] === 'order' && isset($result['trade_state']) && $result['trade_state'] == 'SUCCESS') {
|
||||
if ($data['order'] !== $result['out_trade_no']) return response('error', 500);
|
||||
$extra = [
|
||||
'openid' => $result['payer']['openid'] ?? null,
|
||||
'payment_bank' => $result['bank_type'],
|
||||
'payment_time' => date('Y-m-d H:i:s', strtotime($result['success_time'])),
|
||||
'payment_remark' => $result['trade_state_desc'] ?? null,
|
||||
'payment_notify' => json_encode($result, 64 | 256),
|
||||
];
|
||||
$pAmount = strval($result['amount']['payer_total'] / 100);
|
||||
if (!static::updatePaymentAction($result['out_trade_no'], $result['transaction_id'], $pAmount, $extra)) {
|
||||
return response('error', 500);
|
||||
}
|
||||
} elseif ($data['scen'] === 'refund' && isset($result['refund_status']) && $result['refund_status'] == 'SUCCESS') {
|
||||
if ($data['order'] !== $result['out_refund_no']) return response('error', 500);
|
||||
$refund = WechatPaymentRefund::mk()->where(['code' => $result['out_refund_no']])->findOrEmpty();
|
||||
if ($refund->isEmpty()) return response('error', 500);
|
||||
$refund->save([
|
||||
'refund_time' => date('Y-m-d H:i:s', strtotime($result['success_time'])),
|
||||
'refund_trade' => $result['refund_id'],
|
||||
'refund_scode' => $result['refund_status'],
|
||||
'refund_status' => 1,
|
||||
'refund_notify' => json_encode($result, 64 | 256),
|
||||
'refund_account' => $result['user_received_account'] ?? '',
|
||||
]);
|
||||
static::refundSync($refund->getAttr('record_code'));
|
||||
}
|
||||
return response('success');
|
||||
} catch (\Exception $exception) {
|
||||
empty($data['order']) || self::query($data['order']);
|
||||
return json(['code' => 'FAIL', 'message' => $exception->getMessage()])->code(500);
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* 创建支付单退款
|
||||
* @param string $pcode 支付单号
|
||||
* @param string $amount 退款金额
|
||||
* @param string $reason 退款原因
|
||||
* @return array
|
||||
* @throws \WeChat\Exceptions\InvalidResponseException
|
||||
* @throws \WeChat\Exceptions\LocalCacheException
|
||||
* @throws \think\admin\Exception
|
||||
*/
|
||||
public static function refund(string $pcode, string $amount, string $reason = ''): array
|
||||
{
|
||||
// 同步已退款状态
|
||||
$record = static::refundSync($pcode);
|
||||
if ($record->getAttr('refund_amount') >= $record->getAttr('payment_amount')) {
|
||||
return [1, '该订单已完成退款!'];
|
||||
}
|
||||
if ($record->getAttr('refund_amount') + floatval($amount) > $record->getAttr('payment_amount')) {
|
||||
return [0, '退款大于支付金额!'];
|
||||
}
|
||||
// 创建支付退款申请
|
||||
do $check = ['code' => $rcode = CodeExtend::uniqidNumber(16, 'R')];
|
||||
while (($model = WechatPaymentRefund::mk()->master()->where($check)->findOrEmpty())->isExists());
|
||||
// 初始化退款申请记录
|
||||
$model->save(['code' => $rcode, 'record_code' => $pcode, 'refund_amount' => $amount, 'refund_remark' => $reason]);
|
||||
$options = [
|
||||
'out_trade_no' => $pcode,
|
||||
'out_refund_no' => $rcode,
|
||||
'notify_url' => static::withNotifyUrl($rcode, 'refund'),
|
||||
'amount' => [
|
||||
'refund' => intval(floatval($amount) * 100),
|
||||
'total' => intval($record->getAttr('payment_amount') * 100),
|
||||
'currency' => 'CNY'
|
||||
]
|
||||
];
|
||||
if (strlen($reason) > 0) $options['reason'] = $reason;
|
||||
$result = static::withPayment()->createRefund($options);
|
||||
if (in_array($result['code'] ?? $result['status'], ['SUCCESS', 'PROCESSING'])) {
|
||||
return self::refundSyncByQuery($rcode);
|
||||
} else {
|
||||
return [0, $result['message'] ?? $result['status']];
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* 同步退款统计状态
|
||||
* @param string $pCode
|
||||
* @return \app\wechat\model\WechatPaymentRecord
|
||||
* @throws \WeChat\Exceptions\InvalidResponseException
|
||||
* @throws \WeChat\Exceptions\LocalCacheException
|
||||
* @throws \think\admin\Exception
|
||||
*/
|
||||
public static function refundSync(string $pCode): WechatPaymentRecord
|
||||
{
|
||||
$record = WechatPaymentRecord::mk()->where(['code' => $pCode])->findOrEmpty();
|
||||
if ($record->isEmpty()) throw new Exception('支付单不存在!');
|
||||
if ($record->getAttr('payment_status') < 1) throw new Exception("支付未完成!");
|
||||
// 最近一条记录,同步查询刷新
|
||||
$map = ['record_code' => $pCode];
|
||||
$last = WechatPaymentRefund::mk()->where($map)->order('id desc')->findOrEmpty();
|
||||
if ($last->isExists() && $last->getAttr('refund_status') === 0) {
|
||||
static::refundSyncByQuery($last->getAttr('code'));
|
||||
}
|
||||
// 统计刷新退款金额
|
||||
$where = ['record_code' => $pCode, 'refund_status' => 1];
|
||||
$amount = WechatPaymentRefund::mk()->where($where)->sum('refund_amount');
|
||||
$record->save(['refund_amount' => $amount, 'refund_status' => intval($amount > 0)]);
|
||||
return $record;
|
||||
}
|
||||
|
||||
/**
|
||||
* 同步退款状态
|
||||
* @param string $rCode
|
||||
* @return array
|
||||
* @throws \WeChat\Exceptions\InvalidResponseException
|
||||
* @throws \WeChat\Exceptions\LocalCacheException
|
||||
* @throws \think\admin\Exception
|
||||
*/
|
||||
public static function refundSyncByQuery(string $rCode): array
|
||||
{
|
||||
$refund = WechatPaymentRefund::mk()->where(['code' => $rCode])->findOrEmpty();
|
||||
if ($refund->isEmpty()) return [0, '退款不存在!'];
|
||||
if ($refund->getAttr('refund_status')) return [1, '退款已完成!'];
|
||||
$result = static::withPayment()->queryRefund($rCode);
|
||||
$extra = [
|
||||
'refund_trade' => $result['refund_id'],
|
||||
'refund_scode' => $result['status'],
|
||||
'refund_status' => intval($result['status'] === 'SUCCESS'),
|
||||
'refund_notify' => json_encode($result, 64 | 256),
|
||||
'refund_account' => $result['user_received_account'] ?? '',
|
||||
];
|
||||
if (isset($result['success_time'])) {
|
||||
$extra['refund_time'] = date('Y-m-d H:i:s', strtotime($result['success_time']));
|
||||
}
|
||||
$refund->save($extra);
|
||||
// 同步支付订单
|
||||
static::refundSync($refund->getAttr('record_code'));
|
||||
if ($result['status'] === 'SUCCESS') return [1, '退款已完成!'];
|
||||
if ($result['status'] === 'PROCESSING') return [1, '退款处理中!'];
|
||||
return [0, $result['message'] ?? $result['status']];
|
||||
}
|
||||
|
||||
/**
|
||||
* 判断是否完成支付
|
||||
* @param string $oCode 原订单单号
|
||||
* @param string $oAmount 需支付金额
|
||||
* @param ?float $oPayed 已支付金额[赋值]
|
||||
* @return boolean
|
||||
*/
|
||||
public static function isPayed(string $oCode, string $oAmount, ?float &$oPayed = null): bool
|
||||
{
|
||||
return self::withPayed($oCode, $oPayed) >= $oAmount;
|
||||
}
|
||||
|
||||
/**
|
||||
* 获取已支付金额
|
||||
* @param string $oCode 原订单单号
|
||||
* @param ?float $oPayed 已支付金额[赋值]
|
||||
* @return float
|
||||
*/
|
||||
public static function withPayed(string $oCode, ?float &$oPayed = null): float
|
||||
{
|
||||
$where = ['order_code' => $oCode, 'payment_status' => 1];
|
||||
return $oPayed = WechatPaymentRecord::mk()->where($where)->sum('payment_amount');
|
||||
}
|
||||
|
||||
/**
|
||||
* 初始化支付实现
|
||||
* @param array|null $config
|
||||
* @return \WePayV3\Order
|
||||
* @throws \WeChat\Exceptions\InvalidResponseException
|
||||
* @throws \WeChat\Exceptions\LocalCacheException
|
||||
* @throws \think\admin\Exception
|
||||
*/
|
||||
protected static function withPayment(?array $config = null): Order
|
||||
{
|
||||
return Order::instance($config ?: WechatService::getConfig(true));
|
||||
}
|
||||
|
||||
/**
|
||||
* 获取支付通知地址
|
||||
* @param string $order 订单单号
|
||||
* @param string $scene 支付场景
|
||||
* @param array $extra
|
||||
* @return string
|
||||
*/
|
||||
protected static function withNotifyUrl(string $order, string $scene = 'order', array $extra = []): string
|
||||
{
|
||||
$data = ['scen' => $scene, 'order' => $order];
|
||||
$vars = CodeExtend::enSafe64(json_encode($extra + $data, 64 | 256));
|
||||
return sysuri('@plugin-wxpay-notify', [], false, true) . "/{$vars}";
|
||||
}
|
||||
|
||||
/**
|
||||
* 创建支付行为
|
||||
* @param string $openid 用户编号
|
||||
* @param string $oCode 订单单号
|
||||
* @param string $oName 订单标题
|
||||
* @param string $oAmount 订单总金额
|
||||
* @param string $pType 支付平台
|
||||
* @param string $pCode 子支付单号
|
||||
* @param string $pAmount 子支付金额
|
||||
* @return array
|
||||
* @throws \think\admin\Exception
|
||||
*/
|
||||
protected static function createPaymentAction(string $openid, string $oCode, string $oName, string $oAmount, string $pType, string $pCode, string $pAmount): array
|
||||
{
|
||||
// 检查是否已经支付
|
||||
if (static::withPayed($oCode, $oPayed) >= floatval($oAmount)) {
|
||||
throw new Exception("已经完成支付", 1);
|
||||
}
|
||||
if ($oPayed + floatval($pAmount) > floatval($oAmount)) {
|
||||
throw new Exception('总支付超出金额', 0);
|
||||
}
|
||||
$map = ['order_code' => $oCode, 'payment_status' => 1];
|
||||
$model = WechatPaymentRecord::mk()->where($map)->findOrEmpty();
|
||||
if ($model->isExists()) throw new Exception("已经完成支付", 1);
|
||||
// 写入订单支付行为
|
||||
$model->save([
|
||||
'type' => $pType,
|
||||
'code' => $pCode,
|
||||
'appid' => WechatService::getAppid(),
|
||||
'openid' => $openid,
|
||||
'order_code' => $oCode,
|
||||
'order_name' => $oName,
|
||||
'order_amount' => $oAmount,
|
||||
]);
|
||||
return $model->toArray();
|
||||
}
|
||||
|
||||
/**
|
||||
* 更新创建支付行为
|
||||
* @param string $pCode 商户订单单号
|
||||
* @param string $pTrade 平台交易单号
|
||||
* @param string $pAmount 实际到账金额
|
||||
* @param array $extra 订单扩展数据
|
||||
* @return boolean|array
|
||||
*/
|
||||
protected static function updatePaymentAction(string $pCode, string $pTrade, string $pAmount, array $extra = [])
|
||||
{
|
||||
// 更新支付记录
|
||||
$model = WechatPaymentRecord::mk()->where(['code' => $pCode])->findOrEmpty();
|
||||
if ($model->isEmpty()) return false;
|
||||
// 更新支付行为
|
||||
foreach ($extra as $k => $v) if (is_null($v)) unset($extra[$k]);
|
||||
$model->save($extra + ['payment_trade' => $pTrade, 'payment_status' => 1, 'payment_amount' => $pAmount]);
|
||||
// 触发支付成功事件
|
||||
Library::$sapp->event->trigger('WechatPaymentSuccess', $model->refresh()->toArray());
|
||||
// 返回支付行为数据
|
||||
return $model->toArray();
|
||||
}
|
||||
}
|
366
app/wechat/service/WechatService.php
Normal file
366
app/wechat/service/WechatService.php
Normal file
@ -0,0 +1,366 @@
|
||||
<?php
|
||||
|
||||
// +----------------------------------------------------------------------
|
||||
// | Wechat Plugin for ThinkAdmin
|
||||
// +----------------------------------------------------------------------
|
||||
// | 版权所有 2014~2025 Anyon <zoujingli@qq.com>
|
||||
// +----------------------------------------------------------------------
|
||||
// | 官方网站: https://thinkadmin.top
|
||||
// +----------------------------------------------------------------------
|
||||
// | 开源协议 ( https://mit-license.org )
|
||||
// | 免责声明 ( https://thinkadmin.top/disclaimer )
|
||||
// +----------------------------------------------------------------------
|
||||
// | gitee 代码仓库:https://gitee.com/zoujingli/think-plugs-wechat
|
||||
// | github 代码仓库:https://github.com/zoujingli/think-plugs-wechat
|
||||
// +----------------------------------------------------------------------
|
||||
|
||||
namespace app\wechat\service;
|
||||
|
||||
use think\admin\Exception;
|
||||
use think\admin\extend\JsonRpcClient;
|
||||
use think\admin\Library;
|
||||
use think\admin\Service;
|
||||
use think\admin\storage\LocalStorage;
|
||||
use think\exception\HttpResponseException;
|
||||
use think\Response;
|
||||
|
||||
/**
|
||||
* 微信接口调度服务
|
||||
* @class WechatService
|
||||
* @package app\wechat\serivce
|
||||
*
|
||||
* @method \WeChat\Card WeChatCard() static 微信卡券管理
|
||||
* @method \WeChat\Custom WeChatCustom() static 微信客服消息
|
||||
* @method \WeChat\Limit WeChatLimit() static 接口调用频次限制
|
||||
* @method \WeChat\Media WeChatMedia() static 微信素材管理
|
||||
* @method \WeChat\Draft WeChatDraft() static 微信草稿箱管理
|
||||
* @method \WeChat\Menu WeChatMenu() static 微信菜单管理
|
||||
* @method \WeChat\Oauth WeChatOauth() static 微信网页授权
|
||||
* @method \WeChat\Pay WeChatPay() static 微信支付商户
|
||||
* @method \WeChat\Product WeChatProduct() static 微信商店管理
|
||||
* @method \WeChat\Qrcode WeChatQrcode() static 微信二维码管理
|
||||
* @method \WeChat\Receive WeChatReceive() static 微信推送管理
|
||||
* @method \WeChat\Scan WeChatScan() static 微信扫一扫接入管理
|
||||
* @method \WeChat\Script WeChatScript() static 微信前端支持
|
||||
* @method \WeChat\Shake WeChatShake() static 微信揺一揺周边
|
||||
* @method \WeChat\Tags WeChatTags() static 微信用户标签管理
|
||||
* @method \WeChat\Template WeChatTemplate() static 微信模板消息
|
||||
* @method \WeChat\User WeChatUser() static 微信粉丝管理
|
||||
* @method \WeChat\Wifi WeChatWifi() static 微信门店WIFI管理
|
||||
* @method \WeChat\Freepublish WeChatFreepublish() static 发布能力
|
||||
*
|
||||
* ----- WeMini -----
|
||||
* @method \WeMini\Account WeMiniAccount() static 小程序账号管理
|
||||
* @method \WeMini\Basic WeMiniBasic() static 小程序基础信息设置
|
||||
* @method \WeMini\Code WeMiniCode() static 小程序代码管理
|
||||
* @method \WeMini\Domain WeMiniDomain() static 小程序域名管理
|
||||
* @method \WeMini\Tester WeMinitester() static 小程序成员管理
|
||||
* @method \WeMini\User WeMiniUser() static 小程序帐号管理
|
||||
* --------------------
|
||||
* @method \WeMini\Crypt WeMiniCrypt() static 小程序数据加密处理
|
||||
* @method \WeMini\Delivery WeMiniDelivery() static 小程序即时配送
|
||||
* @method \WeMini\Guide WeMiniGuide() static 小程序导购助手
|
||||
* @method \WeMini\Image WeMiniImage() static 小程序图像处理
|
||||
* @method \WeMini\Live WeMiniLive() static 小程序直播接口
|
||||
* @method \WeMini\Logistics WeMiniLogistics() static 小程序物流助手
|
||||
* @method \WeMini\Newtmpl WeMiniNewtmpl() static 公众号小程序订阅消息支持
|
||||
* @method \WeMini\Message WeMiniMessage() static 小程序动态消息
|
||||
* @method \WeMini\Operation WeMiniOperation() static 小程序运维中心
|
||||
* @method \WeMini\Ocr WeMiniOcr() static 小程序ORC服务
|
||||
* @method \WeMini\Plugs WeMiniPlugs() static 小程序插件管理
|
||||
* @method \WeMini\Poi WeMiniPoi() static 小程序地址管理
|
||||
* @method \WeMini\Qrcode WeMiniQrcode() static 小程序二维码管理
|
||||
* @method \WeMini\Security WeMiniSecurity() static 小程序内容安全
|
||||
* @method \WeMini\Soter WeMiniSoter() static 小程序生物认证
|
||||
* @method \WeMini\Template WeMiniTemplate() static 小程序模板消息支持
|
||||
* @method \WeMini\Total WeMiniTotal() static 小程序数据接口
|
||||
* @method \WeMini\Scheme WeMiniScheme() static 小程序URL-Scheme
|
||||
* @method \WeMini\Search WeMiniSearch() static 小程序搜索
|
||||
* @method \WeMini\Shipping WeMiniShipping() static 小程序发货信息管理服务
|
||||
*
|
||||
* ----- WePay -----
|
||||
* @method \WePay\Bill WePayBill() static 微信商户账单及评论
|
||||
* @method \WePay\Order WePayOrder() static 微信商户订单
|
||||
* @method \WePay\Refund WePayRefund() static 微信商户退款
|
||||
* @method \WePay\Coupon WePayCoupon() static 微信商户代金券
|
||||
* @method \WePay\Custom WePayCustom() static 微信扩展上报海关
|
||||
* @method \WePay\ProfitSharing WePayProfitSharing() static 微信分账
|
||||
* @method \WePay\Redpack WePayRedpack() static 微信红包支持
|
||||
* @method \WePay\Transfers WePayTransfers() static 微信商户打款到零钱
|
||||
* @method \WePay\TransfersBank WePayTransfersBank() static 微信商户打款到银行卡
|
||||
*
|
||||
* ----- WePayV3 -----
|
||||
* @method \WePayV3\Order WePayV3Order() static 直连商户|订单支付接口
|
||||
* @method \WePayV3\Transfers WePayV3Transfers() static 微信商家转账到零钱
|
||||
* @method \WePayV3\ProfitSharing WePayV3ProfitSharing() static 微信商户分账
|
||||
*
|
||||
* ----- WeOpen -----
|
||||
* @method \WeOpen\Login WeOpenLogin() static 第三方微信登录
|
||||
* @method \WeOpen\Service WeOpenService() static 第三方服务
|
||||
*
|
||||
* ----- ThinkService -----
|
||||
* @method mixed ThinkServiceConfig() static 平台服务配置
|
||||
*/
|
||||
class WechatService extends Service
|
||||
{
|
||||
|
||||
/**
|
||||
* 静态初始化对象
|
||||
* @param string $name
|
||||
* @param array $arguments
|
||||
* @return mixed
|
||||
* @throws \think\admin\Exception
|
||||
*/
|
||||
public static function __callStatic(string $name, array $arguments)
|
||||
{
|
||||
[$type, $base, $class] = static::parseName($name);
|
||||
if ("{$type}{$base}" !== $name) {
|
||||
throw new Exception("抱歉,实例 {$name} 不符合规则!");
|
||||
}
|
||||
if (sysconf('wechat.type') === 'api' || in_array($type, ['WePay', 'WePayV3'])) {
|
||||
if (class_exists($class)) {
|
||||
return new $class($type === 'WeMini' ? static::getWxconf() : static::getConfig());
|
||||
} else {
|
||||
throw new Exception("抱歉,接口模式无法实例 {$class} 对象!");
|
||||
}
|
||||
} else {
|
||||
[$appid, $appkey] = [sysconf('wechat.thr_appid'), sysconf('wechat.thr_appkey')];
|
||||
$data = ['class' => $name, 'appid' => $appid, 'time' => time(), 'nostr' => uniqid()];
|
||||
$data['sign'] = md5("{$data['class']}#{$appid}#{$appkey}#{$data['time']}#{$data['nostr']}");
|
||||
// 创建远程连接,默认使用 JSON-RPC 方式调用接口
|
||||
$token = enbase64url(json_encode($data, JSON_UNESCAPED_UNICODE));
|
||||
$jsonrpc = sysconf('wechat.service_jsonrpc|raw') ?: 'https://open.cuci.cc/plugin-wechat-service/api.client/jsonrpc?token=TOKEN';
|
||||
return new JsonRpcClient(str_replace('token=TOKEN', "token={$token}", $jsonrpc));
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* 解析调用对象名称
|
||||
* @param string $name
|
||||
* @return array
|
||||
*/
|
||||
private static function parseName(string $name): array
|
||||
{
|
||||
foreach (['WeChat', 'WeMini', 'WeOpen', 'WePayV3', 'WePay', 'ThinkService'] as $type) {
|
||||
if (strpos($name, $type) === 0) {
|
||||
[, $base] = explode($type, $name);
|
||||
return [$type, $base, "\\{$type}\\{$base}"];
|
||||
}
|
||||
}
|
||||
return ['-', '-', $name];
|
||||
}
|
||||
|
||||
/**
|
||||
* 获取当前微信APPID
|
||||
* @return string
|
||||
* @throws \think\admin\Exception
|
||||
*/
|
||||
public static function getAppid(): string
|
||||
{
|
||||
if (static::getType() === 'api') {
|
||||
return sysconf('wechat.appid');
|
||||
} else {
|
||||
return sysconf('wechat.thr_appid');
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* 获取接口授权模式
|
||||
* @return string
|
||||
* @throws \think\admin\Exception
|
||||
*/
|
||||
public static function getType(): string
|
||||
{
|
||||
$type = strtolower(sysconf('wechat.type'));
|
||||
if (in_array($type, ['api', 'thr'])) return $type;
|
||||
throw new Exception('请在后台配置微信对接授权模式');
|
||||
}
|
||||
|
||||
/**
|
||||
* 获取公众号配置参数
|
||||
* @param boolean $ispay 获取支付参数
|
||||
* @return array
|
||||
* @throws \think\admin\Exception
|
||||
*/
|
||||
public static function getConfig(bool $ispay = false): array
|
||||
{
|
||||
$config = [
|
||||
'appid' => static::getAppid(),
|
||||
'token' => sysconf('wechat.token'),
|
||||
'appsecret' => sysconf('wechat.appsecret'),
|
||||
'encodingaeskey' => sysconf('wechat.encodingaeskey'),
|
||||
'cache_path' => syspath('runtime/wechat'),
|
||||
];
|
||||
return $ispay ? static::withWxpayCert($config) : $config;
|
||||
}
|
||||
|
||||
/**
|
||||
* 获取小程序配置参数
|
||||
* @param boolean $ispay 获取支付参数
|
||||
* @return array
|
||||
* @throws \think\admin\Exception
|
||||
*/
|
||||
public static function getWxconf(bool $ispay = false): array
|
||||
{
|
||||
$wxapp = sysdata('plugin.wechat.wxapp');
|
||||
$config = [
|
||||
'appid' => $wxapp['appid'] ?? '',
|
||||
'appsecret' => $wxapp['appkey'] ?? '',
|
||||
'cache_path' => syspath('runtime/wechat'),
|
||||
];
|
||||
return $ispay ? static::withWxpayCert($config) : $config;
|
||||
}
|
||||
|
||||
/**
|
||||
* 处理支付证书配置
|
||||
* @param array $options
|
||||
* @return array
|
||||
* @throws \think\admin\Exception
|
||||
*/
|
||||
public static function withWxpayCert(array $options): array
|
||||
{
|
||||
// 文本模式主要是为了解决分布式部署
|
||||
$data = sysdata('plugin.wechat.payment');
|
||||
if (empty($data['mch_id'])) {
|
||||
throw new Exception('无效的支付配置!');
|
||||
}
|
||||
$name1 = sprintf("wxpay/%s_%s_cer.pem", $data['mch_id'], md5($data['ssl_cer_text']));
|
||||
$name2 = sprintf("wxpay/%s_%s_key.pem", $data['mch_id'], md5($data['ssl_key_text']));
|
||||
$local = LocalStorage::instance();
|
||||
if ($local->has($name1, true) && $local->has($name2, true)) {
|
||||
$sslCer = $local->path($name1, true);
|
||||
$sslKey = $local->path($name2, true);
|
||||
} else {
|
||||
$sslCer = $local->set($name1, $data['ssl_cer_text'], true)['file'];
|
||||
$sslKey = $local->set($name2, $data['ssl_key_text'], true)['file'];
|
||||
}
|
||||
$options['mch_id'] = $data['mch_id'];
|
||||
$options['mch_key'] = $data['mch_key'];
|
||||
$options['mch_v3_key'] = $data['mch_v3_key'];
|
||||
$options['ssl_cer'] = $sslCer;
|
||||
$options['ssl_key'] = $sslKey;
|
||||
$options['cert_public'] = $sslCer;
|
||||
$options['cert_private'] = $sslKey;
|
||||
$options['mp_cert_serial'] = $data['mch_pay_sid'] ?? '';
|
||||
$options['mp_cert_content'] = $data['ssl_pay_text'] ?? '';
|
||||
return $options;
|
||||
}
|
||||
|
||||
/**
|
||||
* 获取会话名称
|
||||
* @return string
|
||||
*/
|
||||
public static function getSsid(): string
|
||||
{
|
||||
$conf = Library::$sapp->session->getConfig();
|
||||
$ssid = Library::$sapp->request->get($conf['name'] ?? 'ssid');
|
||||
return empty($ssid) ? Library::$sapp->session->getId() : $ssid;
|
||||
}
|
||||
|
||||
/**
|
||||
* 通过网页授权获取粉丝信息
|
||||
* @param string $source 回跳URL地址
|
||||
* @param integer $isfull 获取资料模式
|
||||
* @param boolean $redirect 是否直接跳转
|
||||
* @return array
|
||||
* @throws \WeChat\Exceptions\InvalidResponseException
|
||||
* @throws \WeChat\Exceptions\LocalCacheException
|
||||
* @throws \think\admin\Exception
|
||||
*/
|
||||
public static function getWebOauthInfo(string $source, int $isfull = 0, bool $redirect = true): array
|
||||
{
|
||||
[$ssid, $appid] = [static::getSsid(), static::getAppid()];
|
||||
$openid = Library::$sapp->cache->get("{$ssid}_openid");
|
||||
$userinfo = Library::$sapp->cache->get("{$ssid}_fansinfo");
|
||||
if ((empty($isfull) && !empty($openid)) || (!empty($isfull) && !empty($openid) && !empty($userinfo))) {
|
||||
empty($userinfo) || FansService::set($userinfo, $appid);
|
||||
return ['openid' => $openid, 'fansinfo' => $userinfo];
|
||||
}
|
||||
if (static::getType() === 'api') {
|
||||
// 解析 GET 参数
|
||||
parse_str(parse_url($source, PHP_URL_QUERY), $params);
|
||||
$getVars = [
|
||||
'code' => $params['code'] ?? input('code', ''),
|
||||
'rcode' => $params['rcode'] ?? input('rcode', ''),
|
||||
'state' => $params['state'] ?? input('state', ''),
|
||||
];
|
||||
$wechat = static::WeChatOauth();
|
||||
if ($getVars['state'] !== $appid || empty($getVars['code'])) {
|
||||
$params['rcode'] = enbase64url($source);
|
||||
$location = strstr("{$source}?", '?', true) . '?' . http_build_query($params);
|
||||
$oauthurl = $wechat->getOauthRedirect($location, $appid, $isfull ? 'snsapi_userinfo' : 'snsapi_base');
|
||||
throw new HttpResponseException(static::createRedirect($oauthurl, $redirect));
|
||||
} elseif (($token = $wechat->getOauthAccessToken($getVars['code'])) && isset($token['openid'])) {
|
||||
$openid = $token['openid'];
|
||||
// 如果是虚拟账号,不保存会话信息,下次重新授权
|
||||
if (empty($token['is_snapshotuser'])) {
|
||||
Library::$sapp->cache->set("{$ssid}_openid", $openid, 3600);
|
||||
}
|
||||
if ($isfull && isset($token['access_token'])) {
|
||||
$userinfo = $wechat->getUserInfo($token['access_token'], $openid);
|
||||
// 如果是虚拟账号,不保存会话信息,下次重新授权
|
||||
if (empty($token['is_snapshotuser'])) {
|
||||
$userinfo['is_snapshotuser'] = 0;
|
||||
// 缓存用户信息
|
||||
Library::$sapp->cache->set("{$ssid}_fansinfo", $userinfo, 3600);
|
||||
empty($userinfo) || FansService::set($userinfo, $appid);
|
||||
} else {
|
||||
$userinfo['is_snapshotuser'] = 1;
|
||||
}
|
||||
}
|
||||
}
|
||||
if ($getVars['rcode']) {
|
||||
throw new HttpResponseException(static::createRedirect(debase64url($getVars['rcode']), $redirect));
|
||||
} elseif ((empty($isfull) && !empty($openid)) || (!empty($isfull) && !empty($openid) && !empty($userinfo))) {
|
||||
return ['openid' => $openid, 'fansinfo' => $userinfo];
|
||||
} else {
|
||||
throw new Exception('Query params [rcode] not find.');
|
||||
}
|
||||
} else {
|
||||
$result = static::ThinkServiceConfig()->oauth(self::getSsid(), $source, $isfull);
|
||||
[$openid, $userinfo] = [$result['openid'] ?? '', $result['fans'] ?? []];
|
||||
// 如果是虚拟账号,不保存会话信息,下次重新授权
|
||||
if (empty($result['token']['is_snapshotuser'])) {
|
||||
Library::$sapp->cache->set("{$ssid}_openid", $openid, 3600);
|
||||
Library::$sapp->cache->set("{$ssid}_fansinfo", $userinfo, 3600);
|
||||
}
|
||||
if ((empty($isfull) && !empty($openid)) || (!empty($isfull) && !empty($openid) && !empty($userinfo))) {
|
||||
empty($result['token']['is_snapshotuser']) && empty($userinfo) || FansService::set($userinfo, $appid);
|
||||
return ['openid' => $openid, 'fansinfo' => $userinfo];
|
||||
}
|
||||
throw new HttpResponseException(static::createRedirect($result['url'], $redirect));
|
||||
}
|
||||
}
|
||||
|
||||
/**
|
||||
* 网页授权链接跳转
|
||||
* @param string $location 跳转链接
|
||||
* @param boolean $redirect 强制跳转
|
||||
* @return \think\Response
|
||||
*/
|
||||
private static function createRedirect(string $location, bool $redirect = true): Response
|
||||
{
|
||||
return $redirect ? redirect($location) : response(join(";\n", [
|
||||
sprintf("sessionStorage.setItem('wechat.session','%s')", self::getSsid()),
|
||||
sprintf("location.replace('%s')", $location), ''
|
||||
]));
|
||||
}
|
||||
|
||||
/**
|
||||
* 获取微信网页JSSDK签名参数
|
||||
* @param null|string $location 签名地址
|
||||
* @return array
|
||||
* @throws \WeChat\Exceptions\InvalidResponseException
|
||||
* @throws \WeChat\Exceptions\LocalCacheException
|
||||
* @throws \think\admin\Exception
|
||||
*/
|
||||
public static function getWebJssdkSign(?string $location = null): array
|
||||
{
|
||||
$location = $location ?: Library::$sapp->request->url(true);
|
||||
if (static::getType() === 'api') {
|
||||
return static::WeChatScript()->getJsSign($location);
|
||||
} else {
|
||||
return static::ThinkServiceConfig()->jsSign($location);
|
||||
}
|
||||
}
|
||||
}
|
36
app/wechat/view/api/login/failed.html
Normal file
36
app/wechat/view/api/login/failed.html
Normal file
@ -0,0 +1,36 @@
|
||||
<!DOCTYPE html>
|
||||
<html lang="zh">
|
||||
<head>
|
||||
<title>扫码登录失败</title>
|
||||
<meta charset="utf-8">
|
||||
<meta http-equiv="X-UA-Compatible" content="IE=edge">
|
||||
<meta name="viewport" content="width=device-width, initial-scale=1, user-scalable=no">
|
||||
<link rel="stylesheet" href="__ROOT__/static/theme/css/mobile.css">
|
||||
<style>
|
||||
header {
|
||||
padding: 16vh 0;
|
||||
text-align: center;
|
||||
}
|
||||
|
||||
header img {
|
||||
width: 40vw;
|
||||
}
|
||||
|
||||
header div {
|
||||
color: rgb(240, 91, 41);
|
||||
font-size: 7vw;
|
||||
margin-top: 8vh;
|
||||
font-weight: bold;
|
||||
letter-spacing: 2px;
|
||||
}
|
||||
</style>
|
||||
</head>
|
||||
<body>
|
||||
<div>
|
||||
<header>
|
||||
<img alt="img" src="__ROOT__/static/theme/img/wechat/m-icon-error.png">
|
||||
<div>{$message|default='授权失败'}</div>
|
||||
</header>
|
||||
</div>
|
||||
</body>
|
||||
</html>
|
53
app/wechat/view/api/login/success.html
Normal file
53
app/wechat/view/api/login/success.html
Normal file
@ -0,0 +1,53 @@
|
||||
<!DOCTYPE html>
|
||||
<html lang="zh">
|
||||
<head>
|
||||
<title>扫码登录成功</title>
|
||||
<meta charset="utf-8">
|
||||
<meta http-equiv="X-UA-Compatible" content="IE=edge">
|
||||
<meta name="viewport" content="width=device-width, initial-scale=1, user-scalable=no">
|
||||
<link rel="stylesheet" href="__ROOT__/static/theme/css/mobile.css">
|
||||
<style>
|
||||
header {
|
||||
padding: 16vh 0;
|
||||
text-align: center;
|
||||
}
|
||||
|
||||
header img {
|
||||
width: 40vw;
|
||||
}
|
||||
|
||||
header div {
|
||||
font-size: 7vw;
|
||||
margin-top: 8vh;
|
||||
color: rgb(49, 131, 238);
|
||||
font-weight: bold;
|
||||
letter-spacing: 2px;
|
||||
}
|
||||
|
||||
.container {
|
||||
font-size: 4vw;
|
||||
margin-top: 10vh;
|
||||
}
|
||||
|
||||
.container p {
|
||||
text-indent: 10vw;
|
||||
line-height: 2em;
|
||||
}
|
||||
|
||||
.container p span {
|
||||
color: #888
|
||||
}
|
||||
</style>
|
||||
</head>
|
||||
<body>
|
||||
<div>
|
||||
<header>
|
||||
<img alt="img" src="__ROOT__/static/theme/img/wechat/m-icon-success.png">
|
||||
<div>{$message|default='授权成功'}</div>
|
||||
</header>
|
||||
<div class="container">
|
||||
<p>授权时间:<span>{:date('Y年m月d日H:i')}</span></p>
|
||||
</div>
|
||||
</div>
|
||||
</body>
|
||||
</html>
|
66
app/wechat/view/api/test/jsapi.html
Normal file
66
app/wechat/view/api/test/jsapi.html
Normal file
@ -0,0 +1,66 @@
|
||||
<!DOCTYPE html>
|
||||
<html lang="zh">
|
||||
<head>
|
||||
<meta charset="utf-8">
|
||||
<title>微信 JSAPI 支付测试</title>
|
||||
<meta http-equiv="X-UA-Compatible" content="IE=edge">
|
||||
<meta name="viewport" content="width=device-width, initial-scale=1, user-scalable=no">
|
||||
<link rel="stylesheet" href="__ROOT__/static/theme/css/mobile.css">
|
||||
<style>
|
||||
.test-show {
|
||||
padding: 10px;
|
||||
font-size: 14px;
|
||||
margin-top: 10px;
|
||||
background: rgba(0, 0, 0, 0.1);
|
||||
border-radius: 10px;
|
||||
}
|
||||
|
||||
.test-show pre {
|
||||
overflow: auto;
|
||||
padding: 10px;
|
||||
}
|
||||
|
||||
.test-payment {
|
||||
padding: 30px 0 10px 0;
|
||||
text-align: center
|
||||
}
|
||||
|
||||
.test-payment button {
|
||||
border: none;
|
||||
color: #fff;
|
||||
padding: 15px;
|
||||
background: #a233c6;
|
||||
border-radius: 4px;
|
||||
}
|
||||
</style>
|
||||
</head>
|
||||
<body>
|
||||
<div style="padding:15px">
|
||||
<h3>1. 用户 OPENID</h3>
|
||||
<div class="test-show">
|
||||
<pre>{$user['openid']}</pre>
|
||||
</div>
|
||||
<br>
|
||||
<h3>2. 微信 JSAPI 支付参数</h3>
|
||||
<div class="test-show">
|
||||
<pre>{:json_encode($result,64|128|256)}</pre>
|
||||
</div>
|
||||
<div class="test-payment">
|
||||
<button id='paytest' type='button'>发起 JSAPI 支付 0.01 元</button>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<script src='//res.wx.qq.com/open/js/jweixin-1.6.0.js'></script>
|
||||
<script>
|
||||
wx.config(JSON.parse('{$configJson|raw}'));
|
||||
document.getElementById('paytest').onclick = function () {
|
||||
let options = JSON.parse('{$optionJson|raw}');
|
||||
options.success = function () {
|
||||
alert('支付成功');
|
||||
};
|
||||
console.log("OPTIONS:", options);
|
||||
wx.chooseWXPay(options);
|
||||
}
|
||||
</script>
|
||||
</body>
|
||||
</html>
|
64
app/wechat/view/api/test/jssdk.html
Normal file
64
app/wechat/view/api/test/jssdk.html
Normal file
@ -0,0 +1,64 @@
|
||||
<!DOCTYPE html>
|
||||
<html lang="zh">
|
||||
<head>
|
||||
<meta charset="utf-8">
|
||||
<title>JSSDK 功能测试</title>
|
||||
<meta http-equiv="X-UA-Compatible" content="IE=edge">
|
||||
<meta name="viewport" content="width=device-width, initial-scale=1, user-scalable=no">
|
||||
<link rel="stylesheet" href="__ROOT__/static/theme/css/mobile.css">
|
||||
</head>
|
||||
|
||||
<body>
|
||||
|
||||
<header>
|
||||
<h1>JSSDK 功能测试</h1>
|
||||
</header>
|
||||
|
||||
<div class="content">
|
||||
<button type="button" id='show-alert'>调起摄像头扫码</button>
|
||||
</div>
|
||||
|
||||
<style>
|
||||
header {
|
||||
padding: 35px 0
|
||||
}
|
||||
|
||||
header h1 {
|
||||
color: #3cc51f;
|
||||
font-size: 34px;
|
||||
text-align: center;
|
||||
font-weight: 400;
|
||||
}
|
||||
|
||||
.content {
|
||||
padding: 15px;
|
||||
text-align: center;
|
||||
}
|
||||
|
||||
.content button {
|
||||
padding: 5px
|
||||
}
|
||||
</style>
|
||||
|
||||
<script src="//res.wx.qq.com/open/js/jweixin-1.4.0.js"></script>
|
||||
<script>
|
||||
wx.error(function (err) {
|
||||
alert(err.errMsg || '配置出错');
|
||||
});
|
||||
wx.config(JSON.parse('{$options|json_encode|raw}'));
|
||||
wx.ready(function () {
|
||||
alert('- 初始化成功 -');
|
||||
document.getElementById('show-alert').onclick = function () {
|
||||
wx.scanQRCode({
|
||||
needResult: 1,
|
||||
scanType: ["qrCode", "barCode"],
|
||||
success: function (res) {
|
||||
alert(res.resultStr);
|
||||
}
|
||||
});
|
||||
}
|
||||
});
|
||||
</script>
|
||||
</body>
|
||||
|
||||
</html>
|
55
app/wechat/view/api/test/oauth.html
Normal file
55
app/wechat/view/api/test/oauth.html
Normal file
@ -0,0 +1,55 @@
|
||||
<!DOCTYPE html>
|
||||
<html lang="zh">
|
||||
|
||||
<head>
|
||||
<meta charset="UTF-8">
|
||||
<title>微信网页授权测试</title>
|
||||
<meta http-equiv="X-UA-Compatible" content="IE=edge">
|
||||
<meta name="viewport" content="width=device-width, initial-scale=1, user-scalable=no">
|
||||
<link rel="stylesheet" href="__ROOT__/static/theme/css/mobile.css">
|
||||
<script src="//res.wx.qq.com/open/js/jweixin-1.2.0.js"></script>
|
||||
</head>
|
||||
|
||||
<body>
|
||||
{if empty($fans.fansinfo)}
|
||||
<fieldset style="width:80%;margin:1rem auto">
|
||||
<legend style="padding:0.2rem 0.3rem;font-size:0.9rem;color:#666">操作失败</legend>
|
||||
<div style="padding-top:0.4rem">
|
||||
<p style="font-size:0.6rem;margin-bottom:0.8rem">通过网页授权获取用户资料失败,请检查权限再试!</p>
|
||||
<p style="font-size:0.6rem;margin-bottom:0.8rem">
|
||||
<a style="cursor:pointer" onclick="location.reload()">重新获取(刷新)</a>
|
||||
</p>
|
||||
</div>
|
||||
</fieldset>
|
||||
{else}
|
||||
<div class="header" style="padding:1rem 0 0.6rem 0;text-align:center">
|
||||
<img alt="img" style='width:132px;height:132px;border-radius:50%;border:0.6rem solid rgba(150,40,40,0.6)' src="{$fans.fansinfo.headimgurl}">
|
||||
<h1>{$fans.fansinfo.nickname}</h1>
|
||||
</div>
|
||||
<fieldset style="width:80%;margin:0.6rem auto">
|
||||
<legend style="padding:0.2rem 0.3rem;font-size:0.9rem;color:#666">用户标识</legend>
|
||||
<div style="padding-top:0.4rem">
|
||||
<div style="font-size:0.8rem;margin-bottom:0.3rem;color:#999">OPENID</div>
|
||||
<p style="font-size:0.6rem;margin-bottom:0.8rem">{$fans.fansinfo.openid}</p>
|
||||
<div style="font-size:0.8rem;margin-bottom:0.3rem;color:#999">UNIONID</div>
|
||||
<p style="font-size:0.6rem;margin-bottom:0.8rem">{$fans.fansinfo.unionid|default='未获取到'}</p>
|
||||
</div>
|
||||
</fieldset>
|
||||
<fieldset style="width:80%;margin:0.6rem auto">
|
||||
<legend style="padding:0.2rem 0.3rem;font-size:0.9rem;color:#666">详细资料</legend>
|
||||
<div style="padding-top:0.4rem">
|
||||
<div style="font-size:0.8rem;margin-bottom:0.3rem;color:#999">性别</div>
|
||||
<p style="font-size:0.6rem;margin-bottom:0.8rem">{:[1=>'男',2=>'女'][$fans.fansinfo.sex]??'未知'}</p>
|
||||
<div style="font-size:0.8rem;margin-bottom:0.3rem;color:#999">系统语言</div>
|
||||
<p style="font-size:0.6rem;margin-bottom:0.8rem">{$fans.fansinfo.language}</p>
|
||||
<div style="font-size:0.8rem;margin-bottom:0.3rem;color:#999">所在区域</div>
|
||||
<p style="font-size:0.6rem;margin-bottom:0.8rem">{$fans.fansinfo.country} - {$fans.fansinfo.province} - {$fans.fansinfo.city}</p>
|
||||
{if isset($fans.fansinfo.privilege.0)}
|
||||
<div style="font-size:0.8rem;margin-bottom:0.3rem;color:#999">设备网络</div>
|
||||
<p style="font-size:0.6rem;margin-bottom:0.8rem">{$fans.fansinfo.privilege.0|default='未获取到网络信息'}</p>
|
||||
{/if}
|
||||
</div>
|
||||
</fieldset>
|
||||
{/if}
|
||||
</body>
|
||||
</html>
|
28
app/wechat/view/api/view/image.html
Normal file
28
app/wechat/view/api/view/image.html
Normal file
@ -0,0 +1,28 @@
|
||||
{extend name='api/view/main'}
|
||||
|
||||
{block name='content'}
|
||||
<div class="header"><span>{:date('H:i')}</span></div>
|
||||
<div class="container">
|
||||
<div class="logo">Ta</div>
|
||||
<div class="content"><img src="{$content|default=''}" alt="img"></div>
|
||||
</div>
|
||||
{/block}
|
||||
|
||||
{block name='style'}
|
||||
<style>
|
||||
.container .content {
|
||||
border: none !important;
|
||||
display: inline-block;
|
||||
padding: 0 !important;
|
||||
position: relative;
|
||||
border-radius: 5px;
|
||||
background: none;
|
||||
}
|
||||
|
||||
.container .content img {
|
||||
max-width: 100%;
|
||||
border-radius: 5px;
|
||||
box-shadow: 0 0 3px #666;
|
||||
}
|
||||
</style>
|
||||
{/block}
|
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