修改微信授权模块

This commit is contained in:
Anyon 2019-12-09 17:36:15 +08:00
parent 88d4e1c409
commit 21071361fe
29 changed files with 3571 additions and 8 deletions

View File

@ -0,0 +1,111 @@
<?php
// +----------------------------------------------------------------------
// | ThinkAdmin
// +----------------------------------------------------------------------
// | 版权所有 2014~2019 广州楚才信息科技有限公司 [ http://www.cuci.cc ]
// +----------------------------------------------------------------------
// | 官方网站: http://demo.thinkadmin.top
// +----------------------------------------------------------------------
// | 开源协议 ( https://mit-license.org )
// +----------------------------------------------------------------------
// | gitee 代码仓库https://gitee.com/zoujingli/ThinkAdmin
// | github 代码仓库https://github.com/zoujingli/ThinkAdmin
// +----------------------------------------------------------------------
namespace app\wechat\controller;
use app\wechat\service\WechatService;
use think\admin\Controller;
use think\admin\Storage;
/**
* 微信授权绑定
* Class Config
* @package app\wechat\controller
*/
class Config extends Controller
{
/**
* 微信授权绑定
* @auth true
* @menu true
* @throws \think\db\exception\DataNotFoundException
* @throws \think\db\exception\DbException
* @throws \think\db\exception\ModelNotFoundException
*/
public function options()
{
$this->_applyFormToken();
$this->thrNotify = url('@wechat/api.push', [], false, true);
if ($this->request->isGet()) {
$this->title = '微信授权绑定';
$this->geoip = $this->app->cache->get('mygeoip', '');
if (empty($this->geoip)) {
$this->geoip = gethostbyname($this->request->host());
$this->app->cache->set('mygeoip', $this->geoip, 360);
}
try {
if (input('?appid') && input('?appkey')) {
sysconf('wechat.type', 'thr');
sysconf('wechat.appid', input('appid'));
sysconf('wechat.appkey', input('appkey'));
WechatService::ThinkAdminConfig(input('appid'))->setApiNotifyUri($this->thrNotify);
}
$source = enbase64url(url('@admin', [], false, true) . '#' . $this->request->url());
$this->authurl = "http://open.cuci.cc/service/api.push/auth?source={$source}";
$this->wechat = WechatService::ThinkAdminConfig(WechatService::instance()->getAppid())->getConfig();
} catch (\Exception $e) {
$this->wechat = [];
}
$this->fetch();
} else {
foreach ($this->request->post() as $k => $v) sysconf($k, $v);
if ($this->request->post('wechat.type') === 'thr') {
WechatService::ThinkAdminConfig(input('appid'))->setApiNotifyUri($this->thrNotify);
}
sysoplog('微信管理', '修改微信授权配置成功');
$uri = url('wechat/config/options');
$this->success('微信参数修改成功!', url('@admin') . "#{$uri}");
}
}
/**
* 微信支付配置
* @auth true
* @menu true
* @throws \think\Exception
* @throws \think\db\exception\DataNotFoundException
* @throws \think\db\exception\DbException
* @throws \think\db\exception\ModelNotFoundException
*/
public function payment()
{
$this->_applyFormToken();
if ($this->request->isGet()) {
$this->title = '微信支付配置';
$file = Storage::instance('local');
$this->wechat_mch_ssl_cer = sysconf('wechat_mch_ssl_cer');
$this->wechat_mch_ssl_key = sysconf('wechat_mch_ssl_key');
$this->wechat_mch_ssl_p12 = sysconf('wechat_mch_ssl_p12');
if (!$file->has($this->wechat_mch_ssl_cer, true)) $this->wechat_mch_ssl_cer = '';
if (!$file->has($this->wechat_mch_ssl_key, true)) $this->wechat_mch_ssl_key = '';
if (!$file->has($this->wechat_mch_ssl_p12, true)) $this->wechat_mch_ssl_p12 = '';
$this->fetch();
} else {
if ($this->request->post('wechat_mch_ssl_type') === 'p12') {
if (!($sslp12 = $this->request->post('wechat_mch_ssl_p12'))) {
$mchid = $this->request->post('wechat_mch_id');
$content = Storage::instance('local')->get($sslp12, true);
if (!openssl_pkcs12_read($content, $certs, $mchid)) {
$this->error('商户MCH_ID与支付P12证书不匹配');
}
}
}
foreach ($this->request->post() as $k => $v) sysconf($k, $v);
sysoplog('微信管理', '修改微信支付配置成功');
$this->success('微信支付配置成功!');
}
}
}

View File

@ -15,7 +15,9 @@
namespace app\wechat\controller;
use app\wechat\service\WechatService;
use think\admin\Controller;
use think\exception\HttpResponseException;
/**
* 微信粉丝管理
@ -28,12 +30,13 @@ class Fans extends Controller
* 绑定数据表
* @var string
*/
protected $table = 'wechatFans';
protected $table = 'WechatFans';
/**
* 微信粉丝管理
* @auth true
* @menu true
* @throws \think\Exception
* @throws \think\db\exception\DataNotFoundException
* @throws \think\db\exception\DbException
* @throws \think\db\exception\ModelNotFoundException
@ -41,7 +44,76 @@ class Fans extends Controller
public function index()
{
$this->title = '微信粉丝管理';
$this->_query($this->table)->order('id desc')->page();
$this->where = ['appid' => WechatService::instance()->getAppid()];
$query = $this->_query($this->table)->like('nickname')->equal('subscribe,is_black');
$query->dateBetween('subscribe_at')->where($this->where)->order('subscribe_time desc')->page();
}
}
/**
* 列表数据处理
* @param array $data
*/
protected function _index_page_filter(array &$data)
{
$tags = $this->app->db->name('WechatFansTags')->column('id,name');
foreach ($data as &$user) {
$user['tags'] = [];
foreach (explode(',', $user['tagid_list']) as $tagid) {
if (isset($tags[$tagid])) $user['tags'][] = $tags[$tagid];
}
}
}
/**
* 批量拉黑粉丝
* @auth true
*/
public function setBlack()
{
try {
$this->_applyFormToken();
foreach (array_chunk(explode(',', $this->request->post('openid')), 20) as $openids) {
WechatService::WeChatUser()->batchBlackList($openids);
$this->app->db->name('WechatFans')->whereIn('openid', $openids)->update(['is_black' => '1']);
}
$this->success('拉黑粉丝信息成功!');
} catch (HttpResponseException $exception) {
throw $exception;
} catch (\Exception $e) {
$this->error("拉黑粉丝信息失败,请稍候再试!{$e->getMessage()}");
}
}
/**
* 取消拉黑粉丝
* @auth true
*/
public function delBlack()
{
try {
$this->_applyFormToken();
foreach (array_chunk(explode(',', $this->request->post('openid')), 20) as $openids) {
WechatService::WeChatUser()->batchUnblackList($openids);
$this->app->db->name('WechatFans')->whereIn('openid', $openids)->update(['is_black' => '0']);
}
$this->success('取消拉黑粉丝信息成功!');
} catch (HttpResponseException $exception) {
throw $exception;
} catch (\Exception $e) {
$this->error("取消拉黑粉丝信息失败,请稍候再试!{$e->getMessage()}");
}
}
/**
* 删除粉丝信息
* @auth true
* @throws \think\db\exception\DbException
*/
public function remove()
{
$this->_applyFormToken();
$this->_delete($this->table);
}
}

View File

@ -0,0 +1,210 @@
<?php
// +----------------------------------------------------------------------
// | ThinkAdmin
// +----------------------------------------------------------------------
// | 版权所有 2014~2019 广州楚才信息科技有限公司 [ http://www.cuci.cc ]
// +----------------------------------------------------------------------
// | 官方网站: http://demo.thinkadmin.top
// +----------------------------------------------------------------------
// | 开源协议 ( https://mit-license.org )
// +----------------------------------------------------------------------
// | gitee 代码仓库https://gitee.com/zoujingli/ThinkAdmin
// | github 代码仓库https://github.com/zoujingli/ThinkAdmin
// +----------------------------------------------------------------------
namespace app\wechat\controller;
use app\wechat\service\WechatService;
use think\admin\Controller;
use think\exception\HttpResponseException;
/**
* 回复规则管理
* Class Keys
* @package app\wechat\controller
*/
class Keys extends Controller
{
/**
* 绑定数据表
* @var string
*/
protected $table = 'WechatKeys';
/**
* 消息类型
* @var array
*/
public $types = [
'text' => '文字', 'news' => '图文', 'image' => '图片', 'music' => '音乐',
'video' => '视频', 'voice' => '语音', 'customservice' => '转客服',
];
/**
* 回复规则管理
* @auth true
* @menu true
* @throws \think\db\exception\DataNotFoundException
* @throws \think\db\exception\DbException
* @throws \think\db\exception\ModelNotFoundException
*/
public function index()
{
// 关键字二维码生成
if ($this->request->get('action') === 'qrc') {
try {
$wechat = WechatService::WeChatQrcode();
$result = $wechat->create($this->request->get('keys', ''));
$this->success('生成二维码成功!', "javascript:$.previewImage('{$wechat->url($result['ticket'])}')");
} catch (HttpResponseException $exception) {
throw $exception;
} catch (\Exception $e) {
$this->error("生成二维码失败,请稍候再试!<br> {$e->getMessage()}");
}
}
// 关键字列表显示
$this->title = '回复规则管理';
$query = $this->_query($this->table)->like('keys,type')->equal('status')->dateBetween('create_at');
$query->whereNotIn('keys', ['subscribe', 'default'])->order('sort desc,id desc')->page();
}
/**
* 列表数据处理
* @param array $data
*/
protected function _index_page_filter(&$data)
{
foreach ($data as &$vo) {
$vo['qrc'] = url('@wechat/keys/index') . "?action=qrc&keys={$vo['keys']}";
$vo['type'] = isset($this->types[$vo['type']]) ? $this->types[$vo['type']] : $vo['type'];
}
}
/**
* 添加关键字
* @auth true
* @throws \think\db\exception\DataNotFoundException
* @throws \think\db\exception\DbException
* @throws \think\db\exception\ModelNotFoundException
*/
public function add()
{
$this->_applyFormToken();
$this->title = '添加关键字规则';
$this->_form($this->table, 'form');
}
/**
* 编辑关键字
* @auth true
* @throws \think\db\exception\DataNotFoundException
* @throws \think\db\exception\DbException
* @throws \think\db\exception\ModelNotFoundException
*/
public function edit()
{
$this->_applyFormToken();
$this->title = '编辑关键字规则';
$this->_form($this->table, 'form');
}
/**
* 删除关键字
* @auth true
* @throws \think\db\exception\DbException
*/
public function remove()
{
$this->_applyFormToken();
$this->_delete($this->table);
}
/**
* 禁用关键字
* @auth true
* @throws \think\db\exception\DbException
*/
public function forbid()
{
$this->_applyFormToken();
$this->_save($this->table, ['status' => '0']);
}
/**
* 启用关键字
* @auth true
* @throws \think\db\exception\DbException
*/
public function resume()
{
$this->_applyFormToken();
$this->_save($this->table, ['status' => '1']);
}
/**
* 配置关注回复
* @auth true
* @throws \think\db\exception\DataNotFoundException
* @throws \think\db\exception\DbException
* @throws \think\db\exception\ModelNotFoundException
*/
public function subscribe()
{
$this->_applyFormToken();
$this->title = '编辑关注回复规则';
$this->_form($this->table, 'form', 'keys', [], ['keys' => 'subscribe']);
}
/**
* 配置默认回复
* @auth true
* @throws \think\db\exception\DataNotFoundException
* @throws \think\db\exception\DbException
* @throws \think\db\exception\ModelNotFoundException
*/
public function defaults()
{
$this->_applyFormToken();
$this->title = '编辑默认回复规则';
$this->_form($this->table, 'form', 'keys', [], ['keys' => 'default']);
}
/**
* 添加数据处理
* @param array $data
*/
protected function _form_filter(array &$data)
{
if ($this->request->isPost() && isset($data['keys'])) {
$db = $this->app->db->name($this->table)->where('keys', $data['keys']);
empty($data['id']) || $db->where('id', 'neq', $data['id']);
if ($db->count() > 0) {
$this->error('关键字已经存在,请使用其它关键字!');
}
}
if ($this->request->isGet()) {
$this->msgTypes = $this->types;
$root = rtrim(dirname(request()->basefile(true)), '\\/');
$this->defaultImage = "{$root}/static/theme/img/image.png";
}
}
/**
* 表单结果处理
* @param boolean $result
*/
protected function _form_result($result)
{
if ($result !== false) {
list($url, $keys) = ['', $this->request->post('keys')];
if (!in_array($keys, ['subscribe', 'default'])) {
$url = url('@admin') . '#' . url('wechat/keys/index') . '?spm=' . $this->request->get('spm');
}
$this->success('恭喜, 关键字保存成功!', $url);
} else {
$this->error('关键字保存失败, 请稍候再试!');
}
}
}

View File

@ -0,0 +1,167 @@
<?php
// +----------------------------------------------------------------------
// | ThinkAdmin
// +----------------------------------------------------------------------
// | 版权所有 2014~2019 广州楚才信息科技有限公司 [ http://www.cuci.cc ]
// +----------------------------------------------------------------------
// | 官方网站: http://demo.thinkadmin.top
// +----------------------------------------------------------------------
// | 开源协议 ( https://mit-license.org )
// +----------------------------------------------------------------------
// | gitee 代码仓库https://gitee.com/zoujingli/ThinkAdmin
// | github 代码仓库https://github.com/zoujingli/ThinkAdmin
// +----------------------------------------------------------------------
namespace app\wechat\controller;
use app\wechat\service\WechatService;
use think\admin\Controller;
use think\exception\HttpResponseException;
/**
* 微信菜单管理
* Class Menu
* @package app\wechat\controller
*/
class Menu extends Controller
{
/**
* 微信菜单的类型
* @var array
*/
protected $menuType = [
'click' => '匹配规则',
'view' => '跳转网页',
'miniprogram' => '打开小程序',
// 'customservice' => '转多客服',
'scancode_push' => '扫码推事件',
'scancode_waitmsg' => '扫码推事件且弹出“消息接收中”提示框',
'pic_sysphoto' => '弹出系统拍照发图',
'pic_photo_or_album' => '弹出拍照或者相册发图',
'pic_weixin' => '弹出微信相册发图器',
'location_select' => '弹出地理位置选择器',
];
/**
* 微信菜单管理
* @auth true
* @menu true
* @throws \think\db\exception\DataNotFoundException
* @throws \think\db\exception\DbException
* @throws \think\db\exception\ModelNotFoundException
*/
public function index()
{
if ($this->request->get('output') === 'json') {
$where = [['keys', 'notin', ['subscribe', 'default']], ['status', 'eq', '1']];
$keys = $this->app->db->name('WechatKeys')->where($where)->order('sort desc,id desc')->select();
$this->success('获取数据成功!', ['menudata' => sysdata('menudata'), 'keysdata' => $keys]);
} else {
$this->title = '微信菜单定制';
$this->menuTypes = $this->menuType;
$this->fetch();
}
}
/**
* 编辑微信菜单
* @auth true
*/
public function edit()
{
if ($this->request->isPost()) {
$data = $this->request->post('data');
if (empty($data)) { // 删除菜单
try {
WechatService::WeChatMenu()->delete();
sysoplog('微信管理', '删除微信菜单成功');
$this->success('删除微信菜单成功!', '');
} catch (HttpResponseException $exception) {
throw $exception;
} catch (\Exception $e) {
sysoplog('微信管理', "删除微信菜单失败:{$e->getMessage()}");
$this->error('删除微信菜单失败,请稍候再试!' . $e->getMessage());
}
} else {
try {
sysdata('menudata', $this->_buildMenuData($menudata = json_decode($data, true)));
WechatService::WeChatMenu()->create(['button' => sysdata('menudata')]);
sysoplog('微信管理', '发布微信菜单成功');
$this->success('保存发布菜单成功!', '');
} catch (HttpResponseException $exception) {
throw $exception;
} catch (\Exception $e) {
sysoplog('微信管理', "发布微信菜单失败:{$e->getMessage()}");
$this->error("微信菜单发布失败,请稍候再试!<br> {$e->getMessage()}");
}
}
}
}
/**
* 菜单数据处理
* @param array $list
* @return array
*/
private function _buildMenuData(array $list)
{
foreach ($list as &$vo) {
unset($vo['active'], $vo['show']);
if (empty($vo['sub_button'])) {
$vo = $this->_buildMenuItemData($vo);
} else {
$item = ['name' => $vo['name'], 'sub_button' => []];
foreach ($vo['sub_button'] as &$sub) {
unset($sub['active'], $sub['show']);
array_push($item['sub_button'], $this->_buildMenuItemData($sub));
}
$vo = $item;
}
}
return $list;
}
/**
* 单个微信菜单数据处理
* @param array $item
* @return array
*/
private function _buildMenuItemData(array $item)
{
switch (strtolower($item['type'])) {
case 'pic_weixin':
case 'pic_sysphoto':
case 'scancode_push':
case 'location_select':
case 'scancode_waitmsg':
case 'pic_photo_or_album':
return ['name' => $item['name'], 'type' => $item['type'], 'key' => isset($item['key']) ? $item['key'] : $item['type']];
case 'click':
if (empty($item['key'])) $this->error('匹配规则存在空的选项');
return ['name' => $item['name'], 'type' => $item['type'], 'key' => $item['key']];
case 'view':
return ['name' => $item['name'], 'type' => $item['type'], 'url' => $item['url']];
case 'miniprogram':
return ['name' => $item['name'], 'type' => $item['type'], 'url' => $item['url'], 'appid' => $item['appid'], 'pagepath' => $item['pagepath']];
}
}
/**
* 取消微信菜单
* @auth true
*/
public function cancel()
{
try {
WechatService::WeChatMenu()->delete();
$this->success('菜单取消成功,重新关注可立即生效!', '');
} catch (HttpResponseException $exception) {
sysoplog('微信管理', '取消微信菜单成功');
throw $exception;
} catch (\Exception $e) {
$this->error("菜单取消失败,请稍候再试!<br> {$e->getMessage()}");
}
}
}

