'; + echo "当前用户OPENID: {$openid}"; + echo "\n--- 创建预支付码 ---\n"; + var_export($result); + echo ''; + + echo '
'; + echo "\n\n--- JSAPI 及 H5 参数 ---\n"; + var_export($options); + echo ''; + echo ""; + echo " + + "; + } + + /** + * 支付通知接收处理 + * @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(); + } + +} diff --git a/app/wechat/service/FansService.php b/app/wechat/service/FansService.php new file mode 100644 index 000000000..ee8915dcd --- /dev/null +++ b/app/wechat/service/FansService.php @@ -0,0 +1,63 @@ +app->db->name('WechatFans')->where(['openid' => $openid])->find(); + } + +} diff --git a/app/wechat/service/MediaService.php b/app/wechat/service/MediaService.php new file mode 100644 index 000000000..ffab5a176 --- /dev/null +++ b/app/wechat/service/MediaService.php @@ -0,0 +1,89 @@ +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']); + } + } +} diff --git a/app/wechat/service/WechatService.php b/app/wechat/service/WechatService.php index 299901df8..b941aa026 100644 --- a/app/wechat/service/WechatService.php +++ b/app/wechat/service/WechatService.php @@ -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); + } + } } \ No newline at end of file diff --git a/app/wechat/view/config/options.html b/app/wechat/view/config/options.html new file mode 100644 index 000000000..c1b763245 --- /dev/null +++ b/app/wechat/view/config/options.html @@ -0,0 +1,43 @@ +{extend name="../../admin/view/main"} + +{block name="content"} + +
请选择微信对接方式,其中第三方平台授权需要微信开放平台支持,同时需要搭建 SERVICE 服务!
++ | 微信昵称 | +粉丝标签 | +性别语言 | +关注时间 | ++ | + |
---|---|---|---|---|---|---|
+ + | +
+
+ 昵称:{$vo.nickname|default='--'}
+
+ + 区域:{$vo.country|default='--'} {$vo.province} {$vo.city} + |
+
+ {foreach $vo.tags as $t}
+ {$t|default='--'} {/foreach} |
+
+ 性别:{switch name='vo.sex'}{case value='1'}男{/case}{case value='2'}女{/case}{default}未知{/switch}
+ + 语言:{$vo.language|raw} + |
+
+ 日期:{$vo.subscribe_at|format_datetime|str_replace=' ',' 时间:',###|raw} + |
+
+ {eq name='vo.subscribe' value='0'}
+ 未关注
+ {else}
+ 已关注
+ {/eq}
+ + {eq name='vo.is_black' value='0'} + 未拉黑 + {else} + 已拉黑 + {/eq} + |
+ + + {eq name='vo.is_black' value='0'} + + 拉 黑 + + {else} + + 拉 白 + + {/eq} + + {if auth("remove")} + 删 除 + {/if} + | +
+ + | ++ + | +关键字 | +类型 | +预览 | +添加时间 | +状态 | ++ |
---|---|---|---|---|---|---|---|
+ + | ++ + | ++ {notempty name='vo.qrc'}{/notempty} + {$vo.keys} + | +{$vo.type} | ++ {if $vo.type eq '音乐'} + 预览 + {elseif in_array($vo.type,['文字','转客服'])} + 预览 + {elseif $vo.type eq '图片'} + 预览 + {elseif $vo.type eq '图文'} + 预览 + {elseif $vo.type eq '视频'} + 预览 + {elseif $vo.type eq '语音'} + 预览 + {else} + {$vo.content} + {/if} + | +{$vo.create_at|format_datetime} | +{if $vo.status eq 0}已禁用{elseif $vo.status eq 1}使用中{/if} | ++ + {if auth("edit")} + 编 辑 + {/if} + + {if $vo.status eq 1 and auth("forbid")} + 禁 用 + {elseif auth("resume")} + 启 用 + {/if} + + {if auth("remove")} + 删 除 + {/if} + + | +
请在左侧创建菜单...+
{$v.title}
{/if} +没有记录哦!
{/if} +