xxxrrrdddd hace 3 años
padre
commit
bd8a2bd181
Se han modificado 46 ficheros con 3180 adiciones y 817 borrados
  1. 0 1
      .gitignore
  2. 1 0
      addons/.htaccess
  3. 1 0
      addons/command/.addonrc
  4. 69 0
      addons/command/Command.php
  5. 4 0
      addons/command/config.php
  6. 15 0
      addons/command/controller/Index.php
  7. 10 0
      addons/command/info.ini
  8. 12 0
      addons/command/install.sql
  9. 28 0
      addons/command/library/Output.php
  10. 1 0
      addons/third/.addonrc
  11. 79 0
      addons/third/Third.php
  12. 13 0
      addons/third/bootstrap.js
  13. 130 0
      addons/third/config.php
  14. 141 0
      addons/third/controller/Api.php
  15. 137 0
      addons/third/controller/Index.php
  16. 10 0
      addons/third/info.ini
  17. 28 0
      addons/third/install.sql
  18. 68 0
      addons/third/library/Application.php
  19. 141 0
      addons/third/library/Qq.php
  20. 155 0
      addons/third/library/Service.php
  21. 127 0
      addons/third/library/Wechat.php
  22. 121 0
      addons/third/library/Weibo.php
  23. 26 0
      addons/third/model/Third.php
  24. 139 0
      addons/third/view/index/index.html
  25. 236 0
      application/admin/controller/Command.php
  26. 66 0
      application/admin/controller/Third.php
  27. 16 0
      application/admin/lang/zh-cn/command.php
  28. 16 0
      application/admin/lang/zh-cn/third.php
  29. 59 0
      application/admin/model/Command.php
  30. 56 0
      application/admin/model/Third.php
  31. 27 0
      application/admin/validate/Command.php
  32. 26 0
      application/admin/validate/Third.php
  33. 400 0
      application/admin/view/command/add.html
  34. 42 0
      application/admin/view/command/detail.html
  35. 25 0
      application/admin/view/command/index.html
  36. 23 0
      application/admin/view/third/index.html
  37. 0 73
      application/api/controller/Demo.php
  38. 3 3
      application/common.php
  39. 1 1
      application/config.php
  40. 12 2
      application/extra/addons.php
  41. 112 0
      application/index/controller/Third.php
  42. 128 0
      application/index/view/third/prepare.html
  43. 170 736
      public/api.html
  44. 14 1
      public/assets/js/addons.js
  45. 234 0
      public/assets/js/backend/command.js
  46. 58 0
      public/assets/js/backend/third.js

+ 0 - 1
.gitignore

@@ -2,7 +2,6 @@
 /thinkphp/
 /vendor/
 /runtime/*
-/addons/*
 /application/admin/command/Install/*.lock
 /public/assets/libs/
 /public/assets/addons/*

+ 1 - 0
addons/.htaccess

@@ -0,0 +1 @@
+deny from all

+ 1 - 0
addons/command/.addonrc

@@ -0,0 +1 @@
+{"license":"regular","licenseto":"39487","licensekey":"YNMEsQF8LKjIvDAt sEV\/cCS3B6Y8jnDDnvhNww==","menus":["command","command\/index","command\/add","command\/detail","command\/execute","command\/del","command\/multi"],"files":["application\\admin\\controller\\Command.php","application\\admin\\lang\\zh-cn\\command.php","application\\admin\\model\\Command.php","application\\admin\\validate\\Command.php","application\\admin\\view\\command\\add.html","application\\admin\\view\\command\\detail.html","application\\admin\\view\\command\\index.html","public\\assets\\js\\backend\\command.js"]}

+ 69 - 0
addons/command/Command.php

@@ -0,0 +1,69 @@
+<?php
+
+namespace addons\command;
+
+use app\common\library\Menu;
+use think\Addons;
+
+/**
+ * 在线命令插件
+ */
+class Command extends Addons
+{
+
+    /**
+     * 插件安装方法
+     * @return bool
+     */
+    public function install()
+    {
+        $menu = [
+            [
+                'name'    => 'command',
+                'title'   => '在线命令管理',
+                'icon'    => 'fa fa-terminal',
+                'sublist' => [
+                    ['name' => 'command/index', 'title' => '查看'],
+                    ['name' => 'command/add', 'title' => '添加'],
+                    ['name' => 'command/detail', 'title' => '详情'],
+                    ['name' => 'command/execute', 'title' => '运行'],
+                    ['name' => 'command/del', 'title' => '删除'],
+                    ['name' => 'command/multi', 'title' => '批量更新'],
+                ]
+            ]
+        ];
+        Menu::create($menu);
+        return true;
+    }
+
+    /**
+     * 插件卸载方法
+     * @return bool
+     */
+    public function uninstall()
+    {
+        Menu::delete('command');
+        return true;
+    }
+
+    /**
+     * 插件启用方法
+     * @return bool
+     */
+    public function enable()
+    {
+        Menu::enable('command');
+        return true;
+    }
+
+    /**
+     * 插件禁用方法
+     * @return bool
+     */
+    public function disable()
+    {
+        Menu::disable('command');
+        return true;
+    }
+
+}

+ 4 - 0
addons/command/config.php

@@ -0,0 +1,4 @@
+<?php
+
+return [
+];

+ 15 - 0
addons/command/controller/Index.php

@@ -0,0 +1,15 @@
+<?php
+
+namespace addons\command\controller;
+
+use think\addons\Controller;
+
+class Index extends Controller
+{
+
+    public function index()
+    {
+        $this->error("当前插件暂无前台页面");
+    }
+
+}

+ 10 - 0
addons/command/info.ini

@@ -0,0 +1,10 @@
+name = command
+title = 在线命令
+intro = 可在线执行FastAdmin的命令行相关命令
+author = Karson
+website = https://www.fastadmin.net
+version = 1.0.6
+state = 1
+url = /addons/command
+license = regular
+licenseto = 39487

+ 12 - 0
addons/command/install.sql

@@ -0,0 +1,12 @@
+CREATE TABLE IF NOT EXISTS `__PREFIX__command`  (
+  `id` int(10) UNSIGNED NOT NULL AUTO_INCREMENT COMMENT 'ID',
+  `type` varchar(30) CHARACTER SET utf8 COLLATE utf8_general_ci NOT NULL DEFAULT '' COMMENT '类型',
+  `params` varchar(1500) CHARACTER SET utf8 COLLATE utf8_general_ci NOT NULL DEFAULT '' COMMENT '参数',
+  `command` varchar(1500) CHARACTER SET utf8 COLLATE utf8_general_ci NOT NULL DEFAULT '' COMMENT '命令',
+  `content` text COMMENT '返回结果',
+  `executetime` int(10) UNSIGNED DEFAULT NULL COMMENT '执行时间',
+  `createtime` int(10) UNSIGNED DEFAULT NULL COMMENT '创建时间',
+  `updatetime` int(10) UNSIGNED DEFAULT NULL COMMENT '更新时间',
+  `status` enum('successed','failured') CHARACTER SET utf8 COLLATE utf8_general_ci NOT NULL DEFAULT 'failured' COMMENT '状态',
+  PRIMARY KEY (`id`) USING BTREE
+) ENGINE = InnoDB AUTO_INCREMENT = 1 CHARACTER SET = utf8 COLLATE = utf8_general_ci COMMENT = '在线命令表';

+ 28 - 0
addons/command/library/Output.php

@@ -0,0 +1,28 @@
+<?php
+
+namespace addons\command\library;
+
+/**
+ * Class Output
+ */
+class Output extends \think\console\Output
+{
+
+    protected $message = [];
+
+    public function __construct($driver = 'console')
+    {
+        parent::__construct($driver);
+    }
+
+    protected function block($style, $message)
+    {
+        $this->message[] = $message;
+    }
+
+    public function getMessage()
+    {
+        return $this->message;
+    }
+
+}

+ 1 - 0
addons/third/.addonrc

@@ -0,0 +1 @@
+{"license":"regular","licenseto":"39487","licensekey":"LXlUgRKiuBWfSo7V awr0LgqqG7sT5Fl3IHaG+w==","menus":["third","third\/index","third\/del"],"files":["application\\admin\\controller\\Third.php","application\\admin\\lang\\zh-cn\\third.php","application\\admin\\model\\Third.php","application\\admin\\validate\\Third.php","application\\admin\\view\\third\\index.html","application\\index\\controller\\Third.php","application\\index\\view\\third\\prepare.html","public\\assets\\js\\backend\\third.js"]}

+ 79 - 0
addons/third/Third.php

@@ -0,0 +1,79 @@
+<?php
+
+namespace addons\third;
+
+use app\common\library\Menu;
+use think\Addons;
+
+/**
+ * 第三方登录
+ */
+class Third extends Addons
+{
+
+    /**
+     * 插件安装方法
+     * @return bool
+     */
+    public function install()
+    {
+        $menu = [
+            [
+                'name'    => 'third',
+                'title'   => '第三方登录管理',
+                'icon'    => 'fa fa-users',
+                'sublist' => [
+                    [
+                        "name"  => "third/index",
+                        "title" => "查看"
+                    ],
+                    [
+                        "name"  => "third/del",
+                        "title" => "删除"
+                    ]
+                ]
+            ]
+        ];
+        Menu::create($menu);
+        return true;
+    }
+
+    /**
+     * 插件卸载方法
+     * @return bool
+     */
+    public function uninstall()
+    {
+        Menu::delete("third");
+        return true;
+    }
+
+    /**
+     * 插件启用方法
+     * @return bool
+     */
+    public function enable()
+    {
+        Menu::enable("third");
+        return true;
+    }
+
+    /**
+     * 插件禁用方法
+     * @return bool
+     */
+    public function disable()
+    {
+        Menu::disable("third");
+        return true;
+    }
+
+    /**
+     * @param $params
+     */
+    public function configInit(&$params)
+    {
+        $config = $this->getConfig();
+        $params['third'] = ['status' => explode(',', $config['status'])];
+    }
+}

+ 13 - 0
addons/third/bootstrap.js

@@ -0,0 +1,13 @@
+if (Config.modulename === 'index' && Config.controllername === 'user' && ['login', 'register'].indexOf(Config.actionname) > -1 && $("#register-form,#login-form").size() > 0) {
+    $('<style>.social-login{display:flex}.social-login a{flex:1;margin:0 2px;}.social-login a:first-child{margin-left:0;}.social-login a:last-child{margin-right:0;}</style>').appendTo("head");
+    $("#register-form,#login-form").append('<div class="form-group social-login"></div>');
+    if (Config.third.status.indexOf("wechat") > -1) {
+        $('<a class="btn btn-success" href="' + Fast.api.fixurl('/third/connect/wechat') + '"><i class="fa fa-wechat"></i> 微信登录</a>').appendTo(".social-login");
+    }
+    if (Config.third.status.indexOf("qq") > -1) {
+        $('<a class="btn btn-info" href="' + Fast.api.fixurl('/third/connect/qq') + '"><i class="fa fa-qq"></i> QQ登录</a>').appendTo(".social-login");
+    }
+    if (Config.third.status.indexOf("weibo") > -1) {
+        $('<a class="btn btn-danger" href="' + Fast.api.fixurl('/third/connect/weibo') + '"><i class="fa fa-weibo"></i> 微博登录</a>').appendTo(".social-login");
+    }
+}

+ 130 - 0
addons/third/config.php

@@ -0,0 +1,130 @@
+<?php
+
+return array(
+    0 =>
+        array(
+            'name'    => 'qq',
+            'title'   => 'QQ',
+            'type'    => 'array',
+            'content' =>
+                array(
+                    'app_id'     => '',
+                    'app_secret' => '',
+                    'scope'      => 'get_user_info',
+                ),
+            'value'   =>
+                array(
+                    'app_id'     => '100000000',
+                    'app_secret' => '123456',
+                    'scope'      => 'get_user_info',
+                ),
+            'rule'    => 'required',
+            'msg'     => '',
+            'tip'     => '',
+            'ok'      => '',
+            'extend'  => '',
+        ),
+    1 =>
+        array(
+            'name'    => 'wechat',
+            'title'   => '微信',
+            'type'    => 'array',
+            'content' =>
+                array(
+                    'app_id'     => '',
+                    'app_secret' => '',
+                    'callback'   => '',
+                    'scope'      => 'snsapi_base',
+                ),
+            'value'   =>
+                array(
+                    'app_id'     => '100000000',
+                    'app_secret' => '123456',
+                    'scope'      => 'snsapi_userinfo',
+                ),
+            'rule'    => 'required',
+            'msg'     => '',
+            'tip'     => '',
+            'ok'      => '',
+            'extend'  => '',
+        ),
+    2 =>
+        array(
+            'name'    => 'weibo',
+            'title'   => '微博',
+            'type'    => 'array',
+            'content' =>
+                array(
+                    'app_id'     => '',
+                    'app_secret' => '',
+                    'scope'      => 'get_user_info',
+                ),
+            'value'   =>
+                array(
+                    'app_id'     => '100000000',
+                    'app_secret' => '123456',
+                    'scope'      => 'get_user_info',
+                ),
+            'rule'    => 'required',
+            'msg'     => '',
+            'tip'     => '',
+            'ok'      => '',
+            'extend'  => '',
+        ),
+    3 =>
+        array(
+            'name'    => 'bindaccount',
+            'title'   => '账号绑定',
+            'type'    => 'radio',
+            'content' =>
+                array(
+                    1 => '开启',
+                    0 => '关闭',
+                ),
+            'value'   => '1',
+            'rule'    => 'required',
+            'msg'     => '',
+            'tip'     => '',
+            'ok'      => '是否开启账号绑定',
+            'extend'  => '',
+        ),
+    4 =>
+        array(
+            'name'    => 'status',
+            'title'   => '前台第三方登录开关',
+            'type'    => 'checkbox',
+            'content' =>
+                array(
+                    'qq'     => 'QQ',
+                    'wechat' => '微信',
+                    'weibo'  => '微博',
+                ),
+            'value'   => 'qq,wechat,weibo',
+            'rule'    => '',
+            'msg'     => '',
+            'tip'     => '',
+            'ok'      => '前台第三方登录的开关',
+            'extend'  => '',
+        ),
+    5 =>
+        array(
+            'name'    => 'rewrite',
+            'title'   => '伪静态',
+            'type'    => 'array',
+            'content' =>
+                array(),
+            'value'   =>
+                array(
+                    'index/index'    => '/third$',
+                    'index/connect'  => '/third/connect/[:platform]',
+                    'index/callback' => '/third/callback/[:platform]',
+                    'index/bind'     => '/third/bind/[:platform]',
+                    'index/unbind'   => '/third/unbind/[:platform]',
+                ),
+            'rule'    => 'required',
+            'msg'     => '',
+            'tip'     => '',
+            'ok'      => '',
+            'extend'  => '',
+        ),
+);