View File

@ -0,0 +1,176 @@
<?php
// +----------------------------------------------------------------------
// | ThinkAdmin
// +----------------------------------------------------------------------
// | 版权所有 2014~2019 广州楚才信息科技有限公司 [ http://www.cuci.cc ]
// +----------------------------------------------------------------------
// | 官方网站: http://demo.thinkadmin.top
// +----------------------------------------------------------------------
// | 开源协议 ( https://mit-license.org )
// +----------------------------------------------------------------------
// | gitee 代码仓库https://gitee.com/zoujingli/ThinkAdmin
// | github 代码仓库https://github.com/zoujingli/ThinkAdmin
// +----------------------------------------------------------------------
namespace app\wechat\controller;
use app\wechat\service\MediaService;
use think\admin\Controller;
/**
* 微信图文管理
* Class News
* @package app\wechat\controller
*/
class News extends Controller
{
/**
* 设置默认操作表
* @var string
*/
protected $table = 'WechatNews';
/**
* 微信图文管理
* @auth true
* @menu true
* @throws \think\db\exception\DataNotFoundException
* @throws \think\db\exception\DbException
* @throws \think\db\exception\ModelNotFoundException
*/
public function index()
{
$this->title = '微信图文列表';
$this->_query($this->table)->where(['is_deleted' => '0'])->order('id desc')->page();
}
/**
* 图文列表数据处理
* @param array $data
* @throws \think\db\exception\DataNotFoundException
* @throws \think\db\exception\ModelNotFoundException
* @throws \think\exception\DbException
*/
protected function _index_page_filter(&$data)
{
foreach ($data as &$vo) $vo = MediaService::news($vo['id']);
}
/**
* 图文选择器
* @return string
* @throws \think\db\exception\DataNotFoundException
* @throws \think\db\exception\DbException
* @throws \think\db\exception\ModelNotFoundException
* @auth true
*/
public function select()
{
$this->index();
}
/**
* 图文列表数据处理
* @param array $data
* @throws \think\db\exception\DataNotFoundException
* @throws \think\db\exception\ModelNotFoundException
* @throws \think\exception\DbException
*/
protected function _select_page_filter(&$data)
{
foreach ($data as &$vo) $vo = MediaService::news($vo['id']);
}
/**
* 添加微信图文
* @auth true
* @throws \think\Exception
* @throws \think\db\exception\DataNotFoundException
* @throws \think\db\exception\DbException
* @throws \think\db\exception\ModelNotFoundException
*/
public function add()
{
if ($this->request->isGet()) {
$this->title = '新建图文';
$this->fetch('form');
} else {
$data = $this->request->post();
if (($ids = $this->_apply_news_article($data['data'])) && !empty($ids)) {
if (data_save($this->table, ['article_id' => $ids, 'create_by' => session('user.id')], 'id') !== false) {
$url = url('@admin') . '#' . url('@wechat/news/index') . '?spm=' . $this->request->get('spm');
$this->success('图文添加成功!', $url);
}
}
$this->error('图文添加失败,请稍候再试!');
}
}
/**
* 编辑微信图文
* @auth true
* @throws \think\Exception
* @throws \think\db\exception\DataNotFoundException
* @throws \think\db\exception\DbException
* @throws \think\db\exception\ModelNotFoundException
*/
public function edit()
{
if (($this->id = $this->request->get('id')) < 1) {
$this->error('参数错误,请稍候再试!');
}
if ($this->request->isGet()) {
if ($this->request->get('output') === 'json') {
$this->success('获取数据成功!', MediaService::news($this->id));
} else {
$this->fetch('form', ['title' => '编辑图文']);
}
} else {
$post = $this->request->post();
if (isset($post['data']) && ($ids = $this->_apply_news_article($post['data']))) {
if (data_save('wechat_news', ['id' => $this->id, 'article_id' => $ids], 'id')) {
$this->success('图文更新成功!', url('@admin') . '#' . url('@wechat/news/index'));
}
}
$this->error('图文更新失败,请稍候再试!');
}
}
/**
* 图文更新操作
* @param array $data
* @param array $ids
* @return string
*/
private function _apply_news_article($data, $ids = [])
{
foreach ($data as &$vo) {
if (empty($vo['digest'])) {
$vo['digest'] = mb_substr(strip_tags(str_replace(["\s", ' '], '', $vo['content'])), 0, 120);
}
$vo['create_at'] = date('Y-m-d H:i:s');
if (empty($vo['id'])) {
$result = $id = Db::name('WechatNewsArticle')->insertGetId($vo);
} else {
$id = intval($vo['id']);
$result = Db::name('WechatNewsArticle')->where('id', $id)->update($vo);
}
if ($result !== false) {
array_push($ids, $id);
}
}
return join(',', $ids);
}
/**
* 删除微信图文
* @throws \think\db\exception\DbException
*/
public function remove()
{
$this->_delete($this->table);
}
}

View File

@ -0,0 +1,62 @@
<?php
// +----------------------------------------------------------------------
// | ThinkAdmin
// +----------------------------------------------------------------------
// | 版权所有 2014~2019 广州楚才信息科技有限公司 [ http://www.cuci.cc ]
// +----------------------------------------------------------------------
// | 官方网站: http://demo.thinkadmin.top
// +----------------------------------------------------------------------
// | 开源协议 ( https://mit-license.org )
// +----------------------------------------------------------------------
// | gitee 代码仓库https://gitee.com/zoujingli/ThinkAdmin
// | github 代码仓库https://github.com/zoujingli/ThinkAdmin
// +----------------------------------------------------------------------
namespace app\wechat\controller\api;
use app\wechat\service\WechatService;
use think\admin\Controller;
use think\Response;
/**
* 前端JS获取控制器
* Class Js
* @package app\wechat\controller\api
*/
class Js extends Controller
{
/**
* 返回生成的JS内容
* @return \think\Response
* @throws \WeChat\Exceptions\InvalidResponseException
* @throws \WeChat\Exceptions\LocalCacheException
* @throws \think\Exception
* @throws \think\db\exception\DataNotFoundException
* @throws \think\db\exception\DbException
* @throws \think\db\exception\ModelNotFoundException
*/
public function index()
{
$url = $this->request->server('http_referer', $this->request->url(true));
$wechat = WechatService::instance()->getWebOauthInfo($url, $this->request->get('mode', 1), false);
$openid = isset($wechat['openid']) ? $wechat['openid'] : '';
$unionid = empty($wechat['fansinfo']['unionid']) ? '' : $wechat['fansinfo']['unionid'];
$configJson = json_encode(WechatService::instance()->getWebJssdkSign($url), JSON_UNESCAPED_UNICODE);
$fansinfoJson = json_encode(isset($wechat['fansinfo']) ? $wechat['fansinfo'] : [], JSON_UNESCAPED_UNICODE);
$html = <<<EOF
if(typeof wx === 'object'){
wx.openid="{$openid}";
wx.unionid="{$unionid}";
wx.config({$configJson});
wx.fansinfo={$fansinfoJson};
wx.ready(function(){
wx.hideOptionMenu();
wx.hideAllNonBaseMenuItem();
});
}
EOF;
return Response::create($html)->contentType('application/x-javascript');
}
}

View File

@ -0,0 +1,333 @@
<?php
// +----------------------------------------------------------------------
// | ThinkAdmin
// +----------------------------------------------------------------------
// | 版权所有 2014~2019 广州楚才信息科技有限公司 [ http://www.cuci.cc ]
// +----------------------------------------------------------------------
// | 官方网站: http://demo.thinkadmin.top
// +----------------------------------------------------------------------
// | 开源协议 ( https://mit-license.org )
// +----------------------------------------------------------------------
// | gitee 代码仓库https://gitee.com/zoujingli/ThinkAdmin
// | github 代码仓库https://github.com/zoujingli/ThinkAdmin
// +----------------------------------------------------------------------
namespace app\wechat\controller\api;
use app\wechat\service\FansService;
use app\wechat\service\MediaService;
use app\wechat\service\WechatService;
use think\admin\Controller;
/**
* 微信消息推送处理
* Class Push
* @package app\wechat\controller\api
*/
class Push extends Controller
{
/**
* 微信APPID
* @var string
*/
protected $appid;
/**
* 微信用户OPENID
* @var string
*/
protected $openid;
/**
* 消息是否加密码
* @var boolean
*/
protected $encrypt;
/**
* 微信OPENID
* @var string
*/
protected $fromOpenid;
/**
* 微信消息对象
* @var array
*/
protected $receive;
/**
* 微信实例对象
* @var \WeChat\Receive
*/
protected $wechat;
/**
* 强制返回JSON消息
* @var boolean
*/
protected $forceJson = false;
/**
* 强制客服消息回复
* @var boolean
*/
protected $forceCustom = false;
/**
* 获取网络出口IP
* @return mixed
*/
public function geoip()
{
return $this->request->ip();
}
/**
* 消息推送处理接口
* @return string
*/
public function index()
{
try {
if ($this->request->has('receive', 'post') && WechatService::instance()->getType() === 'thr') {
$this->forceJson = true; // 强制返回JSON到Service转发
$this->forceCustom = false; // 强制使用客服消息模式推送
$this->appid = $this->request->post('appid', '', null);
$this->openid = $this->request->post('openid', '', null);
$this->encrypt = boolval($this->request->post('encrypt', 0));
$this->receive = $this->toLower(unserialize($this->request->post('receive', '', null)));
if (empty($this->appid) || empty($this->openid) || empty($this->receive)) {
throw new \think\Exception('微信API实例缺失必要参数[appid,openid,receive]');
}
} else {
$this->forceJson = false; // 暂停返回JSON消息对象
$this->forceCustom = false; // 暂停使用客户消息模式
$this->wechat = WechatService::WeChatReceive();
$this->appid = WechatService::instance()->getAppid();
$this->openid = $this->wechat->getOpenid();
$this->encrypt = $this->wechat->isEncrypt();
$this->receive = $this->toLower($this->wechat->getReceive());
}
$this->fromOpenid = $this->receive['tousername'];
// text, event, image, location
if (method_exists($this, ($method = $this->receive['msgtype']))) {
if (is_string(($result = $this->$method()))) return $result;
}
} catch (\Exception $e) {
sysoplog('微信接口', "{$e->getFile()}:{$e->getLine()} [{$e->getCode()}] {$e->getMessage()}");
}
return 'success';
}
/**
* 数组KEY全部转小写
* @param array $data
* @return array
*/
private function toLower(array $data)
{
$data = array_change_key_case($data, CASE_LOWER);
foreach ($data as $key => $vo) if (is_array($vo)) {
$data[$key] = $this->toLower($vo);
}
return $data;
}
/**
* 文件消息处理
* @return boolean|string
* @throws \WeChat\Exceptions\InvalidDecryptException
* @throws \WeChat\Exceptions\InvalidResponseException
* @throws \WeChat\Exceptions\LocalCacheException
* @throws \think\Exception
* @throws \think\db\exception\DataNotFoundException
* @throws \think\db\exception\ModelNotFoundException
*/
protected function text()
{
return $this->keys("wechat_keys#keys#{$this->receive['content']}", false, $this->forceCustom);
}
/**
* 事件消息处理
* @return boolean|string
* @throws \WeChat\Exceptions\InvalidDecryptException
* @throws \WeChat\Exceptions\InvalidResponseException
* @throws \WeChat\Exceptions\LocalCacheException
* @throws \think\Exception
* @throws \think\db\exception\DataNotFoundException
* @throws \think\db\exception\DbException
* @throws \think\db\exception\ModelNotFoundException
*/
protected function event()
{
switch (strtolower($this->receive['event'])) {
case 'subscribe':
$this->updateFansinfo(true);
if (isset($this->receive['eventkey']) && is_string($this->receive['eventkey'])) {
if (($key = preg_replace('/^qrscene_/i', '', $this->receive['eventkey']))) {
return $this->keys("wechat_keys#keys#{$key}", false, true);
}
}
return $this->keys('wechat_keys#keys#subscribe', true, $this->forceCustom);
case 'unsubscribe':
return $this->updateFansinfo(false);
case 'click':
return $this->keys("wechat_keys#keys#{$this->receive['eventkey']}", false, $this->forceCustom);
case 'scancode_push':
case 'scancode_waitmsg':
if (empty($this->receive['scancodeinfo'])) return false;
if (empty($this->receive['scancodeinfo']['scanresult'])) return false;
return $this->keys("wechat_keys#keys#{$this->receive['scancodeinfo']['scanresult']}", false, $this->forceCustom);
case 'scan':
if (empty($this->receive['eventkey'])) return false;
return $this->keys("wechat_keys#keys#{$this->receive['eventkey']}", false, $this->forceCustom);
default:
return false;
}
}
/**
* 关键字处理
* @param string $rule 关键字规则
* @param boolean $isLast 重复回复消息处理
* @param boolean $isCustom 是否使用客服消息发送
* @return boolean|string
* @throws \WeChat\Exceptions\InvalidDecryptException
* @throws \WeChat\Exceptions\InvalidResponseException
* @throws \WeChat\Exceptions\LocalCacheException
* @throws \think\Exception
* @throws \think\db\exception\DataNotFoundException
* @throws \think\db\exception\ModelNotFoundException
* @throws \think\exception\DbException
* @throws \think\exception\PDOException
*/
private function keys($rule, $isLast = false, $isCustom = false)
{
list($table, $field, $value) = explode('#', $rule . '##');
$data = Db::name($table)->where([$field => $value])->find();
if (empty($data['type']) || (array_key_exists('status', $data) && empty($data['status']))) {
return $isLast ? false : $this->keys('wechat_keys#keys#default', true, $isCustom);
}
switch (strtolower($data['type'])) {
case 'keys':
$content = empty($data['content']) ? $data['name'] : $data['content'];
return $this->keys("wechat_keys#keys#{$content}", $isLast, $isCustom);
case 'text':
return $this->sendMessage('text', ['content' => $data['content']], $isCustom);
case 'customservice':
return $this->sendMessage('customservice', ['content' => $data['content']], false);
case 'voice':
if (empty($data['voice_url']) || !($mediaId = MediaService::upload($data['voice_url'], 'voice'))) return false;
return $this->sendMessage('voice', ['media_id' => $mediaId], $isCustom);
case 'image':
if (empty($data['image_url']) || !($mediaId = MediaService::upload($data['image_url'], 'image'))) return false;
return $this->sendMessage('image', ['media_id' => $mediaId], $isCustom);
case 'news':
list($news, $articles) = [MediaService::news($data['news_id']), []];
if (empty($news['articles'])) return false;
foreach ($news['articles'] as $vo) array_push($articles, [
'url' => url("@wechat/api.review/view", '', false, true) . "?id={$vo['id']}",
'title' => $vo['title'], 'picurl' => $vo['local_url'], 'description' => $vo['digest'],
]);
return $this->sendMessage('news', ['articles' => $articles], $isCustom);
case 'music':
if (empty($data['music_url']) || empty($data['music_title']) || empty($data['music_desc'])) return false;
return $this->sendMessage('music', [
'thumb_media_id' => empty($data['music_image']) ? '' : MediaService::upload($data['music_image'], 'image'),
'description' => $data['music_desc'], 'title' => $data['music_title'],
'hqmusicurl' => $data['music_url'], 'musicurl' => $data['music_url'],
], $isCustom);
case 'video':
if (empty($data['video_url']) || empty($data['video_desc']) || empty($data['video_title'])) return false;
$videoData = ['title' => $data['video_title'], 'introduction' => $data['video_desc']];
if (!($mediaId = MediaService::upload($data['video_url'], 'video', $videoData))) return false;
return $this->sendMessage('video', ['media_id' => $mediaId, 'title' => $data['video_title'], 'description' => $data['video_desc']], $isCustom);
default:
return false;
}
}
/**
* 发送消息到微信
* @param string $type 消息类型text|image|voice|video|music|news|mpnews|wxcard
* @param array $data 消息内容数据对象
* @param boolean $isCustom 是否使用客服消息发送
* @return array|boolean
* @throws \WeChat\Exceptions\InvalidDecryptException
* @throws \WeChat\Exceptions\InvalidResponseException
* @throws \WeChat\Exceptions\LocalCacheException
* @throws \think\Exception
* @throws \think\exception\PDOException
*/
private function sendMessage($type, $data, $isCustom = false)
{
if ($isCustom) {
WechatService::WeChatCustom()->send(['touser' => $this->openid, 'msgtype' => $type, "{$type}" => $data]);
} else switch (strtolower($type)) {
case 'text': // 发送文本消息
$reply = ['CreateTime' => time(), 'MsgType' => 'text', 'ToUserName' => $this->openid, 'FromUserName' => $this->fromOpenid, 'Content' => $data['content']];
return $this->forceJson ? json_encode($reply, JSON_UNESCAPED_UNICODE) : WechatService::WeChatReceive()->reply($reply, true, $this->encrypt);
case 'image': // 发送图片消息
return $this->buildMessage($type, ['MediaId' => $data['media_id']]);
case 'voice': // 发送语言消息
return $this->buildMessage($type, ['MediaId' => $data['media_id']]);
case 'video': // 发送视频消息
return $this->buildMessage($type, ['Title' => $data['title'], 'MediaId' => $data['media_id'], 'Description' => $data['description']]);
case 'music': // 发送音乐消息
return $this->buildMessage($type, ['Title' => $data['title'], 'Description' => $data['description'], 'MusicUrl' => $data['musicurl'], 'HQMusicUrl' => $data['musicurl'], 'ThumbMediaId' => $data['thumb_media_id']]);
case 'customservice': // 转交客服消息
if ($data['content']) $this->sendMessage('text', $data, true);
return $this->buildMessage('transfer_customer_service');
case 'news': // 发送图文消息
$articles = [];
foreach ($data['articles'] as $article) array_push($articles, ['PicUrl' => $article['picurl'], 'Title' => $article['title'], 'Description' => $article['description'], 'Url' => $article['url']]);
$reply = ['CreateTime' => time(), 'MsgType' => 'news', 'ToUserName' => $this->openid, 'FromUserName' => $this->fromOpenid, 'Articles' => $articles, 'ArticleCount' => count($articles)];
return $this->forceJson ? json_encode($reply, JSON_UNESCAPED_UNICODE) : WechatService::WeChatReceive()->reply($reply, true, $this->encrypt);
default:
return 'success';
}
}
/**
* 消息数据生成
* @param string $type 消息类型
* @param string|array $data 消息数据
* @return string
* @throws \WeChat\Exceptions\InvalidDecryptException
*/
private function buildMessage($type, $data = [])
{
$reply = ['CreateTime' => time(), 'MsgType' => strtolower($type), 'ToUserName' => $this->openid, 'FromUserName' => $this->fromOpenid];
if (!empty($data)) $reply[ucfirst(strtolower($type))] = $data;
return $this->forceJson ? json_encode($reply, JSON_UNESCAPED_UNICODE) : WechatService::WeChatReceive()->reply($reply, true, $this->encrypt);
}
/**
* 同步粉丝状态
* @param boolean $subscribe 关注状态
* @return boolean
* @throws \think\db\exception\DataNotFoundException
* @throws \think\db\exception\DbException
* @throws \think\db\exception\ModelNotFoundException
*/
private function updateFansinfo($subscribe = true)
{
if ($subscribe) {
try {
$user = WechatService::WeChatUser()->getUserInfo($this->openid);
return FansService::instance()->set(array_merge($user, ['subscribe' => '1', 'appid' => $this->appid]));
} catch (\Exception $e) {
sysoplog('微信接口', __METHOD__ . " {$this->openid} get userinfo faild. {$e->getMessage()}");
return false;
}
} else {
return FansService::instance()->set(['subscribe' => '0', 'openid' => $this->openid, 'appid' => $this->appid]);
}
}
}

