1
0
mirror of https://gitee.com/koogua/course-tencent-cloud.git synced 2025-06-25 04:07:17 +08:00

!54 v1.2.4阶段性合并

Merge pull request !54 from koogua/develop
This commit is contained in:
koogua 2021-01-09 18:26:29 +08:00 committed by Gitee
commit 1fd957a6e2
79 changed files with 2894 additions and 1298 deletions

View File

@ -1,3 +1,19 @@
### [v1.2.4](https://gitee.com/koogua/course-tencent-cloud/releases/v1.2.4)(2021-01-10)
#### 增加
- 后台增加上传logo和favicon图标
- 后台增加公众号自定义菜单配置
- 课程页增加咨询
### 优化
- oauth中state参数为安全base64加解码
- findById参数类型不对时抛出异常
- task表增加索引加快数据查找
- markdown内容解析改由后端完成
- 公众号应答处理逻辑
### [v1.2.3](https://gitee.com/koogua/course-tencent-cloud/releases/v1.2.3)(2021-01-03) ### [v1.2.3](https://gitee.com/koogua/course-tencent-cloud/releases/v1.2.3)(2021-01-03)
#### 增加 #### 增加

View File

@ -2,16 +2,16 @@
![酷瓜云网课GPL协议开源](https://images.gitee.com/uploads/images/2020/1127/092621_3805cf8f_23592.png) ![酷瓜云网课GPL协议开源](https://images.gitee.com/uploads/images/2020/1127/092621_3805cf8f_23592.png)
#### 项目介绍 ### 项目介绍
酷瓜云课堂依托腾讯云基础服务架构采用C扩展框架Phalcon开发GPL-2.0开源协议,致力开源网课系统,开源网校系统,开源在线教育系统。 酷瓜云课堂依托腾讯云基础服务架构采用C扩展框架Phalcon开发GPL-2.0开源协议,致力开源网课系统,开源网校系统,开源在线教育系统。
![](https://img.shields.io/static/v1?label=release&message=1.2.3&color=blue) ![](https://img.shields.io/static/v1?label=release&message=1.2.4&color=blue)
![](https://img.shields.io/static/v1?label=stars&message=160&color=blue) ![](https://img.shields.io/static/v1?label=stars&message=168&color=blue)
![](https://img.shields.io/static/v1?label=forks&message=60&color=blue) ![](https://img.shields.io/static/v1?label=forks&message=63&color=blue)
![](https://img.shields.io/static/v1?label=license&message=GPL-2.0&color=blue) ![](https://img.shields.io/static/v1?label=license&message=GPL-2.0&color=blue)
#### 系统功能 ### 系统功能
实现了点播、直播、专栏、会员、微聊等,是一个完整的产品,具体功能我也不想写一大堆,自己体验吧! 实现了点播、直播、专栏、会员、微聊等,是一个完整的产品,具体功能我也不想写一大堆,自己体验吧!
@ -21,17 +21,19 @@
- 课程数据来源于网络(无实质内容),切莫购买 - 课程数据来源于网络(无实质内容),切莫购买
- 管理后台已禁止数据提交,私密配置已过滤 - 管理后台已禁止数据提交,私密配置已过滤
演示帐号:**13507083515 / 123456** (前后台通用)
桌面端演示: 桌面端演示:
- [前台演示](https://ctc.koogua.com) - [前台演示](https://ctc.koogua.com)
- [后台演示](https://ctc.koogua.com/admin) - [后台演示](https://ctc.koogua.com/admin)
演示帐号100015@163.com / 123456 (前后台通用)
移动端演示: 移动端演示:
![移动端二维码](https://images.gitee.com/uploads/images/2020/1127/093203_265221a2_23592.png) ![移动端二维码](https://images.gitee.com/uploads/images/2020/1127/093203_265221a2_23592.png)
演示帐号13507083515 / 123456
支付流程演示: 支付流程演示:
- [MySQL提升课程全面讲解MySQL架构设计0.01元)](https://ctc.koogua.com/order/confirm?item_id=1390&item_type=1) - [MySQL提升课程全面讲解MySQL架构设计0.01元)](https://ctc.koogua.com/order/confirm?item_id=1390&item_type=1)
@ -40,7 +42,18 @@
Tips: 测试支付请用手机号注册一个新账户,以便接收订单通知,以及避免课程无法购买 Tips: 测试支付请用手机号注册一个新账户,以便接收订单通知,以及避免课程无法购买
#### 项目组件 即时通讯演示:
请使用以下两个帐号在不同终端或者浏览器登录,打开微聊界面
- 帐号A100015@163.com / 123456
- 帐号B100065@163.com / 123456
微信推送演示:
Tips: 请用手机注册一个新账号,用户中心 -> 关注订阅,扫码关注公众号。之后的登录、购买、退款、直播、咨询等会有消息推送。
### 项目组件
- 后台框架:[phalcon 3.4.5](https://phalcon.io) - 后台框架:[phalcon 3.4.5](https://phalcon.io)
- 前端框架:[layui 2.5.6](https://layui.com) [layim 3.9.5](https://www.layui.com/layim)(已授权) - 前端框架:[layui 2.5.6](https://layui.com) [layim 3.9.5](https://www.layui.com/layim)(已授权)
@ -48,45 +61,27 @@ Tips: 测试支付请用手机号注册一个新账户,以便接收订单通
- 即时通讯:[workerman 3.5.22](https://workerman.net) - 即时通讯:[workerman 3.5.22](https://workerman.net)
- 基础依赖:[php7.3](https://php.net) [mysql5.7](https://mysql.com) [redis5.0](https://redis.io) - 基础依赖:[php7.3](https://php.net) [mysql5.7](https://mysql.com) [redis5.0](https://redis.io)
#### 安装指南 ### 安装指南
- [运行环境搭建](https://gitee.com/koogua/course-tencent-cloud-docker) - [运行环境搭建](https://gitee.com/koogua/course-tencent-cloud-docker)
- [系统服务配置](https://gitee.com/koogua/course-tencent-cloud/wikis) - [系统服务配置](https://gitee.com/koogua/course-tencent-cloud/wikis)
- [客户终端配置](https://gitee.com/koogua/course-tencent-cloud-app)
#### 开发计划 ### 意见反馈
- 桌面端:进行中
- 移动端:进行中
- 小程序:待启动
#### 意见反馈
- [在线反馈](https://gitee.com/koogua/course-tencent-cloud/issues)(推荐) - [在线反馈](https://gitee.com/koogua/course-tencent-cloud/issues)(推荐)
- [官方论坛](https://koogua.com/forum)(推荐)
- QQ交流群: 787363898 - QQ交流群: 787363898
#### 通过这个项目能学到什么? ### 有阿里云版吗?
- 项目规划phalcon缓存JWT即时通讯全文检索
- dockersupervisordevops
- gitlinuxphpmysqlredisnginx
#### 有阿里云版吗?
阿里云版规划中,之前阿里云服务过期未续费,所以腾讯云版本先出。 阿里云版规划中,之前阿里云服务过期未续费,所以腾讯云版本先出。
#### 代码有加密吗? ### 代码有加密吗?
所有代码都公开授权代码除外例如layim没有所谓的商业版和付费插件。 所有代码都公开授权代码除外例如layim没有所谓的商业版和付费插件。
#### 有商业服务吗? ### 开源助力
生存才能发展,我们目前提供的服务包括:
- 系统安装
- 系统定制
- 企业授权
#### 开源助力
毫无保留的真开源不容易,如果对你有帮助,请给我们 **STAR** 毫无保留的真开源不容易,如果对你有帮助,请给我们 **STAR**

View File

@ -248,12 +248,12 @@ class DeliverTask extends Task
{ {
$itemType = TaskModel::TYPE_DELIVER; $itemType = TaskModel::TYPE_DELIVER;
$status = TaskModel::STATUS_PENDING; $status = TaskModel::STATUS_PENDING;
$tryCount = self::TRY_COUNT; $createTime = strtotime('-3 days');
return TaskModel::query() return TaskModel::query()
->where('item_type = :item_type:', ['item_type' => $itemType]) ->where('item_type = :item_type:', ['item_type' => $itemType])
->andWhere('status = :status:', ['status' => $status]) ->andWhere('status = :status:', ['status' => $status])
->andWhere('try_count < :try_count:', ['try_count' => $tryCount + 1]) ->andWhere('create_time > :create_time:', ['create_time' => $createTime])
->orderBy('priority ASC') ->orderBy('priority ASC')
->limit($limit) ->limit($limit)
->execute(); ->execute();

View File

@ -125,12 +125,12 @@ class NoticeTask extends Task
$status = TaskModel::STATUS_PENDING; $status = TaskModel::STATUS_PENDING;
$tryCount = self::TRY_COUNT; $createTime = strtotime('-1 days');
return TaskModel::query() return TaskModel::query()
->inWhere('item_type', $itemTypes) ->inWhere('item_type', $itemTypes)
->andWhere('status = :status:', ['status' => $status]) ->andWhere('status = :status:', ['status' => $status])
->andWhere('try_count < :try_count:', ['try_count' => $tryCount + 1]) ->andWhere('create_time > :create_time:', ['create_time' => $createTime])
->orderBy('priority ASC') ->orderBy('priority ASC')
->limit($limit) ->limit($limit)
->execute(); ->execute();

View File

@ -296,12 +296,12 @@ class RefundTask extends Task
{ {
$itemType = TaskModel::TYPE_REFUND; $itemType = TaskModel::TYPE_REFUND;
$status = TaskModel::STATUS_PENDING; $status = TaskModel::STATUS_PENDING;
$tryCount = self::TRY_COUNT; $createTime = strtotime('-3 days');
return TaskModel::query() return TaskModel::query()
->where('item_type = :item_type:', ['item_type' => $itemType]) ->where('item_type = :item_type:', ['item_type' => $itemType])
->andWhere('status = :status:', ['status' => $status]) ->andWhere('status = :status:', ['status' => $status])
->andWhere('try_count < :try_count:', ['try_count' => $tryCount + 1]) ->andWhere('create_time > :create_time:', ['create_time' => $createTime])
->orderBy('priority ASC') ->orderBy('priority ASC')
->limit($limit) ->limit($limit)
->execute(); ->execute();

View File

@ -80,7 +80,7 @@ class UpgradeTask extends Task
$redis->del($statsKey); $redis->del($statsKey);
} }
echo "start reset metadata..." . PHP_EOL; echo "end reset metadata..." . PHP_EOL;
} }
/** /**

View File

@ -328,9 +328,9 @@ class SettingController extends Controller
} }
/** /**
* @Route("/wechat", name="admin.setting.wechat") * @Route("/wechat/oa", name="admin.setting.wechat_oa")
*/ */
public function wechatAction() public function wechatOaAction()
{ {
$settingService = new SettingService(); $settingService = new SettingService();
@ -340,7 +340,7 @@ class SettingController extends Controller
$data = $this->request->getPost(); $data = $this->request->getPost();
$settingService->updateWechatSettings($section, $data); $settingService->updateWechatOASettings($section, $data);
return $this->jsonSuccess(['msg' => '更新配置成功']); return $this->jsonSuccess(['msg' => '更新配置成功']);
@ -348,6 +348,7 @@ class SettingController extends Controller
$oa = $settingService->getWechatOASettings(); $oa = $settingService->getWechatOASettings();
$this->view->pick('setting/wechat_oa');
$this->view->setVar('oa', $oa); $this->view->setVar('oa', $oa);
} }
} }

View File

@ -11,6 +11,7 @@ use App\Services\Mail\Test as TestMailService;
use App\Services\MyStorage as StorageService; use App\Services\MyStorage as StorageService;
use App\Services\Sms\Test as TestSmsService; use App\Services\Sms\Test as TestSmsService;
use App\Services\Vod as VodService; use App\Services\Vod as VodService;
use App\Services\Wechat as WechatService;
/** /**
* @RoutePrefix("/admin/test") * @RoutePrefix("/admin/test")
@ -54,6 +55,24 @@ class TestController extends Controller
} }
} }
/**
* @Post("/wechat/oa", name="admin.test.wechat_oa")
*/
public function wechatOaAction()
{
$wechatService = new WechatService();
$oa = $wechatService->getOfficialAccount();
$result = $oa->qrcode->temporary('foo', 86400);
if (isset($result['ticket'])) {
return $this->jsonSuccess(['msg' => '接口返回成功']);
} else {
return $this->jsonError(['msg' => '接口返回失败,请检查相关配置']);
}
}
/** /**
* @Get("/live/push", name="admin.test.live_push") * @Get("/live/push", name="admin.test.live_push")
*/ */

View File

@ -759,9 +759,9 @@ class AuthNode extends Service
], ],
[ [
'id' => '5-1-13', 'id' => '5-1-13',
'title' => '微信平台', 'title' => '微信公众号',
'type' => 'menu', 'type' => 'menu',
'route' => 'admin.setting.wechat', 'route' => 'admin.setting.wechat_oa',
], ],
], ],
], ],

View File

@ -5,6 +5,7 @@ namespace App\Http\Admin\Services;
use App\Caches\Setting as SettingCache; use App\Caches\Setting as SettingCache;
use App\Repos\Setting as SettingRepo; use App\Repos\Setting as SettingRepo;
use App\Repos\Vip as VipRepo; use App\Repos\Vip as VipRepo;
use App\Services\Wechat as WechatService;
class Setting extends Service class Setting extends Service
{ {
@ -63,6 +64,26 @@ class Setting extends Service
$oa['notify_url'] = $oa['notify_url'] ?: kg_full_url(['for' => 'home.wechat.oa.notify']); $oa['notify_url'] = $oa['notify_url'] ?: kg_full_url(['for' => 'home.wechat.oa.notify']);
$oa['menu'] = json_decode($oa['menu'], true);
/**
* 构造一个35的二维树形菜单
*/
for ($i = 0; $i < 3; $i++) {
if (!isset($oa['menu'][$i])) {
$oa['menu'][$i] = ['name' => sprintf('菜单%s', $i + 1)];
}
for ($j = 0; $j < 5; $j++) {
if (!isset($oa['menu'][$i]['children'][$j])) {
$oa['menu'][$i]['children'][$j] = [
'type' => 'view',
'name' => '',
'url' => '',
];;
}
}
}
return $oa; return $oa;
} }
@ -171,12 +192,36 @@ class Setting extends Service
} }
} }
public function updateWechatSettings($section, $settings) public function updateWechatOASettings($section, $settings)
{ {
if ($section == 'wechat.oa') { if (!empty($settings['notice_template'])) {
if (isset($settings['notice_template'])) { $settings['notice_template'] = kg_json_encode($settings['notice_template']);
$settings['notice_template'] = kg_json_encode($settings['notice_template']); }
$buttons = [];
if (!empty($settings['menu'])) {
foreach ($settings['menu'] as $i => $top) {
$buttons[$i]['name'] = $top['name'];
$buttons[$i]['url'] = $top['url'];
$buttons[$i]['type'] = 'view';
foreach ($top['children'] as $j => $sub) {
if (!empty($sub['name']) && !empty($sub['url'])) {
$buttons[$i]['sub_button'][$j]['name'] = $sub['name'];
$buttons[$i]['sub_button'][$j]['url'] = $sub['url'];
$buttons[$i]['sub_button'][$j]['type'] = 'view';
} else {
unset($settings['menu'][$i]['children'][$j]);
}
}
} }
$settings['menu'] = kg_json_encode($settings['menu']);
}
if (!empty($buttons)) {
$service = new WechatService();
$oa = $service->getOfficialAccount();
$oa->menu->create($buttons);
} }
$this->updateSettings($section, $settings); $this->updateSettings($section, $settings);

View File

@ -31,6 +31,32 @@
<input type="radio" name="index_tpl_type" value="full" title="丰富" {% if site.index_tpl_type == 'full' %}checked="checked"{% endif %}> <input type="radio" name="index_tpl_type" value="full" title="丰富" {% if site.index_tpl_type == 'full' %}checked="checked"{% endif %}>
</div> </div>
</div> </div>
<div class="layui-form-item">
<div class="layui-inline">
<label class="layui-form-label">Logo</label>
<div class="kg-input-inline">
<input class="layui-input" type="text" name="logo" placeholder="请确保存储已正确配置" value="{{ site.logo }}">
</div>
</div>
<div class="layui-inline">
<div class="kg-input-inline">
<button class="layui-btn" type="button" id="upload-logo">上传文件</button>
</div>
</div>
</div>
<div class="layui-form-item">
<div class="layui-inline">
<label class="layui-form-label">Favicon</label>
<div class="kg-input-inline">
<input class="layui-input" type="text" name="favicon" placeholder="请确保存储已正确配置" value="{{ site.favicon }}">
</div>
</div>
<div class="layui-inline">
<div class="kg-input-inline">
<button class="layui-btn" type="button" id="upload-favicon">上传文件</button>
</div>
</div>
</div>
<div class="layui-form-item"> <div class="layui-form-item">
<label class="layui-form-label">网站名称</label> <label class="layui-form-label">网站名称</label>
<div class="layui-input-block"> <div class="layui-input-block">
@ -120,10 +146,11 @@
<script> <script>
layui.use(['jquery', 'form', 'layer'], function () { layui.use(['jquery', 'form', 'layer', 'upload'], function () {
var $ = layui.jquery; var $ = layui.jquery;
var form = layui.form; var form = layui.form;
var upload = layui.upload;
form.on('radio(status)', function (data) { form.on('radio(status)', function (data) {
var block = $('#closed-tips-block'); var block = $('#closed-tips-block');
@ -143,6 +170,38 @@
} }
}); });
upload.render({
elem: '#upload-logo',
url: '/admin/upload/content/img',
exts: 'gif|jpg|png',
before: function () {
layer.load();
},
done: function (res, index, upload) {
$('input[name=logo]').val(res.data.src);
layer.closeAll('loading');
},
error: function (index, upload) {
layer.msg('上传文件失败', {icon: 2});
}
});
upload.render({
elem: '#upload-favicon',
url: '/admin/upload/content/img',
exts: 'gif|jpg|png|ico',
before: function () {
layer.load();
},
done: function (res, index, upload) {
$('input[name=favicon]').val(res.data.src);
layer.closeAll('loading');
},
error: function (index, upload) {
layer.msg('上传文件失败', {icon: 2});
}
});
}); });
</script> </script>

View File

@ -1,16 +0,0 @@
{% extends 'templates/main.volt' %}
{% block content %}
<div class="layui-tab layui-tab-brief">
<ul class="layui-tab-title kg-tab-title">
<li class="layui-this">公众号</li>
</ul>
<div class="layui-tab-content">
<div class="layui-tab-item layui-show">
{{ partial('setting/wechat_oa') }}
</div>
</div>
</div>
{% endblock %}

View File

@ -1,104 +1,24 @@
{% set notice_template = oa.notice_template|json_decode %} {% extends 'templates/main.volt' %}
<form class="layui-form kg-form" method="POST" action="{{ url({'for':'admin.setting.wechat'}) }}"> {% block content %}
<div class="layui-form-item">
<label class="layui-form-label">开启</label> <div class="layui-tab layui-tab-brief">
<div class="layui-input-block"> <ul class="layui-tab-title kg-tab-title">
<input type="radio" name="enabled" value="1" title="是" {% if oa.enabled == "1" %}checked="checked"{% endif %}> <li class="layui-this">基本设置</li>
<input type="radio" name="enabled" value="0" title="否" {% if oa.enabled == "0" %}checked="checked"{% endif %}> <li class="layui">模板消息</li>
<li class="layui">自定义菜单</li>
</ul>
<div class="layui-tab-content">
<div class="layui-tab-item layui-show">
{{ partial('setting/wechat_oa_basic') }}
</div>
<div class="layui-tab-item">
{{ partial('setting/wechat_oa_notice') }}
</div>
<div class="layui-tab-item">
{{ partial('setting/wechat_oa_menu') }}
</div>
</div> </div>
</div> </div>
<div class="layui-form-item">
<label class="layui-form-label">App ID</label> {% endblock %}
<div class="layui-input-block">
<input class="layui-input" type="text" name="app_id" value="{{ oa.app_id }}" lay-verify="required">
</div>
</div>
<div class="layui-form-item">
<label class="layui-form-label">App Secret</label>
<div class="layui-input-block">
<input class="layui-input" type="text" name="app_secret" value="{{ oa.app_secret }}" lay-verify="required">
</div>
</div>
<div class="layui-form-item">
<label class="layui-form-label">App Token</label>
<div class="layui-input-block">
<input class="layui-input" type="text" name="app_token" value="{{ oa.app_token }}" lay-verify="required">
</div>
</div>
<div class="layui-form-item">
<label class="layui-form-label">Aes Key</label>
<div class="layui-input-block">
<input class="layui-input" type="text" name="aes_key" value="{{ oa.aes_key }}" lay-verify="required">
</div>
</div>
<div class="layui-form-item">
<label class="layui-form-label">Notify Url</label>
<div class="layui-input-block">
<input class="layui-input" type="text" name="notify_url" value="{{ oa.notify_url }}" lay-verify="required">
</div>
</div>
<div class="layui-form-item">
<label class="layui-form-label"></label>
<div class="layui-input-block">
<button class="layui-btn" lay-submit="true" lay-filter="go">提交</button>
<button type="button" class="kg-back layui-btn layui-btn-primary">返回</button>
<input type="hidden" name="section" value="wechat.oa">
</div>
</div>
</form>
<fieldset class="layui-elem-field layui-field-title">
<legend>模板配置</legend>
</fieldset>
<form class="layui-form kg-form" method="POST" action="{{ url({'for':'admin.setting.wechat'}) }}">
<table class="layui-table kg-table layui-form">
<colgroup>
<col width="12%">
<col width="40%">
<col>
</colgroup>
<thead>
<tr>
<th>名称</th>
<th>模板编号</th>
<th></th>
</tr>
</thead>
<tbody>
<tr>
<td>登录成功通知</td>
<td><input class="layui-input" type="text" name="notice_template[account_login]" value="{{ notice_template.account_login }}" lay-verify="required"></td>
<td></td>
</tr>
<tr>
<td>购买成功提醒</td>
<td><input class="layui-input" type="text" name="notice_template[order_finish]" value="{{ notice_template.order_finish }}" lay-verify="required"></td>
<td></td>
</tr>
<tr>
<td>退款成功通知</td>
<td><input class="layui-input" type="text" name="notice_template[refund_finish]" value="{{ notice_template.refund_finish }}" lay-verify="required"></td>
<td></td>
</tr>
<tr>
<td>课程直播提醒</td>
<td><input class="layui-input" type="text" name="notice_template[live_begin]" value="{{ notice_template.live_begin }}" lay-verify="required"></td>
<td></td>
</tr>
<tr>
<td>咨询结果通知</td>
<td><input class="layui-input" type="text" name="notice_template[consult_reply]" value="{{ notice_template.consult_reply }}" lay-verify="required"></td>
<td></td>
</tr>
</tbody>
</table>
<br>
<div class="layui-form-item">
<label class="layui-form-label"></label>
<div class="layui-input-block">
<button class="layui-btn" lay-submit="true" lay-filter="go">提交</button>
<button type="button" class="kg-back layui-btn layui-btn-primary">返回</button>
<input type="hidden" name="section" value="wechat.oa">
</div>
</div>
</form>

View File

@ -0,0 +1,66 @@
<form class="layui-form kg-form" method="POST" action="{{ url({'for':'admin.setting.wechat_oa'}) }}">
<div class="layui-form-item">
<label class="layui-form-label">开启</label>
<div class="layui-input-block">
<input type="radio" name="enabled" value="1" title="是" {% if oa.enabled == "1" %}checked="checked"{% endif %}>
<input type="radio" name="enabled" value="0" title="否" {% if oa.enabled == "0" %}checked="checked"{% endif %}>
</div>
</div>
<div class="layui-form-item">
<label class="layui-form-label">App ID</label>
<div class="layui-input-block">
<input class="layui-input" type="text" name="app_id" value="{{ oa.app_id }}" lay-verify="required">
</div>
</div>
<div class="layui-form-item">
<label class="layui-form-label">App Secret</label>
<div class="layui-input-block">
<input class="layui-input" type="text" name="app_secret" value="{{ oa.app_secret }}" lay-verify="required">
</div>
</div>
<div class="layui-form-item">
<label class="layui-form-label">App Token</label>
<div class="layui-input-block">
<input class="layui-input" type="text" name="app_token" value="{{ oa.app_token }}" lay-verify="required">
</div>
</div>
<div class="layui-form-item">
<label class="layui-form-label">Aes Key</label>
<div class="layui-input-block">
<input class="layui-input" type="text" name="aes_key" value="{{ oa.aes_key }}" lay-verify="required">
</div>
</div>
<div class="layui-form-item">
<label class="layui-form-label">Notify Url</label>
<div class="layui-input-block">
<input class="layui-input" type="text" name="notify_url" value="{{ oa.notify_url }}" lay-verify="required">
</div>
</div>
<div class="layui-form-item">
<label class="layui-form-label"></label>
<div class="layui-input-block">
<button class="layui-btn" lay-submit="true" lay-filter="go">提交</button>
<button type="button" class="kg-back layui-btn layui-btn-primary">返回</button>
<input type="hidden" name="section" value="wechat.oa">
</div>
</div>
</form>
<form class="layui-form kg-form" method="POST" action="{{ url({'for':'admin.test.wechat_oa'}) }}">
<fieldset class="layui-elem-field layui-field-title">
<legend>接口测试</legend>
</fieldset>
<div class="layui-form-item">
<label class="layui-form-label">请求方法</label>
<div class="layui-input-block">
<input class="layui-input" type="text" name="file" value="qrcode/create" readonly="readonly">
</div>
</div>
<div class="layui-form-item">
<label class="layui-form-label"></label>
<div class="layui-input-block">
<button class="layui-btn" lay-submit="true" lay-filter="go">提交</button>
<button type="button" class="kg-back layui-btn layui-btn-primary">返回</button>
</div>
</div>
</form>

View File

@ -0,0 +1,42 @@
<form class="layui-form kg-form" method="POST" action="{{ url({'for':'admin.setting.wechat_oa'}) }}">
<table class="layui-table kg-table layui-form">
<colgroup>
<col width="15%">
<col width="20%">
<col>
</colgroup>
<thead>
<tr>
<th>层级</th>
<th>名称</th>
<th>链接</th>
</tr>
</thead>
<tbody>
{% for i,top in oa.menu %}
<tr>
<td>├──</td>
<td><input class="layui-input" type="text" name="menu[{{ i }}][name]" value="{{ top.name }}" placeholder="一级菜单最多4个汉字" lay-verify="required"></td>
<td><input class="layui-input" type="text" name="menu[{{ i }}][url]" value="{{ top.url }}" placeholder="网页链接"></td>
</tr>
{% for j,sub in top.children %}
<tr>
<td><span style="padding: 0 15px;"></span>├──</td>
<td><input class="layui-input" type="text" name="menu[{{ i }}][children][{{ j }}][name]" value="{{ sub.name }}" placeholder="二级菜单最多7个汉字"></td>
<td><input class="layui-input" type="text" name="menu[{{ i }}][children][{{ j }}][url]" value="{{ sub.url }}" placeholder="网页链接"></td>
</tr>
{% endfor %}
{% endfor %}
</tbody>
</table>
<br>
<div class="layui-form-item">
<label class="layui-form-label"></label>
<div class="layui-input-block">
<button class="layui-btn" lay-submit="true" lay-filter="go">提交</button>
<button type="button" class="kg-back layui-btn layui-btn-primary">返回</button>
<input type="hidden" name="section" value="wechat.oa">
</div>
</div>
</form>

View File

@ -0,0 +1,47 @@
{% set notice_template = oa.notice_template|json_decode %}
<form class="layui-form kg-form" method="POST" action="{{ url({'for':'admin.setting.wechat_oa'}) }}">
<table class="layui-table kg-table layui-form">
<colgroup>
<col width="15%">
<col>
</colgroup>
<thead>
<tr>
<th>模板名称</th>
<th>模板编号</th>
</tr>
</thead>
<tbody>
<tr>
<td>登录成功通知</td>
<td><input class="layui-input" type="text" name="notice_template[account_login]" value="{{ notice_template.account_login }}" lay-verify="required"></td>
</tr>
<tr>
<td>购买成功提醒</td>
<td><input class="layui-input" type="text" name="notice_template[order_finish]" value="{{ notice_template.order_finish }}" lay-verify="required"></td>
</tr>
<tr>
<td>退款成功通知</td>
<td><input class="layui-input" type="text" name="notice_template[refund_finish]" value="{{ notice_template.refund_finish }}" lay-verify="required"></td>
</tr>
<tr>
<td>课程直播提醒</td>
<td><input class="layui-input" type="text" name="notice_template[live_begin]" value="{{ notice_template.live_begin }}" lay-verify="required"></td>
</tr>
<tr>
<td>咨询结果通知</td>
<td><input class="layui-input" type="text" name="notice_template[consult_reply]" value="{{ notice_template.consult_reply }}" lay-verify="required"></td>
</tr>
</tbody>
</table>
<br>
<div class="layui-form-item">
<label class="layui-form-label"></label>
<div class="layui-input-block">
<button class="layui-btn" lay-submit="true" lay-filter="go">提交</button>
<button type="button" class="kg-back layui-btn layui-btn-primary">返回</button>
<input type="hidden" name="section" value="wechat.oa">
</div>
</div>
</form>

View File

@ -2,6 +2,8 @@
namespace App\Http\Home\Controllers; namespace App\Http\Home\Controllers;
use App\Models\User as UserModel;
use App\Services\Auth\Home as HomeAuth;
use App\Services\Service as AppService; use App\Services\Service as AppService;
use App\Traits\Response as ResponseTrait; use App\Traits\Response as ResponseTrait;
@ -11,8 +13,27 @@ use App\Traits\Response as ResponseTrait;
class ErrorController extends \Phalcon\Mvc\Controller class ErrorController extends \Phalcon\Mvc\Controller
{ {
/**
* @var array
*/
protected $siteInfo;
/**
* @var UserModel
*/
protected $authUser;
use ResponseTrait; use ResponseTrait;
public function initialize()
{
$this->siteInfo = $this->getSiteInfo();
$this->authUser = $this->getAuthUser();
$this->view->setVar('site_info', $this->siteInfo);
$this->view->setVar('auth_user', $this->authUser);
}
/** /**
* @Get("/400", name="home.error.400") * @Get("/400", name="home.error.400")
*/ */
@ -82,4 +103,21 @@ class ErrorController extends \Phalcon\Mvc\Controller
$this->view->setVar('message', $siteInfo['closed_tips']); $this->view->setVar('message', $siteInfo['closed_tips']);
} }
protected function getSiteInfo()
{
$appService = new AppService();
return $appService->getSettings('site');
}
protected function getAuthUser()
{
/**
* @var HomeAuth $auth
*/
$auth = $this->getDI()->get('auth');
return $auth->getCurrentUser();
}
} }

View File

@ -2,6 +2,7 @@
namespace App\Http\Home\Controllers; namespace App\Http\Home\Controllers;
use App\Http\Home\Services\Index as IndexService;
use App\Services\Logic\Help\HelpInfo as HelpInfoService; use App\Services\Logic\Help\HelpInfo as HelpInfoService;
use App\Services\Logic\Help\HelpList as HelpListService; use App\Services\Logic\Help\HelpList as HelpListService;
@ -12,7 +13,7 @@ class HelpController extends Controller
{ {
/** /**
* @Get("/index", name="home.help.index") * @Get("/", name="home.help.index")
*/ */
public function indexAction() public function indexAction()
{ {
@ -34,9 +35,19 @@ class HelpController extends Controller
$help = $service->handle($id); $help = $service->handle($id);
$featuredCourses = $this->getFeaturedCourses();
$this->seo->prependTitle(['帮助', $help['title']]); $this->seo->prependTitle(['帮助', $help['title']]);
$this->view->setVar('help', $help); $this->view->setVar('help', $help);
$this->view->setVar('featured_courses', $featuredCourses);
}
protected function getFeaturedCourses()
{
$service = new IndexService();
return $service->getSimpleFeaturedCourses();
} }
} }

View File

@ -4,6 +4,7 @@ namespace App\Http\Home\Controllers;
use App\Models\User as UserModel; use App\Models\User as UserModel;
use App\Services\Auth\Home as HomeAuth; use App\Services\Auth\Home as HomeAuth;
use App\Services\Service as AppService;
use App\Traits\Response as ResponseTrait; use App\Traits\Response as ResponseTrait;
use App\Traits\Security as SecurityTrait; use App\Traits\Security as SecurityTrait;
use Phalcon\Mvc\Dispatcher; use Phalcon\Mvc\Dispatcher;
@ -11,6 +12,11 @@ use Phalcon\Mvc\Dispatcher;
class LayerController extends \Phalcon\Mvc\Controller class LayerController extends \Phalcon\Mvc\Controller
{ {
/**
* @var array
*/
protected $siteInfo;
/** /**
* @var UserModel * @var UserModel
*/ */
@ -33,11 +39,20 @@ class LayerController extends \Phalcon\Mvc\Controller
public function initialize() public function initialize()
{ {
$this->siteInfo = $this->getSiteInfo();
$this->authUser = $this->getAuthUser(); $this->authUser = $this->getAuthUser();
$this->view->setVar('site_info', $this->siteInfo);
$this->view->setVar('auth_user', $this->authUser); $this->view->setVar('auth_user', $this->authUser);
} }
protected function getSiteInfo()
{
$appService = new AppService();
return $appService->getSettings('site');
}
protected function getAuthUser() protected function getAuthUser()
{ {
/** /**

View File

@ -2,6 +2,7 @@
namespace App\Http\Home\Controllers; namespace App\Http\Home\Controllers;
use App\Http\Home\Services\Index as IndexService;
use App\Services\Logic\Page\PageInfo as PageInfoService; use App\Services\Logic\Page\PageInfo as PageInfoService;
/** /**
@ -19,9 +20,19 @@ class PageController extends Controller
$page = $service->handle($id); $page = $service->handle($id);
$this->seo->prependTitle(['单页', $page['title']]); $featuredCourses = $this->getFeaturedCourses();
$this->seo->prependTitle($page['title']);
$this->view->setVar('page', $page); $this->view->setVar('page', $page);
$this->view->setVar('featured_courses', $featuredCourses);
}
protected function getFeaturedCourses()
{
$service = new IndexService();
return $service->getSimpleFeaturedCourses();
} }
} }

View File

@ -3,9 +3,9 @@
namespace App\Http\Home\Services; namespace App\Http\Home\Services;
use App\Models\WechatSubscribe as WechatSubscribeModel; use App\Models\WechatSubscribe as WechatSubscribeModel;
use App\Repos\User as UserRepo;
use App\Repos\WechatSubscribe as WechatSubscribeRepo; use App\Repos\WechatSubscribe as WechatSubscribeRepo;
use App\Services\Wechat as WechatService; use App\Services\Wechat as WechatService;
use App\Validators\User as UserValidator;
use EasyWeChat\Kernel\Messages\Text as TextMessage; use EasyWeChat\Kernel\Messages\Text as TextMessage;
class WechatOfficialAccount extends Service class WechatOfficialAccount extends Service
@ -50,7 +50,7 @@ class WechatOfficialAccount extends Service
{ {
$service = new WechatService(); $service = new WechatService();
$service->logger->debug('Received Message ' . json_encode($message)); $service->logger->info('Received Message ' . json_encode($message));
switch ($message['MsgType']) { switch ($message['MsgType']) {
case 'event': case 'event':
@ -74,7 +74,7 @@ class WechatOfficialAccount extends Service
return $this->handleLocationEvent($message); return $this->handleLocationEvent($message);
break; break;
default: default:
return $this->emptyReplyMessage(); return $this->noMatchReply();
break; break;
} }
break; break;
@ -100,7 +100,7 @@ class WechatOfficialAccount extends Service
return $this->handleLinkReply($message); return $this->handleLinkReply($message);
break; break;
default: default:
return $this->emptyReplyMessage(); return $this->noMatchReply();
break; break;
} }
} }
@ -108,16 +108,16 @@ class WechatOfficialAccount extends Service
protected function handleSubscribeEvent($message) protected function handleSubscribeEvent($message)
{ {
$openId = $message['FromUserName'] ?? ''; $openId = $message['FromUserName'] ?? '';
$eventKey = $message['EventKey'] ?? '';
if (!$eventKey) { $subscribeRepo = new WechatSubscribeRepo();
return $this->emptyReplyMessage();
$subscribe = $subscribeRepo->findByOpenId($openId);
if ($subscribe && $subscribe->deleted == 1) {
$subscribe->deleted = 0;
$subscribe->update();
} }
$userId = str_replace('qrscene_', '', $eventKey);
$this->handleSubscribeRelation($userId, $openId);
return new TextMessage('开心呀,我们又多了一个小伙伴!'); return new TextMessage('开心呀,我们又多了一个小伙伴!');
} }
@ -129,7 +129,7 @@ class WechatOfficialAccount extends Service
$subscribe = $subscribeRepo->findByOpenId($openId); $subscribe = $subscribeRepo->findByOpenId($openId);
if ($subscribe) { if ($subscribe && $subscribe->deleted == 0) {
$subscribe->deleted = 1; $subscribe->deleted = 1;
$subscribe->update(); $subscribe->update();
} }
@ -139,100 +139,95 @@ class WechatOfficialAccount extends Service
protected function handleScanEvent($message) protected function handleScanEvent($message)
{ {
/**
* 注意:当已关注过用户扫码时,"EventKey"没有带"qrscene_"前缀
*/
$openId = $message['FromUserName'] ?? ''; $openId = $message['FromUserName'] ?? '';
$eventKey = $message['EventKey'] ?? ''; $eventKey = $message['EventKey'] ?? '';
$userId = $eventKey;
$this->handleSubscribeRelation($userId, $openId); $userId = str_replace('qrscene_', '', $eventKey);
}
protected function handleClickEvent($message) $userRepo = new UserRepo();
{
$this->defaultReplyMessage();
}
protected function handleViewEvent($message) $user = $userRepo->findById($userId);
{
$this->defaultReplyMessage();
}
protected function handleLocationEvent($message) if (!$user) return;
{
$this->defaultReplyMessage();
}
protected function handleTextReply($message)
{
return $this->defaultReplyMessage();
}
protected function handleImageReply($message)
{
return $this->defaultReplyMessage();
}
protected function handleVoiceReply($message)
{
return $this->defaultReplyMessage();
}
protected function handleVideoReply($message)
{
return $this->defaultReplyMessage();
}
protected function handleShortVideoReply($message)
{
return $this->defaultReplyMessage();
}
protected function handleLocationReply($message)
{
return $this->defaultReplyMessage();
}
protected function handleLinkReply($message)
{
return $this->defaultReplyMessage();
}
protected function emptyReplyMessage()
{
return new TextMessage('');
}
protected function defaultReplyMessage()
{
return new TextMessage('没有匹配的服务,如有需要请联系客服!');
}
protected function handleSubscribeRelation($userId, $openId)
{
$validator = new UserValidator();
$validator->checkUser($userId);
$subscribeRepo = new WechatSubscribeRepo(); $subscribeRepo = new WechatSubscribeRepo();
$subscribe = $subscribeRepo->findByOpenId($openId); $subscribe = $subscribeRepo->findByOpenId($openId);
if ($subscribe) { if ($subscribe) {
if ($subscribe->user_id != $userId) {
$subscribe->user_id = $userId;
}
if ($subscribe->deleted == 1) { if ($subscribe->deleted == 1) {
$subscribe->deleted = 0; $subscribe->deleted = 0;
$subscribe->update();
} }
$subscribe->update();
} else { } else {
$subscribe = $subscribeRepo->findSubscribe($userId, $openId); $subscribe = new WechatSubscribeModel();
if (!$subscribe) { $subscribe->user_id = $userId;
$subscribe = new WechatSubscribeModel(); $subscribe->open_id = $openId;
$subscribe->user_id = $userId; $subscribe->create();
$subscribe->open_id = $openId;
$subscribe->create();
}
} }
} }
protected function handleClickEvent($message)
{
return $this->emptyReply();
}
protected function handleViewEvent($message)
{
return $this->emptyReply();
}
protected function handleLocationEvent($message)
{
return $this->emptyReply();
}
protected function handleTextReply($message)
{
return $this->emptyReply();
}
protected function handleImageReply($message)
{
return $this->emptyReply();
}
protected function handleVoiceReply($message)
{
return $this->emptyReply();
}
protected function handleVideoReply($message)
{
return $this->emptyReply();
}
protected function handleShortVideoReply($message)
{
return $this->emptyReply();
}
protected function handleLocationReply($message)
{
return $this->emptyReply();
}
protected function handleLinkReply($message)
{
return $this->emptyReply();
}
protected function emptyReply()
{
return null;
}
protected function noMatchReply()
{
return new TextMessage('没有匹配的服务哦!');
}
} }

View File

@ -21,6 +21,7 @@
<button class="layui-btn" lay-submit="true" lay-filter="go">提交</button> <button class="layui-btn" lay-submit="true" lay-filter="go">提交</button>
<button type="reset" class="layui-btn layui-btn-primary">重置</button> <button type="reset" class="layui-btn layui-btn-primary">重置</button>
<input type="hidden" name="chapter_id" value="{{ request.get('chapter_id') }}"> <input type="hidden" name="chapter_id" value="{{ request.get('chapter_id') }}">
<input type="hidden" name="course_id" value="{{ request.get('course_id') }}">
</div> </div>
</div> </div>
</form> </form>

View File

@ -4,24 +4,24 @@
{% set consult.answer = consult.answer ? consult.answer : '请耐心等待回复吧' %} {% set consult.answer = consult.answer ? consult.answer : '请耐心等待回复吧' %}
<div class="consult-info clearfix"> <div class="consult-info">
{% if consult.course.id is defined %} {% if consult.course.id is defined %}
<div class="item"> <div class="item clearfix">
<div class="label">课程:</div> <div class="label">课程:</div>
<div class="title">{{ consult.course.title }}</div> <div class="title">{{ consult.course.title }}</div>
</div> </div>
{% endif %} {% endif %}
{% if consult.chapter.id is defined %} {% if consult.chapter.id is defined %}
<div class="item"> <div class="item clearfix">
<div class="label">章节:</div> <div class="label">章节:</div>
<div class="title">{{ consult.chapter.title }}</div> <div class="title">{{ consult.chapter.title }}</div>
</div> </div>
{% endif %} {% endif %}
<div class="item"> <div class="item clearfix">
<div class="label">咨询:</div> <div class="label">咨询:</div>
<div class="content">{{ consult.question }}</div> <div class="content">{{ consult.question }}</div>
</div> </div>
<div class="item"> <div class="item clearfix">
<div class="label">回复:</div> <div class="label">回复:</div>
<div class="content">{{ consult.answer }}</div> <div class="content">{{ consult.answer }}</div>
</div> </div>

View File

@ -4,10 +4,11 @@
{{ partial('macros/course') }} {{ partial('macros/course') }}
{% set favorite_title = course.me.favorited ? '取消收藏' : '收藏' %} {% set favorite_title = course.me.favorited ? '取消收藏' : '收藏课程' %}
{% set favorite_star = course.me.favorited ? 'layui-icon-star-fill' : 'layui-icon-star' %} {% set favorite_star = course.me.favorited ? 'layui-icon-star-fill' : 'layui-icon-star' %}
{% set full_course_url = full_url({'for':'home.course.show','id':course.id}) %} {% set full_course_url = full_url({'for':'home.course.show','id':course.id}) %}
{% set favorite_url = url({'for':'home.course.favorite','id':course.id}) %} {% set favorite_url = url({'for':'home.course.favorite','id':course.id}) %}
{% set consult_url = url({'for':'home.consult.add'},{'course_id':course.id}) %}
{% set qrcode_url = url({'for':'home.qrcode'},{'text':full_course_url}) %} {% set qrcode_url = url({'for':'home.qrcode'},{'text':full_course_url}) %}
<div class="breadcrumb"> <div class="breadcrumb">
@ -18,6 +19,9 @@
</span> </span>
<span class="share"> <span class="share">
<a href="javascript:" title="{{ favorite_title }}" data-url="{{ favorite_url }}"><i class="layui-icon {{ favorite_star }} icon-star"></i></a> <a href="javascript:" title="{{ favorite_title }}" data-url="{{ favorite_url }}"><i class="layui-icon {{ favorite_star }} icon-star"></i></a>
{% if course.market_price > 0 %}
<a href="javascript:" title="课程咨询" data-url="{{ consult_url }}"><i class="layui-icon layui-icon-help icon-help"></i></a>
{% endif %}
<a href="javascript:" title="分享到微信" data-url="{{ qrcode_url }}"><i class="layui-icon layui-icon-login-wechat icon-wechat"></i></a> <a href="javascript:" title="分享到微信" data-url="{{ qrcode_url }}"><i class="layui-icon layui-icon-login-wechat icon-wechat"></i></a>
<a href="javascript:" title="分享到QQ空间"><i class="layui-icon layui-icon-login-qq icon-qq"></i></a> <a href="javascript:" title="分享到QQ空间"><i class="layui-icon layui-icon-login-qq icon-qq"></i></a>
<a href="javascript:" title="分享到微博"><i class="layui-icon layui-icon-login-weibo icon-weibo"></i></a> <a href="javascript:" title="分享到微博"><i class="layui-icon layui-icon-login-weibo icon-weibo"></i></a>
@ -54,7 +58,7 @@
{{ partial('course/show_catalog') }} {{ partial('course/show_catalog') }}
</div> </div>
<div class="layui-tab-item"> <div class="layui-tab-item">
<div class="course-details" id="preview">{{ course.details }}</div> <div class="course-details markdown-body">{{ course.details }}</div>
</div> </div>
{% if show_tab_packages %} {% if show_tab_packages %}
{% set packages_url = url({'for':'home.course.packages','id':course.id}) %} {% set packages_url = url({'for':'home.course.packages','id':course.id}) %}
@ -108,14 +112,12 @@
{% block link_css %} {% block link_css %}
{{ css_link('https://cdn.jsdelivr.net/npm/vditor/dist/index.css', false) }} {{ css_link('home/css/markdown.css') }}
{% endblock %} {% endblock %}
{% block include_js %} {% block include_js %}
{{ js_include('https://cdn.jsdelivr.net/npm/vditor/dist/method.min.js', false) }}
{{ js_include('home/js/markdown.preview.js') }}
{{ js_include('home/js/course.show.js') }} {{ js_include('home/js/course.show.js') }}
{{ js_include('home/js/course.share.js') }} {{ js_include('home/js/course.share.js') }}

View File

@ -1,8 +1,12 @@
{% if course.me.owned == 0 and course.market_price > 0 %} {% if course.me.owned == 0 and course.market_price > 0 %}
{% set order_url = url({'for':'home.order.confirm'},{'item_id':course.id,'item_type':1}) %} {% set order_url = url({'for':'home.order.confirm'},{'item_id':course.id,'item_type':1}) %}
<div class="sidebar wrap"> {% set live_model_ok = course.model == 2 and course.attrs.end_date < date('Y-m-d') %}
<button class="layui-btn layui-btn-fluid layui-bg-red btn-buy" data-url="{{ order_url }}">立即购买</button> {% set other_model_ok = course.model != 2 %}
</div> {% if live_model_ok or other_model_ok %}
<div class="sidebar wrap">
<button class="layui-btn layui-btn-fluid layui-bg-red btn-buy" data-url="{{ order_url }}">立即购买</button>
</div>
{% endif %}
{% endif %} {% endif %}
{% if course.market_price == 0 %} {% if course.market_price == 0 %}

View File

@ -2,6 +2,8 @@
{% block content %} {% block content %}
{{ partial('macros/course') }}
<div class="breadcrumb"> <div class="breadcrumb">
<span class="layui-breadcrumb"> <span class="layui-breadcrumb">
<a href="/">首页</a> <a href="/">首页</a>
@ -10,21 +12,32 @@
</span> </span>
</div> </div>
<div class="page-info wrap"> <div class="layout-main clearfix">
<div class="content" id="preview">{{ help.content }}</div> <div class="layout-content">
<div class="page-info wrap">
<div class="content markdown-body">{{ help.content }}</div>
</div>
</div>
<div class="layout-sidebar">
{% if featured_courses %}
<div class="sidebar">
<div class="layui-card">
<div class="layui-card-header">推荐课程</div>
<div class="layui-card-body">
{% for course in featured_courses %}
{{ sidebar_course_card(course) }}
{% endfor %}
</div>
</div>
</div>
{% endif %}
</div>
</div> </div>
{% endblock %} {% endblock %}
{% block link_css %} {% block link_css %}
{{ css_link('https://cdn.jsdelivr.net/npm/vditor/dist/index.css', false) }} {{ css_link('home/css/markdown.css') }}
{% endblock %}
{% block include_js %}
{{ js_include('https://cdn.jsdelivr.net/npm/vditor/dist/method.min.js', false) }}
{{ js_include('home/js/markdown.preview.js') }}
{% endblock %} {% endblock %}

View File

@ -2,29 +2,41 @@
{% block content %} {% block content %}
{{ partial('macros/course') }}
<div class="breadcrumb"> <div class="breadcrumb">
<span class="layui-breadcrumb"> <span class="layui-breadcrumb">
<a href="/">首页</a> <a href="/">首页</a>
<a><cite>单页</cite></a>
<a><cite>{{ page.title }}</cite></a> <a><cite>{{ page.title }}</cite></a>
</span> </span>
</div> </div>
<div class="page-info wrap"> <div class="layout-main clearfix">
<div class="content layui-hide" id="preview">{{ page.content }}</div> <div class="layout-content">
<div class="page-info wrap">
<div class="content markdown-body">{{ page.content }}</div>
</div>
</div>
<div class="layout-sidebar">
{% if featured_courses %}
<div class="sidebar">
<div class="layui-card">
<div class="layui-card-header">推荐课程</div>
<div class="layui-card-body">
{% for course in featured_courses %}
{{ sidebar_course_card(course) }}
{% endfor %}
</div>
</div>
</div>
{% endif %}
</div>
</div> </div>
{% endblock %} {% endblock %}
{% block link_css %} {% block link_css %}
{{ css_link('https://cdn.jsdelivr.net/npm/vditor/dist/index.css', false) }} {{ css_link('home/css/markdown.css') }}
{% endblock %}
{% block include_js %}
{{ js_include('https://cdn.jsdelivr.net/npm/vditor/dist/method.min.js', false) }}
{{ js_include('home/js/markdown.preview.js') }}
{% endblock %} {% endblock %}

View File

@ -1,4 +1,10 @@
<div class="logo"></div> <div class="logo">
{% if site_info.logo %}
{{ image(site_info.logo,false) }}
{% else %}
{{ image('logo.png') }}
{% endif %}
</div>
<div class="top-nav"> <div class="top-nav">
<ul class="layui-nav"> <ul class="layui-nav">
@ -21,7 +27,7 @@
{% set s_query = request.get('query',['trim','striptags'],'') %} {% set s_query = request.get('query',['trim','striptags'],'') %}
{% set s_url = url({'for':'home.search.index'}) %} {% set s_url = url({'for':'home.search.index'}) %}
<div class="user layui-layout-right"> <div class="user">
<ul class="layui-nav"> <ul class="layui-nav">
<li class="layui-nav-item"> <li class="layui-nav-item">
<a href="javascript:" class="nav-search" data-type="{{ s_type }}" data-query="{{ s_query }}" data-url="{{ s_url }}"><i class="layui-icon layui-icon-search"></i> 搜索</a> <a href="javascript:" class="nav-search" data-type="{{ s_type }}" data-query="{{ s_query }}" data-url="{{ s_url }}"><i class="layui-icon layui-icon-search"></i> 搜索</a>

View File

@ -5,7 +5,7 @@
{% set item.title = item.title ? item.title : '小小教书匠' %} {% set item.title = item.title ? item.title : '小小教书匠' %}
{% set item.about = item.about ? item.about : '这个人很懒,什么都没留下' %} {% set item.about = item.about ? item.about : '这个人很懒,什么都没留下' %}
{% set user_url = url({'for':'home.teacher.show','id':item.id}) %} {% set user_url = url({'for':'home.teacher.show','id':item.id}) %}
<div class="layui-col-md2"> <div class="layui-col-md3">
<div class="user-card"> <div class="user-card">
<div class="avatar"> <div class="avatar">
<a href="{{ user_url }}" title="{{ item.about }}"> <a href="{{ user_url }}" title="{{ item.about }}">

View File

@ -4,8 +4,12 @@
<meta http-equiv="Content-Type" content="text/html; charset=UTF-8"> <meta http-equiv="Content-Type" content="text/html; charset=UTF-8">
<meta http-equiv="X-UA-Compatible" content="IE=edge,chrome=1"> <meta http-equiv="X-UA-Compatible" content="IE=edge,chrome=1">
<meta name="viewport" content="width=device-width, initial-scale=1, maximum-scale=1"> <meta name="viewport" content="width=device-width, initial-scale=1, maximum-scale=1">
<title>出错啦</title> <title>出错啦 - {{ site_info.title }}</title>
{{ icon_link("favicon.ico") }} {% if site_info.favicon %}
{{ icon_link(site_info.favicon,false) }}
{% else %}
{{ icon_link('favicon.ico') }}
{% endif %}
{{ css_link("lib/layui/css/layui.css") }} {{ css_link("lib/layui/css/layui.css") }}
{{ css_link("home/css/error.css") }} {{ css_link("home/css/error.css") }}
</head> </head>

View File

@ -5,8 +5,12 @@
<meta http-equiv="X-UA-Compatible" content="IE=edge,chrome=1"> <meta http-equiv="X-UA-Compatible" content="IE=edge,chrome=1">
<meta name="viewport" content="width=device-width, initial-scale=1, maximum-scale=1"> <meta name="viewport" content="width=device-width, initial-scale=1, maximum-scale=1">
<meta name="csrf-token" content="{{ csrfToken.getToken() }}"> <meta name="csrf-token" content="{{ csrfToken.getToken() }}">
<title>酷瓜云课堂</title> <title>{{ site_info.title }}</title>
{{ icon_link('favicon.ico') }} {% if site_info.favicon %}
{{ icon_link(site_info.favicon,false) }}
{% else %}
{{ icon_link('favicon.ico') }}
{% endif %}
{{ css_link('lib/layui/css/layui.css') }} {{ css_link('lib/layui/css/layui.css') }}
{{ css_link('home/css/common.css') }} {{ css_link('home/css/common.css') }}
{% block link_css %}{% endblock %} {% block link_css %}{% endblock %}

View File

@ -8,7 +8,11 @@
<meta name="description" content="{{ seo.getDescription() }}"> <meta name="description" content="{{ seo.getDescription() }}">
<meta name="csrf-token" content="{{ csrfToken.getToken() }}"> <meta name="csrf-token" content="{{ csrfToken.getToken() }}">
<title>{{ seo.getTitle() }}</title> <title>{{ seo.getTitle() }}</title>
{{ icon_link('favicon.ico') }} {% if site_info.favicon %}
{{ icon_link(site_info.favicon,false) }}
{% else %}
{{ icon_link('favicon.ico') }}
{% endif %}
{{ css_link('lib/layui/css/layui.css') }} {{ css_link('lib/layui/css/layui.css') }}
{{ css_link('home/css/common.css') }} {{ css_link('home/css/common.css') }}
{% block link_css %}{% endblock %} {% block link_css %}{% endblock %}

View File

@ -11,7 +11,7 @@ class AppInfo
protected $link = 'https://gitee.com/koogua'; protected $link = 'https://gitee.com/koogua';
protected $version = '1.2.3'; protected $version = '1.2.4';
public function __get($name) public function __get($name)
{ {

View File

@ -277,9 +277,13 @@ function kg_cos_img_style_trim($path)
*/ */
function kg_parse_markdown($content) function kg_parse_markdown($content)
{ {
return preg_replace_callback('/\/img\/content\/(.*?)\)/', function ($matches) { $content = preg_replace_callback('/\/img\/content\/(.*?)\)/', function ($matches) {
return '/img/content/' . trim($matches[1]) . '!content_800'; return sprintf('/img/content/%s!content_800)', trim($matches[1]));
}, $content); }, $content);
$parser = new HyperDown\Parser();
return $parser->makeHtml($content);
} }
/** /**

View File

@ -77,6 +77,7 @@ class FileInfo
'png' => 'image/png', 'png' => 'image/png',
'webp' => 'image/webp', 'webp' => 'image/webp',
'bmp' => 'image/bmp', 'bmp' => 'image/bmp',
'ico' => 'image/x-icon',
'tif' => 'image/tiff', 'tif' => 'image/tiff',
'tiff' => 'image/tiff', 'tiff' => 'image/tiff',
'svg' => 'image/svg+xml', 'svg' => 'image/svg+xml',

View File

@ -16,7 +16,10 @@ class Account extends Repository
*/ */
public function findById($id) public function findById($id)
{ {
return AccountModel::findFirst($id); return AccountModel::findFirst([
'conditions' => 'id = :id:',
'bind' => ['id' => $id],
]);
} }
/** /**

View File

@ -68,7 +68,10 @@ class Audit extends Repository
*/ */
public function findById($id) public function findById($id)
{ {
return AuditModel::findFirst($id); return AuditModel::findFirst([
'conditions' => 'id = :id:',
'bind' => ['id' => $id],
]);
} }
/** /**

View File

@ -51,7 +51,10 @@ class Category extends Repository
*/ */
public function findById($id) public function findById($id)
{ {
return CategoryModel::findFirst($id); return CategoryModel::findFirst([
'conditions' => 'id = :id:',
'bind' => ['id' => $id],
]);
} }
/** /**

View File

@ -50,7 +50,10 @@ class Chapter extends Repository
*/ */
public function findById($id) public function findById($id)
{ {
return ChapterModel::findFirst($id); return ChapterModel::findFirst([
'conditions' => 'id = :id:',
'bind' => ['id' => $id],
]);
} }
/** /**

View File

@ -43,7 +43,10 @@ class Connect extends Repository
*/ */
public function findById($id) public function findById($id)
{ {
return ConnectModel::findFirst($id); return ConnectModel::findFirst([
'conditions' => 'id = :id:',
'bind' => ['id' => $id],
]);
} }
/** /**

View File

@ -78,7 +78,10 @@ class Consult extends Repository
*/ */
public function findById($id) public function findById($id)
{ {
return ConsultModel::findFirst($id); return ConsultModel::findFirst([
'conditions' => 'id = :id:',
'bind' => ['id' => $id],
]);
} }
/** /**

View File

@ -123,7 +123,10 @@ class Course extends Repository
*/ */
public function findById($id) public function findById($id)
{ {
return CourseModel::findFirst($id); return CourseModel::findFirst([
'conditions' => 'id = :id:',
'bind' => ['id' => $id],
]);
} }
/** /**

View File

@ -66,7 +66,10 @@ class Danmu extends Repository
*/ */
public function findById($id) public function findById($id)
{ {
return DanmuModel::findFirst($id); return DanmuModel::findFirst([
'conditions' => 'id = :id:',
'bind' => ['id' => $id],
]);
} }
/** /**

View File

@ -16,7 +16,10 @@ class Help extends Repository
*/ */
public function findById($id) public function findById($id)
{ {
return HelpModel::findFirst($id); return HelpModel::findFirst([
'conditions' => 'id = :id:',
'bind' => ['id' => $id],
]);
} }
/** /**

View File

@ -76,7 +76,10 @@ class ImGroup extends Repository
*/ */
public function findById($id) public function findById($id)
{ {
return ImGroupModel::findFirst($id); return ImGroupModel::findFirst([
'conditions' => 'id = :id:',
'bind' => ['id' => $id],
]);
} }
/** /**

View File

@ -65,7 +65,10 @@ class ImMessage extends Repository
*/ */
public function findById($id) public function findById($id)
{ {
return ImMessageModel::findFirst($id); return ImMessageModel::findFirst([
'conditions' => 'id = :id:',
'bind' => ['id' => $id],
]);
} }
/** /**

View File

@ -57,7 +57,10 @@ class ImNotice extends Repository
*/ */
public function findById($id) public function findById($id)
{ {
return ImNoticeModel::findFirst($id); return ImNoticeModel::findFirst([
'conditions' => 'id = :id:',
'bind' => ['id' => $id],
]);
} }
/** /**

View File

@ -60,7 +60,10 @@ class ImUser extends Repository
*/ */
public function findById($id) public function findById($id)
{ {
return ImUserModel::findFirst($id); return ImUserModel::findFirst([
'conditions' => 'id = :id:',
'bind' => ['id' => $id],
]);
} }
/** /**

View File

@ -60,7 +60,10 @@ class Learning extends Repository
*/ */
public function findById($id) public function findById($id)
{ {
return LearningModel::findFirst($id); return LearningModel::findFirst([
'conditions' => 'id = :id:',
'bind' => ['id' => $id],
]);
} }
/** /**

View File

@ -51,7 +51,10 @@ class Nav extends Repository
*/ */
public function findById($id) public function findById($id)
{ {
return NavModel::findFirst($id); return NavModel::findFirst([
'conditions' => 'id = :id:',
'bind' => ['id' => $id],
]);
} }
/** /**

View File

@ -79,7 +79,10 @@ class Order extends Repository
*/ */
public function findById($id) public function findById($id)
{ {
return OrderModel::findFirst($id); return OrderModel::findFirst([
'conditions' => 'id = :id:',
'bind' => ['id' => $id],
]);
} }
/** /**

View File

@ -60,7 +60,10 @@ class Package extends Repository
*/ */
public function findById($id) public function findById($id)
{ {
return PackageModel::findFirst($id); return PackageModel::findFirst([
'conditions' => 'id = :id:',
'bind' => ['id' => $id],
]);
} }
/** /**

View File

@ -54,7 +54,10 @@ class Page extends Repository
*/ */
public function findById($id) public function findById($id)
{ {
return PageModel::findFirst($id); return PageModel::findFirst([
'conditions' => 'id = :id:',
'bind' => ['id' => $id],
]);
} }
/** /**

View File

@ -66,7 +66,10 @@ class Refund extends Repository
*/ */
public function findById($id) public function findById($id)
{ {
return RefundModel::findFirst($id); return RefundModel::findFirst([
'conditions' => 'id = :id:',
'bind' => ['id' => $id],
]);
} }
/** /**

View File

@ -16,7 +16,10 @@ class Resource extends Repository
*/ */
public function findById($id) public function findById($id)
{ {
return ResourceModel::findFirst($id); return ResourceModel::findFirst([
'conditions' => 'id = :id:',
'bind' => ['id' => $id],
]);
} }
/** /**

View File

@ -77,7 +77,10 @@ class Review extends Repository
*/ */
public function findById($id) public function findById($id)
{ {
return ReviewModel::findFirst($id); return ReviewModel::findFirst([
'conditions' => 'id = :id:',
'bind' => ['id' => $id],
]);
} }
/** /**

View File

@ -33,7 +33,10 @@ class Reward extends Repository
*/ */
public function findById($id) public function findById($id)
{ {
return RewardModel::findFirst($id); return RewardModel::findFirst([
'conditions' => 'id = :id:',
'bind' => ['id' => $id],
]);
} }
/** /**

View File

@ -34,7 +34,10 @@ class Role extends Repository
*/ */
public function findById($id) public function findById($id)
{ {
return RoleModel::findFirst($id); return RoleModel::findFirst([
'conditions' => 'id = :id:',
'bind' => ['id' => $id],
]);
} }
/** /**

View File

@ -54,7 +54,10 @@ class Slide extends Repository
*/ */
public function findById($id) public function findById($id)
{ {
return SlideModel::findFirst($id); return SlideModel::findFirst([
'conditions' => 'id = :id:',
'bind' => ['id' => $id],
]);
} }
/** /**

View File

@ -56,7 +56,10 @@ class Topic extends Repository
*/ */
public function findById($id) public function findById($id)
{ {
return TopicModel::findFirst($id); return TopicModel::findFirst([
'conditions' => 'id = :id:',
'bind' => ['id' => $id],
]);
} }
/** /**

View File

@ -78,7 +78,10 @@ class Trade extends Repository
*/ */
public function findById($id) public function findById($id)
{ {
return TradeModel::findFirst($id); return TradeModel::findFirst([
'conditions' => 'id = :id:',
'bind' => ['id' => $id],
]);
} }
/** /**

View File

@ -16,7 +16,10 @@ class Upload extends Repository
*/ */
public function findById($id) public function findById($id)
{ {
return UploadModel::findFirst($id); return UploadModel::findFirst([
'conditions' => 'id = :id:',
'bind' => ['id' => $id],
]);
} }
/** /**

View File

@ -73,7 +73,10 @@ class User extends Repository
*/ */
public function findById($id) public function findById($id)
{ {
return UserModel::findFirst($id); return UserModel::findFirst([
'conditions' => 'id = :id:',
'bind' => ['id' => $id],
]);
} }
/** /**

View File

@ -33,7 +33,10 @@ class Vip extends Repository
*/ */
public function findById($id) public function findById($id)
{ {
return VipModel::findFirst($id); return VipModel::findFirst([
'conditions' => 'id = :id:',
'bind' => ['id' => $id],
]);
} }
/** /**

View File

@ -27,7 +27,10 @@ class WechatSubscribe extends Repository
*/ */
public function findById($id) public function findById($id)
{ {
return WechatSubscribeModel::findFirst($id); return WechatSubscribeModel::findFirst([
'conditions' => 'id = :id:',
'bind' => ['id' => $id],
]);
} }
/** /**

View File

@ -70,24 +70,19 @@ abstract class OAuth extends Service
*/ */
$crypt = Di::getDefault()->get('crypt'); $crypt = Di::getDefault()->get('crypt');
return $crypt->encryptBase64(rand(1000, 9999)); $text = rand(1000, 9999);
return $crypt->encryptBase64($text, null, true);
} }
public function checkState($state) public function checkState($state)
{ {
/**
* 注意事项:
* callback中的state参数并未做encode处理参数中含有"+"
* 获取参数的时候却自动做了decode处理"+"变成了空格
*/
$state = str_replace(' ', '+', $state);
/** /**
* @var $crypt Crypt * @var $crypt Crypt
*/ */
$crypt = Di::getDefault()->get('crypt'); $crypt = Di::getDefault()->get('crypt');
$value = $crypt->decryptBase64($state); $value = $crypt->decryptBase64($state, null, true);
if ($value < 1000 || $value > 9999) { if ($value < 1000 || $value > 9999) {
throw new \Exception('Invalid OAuth State Value'); throw new \Exception('Invalid OAuth State Value');

View File

@ -23,8 +23,6 @@ class Account extends Validator
$account = $accountRepo->findByEmail($name); $account = $accountRepo->findByEmail($name);
} elseif (CommonValidator::phone($name)) { } elseif (CommonValidator::phone($name)) {
$account = $accountRepo->findByPhone($name); $account = $accountRepo->findByPhone($name);
} else {
$account = $accountRepo->findById($name);
} }
if (!$account) { if (!$account) {
@ -122,6 +120,8 @@ class Account extends Validator
public function checkVerifyLogin($name, $code) public function checkVerifyLogin($name, $code)
{ {
$this->checkLoginName($name);
$account = $this->checkAccount($name); $account = $this->checkAccount($name);
$verify = new Verify(); $verify = new Verify();
@ -135,6 +135,8 @@ class Account extends Validator
public function checkUserLogin($name, $password) public function checkUserLogin($name, $password)
{ {
$this->checkLoginName($name);
$account = $this->checkAccount($name); $account = $this->checkAccount($name);
$hash = PasswordUtil::hash($password, $account->salt); $hash = PasswordUtil::hash($password, $account->salt);

View File

@ -21,7 +21,8 @@
"xiaochong0302/ip2region": "^1.0", "xiaochong0302/ip2region": "^1.0",
"robmorgan/phinx": "^0.12", "robmorgan/phinx": "^0.12",
"lcobucci/jwt": "^3.3", "lcobucci/jwt": "^3.3",
"overtrue/wechat": "^4.2" "overtrue/wechat": "^4.2",
"joyqi/hyper-down": "dev-master"
}, },
"require-dev": { "require-dev": {
"odan/phinx-migrations-generator": "^5.3", "odan/phinx-migrations-generator": "^5.3",

730
composer.lock generated
View File

@ -4,7 +4,7 @@
"Read more about it at https://getcomposer.org/doc/01-basic-usage.md#installing-dependencies", "Read more about it at https://getcomposer.org/doc/01-basic-usage.md#installing-dependencies",
"This file is @generated automatically" "This file is @generated automatically"
], ],
"content-hash": "09a618cffed2c4cfb593c0a791c19b3f", "content-hash": "907178db979a21189806683196c6516b",
"packages": [ "packages": [
{ {
"name": "aferrandini/phpqrcode", "name": "aferrandini/phpqrcode",
@ -56,16 +56,16 @@
], ],
"abandoned": "endroid/qr-code", "abandoned": "endroid/qr-code",
"time": "2013-07-08T09:39:08+00:00" "time": "2013-07-08T09:39:08+00:00"
}, },
{ {
"name": "cakephp/core", "name": "cakephp/core",
"version": "4.1.3", "version": "4.1.3",
"source": { "source": {
"type": "git", "type": "git",
"url": "https://github.com/cakephp/core.git", "url": "https://github.com/cakephp/core.git",
"reference": "c7e88f1cd6dfe17065e71f1af305d415f155e97e" "reference": "c7e88f1cd6dfe17065e71f1af305d415f155e97e"
}, },
"dist": { "dist": {
"type": "zip", "type": "zip",
"url": "https://api.github.com/repos/cakephp/core/zipball/c7e88f1cd6dfe17065e71f1af305d415f155e97e", "url": "https://api.github.com/repos/cakephp/core/zipball/c7e88f1cd6dfe17065e71f1af305d415f155e97e",
"reference": "c7e88f1cd6dfe17065e71f1af305d415f155e97e", "reference": "c7e88f1cd6dfe17065e71f1af305d415f155e97e",
@ -329,14 +329,14 @@
"MIT" "MIT"
], ],
"authors": [ "authors": [
{ {
"name": "Guilherme Blanco", "name": "Guilherme Blanco",
"email": "guilhermeblanco@gmail.com" "email": "guilhermeblanco@gmail.com"
}, },
{ {
"name": "Roman Borschel", "name": "Roman Borschel",
"email": "roman@code-factory.org" "email": "roman@code-factory.org"
}, },
{ {
"name": "Johannes Schmitt", "name": "Johannes Schmitt",
"email": "schmittjoh@gmail.com" "email": "schmittjoh@gmail.com"
@ -428,16 +428,16 @@
] ]
}, },
"require": { "require": {
"doctrine/lexer": "^1.0.1", "doctrine/lexer": "^1.0.1",
"php": ">= 5.5" "php": ">= 5.5"
}, },
"require-dev": { "require-dev": {
"dominicsayers/isemail": "dev-master", "dominicsayers/isemail": "dev-master",
"phpunit/phpunit": "^4.8.35||^5.7||^6.0", "phpunit/phpunit": "^4.8.35||^5.7||^6.0",
"satooshi/php-coveralls": "^1.0.1", "satooshi/php-coveralls": "^1.0.1",
"symfony/phpunit-bridge": "^4.4@dev" "symfony/phpunit-bridge": "^4.4@dev"
}, },
"suggest": { "suggest": {
"ext-intl": "PHP Internationalization Libraries are required to use the SpoofChecking validation" "ext-intl": "PHP Internationalization Libraries are required to use the SpoofChecking validation"
}, },
"type": "library", "type": "library",
@ -856,26 +856,74 @@
} }
], ],
"description": "xunsearch php sdk, include yii, yii2 supports", "description": "xunsearch php sdk, include yii, yii2 supports",
"homepage": "http://www.xunsearch.com/", "homepage": "http://www.xunsearch.com/",
"keywords": [ "keywords": [
"search engine", "search engine",
"xunsearch", "xunsearch",
"yii", "yii",
"yii2" "yii2"
], ],
"time": "2020-09-03T16:46:04+00:00" "time": "2020-09-03T16:46:04+00:00"
}, },
{
"name": "joyqi/hyper-down",
"version": "dev-master",
"source": {
"type": "git",
"url": "https://github.com/SegmentFault/HyperDown.git",
"reference": "1774a7bb8a3853503e44cfa5a2186b1943f6493f"
},
"dist": {
"type": "zip",
"url": "https://api.github.com/repos/SegmentFault/HyperDown/zipball/1774a7bb8a3853503e44cfa5a2186b1943f6493f",
"reference": "1774a7bb8a3853503e44cfa5a2186b1943f6493f",
"shasum": "",
"mirrors": [
{
"url": "https://mirrors.aliyun.com/composer/dists/%package%/%reference%.%type%",
"preferred": true
}
]
},
"require": {
"ext-mbstring": "*",
"php": ">=5.4.0"
},
"default-branch": true,
"type": "library",
"autoload": {
"psr-4": {
"HyperDown\\": "./"
}
},
"notification-url": "https://packagist.org/downloads/",
"license": [
"BSD"
],
"authors": [
{ {
"name": "lcobucci/jwt", "name": "joyqi",
"version": "3.3.3", "email": "joyqi@segmentfault.com"
"source": { }
"type": "git", ],
"url": "https://github.com/lcobucci/jwt.git", "description": "A light weight markdown parser library",
"reference": "c1123697f6a2ec29162b82f170dd4a491f524773" "support": {
}, "issues": "https://github.com/SegmentFault/HyperDown/issues",
"dist": { "source": "https://github.com/SegmentFault/HyperDown/tree/master"
"type": "zip", },
"url": "https://api.github.com/repos/lcobucci/jwt/zipball/c1123697f6a2ec29162b82f170dd4a491f524773", "time": "2020-11-30T04:05:08+00:00"
},
{
"name": "lcobucci/jwt",
"version": "3.3.3",
"source": {
"type": "git",
"url": "https://github.com/lcobucci/jwt.git",
"reference": "c1123697f6a2ec29162b82f170dd4a491f524773"
},
"dist": {
"type": "zip",
"url": "https://api.github.com/repos/lcobucci/jwt/zipball/c1123697f6a2ec29162b82f170dd4a491f524773",
"reference": "c1123697f6a2ec29162b82f170dd4a491f524773", "reference": "c1123697f6a2ec29162b82f170dd4a491f524773",
"shasum": "", "shasum": "",
"mirrors": [ "mirrors": [
@ -1056,16 +1104,16 @@
"require": { "require": {
"php": ">=5.3.2" "php": ">=5.3.2"
}, },
"require-dev": { "require-dev": {
"phpunit/phpunit": "~4.0|~5.0" "phpunit/phpunit": "~4.0|~5.0"
}, },
"type": "library", "type": "library",
"autoload": { "autoload": {
"psr-4": { "psr-4": {
"Cron\\": "src/Cron/" "Cron\\": "src/Cron/"
} }
}, },
"notification-url": "https://packagist.org/downloads/", "notification-url": "https://packagist.org/downloads/",
"license": [ "license": [
"MIT" "MIT"
], ],
@ -1252,16 +1300,16 @@
] ]
}, },
"require": { "require": {
"php": "^7" "php": "^7"
}, },
"require-dev": { "require-dev": {
"phpunit/phpunit": "4.*|5.*", "phpunit/phpunit": "4.*|5.*",
"vimeo/psalm": "^1" "vimeo/psalm": "^1"
}, },
"suggest": { "suggest": {
"ext-libsodium": "Provides a modern crypto API that can be used to generate random bytes." "ext-libsodium": "Provides a modern crypto API that can be used to generate random bytes."
}, },
"type": "library", "type": "library",
"notification-url": "https://packagist.org/downloads/", "notification-url": "https://packagist.org/downloads/",
"license": [ "license": [
"MIT" "MIT"
@ -1498,16 +1546,16 @@
] ]
}, },
"require": { "require": {
"php": ">=5.3.0" "php": ">=5.3.0"
}, },
"type": "library", "type": "library",
"extra": { "extra": {
"branch-alias": { "branch-alias": {
"dev-master": "1.0.x-dev" "dev-master": "1.0.x-dev"
} }
}, },
"autoload": { "autoload": {
"psr-4": { "psr-4": {
"Psr\\Cache\\": "src/" "Psr\\Cache\\": "src/"
} }
}, },
@ -1703,15 +1751,15 @@
}, },
"dist": { "dist": {
"type": "zip", "type": "zip",
"url": "https://api.github.com/repos/php-fig/simple-cache/zipball/408d5eafb83c57f6365a3ca330ff23aa4a5fa39b", "url": "https://api.github.com/repos/php-fig/simple-cache/zipball/408d5eafb83c57f6365a3ca330ff23aa4a5fa39b",
"reference": "408d5eafb83c57f6365a3ca330ff23aa4a5fa39b", "reference": "408d5eafb83c57f6365a3ca330ff23aa4a5fa39b",
"shasum": "", "shasum": "",
"mirrors": [ "mirrors": [
{ {
"url": "https://mirrors.aliyun.com/composer/dists/%package%/%reference%.%type%", "url": "https://mirrors.aliyun.com/composer/dists/%package%/%reference%.%type%",
"preferred": true "preferred": true
} }
] ]
}, },
"require": { "require": {
"php": ">=5.3.0" "php": ">=5.3.0"
@ -2017,15 +2065,15 @@
"dev-master": "6.2-dev" "dev-master": "6.2-dev"
} }
}, },
"autoload": { "autoload": {
"files": [ "files": [
"lib/swift_required.php" "lib/swift_required.php"
] ]
}, },
"notification-url": "https://packagist.org/downloads/", "notification-url": "https://packagist.org/downloads/",
"license": [ "license": [
"MIT" "MIT"
], ],
"authors": [ "authors": [
{ {
"name": "Chris Corbyn" "name": "Chris Corbyn"
@ -2251,16 +2299,16 @@
] ]
}, },
"require": { "require": {
"php": ">=7.2.5", "php": ">=7.2.5",
"symfony/deprecation-contracts": "^2.1", "symfony/deprecation-contracts": "^2.1",
"symfony/filesystem": "^4.4|^5.0", "symfony/filesystem": "^4.4|^5.0",
"symfony/polyfill-ctype": "~1.8", "symfony/polyfill-ctype": "~1.8",
"symfony/polyfill-php80": "^1.15" "symfony/polyfill-php80": "^1.15"
}, },
"conflict": { "conflict": {
"symfony/finder": "<4.4" "symfony/finder": "<4.4"
}, },
"require-dev": { "require-dev": {
"symfony/event-dispatcher": "^4.4|^5.0", "symfony/event-dispatcher": "^4.4|^5.0",
"symfony/finder": "^4.4|^5.0", "symfony/finder": "^4.4|^5.0",
"symfony/messenger": "^4.4|^5.0", "symfony/messenger": "^4.4|^5.0",
@ -2398,14 +2446,14 @@
"description": "Symfony Console Component", "description": "Symfony Console Component",
"homepage": "https://symfony.com", "homepage": "https://symfony.com",
"funding": [ "funding": [
{ {
"url": "https://symfony.com/sponsor", "url": "https://symfony.com/sponsor",
"type": "custom" "type": "custom"
}, },
{ {
"url": "https://github.com/fabpot", "url": "https://github.com/fabpot",
"type": "github" "type": "github"
}, },
{ {
"url": "https://tidelift.com/funding/github/packagist/symfony/symfony", "url": "https://tidelift.com/funding/github/packagist/symfony/symfony",
"type": "tidelift" "type": "tidelift"
@ -2443,16 +2491,16 @@
}, },
"thanks": { "thanks": {
"name": "symfony/contracts", "name": "symfony/contracts",
"url": "https://github.com/symfony/contracts" "url": "https://github.com/symfony/contracts"
} }
}, },
"autoload": { "autoload": {
"files": [ "files": [
"function.php" "function.php"
] ]
}, },
"notification-url": "https://packagist.org/downloads/", "notification-url": "https://packagist.org/downloads/",
"license": [ "license": [
"MIT" "MIT"
], ],
"authors": [ "authors": [
@ -2468,14 +2516,14 @@
"description": "A generic function and convention to trigger deprecation notices", "description": "A generic function and convention to trigger deprecation notices",
"homepage": "https://symfony.com", "homepage": "https://symfony.com",
"funding": [ "funding": [
{ {
"url": "https://symfony.com/sponsor", "url": "https://symfony.com/sponsor",
"type": "custom" "type": "custom"
}, },
{ {
"url": "https://github.com/fabpot", "url": "https://github.com/fabpot",
"type": "github" "type": "github"
}, },
{ {
"url": "https://tidelift.com/funding/github/packagist/symfony/symfony", "url": "https://tidelift.com/funding/github/packagist/symfony/symfony",
"type": "tidelift" "type": "tidelift"
@ -2513,17 +2561,17 @@
"provide": { "provide": {
"psr/event-dispatcher-implementation": "1.0", "psr/event-dispatcher-implementation": "1.0",
"symfony/event-dispatcher-implementation": "1.1" "symfony/event-dispatcher-implementation": "1.1"
}, },
"require-dev": { "require-dev": {
"psr/log": "~1.0", "psr/log": "~1.0",
"symfony/config": "^3.4|^4.0|^5.0", "symfony/config": "^3.4|^4.0|^5.0",
"symfony/dependency-injection": "^3.4|^4.0|^5.0", "symfony/dependency-injection": "^3.4|^4.0|^5.0",
"symfony/error-handler": "~3.4|~4.4", "symfony/error-handler": "~3.4|~4.4",
"symfony/expression-language": "^3.4|^4.0|^5.0", "symfony/expression-language": "^3.4|^4.0|^5.0",
"symfony/http-foundation": "^3.4|^4.0|^5.0", "symfony/http-foundation": "^3.4|^4.0|^5.0",
"symfony/service-contracts": "^1.1|^2", "symfony/service-contracts": "^1.1|^2",
"symfony/stopwatch": "^3.4|^4.0|^5.0" "symfony/stopwatch": "^3.4|^4.0|^5.0"
}, },
"suggest": { "suggest": {
"symfony/dependency-injection": "", "symfony/dependency-injection": "",
"symfony/http-kernel": "" "symfony/http-kernel": ""
@ -2531,16 +2579,16 @@
"type": "library", "type": "library",
"autoload": { "autoload": {
"psr-4": { "psr-4": {
"Symfony\\Component\\EventDispatcher\\": "" "Symfony\\Component\\EventDispatcher\\": ""
}, },
"exclude-from-classmap": [ "exclude-from-classmap": [
"/Tests/" "/Tests/"
] ]
}, },
"notification-url": "https://packagist.org/downloads/", "notification-url": "https://packagist.org/downloads/",
"license": [ "license": [
"MIT" "MIT"
], ],
"authors": [ "authors": [
{ {
"name": "Fabien Potencier", "name": "Fabien Potencier",
@ -2606,15 +2654,15 @@
"url": "https://github.com/symfony/contracts" "url": "https://github.com/symfony/contracts"
} }
}, },
"autoload": { "autoload": {
"psr-4": { "psr-4": {
"Symfony\\Contracts\\EventDispatcher\\": "" "Symfony\\Contracts\\EventDispatcher\\": ""
} }
}, },
"notification-url": "https://packagist.org/downloads/", "notification-url": "https://packagist.org/downloads/",
"license": [ "license": [
"MIT" "MIT"
], ],
"authors": [ "authors": [
{ {
"name": "Nicolas Grekas", "name": "Nicolas Grekas",
@ -2661,16 +2709,16 @@
}, },
"dist": { "dist": {
"type": "zip", "type": "zip",
"url": "https://api.github.com/repos/symfony/filesystem/zipball/6e4320f06d5f2cce0d96530162491f4465179157", "url": "https://api.github.com/repos/symfony/filesystem/zipball/6e4320f06d5f2cce0d96530162491f4465179157",
"reference": "6e4320f06d5f2cce0d96530162491f4465179157", "reference": "6e4320f06d5f2cce0d96530162491f4465179157",
"shasum": "", "shasum": "",
"mirrors": [ "mirrors": [
{ {
"url": "https://mirrors.aliyun.com/composer/dists/%package%/%reference%.%type%", "url": "https://mirrors.aliyun.com/composer/dists/%package%/%reference%.%type%",
"preferred": true "preferred": true
} }
] ]
}, },
"require": { "require": {
"php": ">=7.2.5", "php": ">=7.2.5",
"symfony/polyfill-ctype": "~1.8" "symfony/polyfill-ctype": "~1.8"
@ -2706,14 +2754,14 @@
"description": "Symfony Filesystem Component", "description": "Symfony Filesystem Component",
"homepage": "https://symfony.com", "homepage": "https://symfony.com",
"funding": [ "funding": [
{ {
"url": "https://symfony.com/sponsor", "url": "https://symfony.com/sponsor",
"type": "custom" "type": "custom"
}, },
{ {
"url": "https://github.com/fabpot", "url": "https://github.com/fabpot",
"type": "github" "type": "github"
}, },
{ {
"url": "https://tidelift.com/funding/github/packagist/symfony/symfony", "url": "https://tidelift.com/funding/github/packagist/symfony/symfony",
"type": "tidelift" "type": "tidelift"
@ -2751,16 +2799,16 @@
"predis/predis": "~1.0", "predis/predis": "~1.0",
"symfony/cache": "^4.4|^5.0", "symfony/cache": "^4.4|^5.0",
"symfony/expression-language": "^4.4|^5.0", "symfony/expression-language": "^4.4|^5.0",
"symfony/mime": "^4.4|^5.0" "symfony/mime": "^4.4|^5.0"
}, },
"suggest": { "suggest": {
"symfony/mime": "To use the file extension guesser" "symfony/mime": "To use the file extension guesser"
}, },
"type": "library", "type": "library",
"autoload": { "autoload": {
"psr-4": { "psr-4": {
"Symfony\\Component\\HttpFoundation\\": "" "Symfony\\Component\\HttpFoundation\\": ""
}, },
"exclude-from-classmap": [ "exclude-from-classmap": [
"/Tests/" "/Tests/"
] ]
@ -3107,14 +3155,14 @@
"shim" "shim"
], ],
"funding": [ "funding": [
{ {
"url": "https://symfony.com/sponsor", "url": "https://symfony.com/sponsor",
"type": "custom" "type": "custom"
}, },
{ {
"url": "https://github.com/fabpot", "url": "https://github.com/fabpot",
"type": "github" "type": "github"
}, },
{ {
"url": "https://tidelift.com/funding/github/packagist/symfony/symfony", "url": "https://tidelift.com/funding/github/packagist/symfony/symfony",
"type": "tidelift" "type": "tidelift"
@ -3152,16 +3200,16 @@
"extra": { "extra": {
"branch-alias": { "branch-alias": {
"dev-main": "1.20-dev" "dev-main": "1.20-dev"
}, },
"thanks": { "thanks": {
"name": "symfony/polyfill", "name": "symfony/polyfill",
"url": "https://github.com/symfony/polyfill" "url": "https://github.com/symfony/polyfill"
} }
}, },
"autoload": { "autoload": {
"psr-4": { "psr-4": {
"Symfony\\Polyfill\\Mbstring\\": "" "Symfony\\Polyfill\\Mbstring\\": ""
}, },
"files": [ "files": [
"bootstrap.php" "bootstrap.php"
] ]
@ -3434,14 +3482,14 @@
"shim" "shim"
], ],
"funding": [ "funding": [
{ {
"url": "https://symfony.com/sponsor", "url": "https://symfony.com/sponsor",
"type": "custom" "type": "custom"
}, },
{ {
"url": "https://github.com/fabpot", "url": "https://github.com/fabpot",
"type": "github" "type": "github"
}, },
{ {
"url": "https://tidelift.com/funding/github/packagist/symfony/symfony", "url": "https://tidelift.com/funding/github/packagist/symfony/symfony",
"type": "tidelift" "type": "tidelift"
@ -3479,16 +3527,16 @@
}, },
"thanks": { "thanks": {
"name": "symfony/polyfill", "name": "symfony/polyfill",
"url": "https://github.com/symfony/polyfill" "url": "https://github.com/symfony/polyfill"
} }
}, },
"autoload": { "autoload": {
"psr-4": { "psr-4": {
"Symfony\\Polyfill\\Php80\\": "" "Symfony\\Polyfill\\Php80\\": ""
}, },
"files": [ "files": [
"bootstrap.php" "bootstrap.php"
], ],
"classmap": [ "classmap": [
"Resources/stubs" "Resources/stubs"
] ]
@ -3654,16 +3702,16 @@
"extra": { "extra": {
"branch-alias": { "branch-alias": {
"dev-master": "2.2-dev" "dev-master": "2.2-dev"
}, },
"thanks": { "thanks": {
"name": "symfony/contracts", "name": "symfony/contracts",
"url": "https://github.com/symfony/contracts" "url": "https://github.com/symfony/contracts"
} }
}, },
"autoload": { "autoload": {
"psr-4": { "psr-4": {
"Symfony\\Contracts\\Service\\": "" "Symfony\\Contracts\\Service\\": ""
} }
}, },
"notification-url": "https://packagist.org/downloads/", "notification-url": "https://packagist.org/downloads/",
"license": [ "license": [
@ -3803,16 +3851,16 @@
"preferred": true "preferred": true
} }
] ]
}, },
"require": { "require": {
"guzzlehttp/guzzle": "^6.3", "guzzlehttp/guzzle": "^6.3",
"guzzlehttp/psr7": "^1.4", "guzzlehttp/psr7": "^1.4",
"php": ">=5.6.0" "php": ">=5.6.0"
}, },
"type": "library", "type": "library",
"autoload": { "autoload": {
"classmap": [ "classmap": [
"src/QcloudApi/QcloudApi.php" "src/QcloudApi/QcloudApi.php"
], ],
"psr-4": { "psr-4": {
"TencentCloud\\": "./src/TencentCloud" "TencentCloud\\": "./src/TencentCloud"
@ -4212,35 +4260,35 @@
} }
] ]
}, },
"require": { "require": {
"ext-json": "*", "ext-json": "*",
"ext-pdo": "*", "ext-pdo": "*",
"php": "^7.2", "php": "^7.2",
"riimu/kit-phpencoder": "^2.4", "riimu/kit-phpencoder": "^2.4",
"robmorgan/phinx": "^0.12", "robmorgan/phinx": "^0.12",
"symfony/console": "^2.8 || ^3.0 || ^4.0 || ^5.0", "symfony/console": "^2.8 || ^3.0 || ^4.0 || ^5.0",
"symfony/polyfill-php73": "^1.18" "symfony/polyfill-php73": "^1.18"
}, },
"require-dev": { "require-dev": {
"friendsofphp/php-cs-fixer": "^2.16", "friendsofphp/php-cs-fixer": "^2.16",
"overtrue/phplint": "^1.1 || ^2.0", "overtrue/phplint": "^1.1 || ^2.0",
"phpstan/phpstan": "^0.12", "phpstan/phpstan": "^0.12",
"phpunit/phpunit": "^8 || ^9", "phpunit/phpunit": "^8 || ^9",
"squizlabs/php_codesniffer": "^3.4" "squizlabs/php_codesniffer": "^3.4"
}, },
"bin": [ "bin": [
"./bin/phinx-migrations" "./bin/phinx-migrations"
], ],
"type": "library", "type": "library",
"autoload": { "autoload": {
"psr-4": { "psr-4": {
"Odan\\Migration\\": "src/Migration/" "Odan\\Migration\\": "src/Migration/"
} }
}, },
"notification-url": "https://packagist.org/downloads/", "notification-url": "https://packagist.org/downloads/",
"license": [ "license": [
"MIT" "MIT"
], ],
"description": "Migration generator for Phinx", "description": "Migration generator for Phinx",
"homepage": "https://github.com/odan/phinx-migrations-generator", "homepage": "https://github.com/odan/phinx-migrations-generator",
"keywords": [ "keywords": [
@ -4257,60 +4305,60 @@
}, },
"time": "2020-12-30T23:59:57+00:00" "time": "2020-12-30T23:59:57+00:00"
}, },
{ {
"name": "phalcon/ide-stubs", "name": "phalcon/ide-stubs",
"version": "v3.4.3", "version": "v3.4.3",
"source": { "source": {
"type": "git", "type": "git",
"url": "https://github.com/phalcon/ide-stubs.git", "url": "https://github.com/phalcon/ide-stubs.git",
"reference": "65144f2b0fad32b182ccb062b1efc1b4edea5d44" "reference": "65144f2b0fad32b182ccb062b1efc1b4edea5d44"
},
"dist": {
"type": "zip",
"url": "https://api.github.com/repos/phalcon/ide-stubs/zipball/65144f2b0fad32b182ccb062b1efc1b4edea5d44",
"reference": "65144f2b0fad32b182ccb062b1efc1b4edea5d44",
"shasum": "",
"mirrors": [
{
"url": "https://mirrors.aliyun.com/composer/dists/%package%/%reference%.%type%",
"preferred": true
}
]
},
"require": {
"php": ">=5.3.0"
},
"type": "library",
"notification-url": "https://packagist.jp/downloads/",
"license": [
"BSD-3-Clause"
],
"authors": [
{
"name": "Phalcon Team",
"email": "team@phalconphp.com",
"homepage": "https://phalconphp.com/en/team"
},
{
"name": "Contributors",
"homepage": "https://github.com/phalcon/ide-stubs/graphs/contributors"
}
],
"description": "The most complete Phalcon Framework IDE stubs library which enables autocompletion in modern IDEs.",
"homepage": "https://phalconphp.com",
"keywords": [
"Devtools",
"Eclipse",
"autocomplete",
"ide",
"netbeans",
"phalcon",
"phpstorm",
"stub",
"stubs"
],
"time": "2018-12-09T14:11:06+00:00"
}, },
"dist": {
"type": "zip",
"url": "https://api.github.com/repos/phalcon/ide-stubs/zipball/65144f2b0fad32b182ccb062b1efc1b4edea5d44",
"reference": "65144f2b0fad32b182ccb062b1efc1b4edea5d44",
"shasum": "",
"mirrors": [
{
"url": "https://mirrors.aliyun.com/composer/dists/%package%/%reference%.%type%",
"preferred": true
}
]
},
"require": {
"php": ">=5.3.0"
},
"type": "library",
"notification-url": "https://packagist.jp/downloads/",
"license": [
"BSD-3-Clause"
],
"authors": [
{
"name": "Phalcon Team",
"email": "team@phalconphp.com",
"homepage": "https://phalconphp.com/en/team"
},
{
"name": "Contributors",
"homepage": "https://github.com/phalcon/ide-stubs/graphs/contributors"
}
],
"description": "The most complete Phalcon Framework IDE stubs library which enables autocompletion in modern IDEs.",
"homepage": "https://phalconphp.com",
"keywords": [
"Devtools",
"Eclipse",
"autocomplete",
"ide",
"netbeans",
"phalcon",
"phpstorm",
"stub",
"stubs"
],
"time": "2018-12-09T14:11:06+00:00"
},
{ {
"name": "riimu/kit-phpencoder", "name": "riimu/kit-phpencoder",
"version": "v2.4.0", "version": "v2.4.0",
@ -4361,26 +4409,28 @@
"homepage": "http://kit.riimu.net", "homepage": "http://kit.riimu.net",
"keywords": [ "keywords": [
"code", "code",
"encoder", "encoder",
"export", "export",
"generator", "generator",
"variable" "variable"
], ],
"time": "2018-07-03T12:46:23+00:00" "time": "2018-07-03T12:46:23+00:00"
} }
], ],
"aliases": [], "aliases": [],
"minimum-stability": "stable", "minimum-stability": "stable",
"stability-flags": [], "stability-flags": {
"prefer-stable": false, "joyqi/hyper-down": 20
"prefer-lowest": false, },
"platform": { "prefer-stable": false,
"ext-phalcon": "~3.4", "prefer-lowest": false,
"ext-redis": "~5.0", "platform": {
"ext-pdo": "*", "ext-phalcon": "~3.4",
"ext-json": "*", "ext-redis": "~5.0",
"ext-fileinfo": "*" "ext-pdo": "*",
}, "ext-json": "*",
"platform-dev": [], "ext-fileinfo": "*"
},
"platform-dev": [],
"plugin-api-version": "2.0.0" "plugin-api-version": "2.0.0"
} }

View File

@ -0,0 +1,16 @@
<?php
class Schema202101051030 extends Phinx\Migration\AbstractMigration
{
public function change()
{
$this->table('kg_task')
->addIndex(['create_time'], [
'name' => 'create_time',
'unique' => false,
])
->save();
}
}

View File

@ -0,0 +1,37 @@
<?php
class Data202101051130 extends Phinx\Migration\AbstractMigration
{
public function up()
{
$rows = [
[
'section' => 'site',
'item_key' => 'logo',
'item_value' => '',
],
[
'section' => 'site',
'item_key' => 'favicon',
'item_value' => '',
],
];
$this->table('kg_setting')->insert($rows)->save();
}
public function down()
{
$this->getQueryBuilder()
->delete('kg_setting')
->where(['section' => 'site', 'item_key' => 'logo'])
->execute();
$this->getQueryBuilder()
->delete('kg_setting')
->where(['section' => 'site', 'item_key' => 'favicon'])
->execute();
}
}

View File

@ -0,0 +1,68 @@
<?php
declare(strict_types=1);
use Phinx\Migration\AbstractMigration;
final class Data202101061830 extends AbstractMigration
{
public function up()
{
$menu = [
[
'name' => '菜单1',
'url' => '',
'children' => [
[
'type' => 'view',
'name' => '菜单1-1',
'url' => 'https://gitee.com/koogua'
],
],
],
[
'name' => '菜单2',
'url' => '',
'children' => [
[
'type' => 'view',
'name' => '菜单2-1',
'url' => 'https://gitee.com/koogua'
],
],
],
[
'name' => '菜单3',
'url' => '',
'children' => [
[
'type' => 'view',
'name' => '菜单3-1',
'url' => 'https://gitee.com/koogua'
],
],
],
];
$rows = [
[
'section' => 'wechat.oa',
'item_key' => 'menu',
'item_value' => json_encode($menu),
],
];
$this->table('kg_setting')->insert($rows)->save();
}
public function down()
{
$this->getQueryBuilder()
->delete('kg_setting')
->where(['section' => 'wechat.oa', 'item_key' => 'menu'])
->execute();
}
}

File diff suppressed because it is too large Load Diff

View File

@ -30,8 +30,9 @@ layui.use(['jquery', 'form', 'element', 'layer', 'dropdown'], function () {
404: function () { 404: function () {
layer.msg('资源不存在', {icon: 2, anim: 6}); layer.msg('资源不存在', {icon: 2, anim: 6});
}, },
500: function () { 500: function (xhr) {
layer.msg('服务器内部错误', {icon: 2, anim: 6}); var res = JSON.parse(xhr.responseText);
layer.msg(res.msg, {icon: 2, anim: 6});
} }
} }
}); });

View File

@ -5,6 +5,7 @@ layui.use(['jquery'], function () {
var $textarea = $('#vditor-textarea'); var $textarea = $('#vditor-textarea');
var vditor = new Vditor('vditor', { var vditor = new Vditor('vditor', {
mode: 'sv',
minHeight: 300, minHeight: 300,
outline: false, outline: false,
resize: { resize: {

View File

@ -127,8 +127,21 @@
font-size: 12px; font-size: 12px;
} }
#header .logo {
top: 0;
left: 0;
position: absolute;
line-height: 60px;
}
.top-nav { .top-nav {
margin-left: 100px; margin-left: 185px;
}
#header .user {
top: 0;
right: 0;
position: absolute;
} }
.layer-search input { .layer-search input {
@ -1522,8 +1535,6 @@
.consult-info .item { .consult-info .item {
margin-bottom: 15px; margin-bottom: 15px;
line-height: 1.5em; line-height: 1.5em;
height: 1.5em;
clear: both;
} }
.consult-info .item .label { .consult-info .item .label {

View File

@ -0,0 +1,972 @@
.markdown-body .octicon {
display: inline-block;
fill: currentColor;
vertical-align: text-bottom;
}
.markdown-body .anchor {
float: left;
line-height: 1;
margin-left: -20px;
padding-right: 4px;
}
.markdown-body .anchor:focus {
outline: none;
}
.markdown-body h1 .octicon-link,
.markdown-body h2 .octicon-link,
.markdown-body h3 .octicon-link,
.markdown-body h4 .octicon-link,
.markdown-body h5 .octicon-link,
.markdown-body h6 .octicon-link {
color: #1b1f23;
vertical-align: middle;
visibility: hidden;
}
.markdown-body h1:hover .anchor,
.markdown-body h2:hover .anchor,
.markdown-body h3:hover .anchor,
.markdown-body h4:hover .anchor,
.markdown-body h5:hover .anchor,
.markdown-body h6:hover .anchor {
text-decoration: none;
}
.markdown-body h1:hover .anchor .octicon-link,
.markdown-body h2:hover .anchor .octicon-link,
.markdown-body h3:hover .anchor .octicon-link,
.markdown-body h4:hover .anchor .octicon-link,
.markdown-body h5:hover .anchor .octicon-link,
.markdown-body h6:hover .anchor .octicon-link {
visibility: visible;
}
.markdown-body h1:hover .anchor .octicon-link:before,
.markdown-body h2:hover .anchor .octicon-link:before,
.markdown-body h3:hover .anchor .octicon-link:before,
.markdown-body h4:hover .anchor .octicon-link:before,
.markdown-body h5:hover .anchor .octicon-link:before,
.markdown-body h6:hover .anchor .octicon-link:before {
width: 16px;
height: 16px;
content: ' ';
display: inline-block;
background-image: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' viewBox='0 0 16 16' version='1.1' width='16' height='16' aria-hidden='true'%3E%3Cpath fill-rule='evenodd' d='M4 9h1v1H4c-1.5 0-3-1.69-3-3.5S2.55 3 4 3h4c1.45 0 3 1.69 3 3.5 0 1.41-.91 2.72-2 3.25V8.59c.58-.45 1-1.27 1-2.09C10 5.22 8.98 4 8 4H4c-.98 0-2 1.22-2 2.5S3 9 4 9zm9-3h-1v1h1c1 0 2 1.22 2 2.5S13.98 12 13 12H9c-.98 0-2-1.22-2-2.5 0-.83.42-1.64 1-2.09V6.25c-1.09.53-2 1.84-2 3.25C6 11.31 7.55 13 9 13h4c1.45 0 3-1.69 3-3.5S14.5 6 13 6z'%3E%3C/path%3E%3C/svg%3E");
}
.markdown-body {
-ms-text-size-adjust: 100%;
-webkit-text-size-adjust: 100%;
line-height: 1.5;
color: #24292e;
font-family: -apple-system, BlinkMacSystemFont, Segoe UI, Helvetica, Arial, sans-serif, Apple Color Emoji, Segoe UI Emoji;
font-size: 14px;
word-wrap: break-word;
}
.markdown-body details {
display: block;
}
.markdown-body summary {
display: list-item;
}
.markdown-body a {
background-color: initial;
}
.markdown-body a:active,
.markdown-body a:hover {
outline-width: 0;
}
.markdown-body strong {
font-weight: bolder;
}
.markdown-body h1 {
font-size: 2em;
margin: .67em 0;
}
.markdown-body img {
border-style: none;
}
.markdown-body code,
.markdown-body kbd,
.markdown-body pre {
font-family: monospace, monospace;
font-size: 1em;
}
.markdown-body hr {
box-sizing: initial;
height: 0;
overflow: visible;
}
.markdown-body input {
font: inherit;
margin: 0;
}
.markdown-body input {
overflow: visible;
}
.markdown-body [type=checkbox] {
box-sizing: border-box;
padding: 0;
}
.markdown-body * {
box-sizing: border-box;
}
.markdown-body input {
font-family: inherit;
font-size: inherit;
line-height: inherit;
}
.markdown-body a {
color: #0366d6;
text-decoration: none;
}
.markdown-body a:hover {
text-decoration: underline;
}
.markdown-body strong {
font-weight: 600;
}
.markdown-body hr {
height: 0;
margin: 15px 0;
overflow: hidden;
background: transparent;
border: 0;
border-bottom: 1px solid #dfe2e5;
}
.markdown-body hr:after,
.markdown-body hr:before {
display: table;
content: "";
}
.markdown-body hr:after {
clear: both;
}
.markdown-body table {
border-spacing: 0;
border-collapse: collapse;
}
.markdown-body td,
.markdown-body th {
padding: 0;
}
.markdown-body details summary {
cursor: pointer;
}
.markdown-body kbd {
display: inline-block;
padding: 3px 5px;
font: 11px SFMono-Regular, Consolas, Liberation Mono, Menlo, monospace;
line-height: 10px;
color: #444d56;
vertical-align: middle;
background-color: #fafbfc;
border: 1px solid #d1d5da;
border-radius: 3px;
box-shadow: inset 0 -1px 0 #d1d5da;
}
.markdown-body h1,
.markdown-body h2,
.markdown-body h3,
.markdown-body h4,
.markdown-body h5,
.markdown-body h6 {
margin-top: 0;
margin-bottom: 0;
}
.markdown-body h1 {
font-size: 32px;
}
.markdown-body h1,
.markdown-body h2 {
font-weight: 600;
}
.markdown-body h2 {
font-size: 24px;
}
.markdown-body h3 {
font-size: 20px;
}
.markdown-body h3,
.markdown-body h4 {
font-weight: 600;
}
.markdown-body h4 {
font-size: 16px;
}
.markdown-body h5 {
font-size: 14px;
}
.markdown-body h5,
.markdown-body h6 {
font-weight: 600;
}
.markdown-body h6 {
font-size: 12px;
}
.markdown-body p {
margin-top: 0;
margin-bottom: 10px;
}
.markdown-body blockquote {
margin: 0;
}
.markdown-body ol,
.markdown-body ul {
padding-left: 0;
margin-top: 0;
margin-bottom: 0;
}
.markdown-body ol ol,
.markdown-body ul ol {
list-style-type: lower-roman;
}
.markdown-body ol ol ol,
.markdown-body ol ul ol,
.markdown-body ul ol ol,
.markdown-body ul ul ol {
list-style-type: lower-alpha;
}
.markdown-body dd {
margin-left: 0;
}
.markdown-body code,
.markdown-body pre {
font-family: SFMono-Regular, Consolas, Liberation Mono, Menlo, monospace;
font-size: 12px;
}
.markdown-body pre {
margin-top: 0;
margin-bottom: 0;
}
.markdown-body input::-webkit-inner-spin-button,
.markdown-body input::-webkit-outer-spin-button {
margin: 0;
-webkit-appearance: none;
appearance: none;
}
.markdown-body :checked + .radio-label {
position: relative;
z-index: 1;
border-color: #0366d6;
}
.markdown-body .border {
border: 1px solid #e1e4e8 !important;
}
.markdown-body .border-0 {
border: 0 !important;
}
.markdown-body .border-bottom {
border-bottom: 1px solid #e1e4e8 !important;
}
.markdown-body .rounded-1 {
border-radius: 3px !important;
}
.markdown-body .bg-white {
background-color: #fff !important;
}
.markdown-body .bg-gray-light {
background-color: #fafbfc !important;
}
.markdown-body .text-gray-light {
color: #6a737d !important;
}
.markdown-body .mb-0 {
margin-bottom: 0 !important;
}
.markdown-body .my-2 {
margin-top: 8px !important;
margin-bottom: 8px !important;
}
.markdown-body .pl-0 {
padding-left: 0 !important;
}
.markdown-body .py-0 {
padding-top: 0 !important;
padding-bottom: 0 !important;
}
.markdown-body .pl-1 {
padding-left: 4px !important;
}
.markdown-body .pl-2 {
padding-left: 8px !important;
}
.markdown-body .py-2 {
padding-top: 8px !important;
padding-bottom: 8px !important;
}
.markdown-body .pl-3,
.markdown-body .px-3 {
padding-left: 16px !important;
}
.markdown-body .px-3 {
padding-right: 16px !important;
}
.markdown-body .pl-4 {
padding-left: 24px !important;
}
.markdown-body .pl-5 {
padding-left: 32px !important;
}
.markdown-body .pl-6 {
padding-left: 40px !important;
}
.markdown-body .f6 {
font-size: 12px !important;
}
.markdown-body .lh-condensed {
line-height: 1.25 !important;
}
.markdown-body .text-bold {
font-weight: 600 !important;
}
.markdown-body .pl-c {
color: #6a737d;
}
.markdown-body .pl-c1,
.markdown-body .pl-s .pl-v {
color: #005cc5;
}
.markdown-body .pl-e,
.markdown-body .pl-en {
color: #6f42c1;
}
.markdown-body .pl-s .pl-s1,
.markdown-body .pl-smi {
color: #24292e;
}
.markdown-body .pl-ent {
color: #22863a;
}
.markdown-body .pl-k {
color: #d73a49;
}
.markdown-body .pl-pds,
.markdown-body .pl-s,
.markdown-body .pl-s .pl-pse .pl-s1,
.markdown-body .pl-sr,
.markdown-body .pl-sr .pl-cce,
.markdown-body .pl-sr .pl-sra,
.markdown-body .pl-sr .pl-sre {
color: #032f62;
}
.markdown-body .pl-smw,
.markdown-body .pl-v {
color: #e36209;
}
.markdown-body .pl-bu {
color: #b31d28;
}
.markdown-body .pl-ii {
color: #fafbfc;
background-color: #b31d28;
}
.markdown-body .pl-c2 {
color: #fafbfc;
background-color: #d73a49;
}
.markdown-body .pl-c2:before {
content: "^M";
}
.markdown-body .pl-sr .pl-cce {
font-weight: 700;
color: #22863a;
}
.markdown-body .pl-ml {
color: #735c0f;
}
.markdown-body .pl-mh,
.markdown-body .pl-mh .pl-en,
.markdown-body .pl-ms {
font-weight: 700;
color: #005cc5;
}
.markdown-body .pl-mi {
font-style: italic;
color: #24292e;
}
.markdown-body .pl-mb {
font-weight: 700;
color: #24292e;
}
.markdown-body .pl-md {
color: #b31d28;
background-color: #ffeef0;
}
.markdown-body .pl-mi1 {
color: #22863a;
background-color: #f0fff4;
}
.markdown-body .pl-mc {
color: #e36209;
background-color: #ffebda;
}
.markdown-body .pl-mi2 {
color: #f6f8fa;
background-color: #005cc5;
}
.markdown-body .pl-mdr {
font-weight: 700;
color: #6f42c1;
}
.markdown-body .pl-ba {
color: #586069;
}
.markdown-body .pl-sg {
color: #959da5;
}
.markdown-body .pl-corl {
text-decoration: underline;
color: #032f62;
}
.markdown-body .mb-0 {
margin-bottom: 0 !important;
}
.markdown-body .my-2 {
margin-bottom: 8px !important;
}
.markdown-body .my-2 {
margin-top: 8px !important;
}
.markdown-body .pl-0 {
padding-left: 0 !important;
}
.markdown-body .py-0 {
padding-top: 0 !important;
padding-bottom: 0 !important;
}
.markdown-body .pl-1 {
padding-left: 4px !important;
}
.markdown-body .pl-2 {
padding-left: 8px !important;
}
.markdown-body .py-2 {
padding-top: 8px !important;
padding-bottom: 8px !important;
}
.markdown-body .pl-3 {
padding-left: 16px !important;
}
.markdown-body .pl-4 {
padding-left: 24px !important;
}
.markdown-body .pl-5 {
padding-left: 32px !important;
}
.markdown-body .pl-6 {
padding-left: 40px !important;
}
.markdown-body .pl-7 {
padding-left: 48px !important;
}
.markdown-body .pl-8 {
padding-left: 64px !important;
}
.markdown-body .pl-9 {
padding-left: 80px !important;
}
.markdown-body .pl-10 {
padding-left: 96px !important;
}
.markdown-body .pl-11 {
padding-left: 112px !important;
}
.markdown-body .pl-12 {
padding-left: 128px !important;
}
.markdown-body hr {
border-bottom-color: #eee;
}
.markdown-body:after,
.markdown-body:before {
display: table;
content: "";
}
.markdown-body:after {
clear: both;
}
.markdown-body > :first-child {
margin-top: 0 !important;
}
.markdown-body > :last-child {
margin-bottom: 0 !important;
}
.markdown-body a:not([href]) {
color: inherit;
text-decoration: none;
}
.markdown-body blockquote,
.markdown-body details,
.markdown-body dl,
.markdown-body ol,
.markdown-body p,
.markdown-body pre,
.markdown-body table,
.markdown-body ul {
margin-top: 0;
margin-bottom: 16px;
}
.markdown-body hr {
height: .25em;
padding: 0;
margin: 24px 0;
background-color: #e1e4e8;
border: 0;
}
.markdown-body blockquote {
padding: 0 1em;
color: #6a737d;
border-left: .25em solid #dfe2e5;
}
.markdown-body blockquote > :first-child {
margin-top: 0;
}
.markdown-body blockquote > :last-child {
margin-bottom: 0;
}
.markdown-body h1,
.markdown-body h2,
.markdown-body h3,
.markdown-body h4,
.markdown-body h5,
.markdown-body h6 {
margin-top: 24px;
margin-bottom: 16px;
font-weight: 600;
line-height: 1.25;
}
.markdown-body h1 {
font-size: 2em;
}
.markdown-body h1,
.markdown-body h2 {
padding-bottom: .3em;
border-bottom: 1px solid #eaecef;
}
.markdown-body h2 {
font-size: 1.5em;
}
.markdown-body h3 {
font-size: 1.25em;
}
.markdown-body h4 {
font-size: 1em;
}
.markdown-body h5 {
font-size: .875em;
}
.markdown-body h6 {
font-size: .85em;
color: #6a737d;
}
.markdown-body ol,
.markdown-body ul {
padding-left: 2em;
}
.markdown-body ol ol,
.markdown-body ol ul,
.markdown-body ul ol,
.markdown-body ul ul {
margin-top: 0;
margin-bottom: 0;
}
.markdown-body li {
list-style: initial;
}
.markdown-body li > p {
margin-top: 16px;
}
.markdown-body li + li {
margin-top: .25em;
}
.markdown-body dl {
padding: 0;
}
.markdown-body dl dt {
padding: 0;
margin-top: 16px;
font-size: 1em;
font-style: italic;
font-weight: 600;
}
.markdown-body dl dd {
padding: 0 16px;
margin-bottom: 16px;
}
.markdown-body table {
display: block;
width: 100%;
overflow: auto;
}
.markdown-body table th {
font-weight: 600;
}
.markdown-body table td,
.markdown-body table th {
padding: 6px 13px;
border: 1px solid #dfe2e5;
}
.markdown-body table tr {
background-color: #fff;
border-top: 1px solid #c6cbd1;
}
.markdown-body table tr:nth-child(2n) {
background-color: #f6f8fa;
}
.markdown-body img {
max-width: 100%;
box-sizing: initial;
background-color: #fff;
}
.markdown-body img[align=right] {
padding-left: 20px;
}
.markdown-body img[align=left] {
padding-right: 20px;
}
.markdown-body code {
padding: .2em .4em;
margin: 0;
font-size: 85%;
background-color: rgba(27, 31, 35, .05);
border-radius: 3px;
}
.markdown-body pre {
word-wrap: normal;
}
.markdown-body pre > code {
padding: 0;
margin: 0;
font-size: 100%;
word-break: normal;
white-space: pre;
background: transparent;
border: 0;
}
.markdown-body .highlight {
margin-bottom: 16px;
}
.markdown-body .highlight pre {
margin-bottom: 0;
word-break: normal;
}
.markdown-body .highlight pre,
.markdown-body pre {
padding: 16px;
overflow: auto;
font-size: 85%;
line-height: 1.45;
background-color: #f6f8fa;
border-radius: 3px;
}
.markdown-body pre code {
display: inline;
max-width: auto;
padding: 0;
margin: 0;
overflow: visible;
line-height: inherit;
word-wrap: normal;
background-color: initial;
border: 0;
}
.markdown-body .commit-tease-sha {
display: inline-block;
font-family: SFMono-Regular, Consolas, Liberation Mono, Menlo, monospace;
font-size: 90%;
color: #444d56;
}
.markdown-body .full-commit .btn-outline:not(:disabled):hover {
color: #005cc5;
border-color: #005cc5;
}
.markdown-body .blob-wrapper {
overflow-x: auto;
overflow-y: hidden;
}
.markdown-body .blob-wrapper-embedded {
max-height: 240px;
overflow-y: auto;
}
.markdown-body .blob-num {
width: 1%;
min-width: 50px;
padding-right: 10px;
padding-left: 10px;
font-family: SFMono-Regular, Consolas, Liberation Mono, Menlo, monospace;
font-size: 12px;
line-height: 20px;
color: rgba(27, 31, 35, .3);
text-align: right;
white-space: nowrap;
vertical-align: top;
cursor: pointer;
-webkit-user-select: none;
-moz-user-select: none;
-ms-user-select: none;
user-select: none;
}
.markdown-body .blob-num:hover {
color: rgba(27, 31, 35, .6);
}
.markdown-body .blob-num:before {
content: attr(data-line-number);
}
.markdown-body .blob-code {
position: relative;
padding-right: 10px;
padding-left: 10px;
line-height: 20px;
vertical-align: top;
}
.markdown-body .blob-code-inner {
overflow: visible;
font-family: SFMono-Regular, Consolas, Liberation Mono, Menlo, monospace;
font-size: 12px;
color: #24292e;
word-wrap: normal;
white-space: pre;
}
.markdown-body .pl-token.active,
.markdown-body .pl-token:hover {
cursor: pointer;
background: #ffea7f;
}
.markdown-body .tab-size[data-tab-size="1"] {
-moz-tab-size: 1;
tab-size: 1;
}
.markdown-body .tab-size[data-tab-size="2"] {
-moz-tab-size: 2;
tab-size: 2;
}
.markdown-body .tab-size[data-tab-size="3"] {
-moz-tab-size: 3;
tab-size: 3;
}
.markdown-body .tab-size[data-tab-size="4"] {
-moz-tab-size: 4;
tab-size: 4;
}
.markdown-body .tab-size[data-tab-size="5"] {
-moz-tab-size: 5;
tab-size: 5;
}
.markdown-body .tab-size[data-tab-size="6"] {
-moz-tab-size: 6;
tab-size: 6;
}
.markdown-body .tab-size[data-tab-size="7"] {
-moz-tab-size: 7;
tab-size: 7;
}
.markdown-body .tab-size[data-tab-size="8"] {
-moz-tab-size: 8;
tab-size: 8;
}
.markdown-body .tab-size[data-tab-size="9"] {
-moz-tab-size: 9;
tab-size: 9;
}
.markdown-body .tab-size[data-tab-size="10"] {
-moz-tab-size: 10;
tab-size: 10;
}
.markdown-body .tab-size[data-tab-size="11"] {
-moz-tab-size: 11;
tab-size: 11;
}
.markdown-body .tab-size[data-tab-size="12"] {
-moz-tab-size: 12;
tab-size: 12;
}
.markdown-body .task-list-item {
list-style-type: none;
}
.markdown-body .task-list-item + .task-list-item {
margin-top: 3px;
}
.markdown-body .task-list-item input {
margin: 0 .2em .25em -1.6em;
vertical-align: middle;
}

View File

@ -29,6 +29,21 @@ layui.use(['jquery', 'layer', 'helper'], function () {
}); });
}); });
/**
* 咨询
*/
$('.icon-help').on('click', function () {
var url = $(this).parent().data('url');
helper.checkLogin(function () {
layer.open({
type: 2,
title: '课程咨询',
content: [url, 'no'],
area: ['640px', '300px']
});
});
});
/** /**
* 打赏 * 打赏
*/ */

BIN
public/static/logo.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 1.1 KiB