+ 141 - 0
addons/third/controller/Api.php

@@ -0,0 +1,141 @@
+<?php
+
+namespace addons\third\controller;
+
+use addons\third\library\Application;
+use app\common\controller\Api as commonApi;
+use addons\third\library\Service;
+use addons\third\model\Third;
+use app\common\library\Sms;
+use fast\Random;
+use think\Lang;
+use think\Config;
+use think\Session;
+use think\Validate;
+
+/**
+ * 第三方登录插件
+ */
+class Api extends commonApi
+{
+    protected $noNeedLogin = ['getAuthUrl', 'callback', 'account']; // 无需登录即可访问的方法,同时也无需鉴权了
+    protected $noNeedRight = ['*']; // 无需鉴权即可访问的方法
+
+    protected $app = null;
+    protected $options = [];
+    protected $config = null;
+
+    public function _initialize()
+    {
+        //跨域检测
+        check_cors_request();
+        //设置session_id
+        Config::set('session.id', $this->request->server("HTTP_SID"));
+
+        parent::_initialize();
+        $this->config = get_addon_config('third');
+        $this->app = new Application($this->config);
+    }
+
+    /**
+     * H5获取授权链接
+     * @return void
+     */
+    public function getAuthUrl()
+    {
+        $url = $this->request->param('url');
+        $platform = $this->request->param('platform');
+        if (!$url || !$platform || !isset($this->config[$platform])) {
+            $this->error('参数错误');
+        }
+        $this->config[$platform]['callback'] = $url;
+        $this->app = new Application($this->config); //
+        if (!$this->app->{$platform}) {
+            $this->error(__('Invalid parameters'));
+        }
+        $this->success('', $this->app->{$platform}->getAuthorizeUrl());
+    }
+
+    /**
+     * 公众号:wechat 授权回调的请求【非第三方,自己的前端请求】
+     * @return void
+     */
+    public function callback()
+    {
+
+        $platform = $this->request->param('platform');
+        if (!$this->app->{$platform}) {
+            $this->error(__('Invalid parameters'));
+        }
+        $userinfo = $this->app->{$platform}->getUserInfo($this->request->param());
+        if (!$userinfo) {
+            $this->error(__('操作失败'));
+        }
+        $userinfo['apptype'] = 'mp';
+        $userinfo['platform'] = $platform;
+
+        $third = [
+            'avatar' => $userinfo['userinfo']['avatar'],
+            'nickname' => $userinfo['userinfo']['nickname']
+        ];
+
+        $user = null;
+        if ($this->auth->isLogin() || Service::isBindThird($userinfo['platform'], $userinfo['openid'], $userinfo['apptype'], $userinfo['unionid'])) {
+            Service::connect($userinfo['platform'], $userinfo);
+            $user = $this->auth->getUserinfo();
+        } else {
+            Session::set('third-userinfo', $userinfo);
+        }
+        $this->success("授权成功!", ['user' => $user, 'third' => $third]);
+    }
+
+    /**
+     * 登录或创建账号
+     */
+    public function account()
+    {
+
+        if ($this->request->isPost()) {
+            $params = Session::get('third-userinfo');
+            $mobile = $this->request->post('mobile', '');
+            $code = $this->request->post('code');
+            $token = $this->request->post('__token__');
+            $rule = [
+                'mobile'    => 'require|regex:/^1\d{10}$/',
+                '__token__' => 'require|token',
+            ];
+            $msg = [
+                'mobile'           => 'Mobile is incorrect',
+            ];
+            $data = [
+                'mobile'    => $mobile,
+                '__token__' => $token,
+            ];
+            $ret = Sms::check($mobile, $code, 'bind');
+            if (!$ret) {
+                $this->error(__('验证码错误'));
+            }
+            $validate = new Validate($rule, $msg);
+            $result = $validate->check($data);
+            if (!$result) {
+                $this->error(__($validate->getError()), ['__token__' => $this->request->token()]);
+            }
+
+            $userinfo = \app\common\model\User::where('mobile', $mobile)->find();
+            if ($userinfo) {
+                $result = $this->auth->direct($userinfo->id);
+            } else {
+                $result = $this->auth->register($mobile, Random::alnum(), '', $mobile);
+            }        
+            
+            if ($result) {
+                Service::connect($params['platform'], $params);
+                $this->success(__('绑定账号成功'), ['userinfo' => $this->auth->getUserinfo()]);
+            } else {
+                $this->error($this->auth->getError(), ['__token__' => $this->request->token()]);
+            }
+        }
+    }
+
+
+}

+ 137 - 0
addons/third/controller/Index.php

@@ -0,0 +1,137 @@
+<?php
+
+namespace addons\third\controller;
+
+use addons\third\library\Application;
+use addons\third\library\Service;
+use addons\third\model\Third;
+use think\addons\Controller;
+use think\Config;
+use think\Cookie;
+use think\Hook;
+use think\Lang;
+use think\Session;
+
+/**
+ * 第三方登录插件
+ */
+class Index extends Controller
+{
+    protected $app = null;
+    protected $options = [];
+
+    public function _initialize()
+    {
+        parent::_initialize();
+        $config = get_addon_config('third');
+        $this->app = new Application($config);
+    }
+
+    /**
+     * 插件首页
+     */
+    public function index()
+    {
+        if (!\app\admin\library\Auth::instance()->id) {
+            $this->error('当前插件暂无前台页面');
+        }
+        $platformList = [];
+        if ($this->auth->id) {
+            $platformList = Third::where('user_id', $this->auth->id)->column('platform');
+        }
+        $this->view->assign('platformList', $platformList);
+        return $this->view->fetch();
+    }
+
+    /**
+     * 发起授权
+     */
+    public function connect()
+    {
+        $platform = $this->request->param('platform');
+        $url = $this->request->request('url', $this->request->server('HTTP_REFERER', '/'), 'trim');        
+        if (!$this->app->{$platform}) {
+            $this->error(__('Invalid parameters'));
+        }
+        if ($url) {
+            Session::set("redirecturl", $url);
+        }
+        // 跳转到登录授权页面
+        $this->redirect($this->app->{$platform}->getAuthorizeUrl());
+        return;
+    }
+
+    /**
+     * 通知回调
+     */
+    public function callback()
+    {
+        $auth = $this->auth;
+
+        //监听注册登录注销的事件
+        Hook::add('user_login_successed', function ($user) use ($auth) {
+            $expire = input('post.keeplogin') ? 30 * 86400 : 0;
+            Cookie::set('uid', $user->id, $expire);
+            Cookie::set('token', $auth->getToken(), $expire);
+        });
+        Hook::add('user_register_successed', function ($user) use ($auth) {
+            Cookie::set('uid', $user->id);
+            Cookie::set('token', $auth->getToken());
+        });
+        Hook::add('user_logout_successed', function ($user) use ($auth) {
+            Cookie::delete('uid');
+            Cookie::delete('token');
+        });
+        $platform = $this->request->param('platform');
+
+        // 成功后返回之前页面
+        $url = Session::has("redirecturl") ? Session::pull("redirecturl") : url('index/user/index');
+
+        // 授权成功后的回调
+        $userinfo = $this->app->{$platform}->getUserInfo();
+        if (!$userinfo) {
+            $this->error(__('操作失败'), $url);
+        }
+
+        Session::set("{$platform}-userinfo", $userinfo);
+        //判断是否启用账号绑定
+        $third = Third::get(['platform' => $platform, 'openid' => $userinfo['openid']]);
+        if (!$third) {
+            $config = get_addon_config('third');
+            //要求绑定账号或会员当前是登录状态
+            if ($config['bindaccount'] || $this->auth->id) {
+                $this->redirect(url('index/third/prepare') . "?" . http_build_query(['platform' => $platform, 'url' => $url]));
+            }
+        }
+
+        $loginret = Service::connect($platform, $userinfo);
+        if ($loginret) {
+            $this->redirect($url);
+        }
+    }
+
+    /**
+     * 绑定账号
+     */
+    public function bind()
+    {
+        $platform = $this->request->request('platform', $this->request->param('platform', ''));
+        $url = $this->request->get('url', $this->request->server('HTTP_REFERER'));
+        $redirecturl = url("index/third/bind") . "?" . http_build_query(['platform' => $platform, 'url' => $url]);
+        $this->redirect($redirecturl);
+        return;
+    }
+
+    /**
+     * 解绑账号
+     */
+    public function unbind()
+    {
+        $platform = $this->request->request('platform', $this->request->param('platform', ''));
+        $url = $this->request->get('url', $this->request->server('HTTP_REFERER'));
+        $redirecturl = url("index/third/unbind") . "?" . http_build_query(['platform' => $platform, 'url' => $url]);
+        $this->redirect($redirecturl);
+        return;
+    }
+
+}

+ 10 - 0
addons/third/info.ini

@@ -0,0 +1,10 @@
+name = third
+title = 第三方登录
+intro = 使用微信、QQ、微博登录插件
+author = FastAdmin
+website = https://www.fastadmin.net
+version = 1.2.2
+state = 1
+url = /addons/third
+license = regular
+licenseto = 39487

+ 28 - 0
addons/third/install.sql

@@ -0,0 +1,28 @@
+
+CREATE TABLE IF NOT EXISTS `__PREFIX__third` (
+  `id` int(10) unsigned NOT NULL AUTO_INCREMENT COMMENT 'ID',
+  `user_id` int(10) unsigned DEFAULT '0' COMMENT '会员ID',
+  `platform` varchar(30) DEFAULT '' COMMENT '第三方应用',
+  `apptype` varchar(50) DEFAULT '' COMMENT '应用类型',
+  `unionid` varchar(100) DEFAULT '' COMMENT '第三方UNIONID',
+  `openid` varchar(100) DEFAULT '' COMMENT '第三方OPENID',
+  `openname` varchar(100) DEFAULT '' COMMENT '第三方会员昵称',
+  `access_token` varchar(255) NULL DEFAULT '' COMMENT 'AccessToken',
+  `refresh_token` varchar(255) DEFAULT 'RefreshToken',
+  `expires_in` int(10) unsigned DEFAULT '0' COMMENT '有效期',
+  `createtime` int(10) unsigned DEFAULT NULL COMMENT '创建时间',
+  `updatetime` int(10) unsigned DEFAULT NULL COMMENT '更新时间',
+  `logintime` int(10) unsigned DEFAULT NULL COMMENT '登录时间',
+  `expiretime` int(10) unsigned DEFAULT NULL COMMENT '过期时间',
+  PRIMARY KEY (`id`),
+  UNIQUE KEY `platform` (`platform`,`openid`),
+  KEY `user_id` (`user_id`,`platform`),
+  KEY `unionid` (`platform`,`unionid`)
+) ENGINE=InnoDB AUTO_INCREMENT=1 DEFAULT CHARSET=utf8 COMMENT='第三方登录表';
+
+ALTER TABLE `__PREFIX__third` ADD COLUMN `apptype` varchar(50) NULL DEFAULT '' COMMENT '应用类型' AFTER `platform`;
+
+ALTER TABLE `__PREFIX__third` ADD COLUMN `unionid` varchar(100) NULL DEFAULT '' COMMENT '第三方UnionID' AFTER `apptype`;
+ALTER TABLE `__PREFIX__third` ADD INDEX `unionid`(`platform`, `unionid`);
+
+

+ 68 - 0
addons/third/library/Application.php

@@ -0,0 +1,68 @@
+<?php
+
+namespace addons\third\library;
+
+class Application
+{
+
+    /**
+     * 配置信息
+     * @var array
+     */
+    private $config = [];
+
+    /**
+     * 服务提供者
+     * @var array
+     */
+    private $providers = [
+        'qq'      => 'Qq',
+        'weibo'   => 'Weibo',
+        'wechat'  => 'Wechat',
+    ];
+
+    /**
+     * 服务对象信息
+     * @var array
+     */
+    protected $services = [];
+
+    public function __construct($options = [])
+    {
+        $options = array_intersect_key($options, $this->providers);
+        $options = array_merge($this->config, is_array($options) ? $options : []);
+        foreach ($options as $key => &$option) {
+            $option['app_id'] = isset($option['app_id']) ? $option['app_id'] : '';
+            $option['app_secret'] = isset($option['app_secret']) ? $option['app_secret'] : '';
+            // 如果未定义回调地址则自动生成
+            $option['callback'] = isset($option['callback']) && $option['callback'] ? $option['callback'] : addon_url('third/index/callback', [':platform' => $key], false, true);
+        }
+        $this->config = $options;
+        //注册服务器提供者
+        $this->registerProviders();
+    }
+
+    /**
+     * 注册服务提供者
+     */
+    private function registerProviders()
+    {
+        foreach ($this->providers as $k => $v) {
+            $this->services[$k] = function () use ($k, $v) {
+                $options = $this->config[$k];
+                $objname = __NAMESPACE__ . "\\{$v}";
+                return new $objname($options);
+            };
+        }
+    }
+
+    public function __set($key, $value)
+    {
+        $this->services[$key] = $value;
+    }
+
+    public function __get($key)
+    {
+        return isset($this->services[$key]) ? $this->services[$key]($this) : null;
+    }
+}