View File

@ -0,0 +1,105 @@
<?php
// +----------------------------------------------------------------------
// | ThinkAdmin
// +----------------------------------------------------------------------
// | 版权所有 2014~2019 广州楚才信息科技有限公司 [ http://www.cuci.cc ]
// +----------------------------------------------------------------------
// | 官方网站: http://demo.thinkadmin.top
// +----------------------------------------------------------------------
// | 开源协议 ( https://mit-license.org )
// +----------------------------------------------------------------------
// | gitee 代码仓库https://gitee.com/zoujingli/ThinkAdmin
// | github 代码仓库https://github.com/zoujingli/ThinkAdmin
// +----------------------------------------------------------------------
namespace app\wechat\controller\api;
use app\wechat\service\MediaService;
use think\admin\Controller;
/**
* Class Review
* @package app\wechat\controller\api
*/
class Review extends Controller
{
/**
* 图文展示
* @param integer $id 图文ID
* @throws \think\db\exception\DataNotFoundException
* @throws \think\db\exception\ModelNotFoundException
* @throws \think\exception\DbException
*/
public function news($id = 0)
{
$this->id = empty($id) ? input('id') : $id;
$this->news = MediaService::news($this->id);
$this->fetch();
}
/**
* 文章展示
* @param integer $id 文章ID
* @throws \think\db\exception\DataNotFoundException
* @throws \think\db\exception\DbException
* @throws \think\db\exception\ModelNotFoundException
*/
public function view($id = 0)
{
$where = ['id' => empty($id) ? input('id') : $id];
$this->app->db->name('WechatNewsArticle')->where($where)->update(['read_num' => $this->app->db->raw('read_num+1')]);
$this->info = $this->app->db->name('WechatNewsArticle')->where($where)->find();
$this->fetch();
}
/**
* 文本展示
*/
public function text()
{
$this->content = strip_tags(input('content', ''), '<a><img>');
$this->fetch();
}
/**
* 图片展示
*/
public function image()
{
$this->content = strip_tags(input('content', ''), '<a><img>');
$this->fetch();
}
/**
* 视频展示
*/
public function video()
{
$this->url = strip_tags(input('url', ''), '<a><img>');
$this->title = strip_tags(input('title', ''), '<a><img>');
$this->fetch();
}
/**
* 语音展示
*/
public function voice()
{
$this->url = strip_tags(input('url', ''), '<a><img>');
$this->fetch();
}
/**
* 音乐展示
*/
public function music()
{
$this->url = strip_tags(input('url', ''), '<a><img>');
$this->desc = strip_tags(input('desc', ''), '<a><img>');
$this->title = strip_tags(input('title', ''), '<a><img>');
$this->fetch();
}
}

View File

@ -0,0 +1,250 @@
<?php
// +----------------------------------------------------------------------
// | ThinkAdmin
// +----------------------------------------------------------------------
// | 版权所有 2014~2019 广州楚才信息科技有限公司 [ http://www.cuci.cc ]
// +----------------------------------------------------------------------
// | 官方网站: http://demo.thinkadmin.top
// +----------------------------------------------------------------------
// | 开源协议 ( https://mit-license.org )
// +----------------------------------------------------------------------
// | gitee 代码仓库https://gitee.com/zoujingli/ThinkAdmin
// | github 代码仓库https://github.com/zoujingli/ThinkAdmin
// +----------------------------------------------------------------------
namespace app\wechat\controller\api;
use app\wechat\service\WechatService;
use think\admin\Controller;
/**
* 微信测试工具
* Class Tools
* @package app\wechat\controller\api
*/
class Tools extends Controller
{
/**
* 网页授权测试
* @return string
*/
public function oauth()
{
$this->url = $this->request->url(true);
$this->fans = WechatService::getWebOauthInfo($this->url, 1);
$this->fetch();
}
/**
* 显示网页授权二维码
* @return \think\Response
* @throws \Endroid\QrCode\Exceptions\ImageFunctionFailedException
* @throws \Endroid\QrCode\Exceptions\ImageFunctionUnknownException
* @throws \Endroid\QrCode\Exceptions\ImageTypeInvalidException
*/
public function oauth_qrc()
{
$url = url('@wechat/api.tools/oauth', '', true, true);
return $this->showQrc($url);
}
/**
* JSSDK测试
* @return string
* @throws \WeChat\Exceptions\InvalidResponseException
* @throws \WeChat\Exceptions\LocalCacheException
* @throws \think\Exception
* @throws \think\exception\PDOException
*/
public function jssdk()
{
$this->options = WechatService::getWebJssdkSign();
$this->fetch();
}
/**
* 显示网页授权二维码
* @throws \Endroid\QrCode\Exceptions\ImageFunctionFailedException
* @throws \Endroid\QrCode\Exceptions\ImageFunctionUnknownException
* @throws \Endroid\QrCode\Exceptions\ImageTypeInvalidException
*/
public function jssdk_qrc()
{
$this->url = url('@wechat/api.tools/jssdk', '', true, true);
return $this->showQrc($this->url);
}
/**
* 微信扫码支付模式一二维码显示
* @throws \Endroid\QrCode\Exceptions\ImageFunctionFailedException
* @throws \Endroid\QrCode\Exceptions\ImageFunctionUnknownException
* @throws \Endroid\QrCode\Exceptions\ImageTypeInvalidException
*/
public function scanOneQrc()
{
$pay = WechatService::WePayOrder();
$result = $pay->qrcParams('8888888');
$this->showQrc($result);
}
/**
* 微信扫码支付模式一通知处理
* -- 注意,需要在微信商户配置支付通知地址
* @return string
* @throws \WeChat\Exceptions\InvalidResponseException
* @throws \WeChat\Exceptions\LocalCacheException
*/
public function scanOneNotify()
{
$pay = WechatService::WePayOrder();
$notify = $pay->getNotify();
p('======= 来自扫码支付1的数据 ======');
p($notify);
// 产品ID 你的业务代码,并实现下面的统一下单操作
$product_id = $notify['product_id'];
// 微信统一下单处理
$options = [
'body' => '测试商品产品ID' . $product_id,
'out_trade_no' => time(),
'total_fee' => '1',
'trade_type' => 'NATIVE',
'notify_url' => url('@wechat/api.tools/notify', '', true, true),
'spbill_create_ip' => request()->ip(),
];
$order = $pay->create($options);
p('======= 来自扫码支付1统一下单结果 ======');
p($order);
// 回复XML文本
$result = [
'return_code' => 'SUCCESS',
'return_msg' => '处理成功',
'appid' => $notify['appid'],
'mch_id' => $notify['mch_id'],
'nonce_str' => \WeChat\Contracts\Tools::createNoncestr(),
'prepay_id' => $order['prepay_id'],
'result_code' => 'SUCCESS',
];
$result['sign'] = $pay->getPaySign($result);
p('======= 来自扫码支付1返回的结果 ======');
p($result);
return \WeChat\Contracts\Tools::arr2xml($result);
}
/**
* 扫码支付模式二测试二维码
* @throws \Endroid\QrCode\Exceptions\ImageFunctionFailedException
* @throws \Endroid\QrCode\Exceptions\ImageFunctionUnknownException
* @throws \Endroid\QrCode\Exceptions\ImageTypeInvalidException
* @throws \WeChat\Exceptions\InvalidResponseException
* @throws \WeChat\Exceptions\LocalCacheException
*/
public function scanQrc()
{
$pay = WechatService::WePayOrder();
$result = $pay->create([
'body' => '测试商品',
'out_trade_no' => time(),
'total_fee' => '1',
'trade_type' => 'NATIVE',
'notify_url' => url('@wechat/api.tools/notify', '', true, true),
'spbill_create_ip' => request()->ip(),
]);
$this->showQrc($result['code_url']);
}
/**
* 微信JSAPI支付二维码
* @return \think\Response
* @throws \Endroid\QrCode\Exceptions\ImageFunctionFailedException
* @throws \Endroid\QrCode\Exceptions\ImageFunctionUnknownException
* @throws \Endroid\QrCode\Exceptions\ImageTypeInvalidException
*/
public function jsapiQrc()
{
$this->url = url('@wechat/api.tools/jsapi', '', true, true);
return $this->showQrc($this->url);
}
/**
* 微信JSAPI支付测试
* @throws \WeChat\Exceptions\InvalidResponseException
* @throws \WeChat\Exceptions\LocalCacheException
* @throws \think\Exception
* @throws \think\exception\PDOException
* @link wx-demo-jsapi
*/
public function jsapi()
{
$pay = WechatService::WePayOrder();
$openid = WechatService::getWebOauthInfo(request()->url(true), 0)['openid'];
$options = [
'body' => '测试商品',
'out_trade_no' => time(),
'total_fee' => '1',
'openid' => $openid,
'trade_type' => 'JSAPI',
'notify_url' => url('@wechat/api.tools/notify', '', true, true),
'spbill_create_ip' => request()->ip(),
];
// 生成预支付码
$result = $pay->create($options);
// 创建JSAPI参数签名
$options = $pay->jsapiParams($result['prepay_id']);
$optionJSON = json_encode($options, JSON_UNESCAPED_UNICODE);
// JSSDK 签名配置
$configJSON = json_encode(WechatService::getWebJssdkSign(), JSON_UNESCAPED_UNICODE);
echo '<pre>';
echo "当前用户OPENID: {$openid}";
echo "\n--- 创建预支付码 ---\n";
var_export($result);
echo '</pre>';
echo '<pre>';
echo "\n\n--- JSAPI 及 H5 参数 ---\n";
var_export($options);
echo '</pre>';
echo "<button id='paytest' type='button'>JSAPI支付测试</button>";
echo "
<script src='//res.wx.qq.com/open/js/jweixin-1.2.0.js'></script>
<script>
wx.config($configJSON);
document.getElementById('paytest').onclick = function(){
var options = $optionJSON;
options.success = function(){
alert('支付成功');
}
wx.chooseWXPay(options);
}
</script>";
}
/**
* 支付通知接收处理
* @return string
* @throws \WeChat\Exceptions\InvalidResponseException
*/
public function notify()
{
$wechat = WechatService::WePayOrder();
p($wechat->getNotify());
return 'SUCCESS';
}
/**
* 创建二维码响应对应
* @param string $url 二维码内容
* @throws \Endroid\QrCode\Exceptions\ImageFunctionFailedException
* @throws \Endroid\QrCode\Exceptions\ImageFunctionUnknownException
* @throws \Endroid\QrCode\Exceptions\ImageTypeInvalidException
*/
protected function showQrc($url)
{
$qrCode = new \Endroid\QrCode\QrCode();
$qrCode->setText($url)->setSize(300)->setPadding(20)->setImageType('png');
response($qrCode->get(), 200, ['Content-Type' => 'image/png'])->send();
}
}

View File

@ -0,0 +1,63 @@
<?php
// +----------------------------------------------------------------------
// | ThinkAdmin
// +----------------------------------------------------------------------
// | 版权所有 2014~2019 广州楚才信息科技有限公司 [ http://www.cuci.cc ]
// +----------------------------------------------------------------------
// | 官方网站: http://demo.thinkadmin.top
// +----------------------------------------------------------------------
// | 开源协议 ( https://mit-license.org )
// +----------------------------------------------------------------------
// | gitee 代码仓库https://gitee.com/zoujingli/ThinkAdmin
// | github 代码仓库https://github.com/zoujingli/ThinkAdmin
// +----------------------------------------------------------------------
namespace app\wechat\service;
use think\admin\Service;
/**
* 微信粉丝信息
* Class FansService
* @package app\wechat\service
*/
class FansService extends Service
{
/**
* 增加或更新粉丝信息
* @param array $user 粉丝信息
* @param string $appid 微信APPID
* @return boolean
* @throws \think\db\exception\DataNotFoundException
* @throws \think\db\exception\DbException
* @throws \think\db\exception\ModelNotFoundException
*/
public function set(array $user, $appid = '')
{
if (!empty($user['subscribe_time'])) {
$user['subscribe_at'] = date('Y-m-d H:i:s', $user['subscribe_time']);
}
if (isset($user['tagid_list']) && is_array($user['tagid_list'])) {
$user['tagid_list'] = is_array($user['tagid_list']) ? join(',', $user['tagid_list']) : '';
}
if ($appid !== '') $user['appid'] = $appid;
unset($user['privilege'], $user['groupid']);
return data_save('WechatFans', $user, 'openid');
}
/**
* 获取粉丝信息
* @param string $openid
* @return array|null
* @throws \think\db\exception\DataNotFoundException
* @throws \think\db\exception\DbException
* @throws \think\db\exception\ModelNotFoundException
*/
public function get($openid)
{
return $this->app->db->name('WechatFans')->where(['openid' => $openid])->find();
}
}

