From 3b4ac3e525bdc51cf5176d3d52cd01d0a9c67d7d Mon Sep 17 00:00:00 2001 From: 三个三 <2029364173@qq.com> Date: 星期二, 27 六月 2023 08:33:06 +0800 Subject: [PATCH] !16 扩展第三方登录授权功能 * add 第三方授权 * add 第三方授权登录 --- src/views/login.vue | 230 ++++++++++++++++++++++++++++++++------------------------- 1 files changed, 130 insertions(+), 100 deletions(-) diff --git a/src/views/login.vue b/src/views/login.vue index e8f8cb7..af98f01 100644 --- a/src/views/login.vue +++ b/src/views/login.vue @@ -1,65 +1,34 @@ <template> <div class="login"> <el-form ref="loginRef" :model="loginForm" :rules="loginRules" class="login-form"> - <h3 class="title">RuoYi-Vue-Plus鍚庡彴绠$悊绯荤粺</h3> - <el-form-item prop="tenantId"> + <h3 class="title">RuoYi-Vue-Plus澶氱鎴风鐞嗙郴缁�</h3> + <el-form-item prop="tenantId" v-if="tenantEnabled"> <el-select v-model="loginForm.tenantId" filterable placeholder="璇烽�夋嫨/杈撳叆鍏徃鍚嶇О" style="width: 100%"> - <el-option - v-for="item in tenantList" - :key="item.tenantId" - :label="item.companyName" - :value="item.tenantId"> - </el-option> + <el-option v-for="item in tenantList" :key="item.tenantId" :label="item.companyName" :value="item.tenantId"></el-option> <template #prefix><svg-icon icon-class="company" class="el-input__icon input-icon" /></template> </el-select> </el-form-item> <el-form-item prop="username"> - <el-input - v-model="loginForm.username" - type="text" - size="large" - auto-complete="off" - placeholder="璐﹀彿" - > + <el-input v-model="loginForm.username" type="text" size="large" auto-complete="off" placeholder="璐﹀彿"> <template #prefix><svg-icon icon-class="user" class="el-input__icon input-icon" /></template> </el-input> </el-form-item> <el-form-item prop="password"> - <el-input - v-model="loginForm.password" - type="password" - size="large" - auto-complete="off" - placeholder="瀵嗙爜" - @keyup.enter="handleLogin" - > + <el-input v-model="loginForm.password" type="password" size="large" auto-complete="off" placeholder="瀵嗙爜" @keyup.enter="handleLogin"> <template #prefix><svg-icon icon-class="password" class="el-input__icon input-icon" /></template> </el-input> </el-form-item> <el-form-item prop="code" v-if="captchaEnabled"> - <el-input - v-model="loginForm.code" - size="large" - auto-complete="off" - placeholder="楠岃瘉鐮�" - style="width: 63%" - @keyup.enter="handleLogin" - > + <el-input v-model="loginForm.code" size="large" auto-complete="off" placeholder="楠岃瘉鐮�" style="width: 63%" @keyup.enter="handleLogin"> <template #prefix><svg-icon icon-class="validCode" class="el-input__icon input-icon" /></template> </el-input> <div class="login-code"> - <img :src="codeUrl" @click="getCode" class="login-code-img"/> + <img :src="codeUrl" @click="getCode" class="login-code-img" /> </div> </el-form-item> <el-checkbox v-model="loginForm.rememberMe" style="margin:0px 0px 25px 0px;">璁颁綇瀵嗙爜</el-checkbox> <el-form-item style="width:100%;"> - <el-button - :loading="loading" - size="large" - type="primary" - style="width:100%;" - @click.prevent="handleLogin" - > + <el-button :loading="loading" size="large" type="primary" style="width:100%;" @click.prevent="handleLogin"> <span v-if="!loading">鐧� 褰�</span> <span v-else>鐧� 褰� 涓�...</span> </el-button> @@ -67,119 +36,171 @@ <router-link class="link-type" :to="'/register'">绔嬪嵆娉ㄥ唽</router-link> </div> </el-form-item> + <div style="display: flex;justify-content: flex-end;flex-direction: row;"> + <el-button circle> + <svg-icon icon-class="qq" @click="doSocialLogin('QQ')" /> + </el-button> + <el-button circle> + <svg-icon icon-class="wechat" @click="doSocialLogin('Wechat')" /> + </el-button> + <el-button circle> + <svg-icon icon-class="gitee" @click="doSocialLogin('gitee')" /> + </el-button> + <el-button circle> + <svg-icon icon-class="github" @click="doSocialLogin('github')" /> + </el-button> + </div> </el-form> <!-- 搴曢儴 --> <div class="el-login-footer"> - <span>Copyright 漏 2018-2023 ruoyi.vip All Rights Reserved.</span> + <span>Copyright 漏 2018-2023 鐤媯鐨勭嫯瀛怢i All Rights Reserved.</span> </div> </div> </template> -<script setup> -import { getCodeImg, getTenantList } from "@/api/login"; -import Cookies from "js-cookie"; -import { encrypt, decrypt } from "@/utils/jsencrypt"; -import useUserStore from '@/store/modules/user' +<script setup lang="ts"> +import { getCodeImg, getTenantList } from '@/api/login'; +import { authBinding } from '@/api/system/social/auth'; +import Cookies from 'js-cookie'; +import { encrypt, decrypt } from '@/utils/jsencrypt'; +import { useUserStore } from '@/store/modules/user'; +import { LoginData, TenantVO } from '@/api/types'; +import { ElForm, FormRules } from 'element-plus'; +import { to } from 'await-to-js'; -const userStore = useUserStore() +const userStore = useUserStore(); const router = useRouter(); -const { proxy } = getCurrentInstance(); -const loginForm = ref({ +const loginForm = ref<LoginData>({ tenantId: "000000", - username: "admin", - password: "admin123", + username: 'admin', + password: 'admin123', rememberMe: false, - code: "", - uuid: "" + code: '', + uuid: '' }); -const loginRules = { +const loginRules: FormRules = { tenantId: [{ required: true, trigger: "blur", message: "璇疯緭鍏ユ偍鐨勭鎴风紪鍙�" }], - username: [{ required: true, trigger: "blur", message: "璇疯緭鍏ユ偍鐨勮处鍙�" }], - password: [{ required: true, trigger: "blur", message: "璇疯緭鍏ユ偍鐨勫瘑鐮�" }], - code: [{ required: true, trigger: "change", message: "璇疯緭鍏ラ獙璇佺爜" }] + username: [{ required: true, trigger: 'blur', message: '璇疯緭鍏ユ偍鐨勮处鍙�' }], + password: [{ required: true, trigger: 'blur', message: '璇疯緭鍏ユ偍鐨勫瘑鐮�' }], + code: [{ required: true, trigger: 'change', message: '璇疯緭鍏ラ獙璇佺爜' }] }; -const codeUrl = ref(""); +const codeUrl = ref(''); const loading = ref(false); // 楠岃瘉鐮佸紑鍏� const captchaEnabled = ref(true); +// 绉熸埛寮�鍏� +const tenantEnabled = ref(true); + + // 娉ㄥ唽寮�鍏� const register = ref(false); const redirect = ref(undefined); +const loginRef = ref(ElForm); // 绉熸埛鍒楄〃 -const tenantList = ref([]); +const tenantList = ref<TenantVO[]>([]); -function handleLogin() { - proxy.$refs.loginRef.validate(valid => { +const handleLogin = () => { + loginRef.value.validate(async (valid: boolean, fields: any) => { if (valid) { loading.value = true; // 鍕鹃�変簡闇�瑕佽浣忓瘑鐮佽缃湪 cookie 涓缃浣忕敤鎴峰悕鍜屽瘑鐮� if (loginForm.value.rememberMe) { Cookies.set("tenantId", loginForm.value.tenantId, { expires: 30 }); - Cookies.set("username", loginForm.value.username, { expires: 30 }); - Cookies.set("password", encrypt(loginForm.value.password), { expires: 30 }); - Cookies.set("rememberMe", loginForm.value.rememberMe, { expires: 30 }); + Cookies.set('username', loginForm.value.username, { expires: 30 }); + Cookies.set('password', String(encrypt(loginForm.value.password)), { expires: 30 }); + Cookies.set('rememberMe', String(loginForm.value.rememberMe), { expires: 30 }); } else { // 鍚﹀垯绉婚櫎 Cookies.remove("tenantId"); - Cookies.remove("username"); - Cookies.remove("password"); - Cookies.remove("rememberMe"); + Cookies.remove('username'); + Cookies.remove('password'); + Cookies.remove('rememberMe'); } // 璋冪敤action鐨勭櫥褰曟柟娉� - userStore.login(loginForm.value).then(() => { - router.push({ path: redirect.value || "/" }); - }).catch(() => { + // prittier-ignore + const [err] = await to(userStore.login(loginForm.value)); + if (!err) { + await router.push({ path: redirect.value || '/' }); + } else { loading.value = false; // 閲嶆柊鑾峰彇楠岃瘉鐮� if (captchaEnabled.value) { - getCode(); + await getCode(); } - }); + } + } else { + console.log('error submit!', fields); } }); -} +}; -function getCode() { - getCodeImg().then(res => { - captchaEnabled.value = res.data.captchaEnabled === undefined ? true : res.data.captchaEnabled; - if (captchaEnabled.value) { - codeUrl.value = "data:image/gif;base64," + res.data.img; - loginForm.value.uuid = res.data.uuid; - } - }); -} +/** + * 鑾峰彇楠岃瘉鐮� + */ +const getCode = async () => { + const res = await getCodeImg(); + const { data } = res; + captchaEnabled.value = data.captchaEnabled === undefined ? true : data.captchaEnabled; + if (captchaEnabled.value) { + codeUrl.value = 'data:image/gif;base64,' + data.img; + loginForm.value.uuid = data.uuid; + } +}; -function initTenantList() { - getTenantList().then(res => { - tenantList.value = res.data; - if (tenantList.value != null && tenantList.value.length !== 0) { - loginForm.value.tenantId = tenantList.value[0].tenantId; - } - }); -} - -function getCookie() { +const getCookie = () => { const tenantId = Cookies.get("tenantId"); - const username = Cookies.get("username"); - const password = Cookies.get("password"); - const rememberMe = Cookies.get("rememberMe"); + const username = Cookies.get('username'); + const password = Cookies.get('password'); + const rememberMe = Cookies.get('rememberMe'); loginForm.value = { tenantId: tenantId === undefined ? loginForm.value.tenantId : tenantId, username: username === undefined ? loginForm.value.username : username, - password: password === undefined ? loginForm.value.password : decrypt(password), + password: password === undefined ? loginForm.value.password : (decrypt(password) as string), rememberMe: rememberMe === undefined ? false : Boolean(rememberMe) }; } -getCode(); -initTenantList(); -getCookie(); + +/** + * 鑾峰彇绉熸埛鍒楄〃 + */ +const initTenantList = async () => { + const { data } = await getTenantList(); + tenantEnabled.value = data.tenantEnabled === undefined ? true : data.tenantEnabled; + if (tenantEnabled.value) { + tenantList.value = data.voList; + if (tenantList.value != null && tenantList.value.length !== 0) { + loginForm.value.tenantId = tenantList.value[0].tenantId; + } + } +} +/** + * 绗笁鏂圭櫥褰� + * @param type + */ +const doSocialLogin = (type: string) => { + authBinding(type).then((res: any) => { + if (res.code === 200) { + window.location.href = res.msg; + } else { + ElMessage.error(res.msg); + } + }); +}; + + + +onMounted(() => { + getCode(); + initTenantList(); + getCookie(); +}); </script> -<style lang='scss' scoped> +<style lang="scss" scoped> .login { display: flex; justify-content: center; @@ -188,6 +209,7 @@ background-image: url("../assets/images/login-background.jpg"); background-size: cover; } + .title { margin: 0px auto 30px auto; text-align: center; @@ -199,32 +221,39 @@ background: #ffffff; width: 400px; padding: 25px 25px 5px 25px; + .el-input { height: 40px; + input { height: 40px; } } + .input-icon { height: 39px; width: 14px; margin-left: 0px; } } + .login-tip { font-size: 13px; text-align: center; color: #bfbfbf; } + .login-code { width: 33%; height: 40px; float: right; + img { cursor: pointer; vertical-align: middle; } } + .el-login-footer { height: 40px; line-height: 40px; @@ -233,10 +262,11 @@ width: 100%; text-align: center; color: #fff; - font-family: Arial; + font-family: Arial, serif; font-size: 12px; letter-spacing: 1px; } + .login-code-img { height: 40px; padding-left: 12px; -- Gitblit v1.9.3