+ 141 - 0
addons/third/library/Qq.php

@@ -0,0 +1,141 @@
+<?php
+
+namespace addons\third\library;
+
+use fast\Http;
+use think\Config;
+use think\Session;
+
+/**
+ * QQ
+ */
+class Qq
+{
+    const GET_AUTH_CODE_URL = "https://graph.qq.com/oauth2.0/authorize";
+    const GET_ACCESS_TOKEN_URL = "https://graph.qq.com/oauth2.0/token";
+    const GET_USERINFO_URL = "https://graph.qq.com/user/get_user_info";
+    const GET_OPENID_URL = "https://graph.qq.com/oauth2.0/me";
+
+    /**
+     * 配置信息
+     * @var array
+     */
+    private $config = [];
+
+    public function __construct($options = [])
+    {
+        if ($config = Config::get('third.qq')) {
+            $this->config = array_merge($this->config, $config);
+        }
+        $this->config = array_merge($this->config, is_array($options) ? $options : []);
+    }
+
+    /**
+     * 登陆
+     */
+    public function login()
+    {
+        header("Location:" . $this->getAuthorizeUrl());
+    }
+
+    /**
+     * 获取authorize_url
+     */
+    public function getAuthorizeUrl()
+    {
+        $state = md5(uniqid(rand(), true));
+        Session::set('state', $state);
+        $queryarr = array(
+            "response_type" => "code",
+            "client_id"     => $this->config['app_id'],
+            "redirect_uri"  => $this->config['callback'],
+            "scope"         => $this->config['scope'],
+            "state"         => $state,
+        );
+        request()->isMobile() && $queryarr['display'] = 'mobile';
+        $url = self::GET_AUTH_CODE_URL . '?' . http_build_query($queryarr);
+        return $url;
+    }
+
+    /**
+     * 获取用户信息
+     * @param array $params
+     * @return array
+     */
+    public function getUserInfo($params = [])
+    {
+        $params = $params ? $params : $_GET;
+        if (isset($params['access_token']) || (isset($params['state']) && $params['state'] == Session::get('state') && isset($params['code']))) {
+            //获取access_token
+            $data = isset($params['code']) ? $this->getAccessToken($params['code']) : $params;
+            $access_token = isset($data['access_token']) ? $data['access_token'] : '';
+            $refresh_token = isset($data['refresh_token']) ? $data['refresh_token'] : '';
+            $expires_in = isset($data['expires_in']) ? $data['expires_in'] : 0;
+            if ($access_token) {
+                $openid = $this->getOpenId($access_token);
+                //获取用户信息
+                $queryarr = [
+                    "access_token"       => $access_token,
+                    "oauth_consumer_key" => $this->config['app_id'],
+                    "openid"             => $openid,
+                ];
+                $ret = Http::get(self::GET_USERINFO_URL, $queryarr);
+                $userinfo = (array)json_decode($ret, true);
+                if (!$userinfo || !isset($userinfo['ret']) || $userinfo['ret'] !== 0) {
+                    return [];
+                }
+                $userinfo = $userinfo ? $userinfo : [];
+                $userinfo['avatar'] = isset($userinfo['figureurl_qq_2']) ? $userinfo['figureurl_qq_2'] : '';
+                $data = [
+                    'access_token'  => $access_token,
+                    'refresh_token' => $refresh_token,
+                    'expires_in'    => $expires_in,
+                    'openid'        => $openid,
+                    'userinfo'      => $userinfo
+                ];
+                return $data;
+            }
+        }
+        return [];
+    }
+
+    /**
+     * 获取access_token
+     * @param string $code
+     * @return array
+     */
+    public function getAccessToken($code = '')
+    {
+        if (!$code) {
+            return [];
+        }
+        $queryarr = array(
+            "grant_type"    => "authorization_code",
+            "client_id"     => $this->config['app_id'],
+            "client_secret" => $this->config['app_secret'],
+            "redirect_uri"  => $this->config['callback'],
+            "code"          => $code,
+        );
+        $ret = Http::get(self::GET_ACCESS_TOKEN_URL, $queryarr);
+        $params = [];
+        parse_str($ret, $params);
+        return $params ? $params : [];
+    }
+
+    /**
+     * 获取open_id
+     * @param string $access_token
+     * @return string
+     */
+    private function getOpenId($access_token = '')
+    {
+        $response = Http::get(self::GET_OPENID_URL, ['access_token' => $access_token]);
+        if (strpos($response, "callback") !== false) {
+            $lpos = strpos($response, "(");
+            $rpos = strrpos($response, ")");
+            $response = substr($response, $lpos + 1, $rpos - $lpos - 1);
+        }
+        $user = (array)json_decode($response, true);
+        return isset($user['openid']) ? $user['openid'] : '';
+    }
+}

+ 155 - 0
addons/third/library/Service.php

@@ -0,0 +1,155 @@
+<?php
+
+namespace addons\third\library;
+
+use addons\third\model\Third;
+use app\common\model\User;
+use fast\Random;
+use think\Db;
+use think\Exception;
+
+/**
+ * 第三方登录服务类
+ *
+ */
+class Service
+{
+
+    /**
+     * 第三方登录
+     * @param string $platform 平台
+     * @param array  $params   参数
+     * @param array  $extend   会员扩展信息
+     * @param int    $keeptime 有效时长
+     * @return boolean
+     */
+    public static function connect($platform, $params = [], $extend = [], $keeptime = 0)
+    {
+
+        $time = time();
+        $nickname = $params['nickname'] ?? ($params['userinfo']['nickname'] ?? '');
+        $avatar = $params['avatar'] ?? ($params['userinfo']['avatar'] ?? '');
+        $values = [
+            'platform'      => $platform,
+            'openid'        => $params['openid'],
+            'openname'      => $nickname,
+            'access_token'  => $params['access_token'],
+            'refresh_token' => $params['refresh_token'],
+            'expires_in'    => $params['expires_in'],
+            'logintime'     => $time,
+            'expiretime'    => $time + $params['expires_in'],
+        ];
+        $values = array_merge($values, $params);
+
+        $auth = \app\common\library\Auth::instance();
+
+        $auth->keeptime($keeptime);
+        //是否有自己的
+        $third = Third::get(['platform' => $platform, 'openid' => $params['openid']], 'user');
+        if ($third) {
+            if (!$third->user) {
+                $third->delete();
+            } else {
+                $third->allowField(true)->save($values);
+                // 写入登录Cookies和Token
+                return $auth->direct($third->user_id);
+            }
+        }
+
+        //存在unionid就需要判断是否需要生成新记录
+        if (isset($params['unionid']) && !empty($params['unionid'])) {
+            $third = Third::get(['platform' => $platform, 'unionid' => $params['unionid']], 'user');
+            if ($third) {
+                if (!$third->user) {
+                    $third->delete();
+                } else {
+                    // 保存第三方信息
+                    $values['user_id'] = $third->user_id;
+                    $third = Third::create($values, true);
+                    // 写入登录Cookies和Token
+                    return $auth->direct($third->user_id);
+                }
+            }
+        }
+
+        if ($auth->id) {
+            if (!$third) {
+                $values['user_id'] = $auth->id;
+                Third::create($values, true);
+            }
+            $user = $auth->getUser();
+        } else {
+            // 先随机一个用户名,随后再变更为u+数字id
+            $username = Random::alnum(20);
+            $password = Random::alnum(6);
+            $domain = request()->host();
+
+            Db::startTrans();
+            try {
+                // 默认注册一个会员
+                $result = $auth->register($username, $password, $username . '@' . $domain, '', $extend);
+                if (!$result) {
+                    throw new Exception($auth->getError());
+                }
+                $user = $auth->getUser();
+                $fields = ['username' => 'u' . $user->id, 'email' => 'u' . $user->id . '@' . $domain];
+                if ($nickname) {
+                    $fields['nickname'] = $nickname;
+                }
+                if ($avatar) {
+                    $fields['avatar'] = htmlspecialchars(strip_tags($avatar));
+                }
+
+                // 更新会员资料
+                $user = User::get($user->id);
+                $user->save($fields);
+
+                // 保存第三方信息
+                $values['user_id'] = $user->id;
+                Third::create($values, true);
+                Db::commit();
+            } catch (\Exception $e) {
+                Db::rollback();
+                $auth->logout();
+                return false;
+            }
+        }
+        // 写入登录Cookies和Token
+        return $auth->direct($user->id);
+    }
+
+
+    public static function isBindThird($platform, $openid, $apptype = '', $unionid = '')
+    {
+        $conddtions = [
+            'platform' => $platform,
+            'openid' => $openid
+        ];
+        if ($apptype) {
+            $conddtions['apptype'] = $apptype;
+        }
+        $third = Third::get($conddtions, 'user');
+        //第三方存在
+        if ($third) {
+            //用户失效
+            if (!$third->user) {
+                $third->delete();
+                return false;
+            }
+            return true;
+        }
+        if ($unionid) {
+            $third = Third::get(['platform' => $platform, 'unionid' => $unionid], 'user');
+            if ($third) {
+                //
+                if (!$third->user) {
+                    $third->delete();
+                    return false;
+                }
+                return true;
+            }
+        }
+
+        return false;
+    }
+}

+ 127 - 0
addons/third/library/Wechat.php

@@ -0,0 +1,127 @@
+<?php
+
+namespace addons\third\library;
+
+use fast\Http;
+use think\Config;
+use think\Session;
+
+/**
+ * 微信
+ */
+class Wechat
+{
+    const GET_AUTH_CODE_URL = "https://open.weixin.qq.com/connect/oauth2/authorize";
+    const GET_ACCESS_TOKEN_URL = "https://api.weixin.qq.com/sns/oauth2/access_token";
+    const GET_USERINFO_URL = "https://api.weixin.qq.com/sns/userinfo";
+
+    /**
+     * 配置信息
+     * @var array
+     */
+    private $config = [];
+
+    public function __construct($options = [])
+    {
+        if ($config = Config::get('third.wechat')) {
+            $this->config = array_merge($this->config, $config);
+        }
+        $this->config = array_merge($this->config, is_array($options) ? $options : []);
+    }
+
+    /**
+     * 登陆
+     */
+    public function login()
+    {
+        header("Location:" . $this->getAuthorizeUrl());
+    }
+
+    /**
+     * 获取authorize_url
+     */
+    public function getAuthorizeUrl()
+    {
+        $state = md5(uniqid(rand(), true));
+        Session::set('state', $state);
+        $queryarr = array(
+            "appid"         => $this->config['app_id'],
+            "redirect_uri"  => $this->config['callback'],
+            "response_type" => "code",
+            "scope"         => $this->config['scope'],
+            "state"         => $state,
+        );
+        request()->isMobile() && $queryarr['display'] = 'mobile';
+        $url = self::GET_AUTH_CODE_URL . '?' . http_build_query($queryarr) . '#wechat_redirect';
+        return $url;
+    }
+
+    /**
+     * 获取用户信息
+     * @param array $params
+     * @return array
+     */
+    public function getUserInfo($params = [])
+    {
+        $params = $params ? $params : request()->get();
+        if (isset($params['access_token']) || (isset($params['state']) && $params['state'] == Session::get('state') && isset($params['code']))) {
+            //获取access_token
+            $data = isset($params['code']) ? $this->getAccessToken($params['code']) : $params;
+            $access_token = isset($data['access_token']) ? $data['access_token'] : '';
+            $refresh_token = isset($data['refresh_token']) ? $data['refresh_token'] : '';
+            $expires_in = isset($data['expires_in']) ? $data['expires_in'] : 0;
+            if ($access_token) {
+                $openid = isset($data['openid']) ? $data['openid'] : '';
+                $unionid = isset($data['unionid']) ? $data['unionid'] : '';
+                if (stripos($this->config['scope'], 'snsapi_userinfo') !== false) {
+                    //获取用户信息
+                    $queryarr = [
+                        "access_token" => $access_token,
+                        "openid"       => $openid,
+                        "lang"         => 'zh_CN'
+                    ];
+                    $ret = Http::get(self::GET_USERINFO_URL, $queryarr);
+                    $userinfo = (array)json_decode($ret, true);
+                    if (!$userinfo || isset($userinfo['errcode'])) {
+                        return [];
+                    }
+                    $userinfo = $userinfo ? $userinfo : [];
+                    $userinfo['avatar'] = isset($userinfo['headimgurl']) ? $userinfo['headimgurl'] : '';
+                } else {
+                    $userinfo = [];
+                }
+                $data = [
+                    'access_token'  => $access_token,
+                    'refresh_token' => $refresh_token,
+                    'expires_in'    => $expires_in,
+                    'openid'        => $openid,
+                    'unionid'       => $unionid,
+                    'userinfo'      => $userinfo
+                ];
+                return $data;
+            }
+        }
+        return [];
+    }
+
+    /**
+     * 获取access_token
+     * @param string code
+     * @return array
+     */
+    public function getAccessToken($code = '')
+    {
+        if (!$code) {
+            return [];
+        }
+        $queryarr = array(
+            "appid"      => $this->config['app_id'],
+            "secret"     => $this->config['app_secret'],
+            "code"       => $code,
+            "grant_type" => "authorization_code",
+        );
+        $response = Http::get(self::GET_ACCESS_TOKEN_URL, $queryarr);
+        $ret = (array)json_decode($response, true);
+        return $ret ? $ret : [];
+    }
+}

+ 121 - 0
addons/third/library/Weibo.php