View File

@ -0,0 +1,89 @@
<?php
// +----------------------------------------------------------------------
// | ThinkAdmin
// +----------------------------------------------------------------------
// | 版权所有 2014~2019 广州楚才信息科技有限公司 [ http://www.cuci.cc ]
// +----------------------------------------------------------------------
// | 官方网站: http://demo.thinkadmin.top
// +----------------------------------------------------------------------
// | 开源协议 ( https://mit-license.org )
// +----------------------------------------------------------------------
// | gitee 代码仓库https://gitee.com/zoujingli/ThinkAdmin
// | github 代码仓库https://github.com/zoujingli/ThinkAdmin
// +----------------------------------------------------------------------
namespace app\wechat\service;
use think\admin\Service;
use think\admin\Storage;
use WeChat\Contracts\MyCurlFile;
/**
* 微信素材管理
* Class MediaService
* @package app\wechat\service
*/
class MediaService extends Service
{
/**
* 通过图文ID读取图文信息
* @param integer $id 本地图文ID
* @param array $where 额外的查询条件
* @return array
* @throws \think\db\exception\DataNotFoundException
* @throws \think\db\exception\DbException
* @throws \think\db\exception\ModelNotFoundException
*/
public function news($id, $where = [])
{
$data = $this->app->db->name('WechatNews')->where(['id' => $id])->where($where)->find();
list($data['articles'], $articleIds) = [[], explode(',', $data['article_id'])];
$articles = $this->app->db->name('WechatNewsArticle')->whereIn('id', $articleIds)->select();
foreach ($articleIds as $article_id) foreach ($articles as $article) {
if (intval($article['id']) === intval($article_id)) array_push($data['articles'], $article);
unset($article['create_by'], $article['create_at']);
}
return $data;
}
/**
* 上传图片永久素材返回素材media_id
* @param string $url 文件URL地址
* @param string $type 文件类型
* @param array $videoInfo 视频信息
* @return string|null
* @throws \WeChat\Exceptions\InvalidResponseException
* @throws \WeChat\Exceptions\LocalCacheException
* @throws \think\Exception
* @throws \think\db\exception\DataNotFoundException
* @throws \think\db\exception\DbException
* @throws \think\db\exception\ModelNotFoundException
*/
public function upload($url, $type = 'image', $videoInfo = [])
{
$where = ['md5' => md5($url), 'appid' => WechatService::instance()->getAppid()];
if (($mediaId = $this->app->db->name('WechatMedia')->where($where)->value('media_id'))) return $mediaId;
$result = WechatService::WeChatMedia()->addMaterial(self::getServerPath($url), $type, $videoInfo);
data_save('WechatMedia', [
'local_url' => $url, 'md5' => $where['md5'], 'appid' => $where['appid'], 'type' => $type,
'media_url' => isset($result['url']) ? $result['url'] : '', 'media_id' => $result['media_id'],
], 'type', $where);
return $result['media_id'];
}
/**
* 文件位置处理
* @param string $local
* @return string
* @throws \WeChat\Exceptions\LocalCacheException
*/
private function getServerPath($local)
{
if (file_exists($local)) {
return new MyCurlFile($local);
} else {
return new MyCurlFile(Storage::down($local)['file']);
}
}
}

View File

@ -105,9 +105,12 @@ class WechatService extends Service
throw new \think\Exception("class {$name} not defined.");
}
$classname = "\\{$type}\\{$class}";
if ($type === 'ThinkAdmin') {
throw new \think\Exception("Interface mode cannot instance {$classname}");
}
return new $classname(self::instance()->getConfig());
} else {
list($appid, $appkey) = [sysconf('wechat.appid'), sysconf('wechat.appkey')];
list($appid, $appkey) = [sysconf('wechat.thr_appid'), sysconf('wechat.thr_appkey')];
$data = ['class' => $name, 'appid' => $appid, 'time' => time(), 'nostr' => uniqid()];
$data['sign'] = md5("{$data['class']}#{$appid}#{$appkey}#{$data['time']}#{$data['nostr']}");
$token = enbase64url(json_encode($data, JSON_UNESCAPED_UNICODE));
@ -130,6 +133,38 @@ class WechatService extends Service
}
}
/**
* 获取当前微信APPID
* @return bool|string
* @throws \think\Exception
* @throws \think\db\exception\DataNotFoundException
* @throws \think\db\exception\DbException
* @throws \think\db\exception\ModelNotFoundException
*/
public function getAppid()
{
if ($this->getType() === 'api') {
return sysconf('wechat.appid');
} else {
return sysconf('wechat.thr_appid');
}
}
/**
* 获取接口授权模式
* @return string
* @throws \think\Exception
* @throws \think\db\exception\DataNotFoundException
* @throws \think\db\exception\DbException
* @throws \think\db\exception\ModelNotFoundException
*/
public function getType()
{
$type = strtolower(sysconf('wechat.type'));
if (in_array($type, ['api', 'thr'])) return $type;
throw new \think\Exception('请在后台配置微信对接授权模式');
}
/**
* 获取公众号配置参数
* @return array
@ -142,9 +177,84 @@ class WechatService extends Service
return [
'token' => sysconf('wechat.token'),
'appid' => sysconf('wechat.appid'),
'appsecret' => sysconf('service.appsecret'),
'encodingaeskey' => sysconf('service.encodingaeskey'),
'appsecret' => sysconf('wechat.appsecret'),
'encodingaeskey' => sysconf('wechat.encodingaeskey'),
'cache_path' => $this->app->getRuntimePath() . 'wechat',
];
}
/**
* 获取网页授权信息
* @param string $url
* @param integer $isfull
* @param boolean $isRedirect
* @return array
* @throws \WeChat\Exceptions\InvalidResponseException
* @throws \WeChat\Exceptions\LocalCacheException
* @throws \think\Exception
* @throws \think\db\exception\DataNotFoundException
* @throws \think\db\exception\DbException
* @throws \think\db\exception\ModelNotFoundException
*/
public function getWebOauthInfo($url, $isfull = 0, $isRedirect = true)
{
$appid = $this->getAppid();
list($openid, $fansinfo) = [$this->app->session->get("{$appid}_openid"), $this->app->session->get("{$appid}_fansinfo")];
if ((empty($isfull) && !empty($openid)) || (!empty($isfull) && !empty($openid) && !empty($fansinfo))) {
empty($fansinfo) || FansService::instance()->set($fansinfo);
return ['openid' => $openid, 'fansinfo' => $fansinfo];
}
if ($this->getType() === 'api') {
$wechat = self::WeChatOauth();
if (input('state') !== $appid) {
$snsapi = empty($isfull) ? 'snsapi_base' : 'snsapi_userinfo';
$param = (strpos($url, '?') !== false ? '&' : '?') . 'rcode=' . encode($url);
$OauthUrl = $wechat->getOauthRedirect($url . $param, $appid, $snsapi);
if ($isRedirect) redirect($OauthUrl, 301)->send();
exit("window.location.href='{$OauthUrl}'");
}
if (($token = $wechat->getOauthAccessToken()) && isset($token['openid'])) {
$this->app->session->set("{$appid}_openid", $openid = $token['openid']);
if (empty($isfull) && input('rcode')) {
redirect(enbase64url(input('rcode')), 301)->send();
}
$this->app->session->set("{$appid}_fansinfo", $fansinfo = $wechat->getUserInfo($token['access_token'], $openid));
empty($fansinfo) || FansService::instance()->set($fansinfo);
}
redirect(enbase64url(input('rcode')), 301)->send();
} else {
$result = self::ThinkAdminConfig()->oauth(session_id(), $url, $isfull);
session("{$appid}_openid", $openid = $result['openid']);
session("{$appid}_fansinfo", $fansinfo = $result['fans']);
if ((empty($isfull) && !empty($openid)) || (!empty($isfull) && !empty($openid) && !empty($fansinfo))) {
empty($fansinfo) || FansService::instance()->set($fansinfo);
return ['openid' => $openid, 'fansinfo' => $fansinfo];
}
if ($isRedirect && !empty($result['url'])) {
redirect($result['url'], 301)->send();
}
exit("window.location.href='{$result['url']}'");
}
}
/**
* 获取微信网页JSSDK
* @param string $url JS签名地址
* @return array
* @throws \WeChat\Exceptions\InvalidResponseException
* @throws \WeChat\Exceptions\LocalCacheException
* @throws \think\Exception
* @throws \think\db\exception\DataNotFoundException
* @throws \think\db\exception\DbException
* @throws \think\db\exception\ModelNotFoundException
*/
public function getWebJssdkSign($url = null)
{
$url = is_null($url) ? $this->app->request->url(true) : $url;
if ($this->getType() === 'api') {
return self::WeChatScript()->getJsSign($url);
} else {
return self::ThinkAdminConfig($this->getAppid())->jsSign($url);
}
}
}

View File

@ -0,0 +1,43 @@
{extend name="../../admin/view/main"}
{block name="content"}
<div class="layui-card layui-form">
<div class="layui-card-body think-box-shadow padding-bottom-5">
<div class="layui-form-item">
<label class="layui-form-label">Auth<br><span class="nowrap color-desc">授权方式</span></label>
<div class="layui-input-block">
{foreach ['api'=>'公众号平台API模式','thr'=>'第三方平台授权模式','test'=>'接口功能测试'] as $k=>$v}
<input type="radio" data-wechat-type="{$k}" name="wechat.type" value="{$k}" title="{$v}" lay-filter="wechat_type">
{/foreach}
<p class="help-block">请选择微信对接方式,其中第三方平台授权需要微信开放平台支持,同时需要搭建 SERVICE 服务!</p>
</div>
</div>
</div>
</div>
<div class="think-box-shadow ">
<div class="layui-anim layui-anim-upbit" data-type="api">{include file='config/options_api'}</div>
<div class="layui-anim layui-anim-upbit" data-type="thr">{include file='config/options_thr'}</div>
<div class="layui-anim layui-anim-upbit" data-type="test">{include file='config/options_help'}</div>
</div>
{/block}
{block name='script'}
<script>
$(function () {
apply('{:sysconf("wechat.type")}');
window.form.render();
window.form.on('radio(wechat_type)', function (data) {
apply(data.value);
});
function apply(value) {
this.$active = $("[data-wechat-type='" + value + "']").trigger('click');
if (this.$active.size() < 1) $("[data-wechat-type]:first").trigger('click');
$('[data-type="' + value + '"]').show().siblings('[data-type]').hide();
}
});
</script>
{/block}

View File

@ -0,0 +1,65 @@
<form onsubmit="return false;" data-auto="true" method="post" class='layui-form layui-card' autocomplete="off">
<div class="layui-card-body">
<div class="color-blue padding-left-40 padding-bottom-20">
使用API直接对接时需要在微信公众号平台配置授权IP及网页授权域名另外再将获取到的参数填写到下面
</div>
<div class="layui-form-item">
<label class="layui-form-label">Token<br><span class="nowrap color-desc">接口认证令牌</span></label>
<div class="layui-input-block">
<input name="wechat.token" required placeholder="请输入消息推送对接认证Token必填" value="{:sysconf('wechat.token')}" class="layui-input">
<p class="help-block">公众号平台与系统对接认证Token请优先填写此参数并保存然后再在微信公众号平台操作对接。</p>
</div>
</div>
<div class="layui-form-item">
<label class="layui-form-label">AppId<br><span class="nowrap color-desc">公众号APPID</span></label>
<div class="layui-input-block">
<input name="wechat.appid" placeholder="请输入以wx开头的18位公众号APPID必填" pattern="^wx[0-9a-z]{16}$" maxlength="18" required="required" value="{:sysconf('wechat.appid')}" class="layui-input">
<p class="help-block">公众号APPID是所有接口必要参数可以在公众号平台 [ 开发 > 基本配置 ] 页面获取。</p>
</div>
</div>
<div class="layui-form-item">
<label class="layui-form-label">AppSecret<br><span class="nowrap color-desc">公众号Secret</span></label>
<div class="layui-input-block">
<input name="wechat.appsecret" required placeholder="请输入32位公众号AppSecret必填" value="{:sysconf('wechat.appsecret')}" maxlength="32" pattern="^[0-9a-z]{32}$" class="layui-input">
<p class="help-block">公众号应用密钥是所有接口必要参数,可以在公众号平台 [ 开发 > 基本配置 ] 页面授权后获取。</p>
</div>
</div>
<div class="layui-form-item">
<label class="layui-form-label">EnAesKey<br><span class="nowrap color-desc">消息加密密钥</span></label>
<div class="layui-input-block">
<input name="wechat.encodingaeskey" placeholder="请输入43位消息公众号加密密钥可选" value="{:sysconf('wechat.encodingaeskey')}" maxlength="43" pattern="^.{43}$" class="layui-input">
<p class="help-block">若开启了消息加密时必需填写,消息加密密钥必需填写并保持与公众号平台一致。</p>
</div>
</div>
<div class="layui-form-item">
<label class="layui-form-label margin-top-15">PushApiUrl<br><span class="nowrap color-desc">消息推送接收</span></label>
<div class="layui-input-block">
<div class="relative">
<input name="wechat.push_url" value="服务器授权IP{$geoip}" readonly class="layui-input layui-bg-gray border-0">
<a data-copy="{$geoip}" class="absolute layui-btn layui-bg-gray" style="top:0;right:0;background:none!important"><i class="fa fa-copy"></i></a>
</div>
<div class="relative margin-top-5">
<input name="wechat.push_url" value="消息推送地址:{$thrNotify}" readonly class="layui-input layui-bg-gray border-0">
<a data-copy="{$thrNotify}" class="absolute layui-btn layui-bg-gray" style="top:0;right:0;background:none!important"><i class="fa fa-copy"></i></a>
</div>
<p class="help-block">公众号服务平台消息推送接口及服务器授权IP地址需在公众号接口开发处配置。</p>
</div>
</div>
<div class="hr-line-dashed"></div>
<input type="hidden" name="wechat.type" value="api">
<div class="layui-form-item text-center">
<button class="layui-btn" type="submit">保存配置</button>
</div>
</div>
</form>

View File

@ -0,0 +1,42 @@
<form onsubmit="return false;" data-auto="true" method="post" class='layui-form layui-card padding-bottom-20' autocomplete="off">
<div class="layui-card-body" style="padding-left:40px">
<div class="layui-elem-quote border-0">
<p class="font-s14 margin-bottom-10 color-text">
<strong>第三方平台授权</strong><br>
JSSDK签名测试需要在开放平台配置当前的授权域名<a data-copy="{:request()->host()}">{:request()->host()}</a>
</p>
<p class="font-s14 margin-bottom-10 color-text">
<strong>普通接口强制绑定</strong><br>
网页授权及JSSDK签名都需要在公众号平台配置授权域名<a data-copy="{:request()->host()}">{:request()->host()}</a>
</p>
<p class="font-s14 color-text">
<strong>支付测试配置</strong><br>
JSAPI支付测试需要在微信商户平台配置支付目录<a data-copy="{:url('@wechat/api.tools/',[],'',true)}">{:url('@wechat/api.tools/',[],'',true)}</a><br>
扫码支付①需要在微信商户平台配置支付通知地址:<a data-copy="{:url('@wechat/api.tools/scanOneNotify',[],'',true)}">{:url('@wechat/api.tools/scanOneNotify',[],'',true)}</a>
</p>
</div>
<div class="layui-clear">
<div class="pull-left padding-right-15 notselect">
<img class="notselect" data-tips-image src="{:url('@wechat/api.tools/oauth_qrc')}" style="width:120px">
<p class="text-center">网页授权</p>
</div>
<div class="pull-left padding-left-0 padding-right-15">
<img class="notselect" data-tips-image src="{:url('@wechat/api.tools/jssdk_qrc')}" style="width:120px;">
<p class="text-center">JSSDK签名</p>
</div>
<div class="pull-left padding-left-0 padding-right-15">
<img onerror="this.src='__ROOT__/static/theme/img/wechat/qrc_pay_error.jpg'" class="notselect" data-tips-image src="{:url('@wechat/api.tools/jsapiqrc')}" style="width:120px;">
<p class="text-center">JSAPI支付</p>
</div>
<div class="pull-left padding-left-0 padding-right-15">
<img onerror="this.src='__ROOT__/static/theme/img/wechat/qrc_pay_error.jpg'" class="notselect" data-tips-image src="{:url('@wechat/api.tools/scanoneqrc')}" style="width:120px;">
<p class="text-center">扫码支付①</p>
</div>
<div class="pull-left padding-left-0">
<img onerror="this.src='__ROOT__/static/theme/img/wechat/qrc_pay_error.jpg'" class="notselect" data-tips-image src="{:url('@wechat/api.tools/scanqrc')}" style="width:120px;">
<p class="text-center">扫码支付②</p>
</div>
</div>
</div>
</form>