@@ -0,0 +1,121 @@
+<?php
+
+namespace addons\third\library;
+
+use fast\Http;
+use think\Config;
+use think\Session;
+
+/**
+ * 微博
+ */
+class Weibo
+{
+    const GET_AUTH_CODE_URL = "https://api.weibo.com/oauth2/authorize";
+    const GET_ACCESS_TOKEN_URL = "https://api.weibo.com/oauth2/access_token";
+    const GET_USERINFO_URL = "https://api.weibo.com/2/users/show.json";
+
+    /**
+     * 配置信息
+     * @var array
+     */
+    private $config = [];
+
+    public function __construct($options = [])
+    {
+        if ($config = Config::get('third.weibo')) {
+            $this->config = array_merge($this->config, $config);
+        }
+        $this->config = array_merge($this->config, is_array($options) ? $options : []);
+    }
+
+    /**
+     * 登陆
+     */
+    public function login()
+    {
+        header("Location:" . $this->getAuthorizeUrl());
+    }
+
+    /**
+     * 获取authorize_url
+     */
+    public function getAuthorizeUrl()
+    {
+        $state = md5(uniqid(rand(), true));
+        Session::set('state', $state);
+        $queryarr = array(
+            "response_type" => "code",
+            "client_id"     => $this->config['app_id'],
+            "redirect_uri"  => $this->config['callback'],
+            "state"         => $state,
+        );
+        request()->isMobile() && $queryarr['display'] = 'mobile';
+        $url = self::GET_AUTH_CODE_URL . '?' . http_build_query($queryarr);
+        return $url;
+    }
+
+    /**
+     * 获取用户信息
+     * @param array $params
+     * @return array
+     */
+    public function getUserInfo($params = [])
+    {
+        $params = $params ? $params : $_GET;
+        if (isset($params['access_token']) || (isset($params['state']) && $params['state'] == Session::get('state') && isset($params['code']))) {
+            //获取access_token
+            $data = isset($params['code']) ? $this->getAccessToken($params['code']) : $params;
+            $access_token = isset($data['access_token']) ? $data['access_token'] : '';
+            $refresh_token = isset($data['refresh_token']) ? $data['refresh_token'] : '';
+            $expires_in = isset($data['expires_in']) ? $data['expires_in'] : 0;
+            if ($access_token) {
+                $uid = isset($data['uid']) ? $data['uid'] : '';
+                //获取用户信息
+                $queryarr = [
+                    "access_token" => $access_token,
+                    "uid"          => $uid,
+                ];
+                $ret = Http::get(self::GET_USERINFO_URL, $queryarr);
+                $userinfo = (array)json_decode($ret, true);
+                if (!$userinfo || isset($userinfo['error_code'])) {
+                    return [];
+                }
+                $userinfo = $userinfo ? $userinfo : [];
+                $userinfo['nickname'] = isset($userinfo['screen_name']) ? $userinfo['screen_name'] : '';
+                $userinfo['avatar'] = isset($userinfo['profile_image_url']) ? $userinfo['profile_image_url'] : '';
+                $data = [
+                    'access_token'  => $access_token,
+                    'refresh_token' => $refresh_token,
+                    'expires_in'    => $expires_in,
+                    'openid'        => $uid,
+                    'userinfo'      => $userinfo
+                ];
+                return $data;
+            }
+        }
+        return [];
+    }
+
+    /**
+     * 获取access_token
+     * @param string code
+     * @return array
+     */
+    public function getAccessToken($code = '')
+    {
+        if (!$code) {
+            return '';
+        }
+        $queryarr = array(
+            "grant_type"    => "authorization_code",
+            "client_id"     => $this->config['app_id'],
+            "client_secret" => $this->config['app_secret'],
+            "redirect_uri"  => $this->config['callback'],
+            "code"          => $code,
+        );
+        $response = Http::post(self::GET_ACCESS_TOKEN_URL, $queryarr);
+        $ret = (array)json_decode($response, true);
+        return $ret ? $ret : [];
+    }
+}

+ 26 - 0
addons/third/model/Third.php

@@ -0,0 +1,26 @@
+<?php
+
+namespace addons\third\model;
+
+use think\Model;
+
+/**
+ * 第三方登录模型
+ */
+class Third extends Model
+{
+
+    // 开启自动写入时间戳字段
+    protected $autoWriteTimestamp = 'int';
+    // 定义时间戳字段名
+    protected $createTime = 'createtime';
+    protected $updateTime = 'updatetime';
+    // 追加属性
+    protected $append = [
+    ];
+
+    public function user()
+    {
+        return $this->belongsTo('\app\common\model\User', 'user_id', 'id', [], 'LEFT');
+    }
+}

+ 139 - 0
addons/third/view/index/index.html

@@ -0,0 +1,139 @@
+<!DOCTYPE html>
+<html>
+<head>
+    <title>第三方登录 - {$site.name}</title>
+    <meta charset="UTF-8">
+    <meta name="viewport" content="width=device-width, initial-scale=1.0">
+
+    <link href="__CDN__/assets/css/frontend.min.css" rel="stylesheet">
+
+    <!-- HTML5 Shim and Respond.js IE8 support of HTML5 elements and media queries -->
+    <!--[if lt IE 9]>
+    <script src="https://cdn.jsdelivr.net/npm/html5shiv@3.7.3/dist/html5shiv.min.js"></script>
+    <script src="https://cdn.jsdelivr.net/npm/respond.js@1.4.2/dest/respond.min.js"></script>
+    <![endif]-->
+
+</head>
+<body>
+<div class="container">
+    <h2>第三方登录</h2>
+    <hr>
+    <div class="well">
+        <div class="row">
+            <div class="col-xs-4">
+                {if $user && in_array('qq', $platformList)}
+                <a href="{:addon_url('third/index/unbind',[':platform'=>'qq'])}" class="btn btn-block btn-info">
+                    <i class="fa fa-qq"></i> 点击解绑
+                </a>
+                {else/}
+                <a href="{:addon_url('third/index/connect',[':platform'=>'qq'])}" class="btn btn-block btn-info">
+                    <i class="fa fa-qq"></i> QQ登录
+                </a>
+                {/if}
+            </div>
+            <div class="col-xs-4">
+                {if $user && in_array('wechat', $platformList)}
+                <a href="{:addon_url('third/index/unbind',[':platform'=>'wechat'])}" class="btn btn-block btn-success">
+                    <i class="fa fa-wechat"></i> 点击解绑
+                </a>
+                {else/}
+                <a href="{:addon_url('third/index/connect',[':platform'=>'wechat'])}" class="btn btn-block btn-success">
+                    <i class="fa fa-wechat"></i> 微信登录
+                </a>
+                {/if}
+            </div>
+            <div class="col-xs-4">
+                {if $user && in_array('weibo', $platformList)}
+                <a href="{:addon_url('third/index/unbind',[':platform'=>'weibo'])}" class="btn btn-block btn-danger">
+                    <i class="fa fa-weibo"></i> 点击解绑
+                </a>
+                {else/}
+                <a href="{:addon_url('third/index/connect',[':platform'=>'weibo'])}" class="btn btn-block btn-danger">
+                    <i class="fa fa-weibo"></i> 微博登录
+                </a>
+                {/if}
+            </div>
+        </div>
+    </div>
+    <h2>相关链接</h2>
+    <hr>
+    <table class="table table-striped table-hover">
+        <thead>
+        <tr>
+            <th>QQ</th>
+            <th>链接</th>
+        </tr>
+        </thead>
+        <tbody>
+        <tr>
+            <td>QQ 连接</td>
+            <td>{:addon_url('third/index/connect',[':platform'=>'qq'], false, true)}</td>
+        </tr>
+        <tr>
+            <td>QQ 绑定</td>
+            <td>{:addon_url('third/index/bind',[':platform'=>'qq'], false, true)}</td>
+        </tr>
+        <tr>
+            <td>QQ 解绑</td>
+            <td>{:addon_url('third/index/unbind',[':platform'=>'qq'], false, true)}</td>
+        </tr>
+        </tbody>
+    </table>
+    <table class="table table-striped table-hover">
+        <thead>
+        <tr>
+            <th>微信</th>
+            <th>链接</th>
+        </tr>
+        </thead>
+        <tbody>
+        <tr>
+            <td>微信 连接</td>
+            <td>{:addon_url('third/index/connect',[':platform'=>'wechat'], false, true)}</td>
+        </tr>
+        <tr>
+            <td>微信 绑定</td>
+            <td>{:addon_url('third/index/bind',[':platform'=>'wechat'], false, true)}</td>
+        </tr>
+        <tr>
+            <td>微信 解绑</td>
+            <td>{:addon_url('third/index/unbind',[':platform'=>'wechat'], false, true)}</td>
+        </tr>
+        </tbody>
+    </table>
+    <table class="table table-striped table-hover">
+        <thead>
+        <tr>
+            <th>微博</th>
+            <th>链接</th>
+        </tr>
+        </thead>
+        <tbody>
+        <tr>
+            <td>微博 连接</td>
+            <td>{:addon_url('third/index/connect',[':platform'=>'weibo'], false, true)}</td>
+        </tr>
+        <tr>
+            <td>微博 绑定</td>
+            <td>{:addon_url('third/index/bind',[':platform'=>'weibo'], false, true)}</td>
+        </tr>
+        <tr>
+            <td>微博 解绑</td>
+            <td>{:addon_url('third/index/unbind',[':platform'=>'weibo'], false, true)}</td>
+        </tr>
+        </tbody>
+    </table>
+</div>
+<!-- jQuery -->
+<script src="https://cdn.jsdelivr.net/npm/jquery@2.1.4/dist/jquery.min.js"></script>
+
+<!-- Bootstrap Core JavaScript -->
+<script src="https://cdn.jsdelivr.net/npm/bootstrap@3.3.7/dist/js/bootstrap.min.js"></script>
+
+<script type="text/javascript">
+    $(function () {
+
+    });
+</script>
+</body>
+</html>

+ 236 - 0
application/admin/controller/Command.php

@@ -0,0 +1,236 @@
+<?php
+
+namespace app\admin\controller;
+
+use app\common\controller\Backend;
+use think\Config;
+use think\console\Input;
+use think\Db;
+use think\Exception;
+
+/**
+ * 在线命令管理
+ *
+ * @icon fa fa-circle-o
+ */
+class Command extends Backend
+{
+
+    /**
+     * Command模型对象
+     */
+    protected $model = null;
+    protected $noNeedRight = ['get_controller_list', 'get_field_list'];
+
+    public function _initialize()
+    {
+        parent::_initialize();
+        $this->model = model('Command');
+        $this->view->assign("statusList", $this->model->getStatusList());
+    }
+
+    /**
+     * 添加
+     */
+    public function add()
+    {
+
+        $tableList = [];
+        $list = \think\Db::query("SHOW TABLES");
+        foreach ($list as $key => $row) {
+            $tableList[reset($row)] = reset($row);
+        }
+
+        $this->view->assign("tableList", $tableList);
+        return $this->view->fetch();
+    }
+
+    /**
+     * 获取字段列表
+     * @internal
+     */
+    public function get_field_list()
+    {
+        $dbname = Config::get('database.database');
+        $prefix = Config::get('database.prefix');
+        $table = $this->request->request('table');
+        //从数据库中获取表字段信息
+        $sql = "SELECT * FROM `information_schema`.`columns` "
+            . "WHERE TABLE_SCHEMA = ? AND table_name = ? "
+            . "ORDER BY ORDINAL_POSITION";
+        //加载主表的列
+        $columnList = Db::query($sql, [$dbname, $table]);
+        $fieldlist = [];
+        foreach ($columnList as $index => $item) {
+            $fieldlist[] = $item['COLUMN_NAME'];
+        }
+        $this->success("", null, ['fieldlist' => $fieldlist]);
+    }
+
+    /**
+     * 获取控制器列表
+     * @internal
+     */
+    public function get_controller_list()
+    {
+        //搜索关键词,客户端输入以空格分开,这里接收为数组
+        $word = (array)$this->request->request("q_word/a");
+        $word = implode('', $word);
+
+        $adminPath = dirname(__DIR__) . DS;
+        $controllerDir = $adminPath . 'controller' . DS;
+        $files = new \RecursiveIteratorIterator(
+            new \RecursiveDirectoryIterator($controllerDir), \RecursiveIteratorIterator::LEAVES_ONLY
+        );
+        $list = [];
+        foreach ($files as $name => $file) {
+            if (!$file->isDir()) {
+                $filePath = $file->getRealPath();
+                $name = str_replace($controllerDir, '', $filePath);
+                $name = str_replace(DS, "/", $name);
+                if (!preg_match("/(.*)\.php\$/", $name)) {
+                    continue;
+                }
+                if (!$word || stripos($name, $word) !== false) {
+                    $list[] = ['id' => $name, 'name' => $name];
+                }
+            }
+        }
+        $pageNumber = $this->request->request("pageNumber");
+        $pageSize = $this->request->request("pageSize");
+        return json(['list' => array_slice($list, ($pageNumber - 1) * $pageSize, $pageSize), 'total' => count($list)]);
+    }
+
+    /**
+     * 详情
+     */
+    public function detail($ids)
+    {
+        $row = $this->model->get($ids);
+        if (!$row) {
+            $this->error(__('No Results were found'));
+        }
+        $this->view->assign("row", $row);
+        return $this->view->fetch();
+    }
+
+    /**
+     * 执行
+     */
+    public function execute($ids)
+    {
+        $row = $this->model->get($ids);
+        if (!$row) {
+            $this->error(__('No Results were found'));
+        }
+        $result = $this->doexecute($row['type'], json_decode($row['params'], true));
+        $this->success("", null, ['result' => $result]);
+    }
+
+    /**
+     * 执行命令
+     */
+    public function command($action = '')
+    {
+        $commandtype = $this->request->request("commandtype");
+        $params = $this->request->request();
+        $allowfields = [
+            'crud' => 'table,controller,model,fields,force,local,delete,menu',
+            'menu' => 'controller,delete',
+            'min'  => 'module,resource,optimize',
+            'api'  => 'url,module,output,template,force,title,author,class,language',
+        ];
+        $argv = [];
+        $allowfields = isset($allowfields[$commandtype]) ? explode(',', $allowfields[$commandtype]) : [];
+        $allowfields = array_filter(array_intersect_key($params, array_flip($allowfields)));
+        if (isset($params['local']) && !$params['local']) {
+            $allowfields['local'] = $params['local'];
+        } else {
+            unset($allowfields['local']);
+        }
+        foreach ($allowfields as $key => $param) {
+            $argv[] = "--{$key}=" . (is_array($param) ? implode(',', $param) : $param);
+        }
+        if ($commandtype == 'crud') {
+            $extend = 'setcheckboxsuffix,enumradiosuffix,imagefield,filefield,intdatesuffix,switchsuffix,citysuffix,selectpagesuffix,selectpagessuffix,ignorefields,sortfield,editorsuffix,headingfilterfield';
+            $extendArr = explode(',', $extend);
+            foreach ($params as $index => $item) {
+                if (in_array($index, $extendArr)) {
+                    foreach (explode(',', $item) as $key => $value) {
+                        if ($value) {
+                            $argv[] = "--{$index}={$value}";
+                        }
+                    }
+                }
+            }
+            $isrelation = (int)$this->request->request('isrelation');
+            if ($isrelation && isset($params['relation'])) {
+                foreach ($params['relation'] as $index => $relation) {
+                    foreach ($relation as $key => $value) {
+                        $argv[] = "--{$key}=" . (is_array($value) ? implode(',', $value) : $value);
+                    }
+                }
+            }
+        } else {
+            if ($commandtype == 'menu') {
+                if (isset($params['allcontroller']) && $params['allcontroller']) {
+                    $argv[] = "--controller=all-controller";
+                } else {
+                    foreach (explode(',', $params['controllerfile']) as $index => $param) {
+                        if ($param) {
+                            $argv[] = "--controller=" . substr($param, 0, -4);
+                        }
+                    }
+                }
+            } else {
+                if ($commandtype == 'min') {
+
+                } else {
+                    if ($commandtype == 'api') {
+
+                    } else {
+
+                    }
+                }
+            }
+        }
+        if ($action == 'execute') {
+            $result = $this->doexecute($commandtype, $argv);
+            $this->success("", null, ['result' => $result]);
+        } else {
+            $this->success("", null, ['command' => "php think {$commandtype} " . implode(' ', $argv)]);
+        }
+
+        return;
+    }
+
+    protected function doexecute($commandtype, $argv)
+    {
+        $commandName = "\\app\\admin\\command\\" . ucfirst($commandtype);
+        $input = new Input($argv);
+        $output = new \addons\command\library\Output();
+        $command = new $commandName($commandtype);
+        $data = [
+            'type'        => $commandtype,
+            'params'      => json_encode($argv),
+            'command'     => "php think {$commandtype} " . implode(' ', $argv),
+            'executetime' => time(),
+        ];
+        $this->model->save($data);
+        try {
+            $command->run($input, $output);
+            $result = implode("\n", $output->getMessage());
+            $this->model->status = 'successed';
+        } catch (Exception $e) {
+            $result = implode("\n", $output->getMessage()) . "\n";
+            $result .= $e->getMessage();
+            $this->model->status = 'failured';
+        }
+        $result = trim($result);
+        $this->model->content = $result;
+        $this->model->save();
+        return $result;
+    }
+
+
+}