View File

@ -0,0 +1,68 @@
<form onsubmit="return false" data-auto="true" method="post" class='layui-form layui-card' autocomplete="off">
<div class="layui-card-body">
<div class="color-blue padding-left-40 padding-bottom-20">
使用第三方授权时,需要单独搭建 SERVICE 服务并将域名配置到 WECHAT 文件,二者需要使用 Yar 或 Soap 通信!
</div>
<!--{if !empty($wechat)}-->
<div class="layui-form-item">
<label class="layui-form-label">QRCode<br><span class="nowrap color-desc">公众号二维码</span></label>
<div class="layui-input-block">
<div class="pull-left notselect"><img data-tips-image src="{$wechat.qrcode_url|local_image}" style="width:100px;margin-left:-7px"></div>
<div class="pull-left padding-left-10">
<p class="nowrap">微信昵称:{$wechat.nick_name|default=''}</p>
<p class="nowrap">微信类型:{if $wechat.service_type eq 2}服务号{elseif $wechat.service_type eq 3}小程序{else}订阅号{/if} / {$wechat.verify_type_info == -1 ? '未认证' : '<span class="color-green">已认证</span>'}</p>
<p class="nowrap">注册公司:{$wechat.principal_name}</p>
<p class="nowrap">授权绑定:{$wechat.create_at|format_datetime}</p>
</div>
</div>
</div>
<!--{/if}-->
<div class="layui-form-item">
<label class="layui-form-label">Authorize<br><span class="nowrap color-desc">公众号授权绑定</span></label>
<div class="layui-input-block">
<button type="button" data-href="{$authurl|default=''}" class="layui-btn layui-btn-primary">重新绑定公众号</button>
<p class="help-block">点击连接将跳转到微信第三方平台进行公众号授权。</p>
</div>
</div>
<div class="layui-form-item">
<label class="layui-form-label">AppId<br><span class="nowrap color-desc">公众号APPID</span></label>
<div class="layui-input-block">
<input name="wechat.thr_appid" placeholder="请输入以wx开头的18位公众号APPID必填" pattern="^wx[0-9a-z]{16}$" maxlength="18" required value="{:sysconf('wechat.thr_appid')}" class="layui-input">
<p class="help-block">众号 appid 通过微信第三方授权自动获取. 若没有值请进行微信第三方授权。</p>
</div>
</div>
<div class="layui-form-item">
<label class="layui-form-label">AppKey<br><span class="nowrap color-desc">第三方服务密钥</span></label>
<div class="layui-input-block">
<input name="wechat.thr_appkey" required placeholder="请输入32位公众号AppSecret必填" value="{:sysconf('wechat.thr_appkey')}" maxlength="32" pattern="^[0-9a-z]{32}$" class="layui-input">
<p class="help-block">公众号服务平台接口密钥, 通过微信第三方授权自动获取, 若没有值请进行微信第三方授权。</p>
</div>
</div>
<div class="layui-form-item">
<label class="layui-form-label">AppPushUri<br><span class="nowrap color-desc">第三方推送接口</span></label>
<div class="layui-input-block">
<div class="relative margin-top-5">
<input name="wechat.thr_appurl" value="消息推送地址:{$thrNotify}" readonly class="layui-input layui-bg-gray border-0">
<a data-copy="{$thrNotify}" class="absolute layui-btn layui-bg-gray" style="top:0;right:0;background:none!important"><i class="fa fa-copy"></i></a>
</div>
<p class="help-block">公众号服务平台接口通知URL, 公众号消息接收与回复等。</p>
</div>
</div>
<div class="hr-line-dashed"></div>
<input type="hidden" name="wechat.type" value="thr">
<div class="layui-form-item text-center">
<button class="layui-btn" type="submit">保存配置</button>
</div>
</div>
</form>

View File

@ -0,0 +1,84 @@
{extend name="../../admin/view/main"}
{block name="content"}
<form onsubmit="return false;" data-auto="true" method="post" class='layui-form layui-card ' autocomplete="off" lay-filter="payment">
<div class="layui-card-body think-box-shadow">
<div class="padding-left-40 padding-right-30">
<label class="layui-form-item margin-top-20 block relative">
<span class="color-green margin-right-10 font-w7">微信商户ID</span><span class="nowrap color-desc">MCH_ID</span>
<input name="wechat.mch_id" required placeholder="请输入微信商户ID必填" value="{:sysconf('wechat.mch_id')}" class="layui-input">
<p class="help-block">微信商户ID需要在微信商户平台获取MCH_ID 与 APPID 匹配</p>
</label>
<label class="layui-form-item margin-top-20 block relative">
<span class="color-green margin-right-10 font-w7">微信商户KEY</span><span class="nowrap color-desc">MCH_KEY</span>
<input name="wechat.mch_key" placeholder="请输入微信商户密钥(必填)" maxlength="32" required value="{:sysconf('wechat.mch_key')}" class="layui-input">
<p class="help-block">微信商户密钥,需要在微信商户平台操作设置密码并获取密钥</p>
</label>
</div>
<div class="hr-line-dashed"></div>
<div class="padding-left-40 padding-right-30">
<span class="color-green margin-right-10 font-w7">微信商户证书</span><span class="nowrap color-desc">MCH_CERT</span>
<div class="">
{foreach ['p12'=>'上传 P12 证书','pem'=>'上传 PEM 证书'] as $k=>$v}
<input type="radio" data-pem-type="{$k}" name="wechat.mch_ssl_type" value="{$k}" title="{$v}" lay-filter="data-mch-type">
{/foreach}
<p class="help-block">请选择需要上传证书类型P12 和 PEM 二选一,证书需要从微信商户平台获取</p>
<div data-mch-type="p12" class="layui-tab-item padding-top-15 padding-bottom-15">
<input name="wechat_mch_ssl_p12" value="{$wechat_mch_ssl_p12|default=''}" type="hidden">
<button data-file="btn" data-uptype="local" data-safe="true" data-type="p12" data-field="wechat_mch_ssl_p12" type="button" class="layui-btn layui-btn-primary">
<i class="layui-icon layui-icon-vercode font-s14"></i> 上传 P12 证书
</button>
<p class="help-block margin-top-10">微信商户支付 P12 证书,实现订单退款、打款、发红包等支出功能都使用证书</p>
</div>
<div data-mch-type="pem" class="layui-tab-item padding-top-15 padding-bottom-15">
<input name="wechat_mch_ssl_key" value="{$wechat_mch_ssl_key|default=''}" type="hidden">
<button data-file="btn" data-uptype="local" data-safe="true" data-type="pem" data-field="wechat_mch_ssl_key" type="button" class="layui-btn layui-btn-primary margin-right-5">
<i class="layui-icon layui-icon-vercode font-s14"></i> 上传 KEY 证书
</button>
<input name="wechat_mch_ssl_cer" value="{$wechat_mch_ssl_cer|default=''}" type="hidden">
<button data-file="btn" data-uptype="local" data-safe="true" data-type="pem" data-field="wechat_mch_ssl_cer" type="button" class="layui-btn layui-btn-primary">
<i class="layui-icon layui-icon-vercode font-s14"></i> 上传CERT证书
</button>
<p class="help-block margin-top-10">微信商户支付 PEM 双向证书,实现订单退款、打款、发红包等支出功能都使用证书</p>
</div>
</div>
</div>
<div class="hr-line-dashed"></div>
<div class="layui-form-item text-center">
<button class="layui-btn" type="submit">保存配置</button>
</div>
</div>
</form>
{/block}
{block name="script"}
<script>
(new function () {
form.render();
this.type = "{:sysconf('wechat_mch_ssl_type')}" || 'p12';
form.val('payment', {wechat_mch_ssl_type: this.type});
form.on('radio(data-mch-type)', function (data) {
apply(data.value);
});
apply.call(this, this.type);
function apply(type) {
$('[data-mch-type="' + type + '"]').show().siblings('[data-mch-type]').hide();
};
// 证书文件上传控制
this.types = ['wechat_mch_ssl_p12', 'wechat_mch_ssl_key', 'wechat_mch_ssl_cer'];
for (var i in this.types) $('input[name="' + this.types[i] + '"]').on('change', function () {
var input = this, $button = $(this).next('button');
setTimeout(function () {
if (typeof input.value === 'string' && input.value.length > 5) {
$button.find('i').addClass('color-green layui-icon-vercode').removeClass('layui-icon-upload-drag');
} else {
$button.find('i').removeClass('color-green layui-icon-vercode').addClass('layui-icon-upload-drag');
}
}, 300);
}).trigger('change');
});
</script>
{/block}

View File