+ 66 - 0
application/admin/controller/Third.php

@@ -0,0 +1,66 @@
+<?php
+
+namespace app\admin\controller;
+
+use app\common\controller\Backend;
+
+/**
+ * 第三方登录管理
+ *
+ * @icon fa fa-circle-o
+ */
+class Third extends Backend
+{
+
+    /**
+     * Third模型对象
+     * @var \app\admin\model\Third
+     */
+    protected $model = null;
+
+    public function _initialize()
+    {
+        parent::_initialize();
+        $this->model = new \app\admin\model\Third;
+    }
+
+    /**
+     * 查看
+     */
+    public function index()
+    {
+        $this->relationSearch = true;
+        //设置过滤方法
+        $this->request->filter(['strip_tags']);
+        if ($this->request->isAjax()) {
+            //如果发送的来源是Selectpage,则转发到Selectpage
+            if ($this->request->request('keyField')) {
+                return $this->selectpage();
+            }
+            list($where, $sort, $order, $offset, $limit) = $this->buildparams();
+            $total = $this->model
+                ->with(['user'])
+                ->where($where)
+                ->order($sort, $order)
+                ->count();
+
+            $list = $this->model
+                ->with(['user'])
+                ->where($where)
+                ->order($sort, $order)
+                ->limit($offset, $limit)
+                ->select();
+            foreach ($list as $index => $item) {
+                if ($item->user) {
+                    $item->user->visible(['nickname']);
+                }
+            }
+            $list = collection($list)->toArray();
+            $result = array("total" => $total, "rows" => $list);
+
+            return json($result);
+        }
+        return $this->view->fetch();
+    }
+
+}

+ 16 - 0
application/admin/lang/zh-cn/command.php

@@ -0,0 +1,16 @@
+<?php
+
+return [
+    'Id'            => 'ID',
+    'Type'          => '类型',
+    'Params'        => '参数',
+    'Command'       => '命令',
+    'Content'       => '返回结果',
+    'Executetime'   => '执行时间',
+    'Createtime'    => '创建时间',
+    'Updatetime'    => '更新时间',
+    'Execute again' => '再次执行',
+    'Successed'     => '成功',
+    'Failured'      => '失败',
+    'Status'        => '状态'
+];

+ 16 - 0
application/admin/lang/zh-cn/third.php

@@ -0,0 +1,16 @@
+<?php
+
+return [
+    'Id'           => 'ID',
+    'User_id'      => '会员ID',
+    'Platform'     => '第三方应用',
+    'Unionid'      => '第三方UnionID',
+    'Openid'       => '第三方OpenID',
+    'Openname'     => '第三方会员昵称',
+    'Access_token' => 'AccessToken',
+    'Expires_in'   => '有效期',
+    'Createtime'   => '创建时间',
+    'Updatetime'   => '更新时间',
+    'Logintime'    => '登录时间',
+    'Expiretime'   => '过期时间'
+];

+ 59 - 0
application/admin/model/Command.php

@@ -0,0 +1,59 @@
+<?php
+
+namespace app\admin\model;
+
+use think\Model;
+
+class Command extends Model
+{
+    // 表名
+    protected $name = 'command';
+
+    // 自动写入时间戳字段
+    protected $autoWriteTimestamp = 'int';
+
+    // 定义时间戳字段名
+    protected $createTime = 'createtime';
+    protected $updateTime = 'updatetime';
+
+    // 追加属性
+    protected $append = [
+        'executetime_text',
+        'type_text',
+        'status_text'
+    ];
+
+
+    public function getStatusList()
+    {
+        return ['successed' => __('Successed'), 'failured' => __('Failured')];
+    }
+
+
+    public function getExecutetimeTextAttr($value, $data)
+    {
+        $value = $value ? $value : $data['executetime'];
+        return is_numeric($value) ? date("Y-m-d H:i:s", $value) : $value;
+    }
+
+    public function getTypeTextAttr($value, $data)
+    {
+        $value = $value ? $value : $data['type'];
+        $list = ['crud' => '一键生成CRUD', 'menu' => '一键生成菜单', 'min' => '一键压缩打包', 'api' => '一键生成文档'];
+        return isset($list[$value]) ? $list[$value] : '';
+    }
+
+    public function getStatusTextAttr($value, $data)
+    {
+        $value = $value ? $value : $data['status'];
+        $list = $this->getStatusList();
+        return isset($list[$value]) ? $list[$value] : '';
+    }
+
+    protected function setExecutetimeAttr($value)
+    {
+        return $value && !is_numeric($value) ? strtotime($value) : $value;
+    }
+
+
+}

+ 56 - 0
application/admin/model/Third.php

@@ -0,0 +1,56 @@
+<?php
+
+namespace app\admin\model;
+
+use think\Model;
+
+class Third extends Model
+{
+
+
+    // 表名
+    protected $name = 'third';
+
+    // 自动写入时间戳字段
+    protected $autoWriteTimestamp = 'int';
+
+    // 定义时间戳字段名
+    protected $createTime = 'createtime';
+    protected $updateTime = 'updatetime';
+    protected $deleteTime = false;
+
+    // 追加属性
+    protected $append = [
+        'logintime_text',
+        'expiretime_text'
+    ];
+
+
+    public function getLogintimeTextAttr($value, $data)
+    {
+        $value = $value ? $value : (isset($data['logintime']) ? $data['logintime'] : '');
+        return is_numeric($value) ? date("Y-m-d H:i:s", $value) : $value;
+    }
+
+
+    public function getExpiretimeTextAttr($value, $data)
+    {
+        $value = $value ? $value : (isset($data['expiretime']) ? $data['expiretime'] : '');
+        return is_numeric($value) ? date("Y-m-d H:i:s", $value) : $value;
+    }
+
+    protected function setLogintimeAttr($value)
+    {
+        return $value === '' ? null : ($value && !is_numeric($value) ? strtotime($value) : $value);
+    }
+
+    protected function setExpiretimeAttr($value)
+    {
+        return $value === '' ? null : ($value && !is_numeric($value) ? strtotime($value) : $value);
+    }
+
+    public function user()
+    {
+        return $this->belongsTo("User", 'user_id', 'id')->setEagerlyType(0);
+    }
+}

+ 27 - 0
application/admin/validate/Command.php

@@ -0,0 +1,27 @@
+<?php
+
+namespace app\admin\validate;
+
+use think\Validate;
+
+class Command extends Validate
+{
+    /**
+     * 验证规则
+     */
+    protected $rule = [
+    ];
+    /**
+     * 提示消息
+     */
+    protected $message = [
+    ];
+    /**
+     * 验证场景
+     */
+    protected $scene = [
+        'add'  => [],
+        'edit' => [],
+    ];
+
+}

+ 26 - 0
application/admin/validate/Third.php

@@ -0,0 +1,26 @@
+<?php
+
+namespace app\admin\validate;
+
+use think\Validate;
+
+class Third extends Validate
+{
+    /**
+     * 验证规则
+     */
+    protected $rule = [
+    ];
+    /**
+     * 提示消息
+     */
+    protected $message = [
+    ];
+    /**
+     * 验证场景
+     */
+    protected $scene = [
+        'add'  => [],
+        'edit' => [],
+    ];
+}

+ 400 - 0
application/admin/view/command/add.html