@ -0,0 +1,98 @@
{extend name="../../admin/view/main"}
{block name="button"}
{if auth("setblack")}
<button data-action='{:url("setblack")}' data-rule="openid#{key}" data-csrf="{:systoken('setblack')}" class='layui-btn layui-btn-sm layui-btn-primary'>批量拉黑</button>
{/if}
{if auth("delblack")}
<button data-action='{:url("delblack")}' data-rule="openid#{key}" data-csrf="{:systoken('delblack')}" class='layui-btn layui-btn-sm layui-btn-primary'>取消拉黑</button>
{/if}
{if auth("sync")}
<button data-load='{:url("sync")}' class='layui-btn layui-btn-sm layui-btn-primary'>同步粉丝</button>
{/if}
{/block}
{block name="content"}
<div class="think-box-shadow">
{include file='fans/index_search'}
<table class="layui-table margin-top-10" lay-skin="line">
{notempty name='list'}
<thead>
<tr>
<th class='list-table-check-td think-checkbox'><input data-auto-none data-check-target='.list-check-box' type='checkbox'></th>
<th width="180px" class='text-left nowrap'>微信昵称</th>
<th width="180px" class="text-left nowrap">粉丝标签</th>
<th width="150px" class='text-left nowrap'>性别语言</th>
<th width="180px" class='text-left nowrap'>关注时间</th>
<th width="80px"></th>
<th></th>
</tr>
</thead>
{/notempty}
<tbody>
{foreach $list as $key=>$vo}
<tr>
<td class='list-table-check-td think-checkbox'>
<input class="list-check-box" value='{$vo.openid}' type='checkbox'>
</td>
<td class='text-left nowrap relative layui-elip'>
<img src="{$vo.headimgurl|default=''}" onerror="$(this).remove()" data-tips-image class="inline-block" style="width:40px;height:40px;vertical-align:top;margin-right:5px">
<div class="inline-block">
昵称:{$vo.nickname|default='--'}
<br>
区域:{$vo.country|default='--'} {$vo.province} {$vo.city}
</div>
</td>
<td class="text-left padding-0">
<div style="max-height:60px;overflow:auto">{foreach $vo.tags as $t}<p><span class="layui-badge layui-bg-cyan margin-right-5">{$t|default='--'}</span></p>{/foreach}</div>
</td>
<td class='text-left nowrap'>
性别:{switch name='vo.sex'}{case value='1'}男{/case}{case value='2'}女{/case}{default}未知{/switch}
<br>
语言:{$vo.language|raw}
</td>
<td class='text-left nowrap'>
日期:{$vo.subscribe_at|format_datetime|str_replace=' ','<br>时间:',###|raw}
</td>
<td class='text-center nowrap'>
{eq name='vo.subscribe' value='0'}
<span class="layui-badge">未关注</span>
{else}
<span class="layui-badge layui-bg-green">已关注</span>
{/eq}
<br>
{eq name='vo.is_black' value='0'}
<span class="layui-badge layui-bg-green">未拉黑</span>
{else}
<span class="layui-badge">已拉黑</span>
{/eq}
</td>
<td class="nowrap">
{eq name='vo.is_black' value='0'}
<!--{if auth("setblack")}-->
<a class="margin-left-10 layui-btn layui-btn-normal layui-btn-sm" data-action="{:url('setblack')}" data-value="openid#{$vo.openid}" data-csrf="{:systoken('setblack')}">拉 黑</a>
<!--{/if}-->
{else}
<!--{if auth("delblack")}-->
<a class="margin-left-10 layui-btn layui-btn-normal layui-btn-sm" data-action="{:url('delblack')}" data-value="openid#{$vo.openid}" data-csrf="{:systoken('delblack')}">拉 白</a>
<!--{/if}-->
{/eq}
{if auth("remove")}
<a class="layui-btn layui-btn-sm layui-btn-danger" data-confirm="确定要删除该粉丝吗?" data-action="{:url('remove')}" data-value="id#{$vo.id}" data-csrf="{:systoken('remove')}">删 除</a>
{/if}
</td>
</tr>
{/foreach}
</tbody>
</table>
{empty name='list'}<span class="notdata">没有记录哦</span>{else}{$pagehtml|raw|default=''}{/empty}
</div>
{/block}

View File

@ -0,0 +1,59 @@
<fieldset>
<legend>条件搜索</legend>
<form class="layui-form layui-form-pane form-search" action="{:request()->url()}" onsubmit="return false" method="get" autocomplete="off">
<div class="layui-form-item layui-inline">
<label class="layui-form-label">微信昵称</label>
<div class="layui-input-inline">
<input name="nickname" value="{$Think.get.nickname|default=''}" placeholder="请输入微信昵称" class="layui-input">
</div>
</div>
<div class="layui-form-item layui-inline">
<label class="layui-form-label">关注状态</label>
<div class="layui-input-inline">
<select class="layui-select" name="subscribe">
{foreach [''=>'-- 全部 --','0'=>'显示未关注的粉丝','1'=>'显示已关注的粉丝'] as $k=>$v}
{eq name='Think.get.subscribe' value='$k.""'}
<option selected value="{$k}">{$v}</option>
{else}
<option value="{$k}">{$v}</option>
{/eq}
{/foreach}
</select>
</div>
</div>
<div class="layui-form-item layui-inline">
<label class="layui-form-label">拉黑状态</label>
<div class="layui-input-inline">
<select class="layui-select" name="is_black">
{foreach [''=>'-- 全部 --','0'=>'显示未拉黑的粉丝','1'=>'显示已拉黑的粉丝'] as $k=>$v}
{eq name='Think.get.is_black' value='$k.""'}
<option selected value="{$k}">{$v}</option>
{else}
<option value="{$k}">{$v}</option>
{/eq}
{/foreach}
</select>
</div>
</div>
<div class="layui-form-item layui-inline">
<label class="layui-form-label">关注时间</label>
<div class="layui-input-inline">
<input name="subscribe_at" value="{$Think.get.subscribe_at|default=''}" placeholder="请选择关注时间" class="layui-input">
</div>
</div>
<div class="layui-form-item layui-inline">
<button class="layui-btn layui-btn-primary"><i class="layui-icon">&#xe615;</i> 搜 索</button>
</div>
</form>
</fieldset>
<script>
window.form.render();
window.laydate.render({range: true, elem: '[name="subscribe_at"]'});
</script>

View File

@ -0,0 +1,271 @@
{extend name='admin@main'}
{block name="style"}
<style>
.keys-container .input-group-addon{top:0;right:0;color:#eee;width:25px;padding:7px;position:absolute;margin-top:-1px;text-align:center;background:#393D49}
.keys-container [data-tips-image]{width:112px;height:auto}
.keys-container .layui-card{width:580px;height:578px;position:absolute;border:1px solid #ccc}
.keys-container .layui-card .layui-card-body{height:515px;padding-right:50px}
</style>
{/block}
{block name="content"}
<div class="nowrap think-box-shadow" style="width:910px">
<div class='mobile-preview inline-block'>
<div class='mobile-header'>公众号</div>
<div class='mobile-body'>
<iframe id="phone-preview" frameborder="0" marginheight="0" marginwidth="0"></iframe>
</div>
</div>
<div class="keys-container inline-block absolute margin-left-10 margin-right-15">
<form class="layui-form" onsubmit="return false" autocomplete="off" data-auto="true" action="{:request()->url()}" method="post">
<div class="layui-card relative">
<div class="layui-card-header layui-bg-gray text-center">编辑关键字</div>
<div class="layui-card-body">
<!--{if !isset($vo.keys) or ($vo.keys neq 'default' and $vo.keys neq 'subscribe')}-->
<div class="layui-form-item margin-top-10">
<label class="layui-form-label">关 键 字</label>
<div class="layui-input-block">
<input required placeholder='请输入关键字' maxlength='20' name='keys' class="layui-input" value='{$vo.keys|default=""}'>
</div>
</div>
<!--{else}-->
<input type="hidden" name="keys" value="{$vo.keys|default=''}">
<!--{/if}-->
<div class="layui-form-item">
<label class="layui-form-label label-required">规则状态</label>
<div class="layui-input-block">
{foreach ['1'=>'启用','0'=>'禁用'] as $k=>$v}
<label class="think-radio">
<!--{if (!isset($vo.status) and $k eq '1') or (isset($vo.status) and $vo.status eq $k)}-->
<input type="radio" checked name="status" value="{$k}"> {$v}
<!--{else}-->
<input type="radio" name="status" value="{$k}"> {$v}
<!--{/if}-->
</label>
{/foreach}
</div>
</div>
<div class="layui-form-item">
<label class="layui-form-label label-required">消息类型</label>
<div class="layui-input-block">
{foreach $msgTypes as $k=>$v}
<label class="think-radio">
<!--{if (!isset($vo.type) and $k eq 'text') or (isset($vo.type) and$vo.type eq $k)}-->
<input name="type" checked type="radio" value="{$k}"> {$v}
<!--{else}-->
<input name="type" type="radio" value="{$k}"> {$v}
<!--{/if}-->
</label>
{/foreach}
</div>
</div>
<div class="layui-form-item" data-keys-type='text'>
<label class="layui-form-label">回复内容</label>
<div class="layui-input-block">
<textarea name="content" maxlength="10000" class="layui-textarea">{$vo.content|raw|default='说点什么吧'}</textarea>
</div>
</div>
<div class="layui-form-item" data-keys-type='news'>
<label class="layui-form-label label-required">选取图文</label>
<div class="layui-input-block">
<input type="hidden" name="news_id" value="{$vo.news_id|default=0}">
<a class="layui-btn layui-btn-primary" data-title="选择图文" data-iframe="{:url('wechat/news/select')}?field={:encode('news_id')}">选择图文</a>
</div>
</div>
<div class="layui-form-item" data-keys-type='image'>
<label class="layui-form-label label-required">图片地址</label>
<div class="layui-input-block">
<input class="layui-input" onchange="$(this).nextAll('img').attr('src', this.value)" value="{$vo.image_url|default=$defaultImage}" name="image_url" required placeholder="请上传图片或输入图片URL地址">
<a data-file="btn" data-type="bmp,png,jpeg,jpg,gif" data-field="image_url" class="input-group-addon"><i class="layui-icon layui-icon-upload"></i></a>
<p class="help-block">文件最大2Mb支持bmp/png/jpeg/jpg/gif格式</p>
<img data-tips-image src='{$vo.image_url|default=$defaultImage}'>
</div>
</div>
<div class="layui-form-item" data-keys-type='voice'>
<label class="layui-form-label">上传语音</label>
<div class="layui-input-block">
<div class="input-group">
<input class='layui-input' value="{$vo.voice_url|default=''}" name="voice_url" required title="请上传语音文件或输入语音URL地址   ">
<a data-file="btn" data-type="mp3,wma,wav,amr" data-field="voice_url" class="input-group-addon"><i class="layui-icon layui-icon-upload"></i></a>
</div>
<p class="help-block">文件最大2Mb播放长度不超过60smp3/wma/wav/amr格式</p>
</div>
</div>
<div class="layui-form-item" data-keys-type='music'>
<label class="layui-form-label">音乐标题</label>
<div class="layui-input-block">
<input class='layui-input' value="{$vo.music_title|default='音乐标题'}" name="music_title" required title="请输入音乐标题">
</div>
</div>
<div class="layui-form-item" data-keys-type='music'>
<label class="layui-form-label label-required">上传音乐</label>
<div class="layui-input-block">
<div class="input-group">
<input class='layui-input' value="{$vo.music_url|default=''}" name="music_url" required title="请上传音乐文件或输入音乐URL地址   ">
<a data-file="btn" data-type="mp3,wma,wav,amr" data-field="music_url" class="input-group-addon"><i class="layui-icon layui-icon-upload"></i></a>
</div>
</div>
</div>
<div class="layui-form-item" data-keys-type='music'>
<label class="layui-form-label">音乐描述</label>
<div class="layui-input-block">
<input name="music_desc" class="layui-input" value="{$vo.music_desc|default='音乐描述'|raw}">
</div>
</div>
<div class="layui-form-item" data-keys-type='music'>
<label class="layui-form-label">音乐图片</label>
<div class="layui-input-block">
<input class="layui-input" value="{$vo.music_image|default=$defaultImage}" name="music_image" required title="请上传音乐图片或输入音乐图片URL地址   ">
<a data-file="btn" data-type="jpg,png" data-field="music_image" class="input-group-addon"><i class="layui-icon layui-icon-upload"></i></a>
<p class="help-block">文件最大64KB只支持JPG格式</p>
</div>
</div>
<div class="layui-form-item" data-keys-type='video'>
<label class="layui-form-label">视频标题</label>
<div class="layui-input-block">
<input class='layui-input' value="{$vo.video_title|default='视频标题'}" name="video_title" required placeholder="请输入视频标题">
</div>
</div>
<div class="layui-form-item" data-keys-type='video'>
<label class="layui-form-label">上传视频</label>
<div class="layui-input-block">
<div class="input-group">
<input class='layui-input' value="{$vo.video_url|default=''}" name="video_url" required title="请上传视频或输入音乐视频URL地址   ">
<a data-file="btn" data-type="mp4" data-field="video_url" class="input-group-addon"><i class="layui-icon layui-icon-upload"></i></a>
</div>
<p class="help-block">文件最大10MB只支持MP4格式</p>
</div>
</div>
<div class="layui-form-item" data-keys-type='video'>
<label class="layui-form-label">视频描述</label>
<div class="layui-input-block">
<input value="{$vo.video_desc|default='视频描述'}" name="video_desc" maxlength="50" class="layui-input">
</div>
</div>
<div class="text-center padding-bottom-10 absolute full-width" style="bottom:0;margin-left:-15px">
<div class="hr-line-dashed margin-top-10 margin-bottom-10"></div>
<button class="layui-btn menu-submit">保存数据</button>
<!--{if !isset($vo.keys) || !in_array($vo.keys,['default','subscribe'])}-->
<button data-cancel-edit class="layui-btn layui-btn-danger" type='button'>取消编辑</button>
<!--{/if}-->
{if isset($vo['id'])}<input type='hidden' value='{$vo.id}' name='id'>{/if}
</div>
</div>
</div>
</form>
</div>
</div>
{/block}
{block name="script"}
<script>
$(function () {
var $body = $('body');
// 取消编辑返回
$('[data-cancel-edit]').on('click', function () {
$.msg.confirm('确定取消编辑吗?', function (index) {
history.back(), $.msg.close(index);
});
});
// 刷新预览显示
function showReview(params) {
var src = '';
if (params['type'] === 'news') {
src = '{:url("@wechat/api.review/news/_id_")}'.replace('_id_', params.content);
} else {
src = '{:url("@wechat/api.review/_type_")}?'.replace('_type_', params.type) + $.param(params || {});
}
$('#phone-preview').attr('src', src);
}
// 图文显示预览
$body.off('change', '[name="news_id"]').on('change', '[name="news_id"]', function () {
showReview({type: 'news', content: this.value});
});
// 文字显示预览
$body.off('change', '[name="content"]').on('change', '[name="content"]', function () {
showReview({type: 'text', content: this.value});
});
// 图片显示预览
$body.off('change', '[name="image_url"]').on('change', '[name="image_url"]', function () {
showReview({type: 'image', content: this.value});
});
// 图片显示预览
$body.off('change', '[name="voice_url"]').on('change', '[name="voice_url"]', function () {
showReview({type: 'voice', content: this.value});
});
// 音乐显示预览
var musicSelector = '[name="music_url"],[name="music_title"],[name="music_desc"],[name="music_image"]';
$body.off('change', musicSelector).on('change', musicSelector, function () {
var params = {type: 'music'}, $parent = $(this).parents('form');
params.url = $parent.find('[name="music_url"]').val();
params.desc = $parent.find('[name="music_desc"]').val();
params.title = $parent.find('[name="music_title"]').val();
params.image = $parent.find('[name="music_image"]').val();
showReview(params);
});
// 视频显示预览
var videoSelector = '[name="video_title"],[name="video_url"],[name="video_desc"]';
$body.off('change', videoSelector).on('change', videoSelector, function () {
var params = {type: 'video'}, $parent = $(this).parents('form');
params.url = $parent.find('[name="video_url"]').val();
params.desc = $parent.find('[name="video_desc"]').val();
params.title = $parent.find('[name="video_title"]').val();
showReview(params);
});
// 默认类型事件
$body.off('click', 'input[name=type]').on('click', 'input[name=type]', function () {
var value = $(this).val(), $form = $(this).parents('form');
if (value === 'customservice') value = 'text';
var $current = $form.find('[data-keys-type="' + value + '"]').removeClass('layui-hide');
$form.find('[data-keys-type]').not($current).addClass('layui-hide');
switch (value) {
case 'news':
return $('[name="news_id"]').trigger('change');
case 'text':
return $('[name="content"]').trigger('change');
case 'image':
return $('[name="image_url"]').trigger('change');
case 'video':
return $('[name="video_url"]').trigger('change');
case 'music':
return $('[name="music_url"]').trigger('change');
case 'voice':
return $('[name="voice_url"]').trigger('change');
case 'customservice':
return $('[name="content"]').trigger('change');
}
});
// 默认事件触发
$('input[name=type]:checked').map(function () {
$(this).trigger('click');
});
});
</script>
{/block}

View File

@ -0,0 +1,180 @@
{extend name="../../admin/view/main"}
{block name="button"}
{if auth("add")}
<button data-open="{:url('add')}" class='layui-btn layui-btn-sm layui-btn-primary'>添加规则</button>
{/if}
{if auth("remove")}
<button data-action='{:url("remove")}' data-rule="id#{key}" data-csrf="{:systoken('remove')}" data-confirm="确定要删除这些规则吗?" class='layui-btn layui-btn-sm layui-btn-primary'>删除规则</button>
{/if}
{/block}
{block name='content'}
<div class="think-box-shadow">
{include file='keys/index_search'}
<table class="layui-table margin-top-10" lay-skin="line">
{notempty name='list'}
<thead>
<tr>
<th class='list-table-check-td think-checkbox'>
<input data-auto-none data-check-target='.list-check-box' type='checkbox'/>
</th>
<th class='list-table-sort-td'>
<button type="button" data-reload class="layui-btn layui-btn-xs">刷 新</button>
</th>
<th class="text-left nowrap">关键字</th>
<th class="text-left nowrap">类型</th>
<th class="text-left nowrap">预览</th>
<th class="text-left nowrap">添加时间</th>
<th class="text-left nowrap">状态</th>
<th></th>
</tr>
</thead>
{/notempty}
<tbody>
{foreach $list as $key=>$vo}
<tr>
<td class='list-table-check-td think-checkbox'>
<input class="list-check-box" value='{$vo.id}' type='checkbox'/>
</td>
<td class='list-table-sort-td'>
<input data-action-blur="{:request()->url()}" data-value="id#{$vo.id};action#sort;sort#{value}" data-loading="false" value="{$vo.sort}" class="list-sort-input">
</td>
<td class="text-left nowrap">
{notempty name='vo.qrc'}<i class="fa fa-qrcode fa-lg pointer margin-right-5" data-load="{$vo.qrc}" data-time="false" data-tips-text="生成关键字二维码"></i>{/notempty}
{$vo.keys}
</td>
<td class="text-left nowrap">{$vo.type}</td>
<td class="text-left nowrap">
{if $vo.type eq '音乐'}
<a data-phone-view='{:url("@wechat/api.review/music")}?title={$vo.music_title|urlencode}&desc={$vo.music_desc|urlencode}'>预览 <i class="fa fa-eye"></i></a>
{elseif in_array($vo.type,['文字','转客服'])}
<a data-phone-view='{:url("@wechat/api.review/text")}?content={$vo.content|urlencode}'>预览 <i class="fa fa-eye"></i></a>
{elseif $vo.type eq '图片'}
<a data-phone-view='{:url("@wechat/api.review/image")}?content={$vo.image_url|urlencode}'>预览 <i class="fa fa-eye"></i></a>
{elseif $vo.type eq '图文'}
<a data-phone-view='{:url("@wechat/api.review/news/".$vo.news_id)}'>预览 <i class="fa fa-eye"></i></a>
{elseif $vo.type eq '视频'}
<a data-phone-view='{:url("@wechat/api.review/video")}?title={$vo.video_title|urlencode}&desc={$vo.video_desc|urlencode}&url={$vo.video_url|urlencode}'>预览 <i class="fa fa-eye"></i></a>
{elseif $vo.type eq '语音'}
<a data-phone-view='{:url("@wechat/api.review/voice")}?content={$vo.voice_url|urlencode}'>预览 <i class="fa fa-eye"></i></a>
{else}
{$vo.content}
{/if}
</td>
<td class="text-left nowrap">{$vo.create_at|format_datetime}</td>
<td class='text-left nowrap'>{if $vo.status eq 0}<span class="color-desc">已禁用</span>{elseif $vo.status eq 1}<span class="color-green">使用中</span>{/if}</td>
<td class='text-left nowrap'>
{if auth("edit")}
<a class="layui-btn layui-btn-sm" data-open='{:url("edit")}?id={$vo.id}'>编 辑</a>
{/if}
{if $vo.status eq 1 and auth("forbid")}
<a class="layui-btn layui-btn-sm layui-btn-warm" data-action="{:url('forbid')}" data-value="id#{$vo.id};status#0" data-csrf="{:systoken('forbid')}">禁 用</a>
{elseif auth("resume")}
<a class="layui-btn layui-btn-sm layui-btn-warm" data-action="{:url('resume')}" data-value="id#{$vo.id};status#1" data-csrf="{:systoken('resume')}">启 用</a>
{/if}
{if auth("remove")}
<a class="layui-btn layui-btn-sm layui-btn-danger" data-confirm="确定要删除该规则吗?" data-action="{:url('remove')}" data-value="id#{$vo.id}" data-csrf="{:systoken('remove')}">删 除</a>
{/if}
</td>
</tr>
{/foreach}
</tbody>
</table>
{empty name='list'}<span class="notdata">没有记录哦</span>{else}{$pagehtml|raw|default=''}{/empty}
</div>
{/block}
{block name="script"}
<script>
$(function () {
/**
* 默认类型事件
* @type String
*/
$('body').off('change', 'select[name=type]').on('change', 'select[name=type]', function () {
var value = $(this).val(), $form = $(this).parents('form');
var $current = $form.find('[data-keys-type="' + value + '"]').removeClass('hide');
$form.find('[data-keys-type]').not($current).addClass('hide');
switch (value) {
case 'news':
return $('[name="news_id"]').trigger('change');
case 'text':
return $('[name="content"]').trigger('change');
case 'image':
return $('[name="image_url"]').trigger('change');
case 'video':
return $('[name="video_url"]').trigger('change');
case 'music':
return $('[name="music_url"]').trigger('change');
case 'voice':
return $('[name="voice_url"]').trigger('change');
}
});
function showReview(params) {
params = params || {};
$('#phone-preview').attr('src', '{"@wechat/review"|app_url}&' + $.param(params));
}
// 图文显示预览
$('body').off('change', '[name="news_id"]').on('change', '[name="news_id"]', function () {
showReview({type: 'news', content: this.value});
});
// 文字显示预览
$('body').off('change', '[name="content"]').on('change', '[name="content"]', function () {
showReview({type: 'text', content: this.value});
});
// 图片显示预览
$('body').off('change', '[name="image_url"]').on('change', '[name="image_url"]', function () {
showReview({type: 'image', content: this.value});
});
// 音乐显示预览
var musicSelector = '[name="music_url"],[name="music_title"],[name="music_desc"],[name="music_image"]';
$('body').off('change', musicSelector).on('change', musicSelector, function () {
var params = {type: 'music'}, $parent = $(this).parents('form');
params.title = $parent.find('[name="music_title"]').val();
params.url = $parent.find('[name="music_url"]').val();
params.image = $parent.find('[name="music_image"]').val();
params.desc = $parent.find('[name="music_desc"]').val();
showReview(params);
});
// 视频显示预览
var videoSelector = '[name="video_title"],[name="video_url"],[name="video_desc"]';
$('body').off('change', videoSelector).on('change', videoSelector, function () {
var params = {type: 'video'}, $parent = $(this).parents('form');
params.title = $parent.find('[name="video_title"]').val();
params.url = $parent.find('[name="video_url"]').val();
params.desc = $parent.find('[name="video_desc"]').val();
showReview(params);
});
// 默认事件触发
$('select[name=type]').map(function () {
$(this).trigger('change');
});
/*! 删除关键字 */
$('[data-delete]').on('click', function () {
var id = this.getAttribute('data-delete');
var url = this.getAttribute('data-action');
var dialogIndex = $.msg.confirm('确定要删除这条记录吗?', function () {
$.form.load(url, {id: id}, 'post', function (ret) {
if (ret.code === "SUCCESS") {
window.location.reload();
}
$.msg.close(dialogIndex);
});
})
});
});
</script>
{/block}

View File

@ -0,0 +1,54 @@
<fieldset>
<legend>条件搜索</legend>
<form class="layui-form layui-form-pane form-search" action="{:request()->url()}" onsubmit="return false" method="get" autocomplete="off">
<div class="layui-form-item layui-inline">
<label class="layui-form-label">匹配规则</label>
<div class="layui-input-inline">
<input name="keys" value="{$Think.get.keys|default=''}" placeholder="请输入匹配规则" class="layui-input">
</div>
</div>
<div class="layui-form-item layui-inline">
<label class="layui-form-label">规则类型</label>
<div class="layui-input-inline">
<select class="layui-select" name="type">
<option value="">-- 全部 --</option>
{foreach $types as $k=>$v}
{eq name='Think.get.type' value='$k.""'}
<option selected value="{$k}">{$v}</option>
{else}
<option value="{$k}">{$v}</option>
{/eq}
{/foreach}
</select>
</div>
</div>
<div class="layui-form-item layui-inline">
<label class="layui-form-label">使用状态</label>
<div class="layui-input-inline">
<select class="layui-select" name="status">
{foreach [''=>'-- 全部 --','0'=>'显示使用的规则','1'=>'显示禁止的规则'] as $k=>$v}
{eq name='Think.get.status' value='$k.""'}
<option selected value="{$k}">{$v}</option>
{else}
<option value="{$k}">{$v}</option>
{/eq}
{/foreach}
</select>
</div>
</div>
<div class="layui-form-item layui-inline">
<label class="layui-form-label">创建时间</label>
<div class="layui-input-inline">
<input name="create_at" value="{$Think.get.create_at|default=''}" placeholder="请选择创建时间" class="layui-input">
</div>
</div>
<div class="layui-form-item layui-inline">
<button class="layui-btn layui-btn-primary"><i class="layui-icon">&#xe615;</i> 搜 索</button>
</div>
</form>
</fieldset>
<script>
window.form.render();
window.laydate.render({range: true, elem: '[name="create_at"]'});
</script>

View File

@ -0,0 +1,203 @@
{extend name="../../admin/view/main"}
{block name='content'}
<div id="MenuEditor" class="layui-hide think-box-shadow" style="width:910px">
<div class='mobile-preview inline-block'>
<div class='mobile-header'>公众号</div>
<div class='mobile-body'></div>
<ul class='mobile-footer notselect'>
<li class="parent-menu" style="{{getItemStyle(list)}}" ng-repeat="one in list">
<a ng-click="setActiveItem(one)" ng-class="{true:'active'}[one.active]">
<i class="icon-sub" ng-if="one.sub_button.length>0"></i> <span ng-bind="one.name"></span>
</a>
<i class="close layui-bg-gray layui-icon" ng-click="delItem(one)">&#x1006;</i>
<div class="sub-menu text-center" ng-if="one.active||one.show">
<ul>
<li ng-repeat="two in one.sub_button">
<a class="bottom-border" ng-click="setActiveItem(one,two)" ng-class="{true:'active'}[two.active]"><span ng-bind="two.name"></span></a>
<i class="close layui-bg-gray layui-icon" ng-click="delItem(one,two)">&#x1006;</i>
</li>
<li class="menu-add" ng-if="one.sub_button.length<5"><a ng-click="addItem(one.sub_button)"><i class="icon-add"></i></a></li>
</ul>
<i class="arrow arrow_out"></i>
<i class="arrow arrow_in"></i>
</div>
</li>
<li class="parent-menu menu-add" style="{{getItemStyle(list)}}" ng-if="list.length<3">
<a ng-click="addItem(list)"><i class="icon-add"></i></a>
</li>
</ul>
</div>
<div class="absolute inline-block layui-card margin-left-10" style="border:1px solid #ccc;height:578px;width:570px">
<div class="layui-card-header layui-bg-gray text-center">菜单编辑</div>
<div class="layui-card-body" ng-if="list.length<1">
<blockquote class="layui-elem-quote border-0 text-center">请在左侧创建菜单...</blockquote>
</div>
<div class="layui-card-body" ng-if="list.length>0">
<form class="layui-form menu-form" autocomplete="off" style="padding-right:80px;">
<div class="layui-form-item margin-top-20">
<label class="layui-form-label">菜单名称</label>
<div class="layui-input-block">
<input required name="menu-name" ng-model="item.name" class="layui-input" placeholder="请输入菜单名称">
<span class="help-block">字数不超过13个汉字或40个字母</span>
</div>
</div>
<div class="layui-form-item margin-top-20" ng-if="!item.sub_button||item.sub_button.length<1">
<label class="layui-form-label label-required">菜单类型</label>
<div class="layui-input-block">
{foreach $menuTypes as $key => $type}
<label class="think-radio layui-elip"><input lay-ignore type="radio" ng-model="item.type" name="menu-type" value="{$key}"> {$type}</label>
{/foreach}
</div>
</div>
<div class="layui-form-item margin-top-20" ng-if="item.type==='customservice'">
<label class="layui-form-label">提示文字</label>
<div class="layui-input-block">
<textarea required class="layui-textarea" ng-model="item.content"></textarea>
</div>
</div>
<div class="layui-form-item margin-top-20" ng-if="(!item.sub_button||item.sub_button.length<1)&&item.type==='click'">
<label class="layui-form-label">匹配规则</label>
<div class="layui-input-block">
<select required class="layui-select" lay-filter="key" lay-search>
<option value="{{x.keys}}" ng-selected="x.keys===item.key" ng-repeat="x in keys" ng-bind="x.keys"></option>
</select>
</div>
</div>
<div class="layui-form-item margin-top-20" ng-if="item.type==='view'">
<label class="layui-form-label">跳转链接</label>
<div class="layui-input-block">
<textarea required class="layui-textarea" ng-model="item.url" placeholder="请输入跳转链接"></textarea>
</div>
</div>
<div ng-if="item.type==='miniprogram'">
<div class="layui-form-item margin-top-20">
<label class="layui-form-label">小程序链接</label>
<div class="layui-input-block">
<input type="text" required class="layui-input" ng-model="item.url" placeholder="请输入小程序链接">
</div>
</div>
<div class="layui-form-item margin-top-20">
<label class="layui-form-label">小程序APPID</label>
<div class="layui-input-block">
<input type="text" required class="layui-input" ng-model="item.appid" placeholder="请输入小程序APPID">
</div>
</div>
<div class="layui-form-item margin-top-20">
<label class="layui-form-label">小程序页面</label>
<div class="layui-input-block">
<input type="text" required class="layui-input" ng-model="item.pagepath" placeholder="请输入小程序页面">
</div>
</div>
</div>
</form>
</div>
</div>
<div class="text-center margin-top-20">
{if auth("edit")}
<button class="layui-btn menu-submit" ng-click="submit()">保存发布</button>
{/if}
{if auth("cancel")}
<button data-load='{:url("cancel")}' class="layui-btn layui-btn-danger">取消发布</button>
{/if}
</div>
</div>
<script>
require(['angular'], function () {
var $form = $('.menu-form').vali();
var $vali = $form.vali().data('validate');
$('#MenuEditor.layui-hide').removeClass('layui-hide');
var app = angular.module("MenuEditor", []).run(callback);
angular.bootstrap(document.getElementById(app.name), [app.name]);
function callback($rootScope) {
$rootScope.item = {};
$rootScope.list = [];
$rootScope.keys = [];
$.form.load('{:url("index")}', {output: 'json'}, 'get', function (ret) {
return $rootScope.$apply(function () {
$rootScope.keys = ret.data.keysdata || [];
$rootScope.list = ret.data.menudata || [];
if ($rootScope.list.length < 1) $rootScope.list = [{name: '请输入名称', type: 'click', sub_button: []}];
for (var i in $rootScope.list) $rootScope.list[i].sub_button = $rootScope.list[i].sub_button || [];
$rootScope.list[0].show = true;
$rootScope.list[0].active = true;
$rootScope.item = $rootScope.list[0];
}), false;
});
// 动态计算宽度
$rootScope.getItemStyle = function (list) {
return 'width:' + (100 / (list.length >= 3 ? 3 : (list.length + 1))) + '%';
};
// 增加菜单选项
$rootScope.addItem = function (list) {
$vali.checkAllInput();
if ($form.find('.validate-error').size() > 0) {
return $.msg.tips('表单验证不成功,请输入需要的内容!');
}
list.push({name: '请输入名称', type: 'click', sub_button: []});
};
// 移除菜单
$rootScope.delItem = function (one, two) {
var tmp = [], _two = null;
if (two) {
for (var i in one.sub_button) if (one.sub_button[i] !== two) {
tmp.push(one.sub_button[i]);
if (one.sub_button[i].active) _two = one.sub_button[i];
}
one.sub_button = tmp;
return $rootScope.setActiveItem(one, _two);
}
for (var i in $rootScope.list) if (one !== $rootScope.list[i]) tmp.push($rootScope.list[i]);
$rootScope.list = tmp;
if ($rootScope.list.length > 1) $rootScope.setActiveItem($rootScope.list[0])
};
// 切换选择菜单
$rootScope.setActiveItem = function (one, two) {
for (var i in $rootScope.list) {
$rootScope.list[i].show = ($rootScope.list[i] === one);
$rootScope.list[i].active = two ? false : ($rootScope.list[i] === one);
}
for (var i in $rootScope.list) for (var j in $rootScope.list[i].sub_button) {
$rootScope.list[i].sub_button[j].active = ($rootScope.list[i].sub_button[j] === two)
}
$rootScope.item = two || one || {};
$rootScope.item.type = $rootScope.item.type || 'click';
if ($rootScope.item.type === 'click') setTimeout(function () {
form.render('select');
$rootScope.item.key = $('[lay-filter="key"]').val();
}, 50);
};
// 下拉列表处理
$rootScope.$watch('item', function () {
if ($rootScope.item.type === 'click') setTimeout(function () {
form.render('select');
}, 50)
}, true);
form.on('select(key)', function (data) {
$rootScope.item.key = data.value;
});
// 提交数据
$rootScope.submit = function () {
$vali.checkAllInput();
if ($form.find('.validate-error').size() > 0) return false;
$.form.load('{:url("edit")}', {data: angular.toJson($rootScope.list)}, 'post');
}
}
})
</script>
{/block}

View File

@ -0,0 +1,84 @@
<style>
.news-left {
width: 300px;
float: left;
margin-right: 15px;
}
.news-right {
overflow: hidden;
width: 980px;
position: relative;
display: inline-block;
}
.news-left .news-item {
height: 150px;
cursor: pointer;
max-width: 270px;
overflow: hidden;
position: relative;
border: 1px solid #ccc;
background-size: cover;
background-position: center center
}
.news-left .news-item.active {
border: 1px solid #44b549 !important
}
.news-left .article-add {
color: #999;
display: block;
font-size: 22px;
text-align: center
}
.news-left .article-add:hover {
color: #666
}
.news-left .news-title {
bottom: 0;
color: #fff;
width: 272px;
display: block;
padding: 0 5px;
max-height: 6em;
overflow: hidden;
margin-left: -1px;
position: absolute;
text-overflow: ellipsis;
background: rgba(0, 0, 0, .7)
}
.news-left .news-item a {
color: #fff;
width: 30px;
height: 30px;
float: right;
font-size: 12px;
margin-top: -1px;
line-height: 34px;
text-align: center;
margin-right: -1px;
background-color: rgba(0, 0, 0, .5)
}
.news-left .news-item:hover a {
display: inline-block !important
}
.news-left .news-item a:hover {
text-decoration: none;
background-color: #0C0C0C
}
.news-right .upload-image-box {
width: 130px;
height: 90px;
border: 1px solid rgba(125, 125, 125, .1);
background: url("__ROOT__/static/plugs/uploader/theme/image.png") no-repeat center center;
background-size: cover;
}
</style>

View File

@ -0,0 +1,215 @@
{extend name='admin@main'}
{block name="style"}{include file='wechat@news/form-style'}{/block}
{block name='content'}
<div id="NewsEditor" class="layui-clear nowrap padding-bottom-30">
<div class="layui-card news-left">
<div class="layui-card-body layui-hide">
<div ng-if="list.length > 0" ng-repeat="x in list">
<div class="news-item transition" ng-click="setItem($index,$event)" style="{{x.style}}" ng-class="x.active?'active':''">
<a ng-click="delItem($index, $event)" class="transition layui-icon layui-hide">&#x1006;</a>
<a ng-click="dnItem($index, $event)" class="transition layui-icon layui-hide">&#xe61a;</a>
<a ng-click="upItem($index, $event)" class="transition layui-icon layui-hide">&#xe619;</a>
<span class="news-title" ng-bind="x.title"></span>
</div>
<hr>
</div>
<div ng-if="list.length<1" class="news-item transition active">
<a ng-click="delItem($index, $event)" class="transition layui-icon layui-hide">&#x1006;</a>
<a ng-click="dnItem($index, $event)" class="transition layui-icon layui-hide">&#xe61a;</a>
<a ng-click="upItem($index, $event)" class="transition layui-icon layui-hide">&#xe619;</a>
<span class="news-title"></span>
<hr>
</div>
<a class='article-add transition' ng-click="addItem()" data-text-tip="添加图文"><i class="fa fa-plus"></i></a>
</div>
</div>
<div class="layui-card news-right">
<div class="layui-card-body">
<form class="layui-form padding-20" role="form" name="news" onsubmit="return false">
<label class="layui-form-item relative block">
<span class="color-green">文章标题</span>
<input maxlength="64" ng-model="item.title" required placeholder="请在这里输入标题" name='title' class="layui-input">
</label>
<label class="layui-form-item relative block">
<span class="color-green">文章作者</span>
<input maxlength="64" ng-model="item.author" required placeholder="请输入文章作者" name='author' class="layui-input">
</label>
<div class="layui-form-item label-required-prev">
<span class="color-green">图文封面大图片</span>
<div class="layui-clear">
<div class="upload-image-box pull-left transition" data-tips-image="{{item.local_url}}" style="{{item.style}}">
<input ng-model="item.local_url" data-auto-none value="{{item.local_url}}" type="hidden" name="local_url">
</div>
<div class="pull-left padding-top-15 padding-left-15">
<button type="button" data-title="上传图片" data-file="btn" data-type="jpg,png,jpeg" data-field="local_url" class="layui-btn layui-btn-sm layui-btn-primary">上传图片</button>
<br>
<label class="think-checkbox notselect margin-top-15">
<input ng-model="item.show_cover_pic" ng-checked="!!item.show_cover_pic" data-auto-none type="checkbox" value="1" name="show_cover_pic"> 在正文顶部显示此图片
</label>
</div>
</div>
<p class="color-desc">封面大图片建议尺寸 900像素 * 500像素</p>
</div>
<div class="layui-form-item label-required-prev">
<span class="color-green">图文文章内容</span>
<textarea ng-model="item.content" name='content'></textarea>
</div>
<label class="layui-form-item relative block">
<span class="help-block">摘要选填如果不填写会默认抓取正文前54个字</span>
<textarea ng-model="item.digest" name="digest" class="layui-textarea" style="height:80px;line-height:18px"></textarea>
</label>
<label class="layui-form-item relative block">
<span class="help-block">原文链接 <span class="color-blue">选填</span>,填写之后在图文左下方会出现此链接</span>
<input pattern="^(http:\/\/|^https:\/\/|^\/\/)((\w|=|\?|\.|\/|&|-)+)" placeholder="请输入跳转链接地址" ng-model="item.content_source_url" maxlength="200" name='content_source_url' class="layui-input">
</label>
<div class="layui-form-item text-center padding-top-30">
<input ng-model="x.read_num" type="hidden">
<button ng-click="submit()" type="button" class="layui-btn">保存图文</button>
</div>
</form>
</div>
</div>
</div>
{/block}
{block name='script'}
<script>
require(['angular', 'ckeditor'], function () {
var $form = $('form[name="news"]');
var $vali = $form.vali().data('validate');
var editor = window.createEditor('[name="content"]');
var app = angular.module("NewsEditor", []).run(callback);
angular.bootstrap(document.getElementById(app.name), [app.name]);
function callback($rootScope) {
$rootScope.list = [];
$rootScope.item = {};
$.form.load('{:request()->url()}', {output: 'json'}, 'get', function (ret) {
return $rootScope.$apply(function () {
apply((ret.data || {articles: []}).articles || []);
}), false;
});
function apply(list) {
if (list.length < 1) list.push({
title: '新建图文',
author: '管理员',
content: '文章内容',
read_num: 0,
local_url: '__ROOT__/static/plugs/uploader/theme/image.png',
});
for (var i in list) {
list[i].active = false;
list[i].style = "background-image:url('" + list[i].local_url + "')";
}
$rootScope.list = list;
$rootScope.item = $rootScope.list[0];
$rootScope.setItemValue('active', true);
setTimeout(function () {
$vali.checkAllInput();
editor.setData($rootScope.item.content);
$('.layui-card-body.layui-hide').removeClass('layui-hide');
}, 100);
}
$rootScope.upItem = function (index, $event) {
$event.stopPropagation();
var tmp = [], cur = $rootScope.list[index];
if (index < 1) return false;
for (var i in $rootScope.list) {
(parseInt(i) === parseInt(index) - 1) && tmp.push(cur);
(parseInt(i) !== parseInt(index)) && tmp.push($rootScope.list[i]);
}
apply(tmp);
};
$rootScope.dnItem = function (index, $event) {
$event.stopPropagation();
var tmp = [], cur = $rootScope.list[index];
if (index > $rootScope.list.length - 2) return false;
for (var i in $rootScope.list) {
(parseInt(i) !== parseInt(index)) && tmp.push($rootScope.list[i]);
(parseInt(i) === parseInt(index) + 1) && tmp.push(cur);
}
apply(tmp);
};
$rootScope.delItem = function (index, $event) {
$event.stopPropagation();
var list = $rootScope.list, temp = [];
for (var i in list) (parseInt(i) !== parseInt(index)) && temp.push(list[i]);
apply(temp);
};
$rootScope.setItem = function (index, $event) {
$event.stopPropagation();
$vali.checkAllInput();
if ($form.find('.validate-error').size() > 0) return 0;
if (editor.getData().length < 1) return $.msg.tips('文章内容不能为空请输入文章内容');
for (var i in $rootScope.list) {
if (parseInt(i) !== parseInt(index)) {
$rootScope.list[i].active = false;
} else {
$rootScope.item.content = editor.getData();
$rootScope.item = $rootScope.list[i];
editor.setData($rootScope.item.content);
$rootScope.setItemValue('active', true);
}
}
};
$rootScope.setItemValue = function (name, value) {
$rootScope.item[name] = value;
$rootScope.item.style = "background-image:url('" + $rootScope.item.local_url + "')";
};
$rootScope.addItem = function () {
if ($rootScope.list.length > 7) return $.msg.tips('最多允许增加7篇文章哦');
$rootScope.list.push({
title: '新建图文',
author: '管理员',
content: '文章内容',
read_num: 0,
local_url: '__ROOT__/static/plugs/uploader/theme/image.png',
style: "background-image:url('__ROOT__/static/plugs/uploader/theme/image.png')"
});
};
$rootScope.submit = function () {
$vali.checkAllInput();
if ($form.find('.validate-error').size() > 0) {
return $.msg.tips('表单验证不成功,请输入需要的内容!');
}
$rootScope.item.content = editor.getData();
var data = [];
for (var i in $rootScope.list) data.push({
id: $rootScope.list[i].id,
title: $rootScope.list[i].title,
author: $rootScope.list[i].author,
digest: $rootScope.list[i].digest,
content: $rootScope.list[i].content,
read_num: $rootScope.list[i].read_num,
local_url: $rootScope.list[i].local_url,
show_cover_pic: $rootScope.list[i].show_cover_pic ? 1 : 0,
content_source_url: $rootScope.list[i].content_source_url,
});
$.form.load('{:request()->url()}', {data: data}, "post");
};
$('[name="local_url"]').on('change', function () {
var value = this.value;
$rootScope.$apply(function () {
$rootScope.setItemValue('local_url', value);
});
});
}
});
</script>
{/block}

View File

@ -0,0 +1,158 @@
{extend name="../../admin/view/main"}
{block name="button"}
<button data-open="{:url('add')}" class='layui-btn layui-btn-sm layui-btn-primary'>添加图文</button>
{/block}
{block name='content'}
<div class="think-box-shadow">
<div id="news-box" class="layui-clear">
{foreach $list as $vo}
<div class="news-item">
<div class='news-tools layui-hide'>
<a data-phone-view="{:url('@wechat/api.review/news/'.$vo.id)}" href='javascript:void(0)'>预览</a>
<a data-open='{:url("edit")}?id={$vo.id}' href='javascript:void(0)'>编辑</a>
<a data-news-del="{$vo.id}" href='javascript:void(0)'>删除</a>
</div>
{foreach $vo.articles as $k => $v}
{if $k < 1}
<div data-tips-image='{$v.local_url}' class='news-articel-item' style='background-image:url("{$v.local_url}")'>
{if $v.title}<p>{$v.title}</p>{/if}
</div>
<div class="hr-line-dashed"></div>
{else}
<div class='news-articel-item other'>
<span>{$v.title}</span>
<div data-tips-image='{$v.local_url}' style='background-image:url("{$v.local_url}");'></div>
</div>
<div class="hr-line-dashed"></div>
{/if}
{/foreach}
</div>
{/foreach}
</div>
{empty name='list'}<span class="notdata">没有记录哦</span>{else}{$pagehtml|raw|default=''}{/empty}
</div>
{/block}
{block name='script'}
<script>
$('body').on('mouseenter', '.news-item', function () {
$(this).find('.news-tools').removeClass('layui-hide');
}).on('mouseleave', '.news-item', function () {
$(this).find('.news-tools').addClass('layui-hide');
});
require(['jquery.masonry'], function (Masonry) {
var item = document.querySelector('#news-box');
var msnry = new Masonry(item, {itemSelector: '.news-item', columnWidth: 0});
msnry.layout();
$('body').on('click', '[data-news-del]', function () {
var that = this;
var dialogIndex = $.msg.confirm('确定要删除图文吗?', function () {
$.form.load('{:url("remove")}', {value: 0, field: 'delete', id: that.getAttribute('data-news-del')}, 'post', function (ret) {
if (ret.code) {
$(that).parents('.news-item').remove();
return $.msg.success(ret.msg), msnry.layout(), false;
}
return $.msg.error(ret.msg), false;
});
$.msg.close(dialogIndex);
});
});
});
</script>
{/block}
{block name="style"}
<style>
#news-box {
position: relative
}
#news-box .news-item {
top: 0;
left: 0;
padding: 5px;
margin: 10px;
width: 300px;
position: relative;
border: 1px solid #ccc;
box-sizing: content-box
}
#news-box .news-item .news-articel-item {
width: 100%;
height: 150px;
overflow: hidden;
position: relative;
background-size: 100%;
background-position: center center
}
#news-box .news-item .news-articel-item p {
bottom: 0;
width: 100%;
color: #fff;
padding: 5px;
max-height: 5em;
font-size: 12px;
overflow: hidden;
position: absolute;
text-overflow: ellipsis;
background: rgba(0, 0, 0, .7)
}
#news-box .news-item .news-articel-item.other {
height: 50px;
padding: 5px 0
}
#news-box .news-item .news-articel-item span {
width: 245px;
overflow: hidden;
line-height: 50px;
white-space: nowrap;
display: inline-block;
text-overflow: ellipsis
}
#news-box .news-item .news-articel-item div {
width: 50px;
height: 50px;
float: right;
overflow: hidden;
position: relative;
background-position: center center;
background-size: cover;
}
#news-box .hr-line-dashed {
margin: 6px 0 1px 0
}
#news-box .news-item .hr-line-dashed:last-child {
display: none
}
#news-box .news-tools {
top: 0;
z-index: 80;
color: #fff;
width: 302px;
padding: 0 5px;
margin-left: -6px;
line-height: 38px;
text-align: right;
position: absolute;
background: rgba(0, 0, 0, .7)
}
#news-box .news-tools a {
color: #fff;
margin-left: 10px
}
</style>
{/block}