@@ -0,0 +1,400 @@
+<style>
+    .relation-item {margin-top:10px;}
+    legend {padding-bottom:5px;font-size:14px;font-weight:600;}
+    label {font-weight:normal;}
+    .form-control{padding:6px 8px;}
+    #extend-zone .col-xs-2 {margin-top:10px;padding-right:0;}
+    #extend-zone .col-xs-2:nth-child(6n+0) {padding-right:15px;}
+</style>
+<div class="panel panel-default panel-intro">
+    <div class="panel-heading">
+        <ul class="nav nav-tabs">
+            <li class="active"><a href="#crud" data-toggle="tab">{:__('一键生成CRUD')}</a></li>
+            <li><a href="#menu" data-toggle="tab">{:__('一键生成菜单')}</a></li>
+            <li><a href="#min" data-toggle="tab">{:__('一键压缩打包')}</a></li>
+            <li><a href="#api" data-toggle="tab">{:__('一键生成API文档')}</a></li>
+        </ul>
+    </div>
+    <div class="panel-body">
+        <div id="myTabContent" class="tab-content">
+            <div class="tab-pane fade active in" id="crud">
+                <div class="row">
+                    <div class="col-xs-12">
+                        <form role="form">
+                            <input type="hidden" name="commandtype" value="crud" />
+                            <div class="form-group">
+                                <div class="row">
+                                    <div class="col-xs-3">
+                                        <input checked="" name="isrelation" type="hidden" value="0">
+                                        <label class="control-label" data-toggle="tooltip" title="当前只支持生成1对1关联模型,选中后请配置关联表和字段">
+                                            <input name="isrelation" type="checkbox" value="1">
+                                            关联模型
+                                        </label>
+                                    </div>
+                                    <div class="col-xs-3">
+                                        <input checked="" name="local" type="hidden" value="1">
+                                        <label class="control-label" data-toggle="tooltip" title="默认模型生成在application/admin/model目录下,选中后将生成在application/common/model目录下">
+                                            <input name="local" type="checkbox" value="0"> 全局模型类
+                                        </label>
+                                    </div>
+                                    <div class="col-xs-3">
+                                        <input checked="" name="delete" type="hidden" value="0">
+                                        <label class="control-label" data-toggle="tooltip" title="删除CRUD生成的相关文件">
+                                            <input name="delete" type="checkbox" value="1"> 删除模式
+                                        </label>
+                                    </div>
+                                    <div class="col-xs-3">
+                                        <input checked="" name="force" type="hidden" value="0">
+                                        <label class="control-label" data-toggle="tooltip" title="选中后,如果已经存在同名文件将被覆盖。如果是删除将不再提醒">
+                                            <input name="force" type="checkbox" value="1">
+                                            强制覆盖模式
+                                        </label>
+                                    </div>
+                                    <!--
+                                    <div class="col-xs-3">
+                                        <input checked="" name="menu" type="hidden" value="0">
+                                        <label class="control-label" data-toggle="tooltip" title="选中后,将同时生成后台菜单规则">
+                                            <input name="menu" type="checkbox" value="1">
+                                            生成菜单
+                                        </label>
+                                    </div>
+                                    -->
+                                </div>
+                            </div>
+                            <div class="form-group">
+                                <legend>主表设置</legend>
+                                <div class="row">
+                                    <div class="col-xs-3">
+                                        <label>请选择主表</label>
+                                        {:build_select('table',$tableList,null,['class'=>'form-control selectpicker', 'data-live-search'=>'true']);}
+                                    </div>
+                                    <div class="col-xs-3">
+                                        <label>自定义控制器名</label>
+                                        <input type="text" class="form-control" name="controller" data-toggle="tooltip" title="默认根据表名自动生成,如果需要放在二级目录请手动填写" placeholder="支持目录层级,以/分隔">
+                                    </div>
+                                    <div class="col-xs-3">
+                                        <label>自定义模型名</label>
+                                        <input type="text" class="form-control" name="model" data-toggle="tooltip" title="默认根据表名自动生成" placeholder="不支持目录层级">
+                                    </div>
+                                    <div class="col-xs-3">
+                                        <label>请选择显示字段(默认全部)</label>
+                                        <select name="fields[]" id="fields" multiple style="height:30px;" class="form-control selectpicker"></select>
+                                    </div>
+
+                                </div>
+
+                            </div>
+
+                            <div class="form-group hide" id="relation-zone">
+                                <legend>关联表设置</legend>
+
+                                <div class="row" style="margin-top:15px;">
+                                    <div class="col-xs-12">
+                                        <a href="javascript:;" class="btn btn-primary btn-sm btn-newrelation" data-index="1">追加关联模型</a>
+                                    </div>
+                                </div>
+                            </div>
+
+                            <hr>
+                            <div class="form-group" id="extend-zone">
+                                <legend>字段识别设置 <span style="font-size:12px;font-weight: normal;">(与之匹配的字段都将生成相应组件)</span></legend>
+                                <div class="row">
+                                    <div class="col-xs-2">
+                                        <label>复选框后缀</label>
+                                        <input type="text" class="form-control" name="setcheckboxsuffix" placeholder="默认为set类型" />
+                                    </div>
+                                    <div class="col-xs-2">
+                                        <label>单选框后缀</label>
+                                        <input type="text" class="form-control" name="enumradiosuffix" placeholder="默认为enum类型" />
+                                    </div>
+                                    <div class="col-xs-2">
+                                        <label>图片类型后缀</label>
+                                        <input type="text" class="form-control" name="imagefield" placeholder="默认为image,images,avatar,avatars" />
+                                    </div>
+                                    <div class="col-xs-2">
+                                        <label>文件类型后缀</label>
+                                        <input type="text" class="form-control" name="filefield" placeholder="默认为file,files" />
+                                    </div>
+                                    <div class="col-xs-2">
+                                        <label>日期时间后缀</label>
+                                        <input type="text" class="form-control" name="intdatesuffix" placeholder="默认为time" />
+                                    </div>
+                                    <div class="col-xs-2">
+                                        <label>开关后缀</label>
+                                        <input type="text" class="form-control" name="switchsuffix" placeholder="默认为switch" />
+                                    </div>
+                                    <div class="col-xs-2">
+                                        <label>城市选择后缀</label>
+                                        <input type="text" class="form-control" name="citysuffix" placeholder="默认为city" />
+                                    </div>
+                                    <div class="col-xs-2">
+                                        <label>动态下拉后缀(单)</label>
+                                        <input type="text" class="form-control" name="selectpagesuffix" placeholder="默认为_id" />
+                                    </div>
+                                    <div class="col-xs-2">
+                                        <label>动态下拉后缀(多)</label>
+                                        <input type="text" class="form-control" name="selectpagessuffix" placeholder="默认为_ids" />
+                                    </div>
+                                    <div class="col-xs-2">
+                                        <label>忽略的字段</label>
+                                        <input type="text" class="form-control" name="ignorefields" placeholder="默认无" />
+                                    </div>
+                                    <div class="col-xs-2">
+                                        <label>排序字段</label>
+                                        <input type="text" class="form-control" name="sortfield" placeholder="默认为weigh" />
+                                    </div>
+                                    <div class="col-xs-2">
+                                        <label>富文本编辑器</label>
+                                        <input type="text" class="form-control" name="editorsuffix" placeholder="默认为content" />
+                                    </div>
+                                    <div class="col-xs-2">
+                                        <label>选项卡过滤字段</label>
+                                        <input type="text" class="form-control" name="headingfilterfield" placeholder="默认为status" />
+                                    </div>
+
+                                </div>
+
+                            </div>
+
+                            <div class="form-group">
+                                <legend>生成命令行</legend>
+                                <textarea class="form-control" data-toggle="tooltip" title="如果在线执行命令失败,可以将命令复制到命令行进行执行" rel="command" rows="1" placeholder="请点击生成命令行"></textarea>
+                            </div>
+
+                            <div class="form-group">
+                                <legend>返回结果</legend>
+                                <textarea class="form-control" rel="result" rows="5" placeholder="请点击立即执行"></textarea>
+                            </div>
+
+                            <div class="form-group">
+                                    <button type="button" class="btn btn-info btn-embossed btn-command">{:__('生成命令行')}</button>
+                                    <button type="button" class="btn btn-success btn-embossed btn-execute">{:__('立即执行')}</button>
+                            </div>
+
+                        </form>
+                    </div>
+                </div>
+            </div>
+            <div class="tab-pane fade" id="menu">
+                <div class="row">
+                    <div class="col-xs-12">
+                        <form role="form">
+                            <input type="hidden" name="commandtype" value="menu" />
+                            <div class="form-group">
+                                <div class="row">
+                                    <div class="col-xs-3">
+                                        <input checked="" name="allcontroller" type="hidden" value="0">
+                                        <label class="control-label">
+                                            <input name="allcontroller" data-toggle="collapse" data-target="#controller" type="checkbox" value="1"> 一键生成全部控制器
+                                        </label>
+                                    </div>
+                                    <div class="col-xs-3">
+                                        <input checked="" name="delete" type="hidden" value="0">
+                                        <label class="control-label">
+                                            <input name="delete" type="checkbox" value="1"> 删除模式
+                                        </label>
+                                    </div>
+                                    <div class="col-xs-3">
+                                        <input checked="" name="force" type="hidden" value="0">
+                                        <label class="control-label">
+                                            <input name="force" type="checkbox" value="1"> 强制覆盖模式
+                                        </label>
+                                    </div>
+                                </div>
+                            </div>
+
+                            <div class="form-group in" id="controller">
+                                <legend>控制器设置</legend>
+
+                                <div class="row" style="margin-top:15px;">
+                                    <div class="col-xs-12">
+                                        <input type="text" name="controllerfile" class="form-control selectpage" style="width:720px;" data-source="command/get_controller_list" data-multiple="true" name="controller" placeholder="请选择控制器" />
+                                    </div>
+                                </div>
+                            </div>
+
+                            <div class="form-group">
+                                <legend>生成命令行</legend>
+                                <textarea class="form-control" rel="command" rows="1" placeholder="请点击生成命令行"></textarea>
+                            </div>
+
+                            <div class="form-group">
+                                <legend>返回结果</legend>
+                                <textarea class="form-control" rel="result" rows="5" placeholder="请点击立即执行"></textarea>
+                            </div>
+
+                            <div class="form-group">
+                                <button type="button" class="btn btn-info btn-embossed btn-command">{:__('生成命令行')}</button>
+                                <button type="button" class="btn btn-success btn-embossed btn-execute">{:__('立即执行')}</button>
+                            </div>
+
+                        </form>
+                    </div>
+                </div>
+            </div>
+            <div class="tab-pane fade" id="min">
+                <div class="row">
+                    <div class="col-xs-12">
+                        <form role="form">
+                            <input type="hidden" name="commandtype" value="min" />
+                            <div class="form-group">
+                                <legend>基础设置</legend>
+                                <div class="row">
+                                    <div class="col-xs-3">
+                                        <label>请选择压缩模块</label>
+                                        <select name="module" class="form-control selectpicker">
+                                            <option value="all" selected>全部</option>
+                                            <option value="backend">后台Backend</option>
+                                            <option value="frontend">前台Frontend</option>
+                                        </select>
+                                    </div>
+                                    <div class="col-xs-3">
+                                        <label>请选择压缩资源</label>
+                                        <select name="resource" class="form-control selectpicker">
+                                            <option value="all" selected>全部</option>
+                                            <option value="js">JS</option>
+                                            <option value="css">CSS</option>
+                                        </select>
+                                    </div>
+                                    <div class="col-xs-3">
+                                        <label>请选择压缩模式</label>
+                                        <select name="optimize" class="form-control selectpicker">
+                                            <option value="">无</option>
+                                            <option value="uglify">uglify</option>
+                                            <option value="closure">closure</option>
+                                        </select>
+                                    </div>
+                                </div>
+                            </div>
+
+                            <div class="form-group in">
+                                <legend>控制器设置</legend>
+
+                                <div class="row" style="margin-top:15px;">
+                                    <div class="col-xs-12">
+
+                                    </div>
+                                </div>
+                            </div>
+
+                            <div class="form-group">
+                                <legend>生成命令行</legend>
+                                <textarea class="form-control" rel="command" rows="1" placeholder="请点击生成命令行"></textarea>
+                            </div>
+
+                            <div class="form-group">
+                                <legend>返回结果</legend>
+                                <textarea class="form-control" rel="result" rows="5" placeholder="请点击立即执行"></textarea>
+                            </div>
+
+                            <div class="form-group">
+                                <button type="button" class="btn btn-info btn-embossed btn-command">{:__('生成命令行')}</button>
+                                <button type="button" class="btn btn-success btn-embossed btn-execute">{:__('立即执行')}</button>
+                            </div>
+
+                        </form>
+                    </div>
+                </div>
+            </div>
+            <div class="tab-pane fade" id="api">
+                <div class="row">
+                    <div class="col-xs-12">
+                        <form role="form">
+                            <input type="hidden" name="commandtype" value="api" />
+                            <div class="form-group">
+                                <div class="row">
+                                    <div class="col-xs-3">
+                                        <input checked="" name="force" type="hidden" value="0">
+                                        <label class="control-label">
+                                            <input name="force" type="checkbox" value="1">
+                                            覆盖模式
+                                        </label>
+                                    </div>
+                                </div>
+                            </div>
+                            <div class="form-group">
+                                <legend>文档设置</legend>
+                                <div class="row">
+                                    <div class="col-xs-3">
+                                        <label>请输入接口URL</label>
+                                        <input type="text" name="url" class="form-control" placeholder="API URL,可留空" />
+                                    </div>
+                                    <div class="col-xs-3">
+                                        <label>接口生成文件</label>
+                                        <input type="text" name="output" class="form-control" placeholder="留空则使用api.html" />
+                                    </div>
+                                    <div class="col-xs-3">
+                                        <label>模板文件</label>
+                                        <input type="text" name="template" class="form-control" placeholder="如果不清楚请留空" />
+                                    </div>
+                                </div>
+                                <div class="row" style="margin-top:10px;">
+                                    <div class="col-xs-3">
+                                        <label>文档标题</label>
+                                        <input type="text" name="title" class="form-control" placeholder="默认为FastAdmin" />
+                                    </div>
+                                    <div class="col-xs-3">
+                                        <label>文档作者</label>
+                                        <input type="text" name="author" class="form-control" placeholder="默认为FastAdmin" />
+                                    </div>
+                                    <div class="col-xs-3">
+                                        <label>文档语言</label>
+                                        <select name="language" class="form-control">
+                                            <option value="" selected>请选择语言</option>
+                                            <option value="zh-cn">中文</option>
+                                            <option value="en">英文</option>
+                                        </select>
+                                    </div>
+                                </div>
+                            </div>
+
+                            <div class="form-group">
+                                <legend>生成命令行</legend>
+                                <textarea class="form-control" rel="command" rows="1" placeholder="请点击生成命令行"></textarea>
+                            </div>
+
+                            <div class="form-group">
+                                <legend>返回结果</legend>
+                                <textarea class="form-control" rel="result" rows="5" placeholder="请点击立即执行"></textarea>
+                            </div>
+
+                            <div class="form-group">
+                                <button type="button" class="btn btn-info btn-embossed btn-command">{:__('生成命令行')}</button>
+                                <button type="button" class="btn btn-success btn-embossed btn-execute">{:__('立即执行')}</button>
+                            </div>
+
+                        </form>
+                    </div>
+                </div>
+            </div>
+        </div>
+    </div>
+</div>
+<script id="relationtpl" type="text/html">
+    <div class="row relation-item">
+        <div class="col-xs-2">
+            <label>请选择关联表</label>
+            <select name="relation[<%=index%>][relation]" class="form-control relationtable" data-live-search="true"></select>
+        </div>
+        <div class="col-xs-2">
+            <label>请选择关联类型</label>
+            <select name="relation[<%=index%>][relationmode]" class="form-control relationmode"></select>
+        </div>
+        <div class="col-xs-2">
+            <label>关联外键</label>
+            <select name="relation[<%=index%>][relationforeignkey]" class="form-control relationforeignkey"></select>
+        </div>
+        <div class="col-xs-2">
+            <label>关联主键</label>
+            <select name="relation[<%=index%>][relationprimarykey]" class="form-control relationprimarykey"></select>
+        </div>
+        <div class="col-xs-2">
+            <label>请选择显示字段</label>
+            <select name="relation[<%=index%>][relationfields][]" multiple class="form-control relationfields"></select>
+        </div>
+        <div class="col-xs-2">
+            <label>&nbsp;</label>
+            <a href="javascript:;" class="btn btn-danger btn-block btn-removerelation">移除</a>
+        </div>
+    </div>
+</script>

+ 42 - 0
application/admin/view/command/detail.html

@@ -0,0 +1,42 @@
+<table class="table table-striped">
+    <thead>
+    <tr>
+        <th>{:__('Title')}</th>
+        <th>{:__('Content')}</th>
+    </tr>
+    </thead>
+    <tbody>
+    <tr>
+        <td>{:__('Type')}</td>
+        <td>{$row.type}({$row.type_text})</td>
+    </tr>
+    <tr>
+        <td>{:__('Params')}</td>
+        <td>{$row.params}</td>
+    </tr>
+    <tr>
+        <td>{:__('Command')}</td>
+        <td>{$row.command}</td>
+    </tr>
+    <tr>
+        <td>{:__('Content')}</td>
+        <td>
+            <textarea class="form-control" name="" id="" cols="60" rows="10">{$row.content}</textarea>
+        </td>
+    </tr>
+    <tr>
+        <td>{:__('Executetime')}</td>
+        <td>{$row.executetime|datetime}</td>
+    </tr>
+    <tr>
+        <td>{:__('Status')}</td>
+        <td>{$row.status_text}</td>
+    </tr>
+    </tbody>
+</table>
+<div class="hide layer-footer">
+    <label class="control-label col-xs-12 col-sm-2"></label>
+    <div class="col-xs-12 col-sm-8">
+        <button type="reset" class="btn btn-primary btn-embossed btn-close" onclick="Layer.closeAll();">{:__('Close')}</button>
+    </div>
+</div>

+ 25 - 0
application/admin/view/command/index.html

@@ -0,0 +1,25 @@
+<div class="panel panel-default panel-intro">
+    {:build_heading()}
+
+    <div class="panel-body">
+        <div id="myTabContent" class="tab-content">
+            <div class="tab-pane fade active in" id="one">
+                <div class="widget-body no-padding">
+                    <div id="toolbar" class="toolbar">
+                        <a href="javascript:;" class="btn btn-primary btn-refresh" title="{:__('Refresh')}" ><i class="fa fa-refresh"></i> </a>
+                        <a href="javascript:;" class="btn btn-success btn-add {:$auth->check('command/add')?'':'hide'}" title="{:__('Add')}" ><i class="fa fa-plus"></i> {:__('Add')}</a>
+                        <a href="javascript:;" class="btn btn-danger btn-del btn-disabled disabled {:$auth->check('command/del')?'':'hide'}" title="{:__('Delete')}" ><i class="fa fa-trash"></i> {:__('Delete')}</a>
+
+                    </div>
+                    <table id="table" class="table table-striped table-bordered table-hover" 
+                           data-operate-detail="{:$auth->check('command/detail')}"
+                           data-operate-execute="{:$auth->check('command/execute')}"
+                           data-operate-del="{:$auth->check('command/del')}"
+                           width="100%">
+                    </table>
+                </div>
+            </div>
+
+        </div>
+    </div>
+</div>

+ 23 - 0
application/admin/view/third/index.html

@@ -0,0 +1,23 @@
+<div class="panel panel-default panel-intro">
+    {:build_heading()}
+
+    <div class="panel-body">
+        <div id="myTabContent" class="tab-content">
+            <div class="tab-pane fade active in" id="one">
+                <div class="widget-body no-padding">
+                    <div id="toolbar" class="toolbar">
+                        <a href="javascript:;" class="btn btn-primary btn-refresh" title="{:__('Refresh')}"><i class="fa fa-refresh"></i> </a>
+                        <a href="javascript:;" class="btn btn-danger btn-del btn-disabled disabled {:$auth->check('third/del')?'':'hide'}" title="{:__('Delete')}"><i class="fa fa-trash"></i> {:__('Delete')}</a>
+
+                    </div>
+                    <table id="table" class="table table-striped table-bordered table-hover table-nowrap"
+                           data-operate-edit=""
+                           data-operate-del="{:$auth->check('third/del')}"
+                           width="100%">
+                    </table>
+                </div>
+            </div>
+
+        </div>
+    </div>
+</div>

+ 0 - 73
application/api/controller/Demo.php

@@ -1,73 +0,0 @@
-<?php
-
-namespace app\api\controller;
-
-use app\common\controller\Api;
-
-/**
- * 示例接口
- */
-class Demo extends Api
-{
-
-    //如果$noNeedLogin为空表示所有接口都需要登录才能请求
-    //如果$noNeedRight为空表示所有接口都需要验证权限才能请求
-    //如果接口已经设置无需登录,那也就无需鉴权了
-    //
-    // 无需登录的接口,*表示全部
-    protected $noNeedLogin = ['test', 'test1'];
-    // 无需鉴权的接口,*表示全部
-    protected $noNeedRight = ['test2'];
-
-    /**
-     * 测试方法
-     *
-     * @ApiTitle    (测试名称)
-     * @ApiSummary  (测试描述信息)
-     * @ApiMethod   (POST)
-     * @ApiRoute    (/api/demo/test/id/{id}/name/{name})
-     * @ApiHeaders  (name=token, type=string, required=true, description="请求的Token")
-     * @ApiParams   (name="id", type="integer", required=true, description="会员ID")
-     * @ApiParams   (name="name", type="string", required=true, description="用户名")
-     * @ApiParams   (name="data", type="object", sample="{'user_id':'int','user_name':'string','profile':{'email':'string','age':'integer'}}", description="扩展数据")
-     * @ApiReturnParams   (name="code", type="integer", required=true, sample="0")
-     * @ApiReturnParams   (name="msg", type="string", required=true, sample="返回成功")
-     * @ApiReturnParams   (name="data", type="object", sample="{'user_id':'int','user_name':'string','profile':{'email':'string','age':'integer'}}", description="扩展数据返回")
-     * @ApiReturn   ({
-         'code':'1',
-         'msg':'返回成功'
-        })
-     */
-    public function test()
-    {
-        $this->success('返回成功', $this->request->param());
-    }
-
-    /**
-     * 无需登录的接口
-     *
-     */
-    public function test1()
-    {
-        $this->success('返回成功', ['action' => 'test1']);
-    }
-
-    /**
-     * 需要登录的接口
-     *
-     */
-    public function test2()
-    {
-        $this->success('返回成功', ['action' => 'test2']);
-    }
-
-    /**
-     * 需要登录且需要验证有相应组的权限
-     *
-     */
-    public function test3()
-    {
-        $this->success('返回成功', ['action' => 'test3']);
-    }
-
-}

+ 3 - 3
application/common.php

@@ -483,10 +483,10 @@ if (!function_exists('check_ip_allowed')) {
     }
 }
 