View File

@ -0,0 +1,112 @@
<form data-auto='true' action='{:request()->url()}' style="position:relative">
<div class="col-xs-2 news-container">
<h5 class="text-center" style="margin:10px 0">微信图文</h5>
<div class="news-box">
{foreach $vo.articles as $key=>$value}
<div class="news-item transition" data-id="{$value.id}">
<div class="news-image" style='background-image:url({$value.local_url})'></div>
<span class="news-title">{$value.title}</span>
</div>
<hr/>
{/foreach}
</div>
</div>
<div class="col-xs-2 list-container">
<h5 class="text-center" style="margin:10px 0">指定粉丝标签推送 <a data-check-all> 全选 </a></h5>
<div class="list-item">
{foreach $fans_tags as $tag}
<label class='control-label layui-form-label' style='text-align:left!important'>
<input name="fans_tags[]" value='{$tag.id}' type="checkbox"/> {$tag.name} ({$tag.count})
</label>
{/foreach}
{literal}
<script id="push" type="text/template">
{{if data}}
{{each data as value key}}
{{if key <= 200}}
<label>{{value}}</label>
{{/if}}
{{/each}}
{{if (data.length > 200)}}
<label>...</label>
{{/if}}
{{else}}
<h5></h5>
{{/if}}
</script>
{/literal}
</div>
<div id='push-tags' class="list-item"></div>
</div>
<div style="clear:both;max-height:60px"></div>
<div class="bottom-btn text-center">
<button class="layui-btn">立即推送图文</button>
<button type='button' data-close='' data-confirm='确定要取消推送图文吗?' class="layui-btn layui-btn-danger">取消推送图文
</button>
</div>
</form>
<script>
require(['jquery', 'template'], function () {
var $allbtn = $('[data-check-all]').on('click', function () {
var check_status = check_checked();
$('input[name*=fans_tags]').map(function () {
this.checked = !check_status;
});
check_checked(), postpush();
});
/*重置和全选的效果处理*/
function check_checked() {
var allcheck = true;
$('input[name*=fans_tags]').map(function () {
(!this.checked) && (allcheck = false);
});
return ((allcheck) ? $allbtn.html('重置') : $allbtn.html('全选')), allcheck;
}
/*点击每一个选项都触发事件*/
$('input[name*=fans_tags]').on('click', function () {
check_checked();
postpush();
});
/*数据异步获取并显示出来*/
function postpush() {
var inp = $('input[name*=fans_tags]');
var group = [];
for (var i = 0; i < inp.length; i++) {
if (inp[i].checked === true) {
group.push(inp[i].value);
}
}
$.post("{:url('wechat/news/push')}?action=getuser", {group: group.join(',')}, function (ret) {
var html = template('push', ret);
document.getElementById('push-tags').innerHTML = html;
});
}
});
</script>
<style>
body{min-width:500px}
#push-tags{max-height:300px;overflow:auto}
.bottom-btn{bottom:0;width:100%;padding:10px;display:block;background:#F7F7F7;position:absolute}
.news-container{width:200px;padding-right:8px}
.list-container{width:578px;padding-left:0;padding-right:8px}
.list-container h5 a{float:right;font-size:12px}
.list-container .list-item{padding:8px;overflow:auto;max-height:300px;border:1px solid #eee}
.list-container .list-item:after{content:'';clear:both;width:100%;display:block}
.list-container .list-item label{width:25%;float:left;display:block;overflow:hidden;white-space:nowrap;text-overflow:ellipsis}
.news-container .news-box{padding:8px;border:1px solid #eee}
.news-container .news-box hr{margin:4px}
.news-container .news-box .container{cursor:pointer;overflow:hidden;position:relative;border-radius:2px;border:1px solid #cecece}
.news-container .news-box .news-image{height:90px;background-size:100%;background-position:center center}
.news-container .news-box .news-title{position:absolute;background:rgba(0,0,0,.5);color:#fff;padding:2px;margin:0;bottom:0;left:0;right:0;text-align:right;white-space:nowrap;text-overflow:ellipsis;overflow:hidden}
</style>

View File

@ -0,0 +1,77 @@
{extend name="../../admin/view/index/index"}
{block name='style'}
<style>
body{min-width:500px}
.news-container{position:relative}
.news-container .news-item{top:0;left:0;width:232px;margin:10px;padding:5px;cursor:pointer;position:relative;border:1px solid #ccc;box-sizing:content-box}
.news-container .news-item.active,.news-container .news-item:hover{border-color:#09c;box-shadow:1px 0 10px #09c}
.news-container .news-item .news-articel-item{width:100%;height:150px;overflow:hidden;position:relative;background-size:100%;background-position:center center}
.news-container .news-item .news-articel-item p{bottom:0;width:100%;color:#fff;padding:5px;max-height:5em;font-size:12px;overflow:hidden;position:absolute;text-overflow:ellipsis;background:rgba(0,0,0,.7)}
.news-container .news-item .news-articel-item.other{height:50px;padding:5px 0}
.news-container .news-item .news-articel-item .left-image{width:50px;height:50px;float:left;overflow:hidden;position:relative;background-size:100%;background-position:center center}
.news-container .news-item .news-articel-item .right-text{float:left;width:172px;overflow:hidden;padding-right:10px;text-overflow:ellipsis}
.news-container .news-item .hr-line-dashed:last-child{display:none}
.news-container .hr-line-dashed{margin:6px 0 1px 0}
.page-style{bottom:0;width:100%;height:50px;padding:0 10px;position:fixed;background:#fff}
.page-style .pagination,.page-style .pagination-trigger{margin:10px 0!important;padding:0 10px!important}
</style>
{/block}
{block name="body"}
<div class="layui-clear news-container layui-hide">
{foreach $list as $vo}
<div class="news-item" data-news-id="{$vo.id}">
{foreach $vo.articles as $k => $v}
{if $k < 1}
<div class='news-articel-item' style='background-image:url("{$v.local_url}")'>
{if $v.title}<p>{$v.title}</p>{/if}
</div>
<div class="hr-line-dashed"></div>
{else}
<div class='news-articel-item other'>
<div class='right-text'>{$v.title}</div>
<div class='left-image' style='background-image:url("{$v.local_url}");'></div>
</div>
<div class="hr-line-dashed"></div>
{/if}
{/foreach}
</div>
{/foreach}
{if empty($list)}<p class="notdata">没有记录哦!</p>{/if}
</div>
<div class="margin-bottom-20 margin-top-20"></div>
<div class="page-style">{if isset($pagehtml)}{$pagehtml|raw}{/if}</div>
{/block}
{block name="script"}
<script>
require(['jquery.masonry'], function (Masonry) {
$('.news-container.layui-hide').removeClass('layui-hide');
var msnry = new Masonry($('.news-container').get(0), {itemSelector: '.news-item', columnWidth: 0});
msnry.layout();
// 事件处理
$('.news-item').on('mouseenter', '.news-container', function () {
$(this).addClass('active');
}).on('mouseleave', '.news-container', function () {
$(this).removeClass('active');
});
// 外部选择器
var seletor = '[name="{$Think.get.field|decode|default=0}"]';
if (seletor) {
$('[data-news-id]').on('click', function () {
window.top.$(seletor).val($(this).attr('data-news-id')).trigger('change');
parent.layer.close(parent.layer.getFrameIndex(window.name))
});
}
// 分页事件处理
$('body').off('change', '.pagination-trigger select').on('change', '.pagination-trigger select', function () {
var urls = this.options[this.selectedIndex].getAttribute('data-url').split('#');
urls.shift();
window.location.href = urls.join('#');
}).off('click', '[data-open]').on('click', '[data-open]', function () {
window.location.href = this.getAttribute('data-open');
});
});
</script>
{/block}

View File

@ -23,12 +23,14 @@
"ext-json": "*",
"ext-curl": "*",
"ext-iconv": "*",
"ext-openssl": "*",
"ext-mbstring": "*",
"topthink/framework": "^6.0",
"topthink/think-view": "^1.0",
"topthink/think-multi-app": "^1.0",
"zoujingli/ip2region": "^1.0",
"zoujingli/think-library": "6.0.*-dev",
"zoujingli/weopen-developer": "dev-master"
"zoujingli/weopen-developer": "dev-master",
"zoujingli/think-library": "6.0.*-dev"
},
"autoload": {
"psr-4": {