-function payment(){
+function payment($appid){
     $config = [
         // 必要配置
-        'app_id'             => 'xxxx',
+        'app_id'             => $appid,
         'mch_id'             => 'your-mch-id',
         'key'                => 'key-for-signature',   // API 密钥
         // 如需使用敏感接口(如退款、发送红包等)需要配置 API 证书路径(登录商户平台下载 API 证书)
@@ -494,5 +494,5 @@ function payment(){
         'key_path'           => 'path/to/your/key',      // XXX: 绝对路径!!!!
         'notify_url'         => '默认的订单回调地址',     // 你也可以在下单时单独设置来想覆盖它
     ];
-    $app = Factory::payment($config);
+    return Factory::payment($config);
 }

+ 1 - 1
application/config.php

@@ -264,7 +264,7 @@ return [
         //会员注册验证码类型email/mobile/wechat/text/false
         'user_register_captcha' => 'text',
         //登录验证码
-        'login_captcha'         => true,
+        'login_captcha'         => false,
         //登录失败超过10次则1天后重试
         'login_failure_retry'   => true,
         //是否同一账号同一时间只能在一个地方登录

+ 12 - 2
application/extra/addons.php

@@ -2,8 +2,18 @@
 
 return [
     'autoload' => false,
-    'hooks' => [],
-    'route' => [],
+    'hooks' => [
+        'config_init' => [
+            'third',
+        ],
+    ],
+    'route' => [
+        '/third$' => 'third/index/index',
+        '/third/connect/[:platform]' => 'third/index/connect',
+        '/third/callback/[:platform]' => 'third/index/callback',
+        '/third/bind/[:platform]' => 'third/index/bind',
+        '/third/unbind/[:platform]' => 'third/index/unbind',
+    ],
     'priority' => [],
     'domain' => '',
 ];

+ 112 - 0
application/index/controller/Third.php

@@ -0,0 +1,112 @@
+<?php
+
+namespace app\index\controller;
+
+use addons\third\library\Application;
+use app\common\controller\Frontend;
+use think\Lang;
+use think\Session;
+
+/**
+ * 第三方登录控制器
+ */
+class Third extends Frontend
+{
+    protected $noNeedLogin = ['prepare'];
+    protected $noNeedRight = ['*'];
+    protected $app = null;
+    protected $options = [];
+    protected $layout = 'default';
+
+    public function _initialize()
+    {
+        parent::_initialize();
+        $config = get_addon_config('third');
+        $this->app = new Application($config);
+    }
+
+    /**
+     * 准备绑定
+     */
+    public function prepare()
+    {
+        $platform = $this->request->request('platform');
+        $url = $this->request->get('url', '/');
+        if ($this->auth->id) {
+            $this->redirect(url("index/third/bind") . "?" . http_build_query(['platform' => $platform, 'url' => $url]));
+        }
+
+        // 授权成功后的回调
+        $userinfo = Session::get("{$platform}-userinfo");
+        if (!$userinfo) {
+            $this->error("操作失败,请返回重度");
+        }
+
+        Lang::load([
+            APP_PATH . 'index' . DS . 'lang' . DS . $this->request->langset() . DS . 'user' . EXT,
+        ]);
+
+        $this->view->assign('userinfo', $userinfo['userinfo']);
+        $this->view->assign('platform', $platform);
+        $this->view->assign('url', $url);
+        $this->view->assign('bindurl', url("index/third/bind") . '?' . http_build_query(['platform' => $platform, 'url' => $url]));
+        $this->view->assign('captchaType', config('fastadmin.user_register_captcha'));
+        $this->view->assign('title', "账号绑定");
+
+        return $this->view->fetch();
+    }
+
+    /**
+     * 绑定账号
+     */
+    public function bind()
+    {
+        $platform = $this->request->request('platform');
+        $url = $this->request->get('url', $this->request->server('HTTP_REFERER'));
+        if (!$platform) {
+            $this->error("参数不正确");
+        }
+
+        // 授权成功后的回调
+        $userinfo = Session::get("{$platform}-userinfo");
+        if (!$userinfo) {
+            $this->redirect(addon_url('third/index/connect', [':platform' => $platform]) . '?url=' . urlencode($url));
+        }
+        $third = \addons\third\model\Third::where('user_id', $this->auth->id)->where('platform', $platform)->find();
+        if ($third) {
+            $this->error("已绑定账号,请勿重复绑定");
+        }
+        $time = time();
+        $values = [
+            'platform'      => $platform,
+            'user_id'       => $this->auth->id,
+            'openid'        => $userinfo['openid'],
+            'openname'      => isset($userinfo['userinfo']['nickname']) ? $userinfo['userinfo']['nickname'] : '',
+            'access_token'  => $userinfo['access_token'],
+            'refresh_token' => $userinfo['refresh_token'],
+            'expires_in'    => $userinfo['expires_in'],
+            'logintime'     => $time,
+            'expiretime'    => $time + $userinfo['expires_in'],
+        ];
+        $third = \addons\third\model\Third::create($values);
+        if ($third) {
+            $this->success("账号绑定成功", $url);
+        } else {
+            $this->error("账号绑定失败,请重试", $url);
+        }
+    }
+
+    /**
+     * 解绑账号
+     */
+    public function unbind()
+    {
+        $platform = $this->request->request('platform');
+        $third = \addons\third\model\Third::where('user_id', $this->auth->id)->where('platform', $platform)->find();
+        if (!$third) {
+            $this->error("未找到指定的账号绑定信息");
+        }
+        $third->delete();
+        $this->success("账号解绑成功");
+    }
+}

+ 128 - 0
application/index/view/third/prepare.html

@@ -0,0 +1,128 @@
+<div id="content-container" class="container">
+    <div class="text-center">
+        <img src="{$userinfo.avatar}" class="img-circle" width="80" height="80" alt=""/>
+        <div style="margin-top:15px;">{$userinfo.nickname}</div>
+    </div>
+    <div class="user-section login-section" style="margin-top:20px;">
+        <div class="logon-tab clearfix">
+            <a href="javascript:" data-type="bind" class="active">绑定已有账号</a>
+            <a href="javascript:" data-type="register">创建新账号</a>
+        </div>
+        <div class="bind-main login-main">
+            <form name="form" id="bind-form" class="form-vertical" method="POST" action="{:url('user/login')}">
+                {:token()}
+                <input type="hidden" name="platform" value="{$platform}"/>
+                <input type="hidden" name="url" value="{$url}"/>
+                <div class="form-group">
+                    <label class="control-label">{:__('Account')}</label>
+                    <div class="controls">
+                        <input type="text" id="account" name="account" data-rule="required" class="form-control input-lg" placeholder="{:__('Email/Mobile/Username')}">
+                        <p class="help-block"></p>
+                    </div>
+                </div>
+                <div class="form-group">
+                    <label class="control-label">{:__('Password')}</label>
+                    <div class="controls">
+                        <input type="password" id="password" name="password" data-rule="required;password" class="form-control input-lg" placeholder="{:__('Password must be 6 to 30 characters')}">
+                        <p class="help-block"></p>
+                    </div>
+                </div>
+                <div class="form-group">
+                    <button type="submit" class="btn btn-primary btn-block btn-lg">{:__('确认绑定')}</button>
+                </div>
+            </form>
+        </div>
+        <div class="register-main login-main hidden">
+            <form name="form" id="register-form" class="form-vertical" method="POST" action="{:url('user/register')}">
+                {:token()}
+                <input type="hidden" name="platform" value="{$platform}"/>
+                <input type="hidden" name="url" value="{$url}"/>
+                <div class="form-group">
+                    <label class="control-label">{:__('Email')}</label>
+                    <div class="controls">
+                        <input type="text" id="email" name="email" data-rule="required;email" class="form-control input-lg" placeholder="{:__('Email')}">
+                        <p class="help-block"></p>
+                    </div>
+                </div>
+                <div class="form-group">
+                    <label class="control-label">{:__('Username')}</label>
+                    <div class="controls">
+                        <input type="text" id="username" name="username" data-rule="required;length(3~30, true)" class="form-control input-lg" placeholder="{:__('Username must be 3 to 30 characters')}">
+                        <p class="help-block"></p>
+                    </div>
+                </div>
+                <div class="form-group">
+                    <label class="control-label">{:__('Password')}</label>
+                    <div class="controls">
+                        <input type="password" id="password" name="password" data-rule="required;password" class="form-control input-lg" placeholder="{:__('Password must be 6 to 30 characters')}">
+                        <p class="help-block"></p>
+                    </div>
+                </div>
+                <div class="form-group">
+                    <label class="control-label">{:__('Mobile')}</label>
+                    <div class="controls">
+                        <input type="text" id="mobile" name="mobile" data-rule="required;mobile" class="form-control input-lg" placeholder="{:__('Mobile')}">
+                        <p class="help-block"></p>
+                    </div>
+                </div>
+                {if $captchaType}
+                <div class="form-group">
+                    <label class="control-label">{:__('Captcha')}</label>
+                    <div class="controls">
+                        <div class="input-group">
+                            {include file="common/captcha" event="register" type="$captchaType" /}
+                        </div>
+                        <p class="help-block"></p>
+                    </div>
+                </div>
+                {/if}
+                <div class="form-group">
+                    <button type="submit" class="btn btn-primary btn-block btn-lg">{:__('创建账号并绑定')}</button>
+                </div>
+            </form>
+        </div>
+    </div>
+</div>
+
+<script>
+    require.callback = function () {
+        define('frontend/third', ['jquery', 'bootstrap', 'frontend', 'template', 'form'], function ($, undefined, Frontend, Template, Form) {
+            var Controller = {
+                prepare: function () {
+                    var validatoroptions = {
+                        invalid: function (form, errors) {
+                            $.each(errors, function (i, j) {
+                                Layer.msg(j);
+                            });
+                        }
+                    };
+
+                    $(".user-section .logon-tab > a").on("click", function () {
+                        $(".bind-main,.register-main").addClass("hidden");
+                        $("." + $(this).data("type") + "-main").removeClass("hidden");
+                        $(".user-section .logon-tab > a").removeClass("active");
+                        $(this).addClass("active");
+                    });
+
+                    //本地验证未通过时提示
+                    $("#register-form").data("validator-options", validatoroptions);
+
+                    //为表单绑定事件
+                    Form.api.bindevent($("#bind-form"), function (data, ret) {
+                        location.href = "{$bindurl}";
+                        return false;
+                    });
+
+                    //为表单绑定事件
+                    Form.api.bindevent($("#register-form"), function (data, ret) {
+                        location.href = "{$bindurl}";
+                        return false;
+                    }, function (data) {
+                        $("input[name=captcha]").next(".input-group-addon").find("img").trigger("click");
+                    });
+                }
+            };
+            return Controller;
+        });
+    }
+</script>

La diferencia del archivo ha sido suprimido porque es demasiado grande
+ 170 - 736
public/api.html


+ 14 - 1
public/assets/js/addons.js

@@ -1,3 +1,16 @@
 define([], function () {
-    
+    if (Config.modulename === 'index' && Config.controllername === 'user' && ['login', 'register'].indexOf(Config.actionname) > -1 && $("#register-form,#login-form").size() > 0) {
+    $('<style>.social-login{display:flex}.social-login a{flex:1;margin:0 2px;}.social-login a:first-child{margin-left:0;}.social-login a:last-child{margin-right:0;}</style>').appendTo("head");
+    $("#register-form,#login-form").append('<div class="form-group social-login"></div>');
+    if (Config.third.status.indexOf("wechat") > -1) {
+        $('<a class="btn btn-success" href="' + Fast.api.fixurl('/third/connect/wechat') + '"><i class="fa fa-wechat"></i> 微信登录</a>').appendTo(".social-login");
+    }
+    if (Config.third.status.indexOf("qq") > -1) {
+        $('<a class="btn btn-info" href="' + Fast.api.fixurl('/third/connect/qq') + '"><i class="fa fa-qq"></i> QQ登录</a>').appendTo(".social-login");
+    }
+    if (Config.third.status.indexOf("weibo") > -1) {
+        $('<a class="btn btn-danger" href="' + Fast.api.fixurl('/third/connect/weibo') + '"><i class="fa fa-weibo"></i> 微博登录</a>').appendTo(".social-login");
+    }
+}
+
 });

+ 234 - 0
public/assets/js/backend/command.js

@@ -0,0 +1,234 @@
+define(['jquery', 'bootstrap', 'backend', 'table', 'form', 'template'], function ($, undefined, Backend, Table, Form, Template) {
+
+    var Controller = {
+        index: function () {
+            // 初始化表格参数配置
+            Table.api.init({
+                extend: {
+                    index_url: 'command/index',
+                    add_url: 'command/add',
+                    edit_url: '',
+                    del_url: 'command/del',
+                    multi_url: 'command/multi',
+                    table: 'command',
+                }
+            });
+
+            var table = $("#table");
+
+            // 初始化表格
+            table.bootstrapTable({
+                url: $.fn.bootstrapTable.defaults.extend.index_url,
+                pk: 'id',
+                sortName: 'id',
+                columns: [
+                    [
+                        {checkbox: true},
+                        {field: 'id', title: __('Id')},
+                        {field: 'type', title: __('Type'), formatter: Table.api.formatter.search},
+                        {field: 'type_text', title: __('Type')},
+                        {
+                            field: 'command', title: __('Command'), formatter: function (value, row, index) {
+                                return '<input type="text" class="form-control" value="' + value + '">';
+                            }
+                        },
+                        {
+                            field: 'executetime',
+                            title: __('Executetime'),
+                            operate: 'RANGE',
+                            addclass: 'datetimerange',
+                            formatter: Table.api.formatter.datetime
+                        },
+                        {
+                            field: 'createtime',
+                            title: __('Createtime'),
+                            operate: 'RANGE',
+                            addclass: 'datetimerange',
+                            formatter: Table.api.formatter.datetime
+                        },
+                        {
+                            field: 'updatetime',
+                            title: __('Updatetime'),
+                            operate: 'RANGE',
+                            addclass: 'datetimerange',
+                            formatter: Table.api.formatter.datetime
+                        },
+                        {
+                            field: 'status',
+                            title: __('Status'),
+                            table: table,
+                            custom: {"successed": 'success', "failured": 'danger'},
+                            searchList: {"successed": __('Successed'), "failured": __('Failured')},
+                            formatter: Table.api.formatter.status
+                        },
+                        {
+                            field: 'operate',
+                            title: __('Operate'),
+                            buttons: [
+                                {
+                                    name: 'execute',
+                                    title: __('Execute again'),
+                                    text: __('Execute again'),
+                                    url: 'command/execute',
+                                    icon: 'fa fa-repeat',
+                                    classname: 'btn btn-success btn-xs btn-execute btn-ajax',
+                                    success: function (data) {
+                                        Layer.alert("<textarea class='form-control' cols='60' rows='5'>" + data.result + "</textarea>", {
+                                            title: __("执行结果"),
+                                            shadeClose: true
+                                        });
+                                        table.bootstrapTable('refresh');
+                                        return false;
+                                    }
+                                },
+                                {
+                                    name: 'execute',
+                                    title: __('Detail'),
+                                    text: __('Detail'),
+                                    url: 'command/detail',
+                                    icon: 'fa fa-list',
+                                    classname: 'btn btn-info btn-xs btn-execute btn-dialog'
+                                }
+                            ],
+                            table: table,
+                            events: Table.api.events.operate,
+                            formatter: Table.api.formatter.operate
+                        }
+                    ]
+                ]
+            });
+
+            // 为表格绑定事件
+            Table.api.bindevent(table);
+        },
+        add: function () {
+            require(['bootstrap-select', 'bootstrap-select-lang']);
+            var mainfields = [];
+            var relationfields = {};
+            var maintable = [];
+            var relationtable = [];
+            var relationmode = ["belongsto", "hasone"];
+
+            var renderselect = function (select, data) {
+                var html = [];
+                for (var i = 0; i < data.length; i++) {
+                    html.push("<option value='" + data[i] + "'>" + data[i] + "</option>");
+                }
+                $(select).html(html.join(""));
+                select.trigger("change");
+                if (select.data("selectpicker")) {
+                    select.selectpicker('refresh');
+                }
+                return select;
+            };
+
+            $("select[name=table] option").each(function () {
+                maintable.push($(this).val());
+            });
+            $(document).on('change', "input[name='isrelation']", function () {
+                $("#relation-zone").toggleClass("hide", !$(this).prop("checked"));
+            });
+            $(document).on('change', "select[name='table']", function () {
+                var that = this;
+                Fast.api.ajax({
+                    url: "command/get_field_list",
+                    data: {table: $(that).val()},
+                }, function (data, ret) {
+                    mainfields = data.fieldlist;
+                    $("#relation-zone .relation-item").remove();
+                    renderselect($("#fields"), mainfields);
+                    return false;
+                });
+                return false;
+            });
+            $(document).on('click', "a.btn-newrelation", function () {
+                var that = this;
+                var index = parseInt($(that).data("index")) + 1;
+                var content = Template("relationtpl", {index: index});
+                content = $(content.replace(/\[index\]/, index));
+                $(this).data("index", index);
+                $(content).insertBefore($(that).closest(".row"));
+                $('select', content).selectpicker();
+                var exists = [$("select[name='table']").val()];
+                $("select.relationtable").each(function () {
+                    exists.push($(this).val());
+                });
+                relationtable = [];
+                $.each(maintable, function (i, j) {
+                    if ($.inArray(j, exists) < 0) {
+                        relationtable.push(j);
+                    }
+                });
+                renderselect($("select.relationtable", content), relationtable);
+                $("select.relationtable", content).trigger("change");
+            });
+            $(document).on('click', "a.btn-removerelation", function () {
+                $(this).closest(".row").remove();
+            });
+            $(document).on('change', "#relation-zone select.relationmode", function () {
+                var table = $("select.relationtable", $(this).closest(".row")).val();
+                var that = this;
+                Fast.api.ajax({
+                    url: "command/get_field_list",
+                    data: {table: table},
+                }, function (data, ret) {
+                    renderselect($(that).closest(".row").find("select.relationprimarykey"), $(that).val() == 'belongsto' ? data.fieldlist : mainfields);
+                    renderselect($(that).closest(".row").find("select.relationforeignkey"), $(that).val() == 'hasone' ? data.fieldlist : mainfields);
+                    return false;
+                });
+            });
+            $(document).on('change', "#relation-zone select.relationtable", function () {
+                var that = this;
+                Fast.api.ajax({
+                    url: "command/get_field_list",
+                    data: {table: $(that).val()},
+                }, function (data, ret) {
+                    renderselect($(that).closest(".row").find("select.relationmode"), relationmode);
+                    renderselect($(that).closest(".row").find("select.relationfields"), mainfields)
+                    renderselect($(that).closest(".row").find("select.relationforeignkey"), data.fieldlist)
+                    renderselect($(that).closest(".row").find("select.relationfields"), data.fieldlist)
+                    return false;
+                });
+            });
+            $(document).on('click', ".btn-command", function () {
+                var form = $(this).closest("form");
+                var textarea = $("textarea[rel=command]", form);
+                textarea.val('');
+                Fast.api.ajax({
+                    url: "command/command/action/command",
+                    data: form.serialize(),
+                }, function (data, ret) {
+                    textarea.val(data.command);
+                    return false;
+                });
+            });
+            $(document).on('click', ".btn-execute", function () {
+                var form = $(this).closest("form");
+                var textarea = $("textarea[rel=result]", form);
+                textarea.val('');
+                Fast.api.ajax({
+                    url: "command/command/action/execute",
+                    data: form.serialize(),
+                }, function (data, ret) {
+                    textarea.val(data.result);
+                    window.parent.$(".toolbar .btn-refresh").trigger('click');
+                    top.window.Fast.api.refreshmenu();
+                    return false;
+                }, function () {
+                    window.parent.$(".toolbar .btn-refresh").trigger('click');
+                });
+            });
+            $("select[name='table']").trigger("change");
+            Controller.api.bindevent();
+        },
+        edit: function () {
+            Controller.api.bindevent();
+        },
+        api: {
+            bindevent: function () {
+                Form.api.bindevent($("form[role=form]"));
+            }
+        }
+    };
+    return Controller;
+});

+ 58 - 0
public/assets/js/backend/third.js

@@ -0,0 +1,58 @@
+define(['jquery', 'bootstrap', 'backend', 'table', 'form'], function ($, undefined, Backend, Table, Form) {
+
+    var Controller = {
+        index: function () {
+            // 初始化表格参数配置
+            Table.api.init({
+                extend: {
+                    index_url: 'third/index' + location.search,
+                    add_url: 'third/add',
+                    edit_url: 'third/edit',
+                    del_url: 'third/del',
+                    multi_url: 'third/multi',
+                    table: 'third',
+                }
+            });
+
+            var table = $("#table");
+
+            // 初始化表格
+            table.bootstrapTable({
+                url: $.fn.bootstrapTable.defaults.extend.index_url,
+                pk: 'id',
+                sortName: 'id',
+                columns: [
+                    [
+                        {checkbox: true},
+                        {field: 'id', title: __('Id')},
+                        {field: 'user_id', title: __('User_id'), formatter: Table.api.formatter.search},
+                        {field: 'user.nickname', title: __('Nickname')},
+                        {field: 'platform', title: __('Platform'), formatter: Table.api.formatter.search},
+                        {field: 'unionid', title: __('Unionid')},
+                        {field: 'openid', title: __('Openid')},
+                        {field: 'openname', title: __('Openname')},
+                        {field: 'createtime', title: __('Createtime'), operate: 'RANGE', addclass: 'datetimerange', formatter: Table.api.formatter.datetime},
+                        {field: 'updatetime', title: __('Updatetime'), operate: 'RANGE', addclass: 'datetimerange', formatter: Table.api.formatter.datetime},
+                        {field: 'logintime', title: __('Logintime'), operate: 'RANGE', addclass: 'datetimerange', formatter: Table.api.formatter.datetime},
+                        {field: 'operate', title: __('Operate'), table: table, events: Table.api.events.operate, formatter: Table.api.formatter.operate}
+                    ]
+                ]
+            });
+
+            // 为表格绑定事件
+            Table.api.bindevent(table);
+        },
+        add: function () {
+            Controller.api.bindevent();
+        },
+        edit: function () {
+            Controller.api.bindevent();
+        },
+        api: {
+            bindevent: function () {
+                Form.api.bindevent($("form[role=form]"));
+            }
+        }
+    };
+    return Controller;
+});

Algunos archivos no se mostraron porque demasiados archivos cambiaron en este cambio