From de59371f50991a0dbee997eb4a13fd3f5f415ffd Mon Sep 17 00:00:00 2001 From: baoshiwei <baoshiwei@shlanbao.cn> Date: 星期五, 21 三月 2025 09:45:21 +0800 Subject: [PATCH] feat(login): 添加 Keycloak 登录支持 --- src/views/login.vue | 284 +++++++++++++++++++++++++++++++++++--------------------- 1 files changed, 175 insertions(+), 109 deletions(-) diff --git a/src/views/login.vue b/src/views/login.vue index e8f8cb7..50d0cb7 100644 --- a/src/views/login.vue +++ b/src/views/login.vue @@ -1,26 +1,18 @@ <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"> - <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> + <div class="title-box"> + <h3 class="title">鍏板疂杞﹂棿璐ㄩ噺绠$悊绯荤粺</h3> + <lang-select /> + </div> + <el-form-item v-if="tenantEnabled" prop="tenantId"> + <el-select v-model="loginForm.tenantId" filterable :placeholder="proxy.$t('login.selectPlaceholder')" style="width: 100%"> + <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="proxy.$t('login.username')"> <template #prefix><svg-icon icon-class="user" class="el-input__icon input-icon" /></template> </el-input> </el-form-item> @@ -30,168 +22,234 @@ type="password" size="large" auto-complete="off" - placeholder="瀵嗙爜" + :placeholder="proxy.$t('login.password')" @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-form-item v-if="captchaEnabled" prop="code"> <el-input v-model="loginForm.code" size="large" auto-complete="off" - placeholder="楠岃瘉鐮�" + :placeholder="proxy.$t('login.code')" 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" class="login-code-img" @click="getCode" /> </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" - > - <span v-if="!loading">鐧� 褰�</span> - <span v-else>鐧� 褰� 涓�...</span> + <el-checkbox v-model="loginForm.rememberMe" style="margin: 0 0 25px 0">{{ proxy.$t('login.rememberPassword') }}</el-checkbox> + <el-form-item style="float: right"> +<!-- <el-button circle :title="proxy.$t('login.social.wechat')" @click="doSocialLogin('wechat')">--> +<!-- <svg-icon icon-class="wechat" />--> +<!-- </el-button>--> + <el-button circle :title="proxy.$t('login.social.keycloak')" @click="doSocialLogin('keycloak')"> + <svg-icon icon-class="keycloak" /> </el-button> - <div style="float: right;" v-if="register"> - <router-link class="link-type" :to="'/register'">绔嬪嵆娉ㄥ唽</router-link> +<!-- <el-button circle :title="proxy.$t('login.social.topiam')" @click="doSocialLogin('topiam')">--> +<!-- <svg-icon icon-class="topiam" />--> +<!-- </el-button>--> +<!-- <el-button circle :title="proxy.$t('login.social.gitee')" @click="doSocialLogin('gitee')">--> +<!-- <svg-icon icon-class="gitee" />--> +<!-- </el-button>--> +<!-- <el-button circle :title="proxy.$t('login.social.github')" @click="doSocialLogin('github')">--> +<!-- <svg-icon icon-class="github" />--> +<!-- </el-button>--> + </el-form-item> + <el-form-item style="width: 100%"> + <el-button :loading="loading" size="large" type="primary" style="width: 100%" @click.prevent="handleLogin"> + <span v-if="!loading">{{ proxy.$t('login.login') }}</span> + <span v-else>{{ proxy.$t('login.logging') }}</span> + </el-button> + <div v-if="register" style="float: right"> + <router-link class="link-type" :to="'/register'">{{ proxy.$t('login.switchRegisterPage') }}</router-link> </div> </el-form-item> </el-form> <!-- 搴曢儴 --> <div class="el-login-footer"> - <span>Copyright 漏 2018-2023 ruoyi.vip All Rights Reserved.</span> + <span>Copyright 漏 2018-2024 涓婃捣鍏版郸鏅鸿兘绉戞妧鏈夐檺鍏徃 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 { useUserStore } from '@/store/modules/user'; +import { LoginData, TenantVO } from '@/api/types'; +import { to } from 'await-to-js'; +import { HttpStatus } from '@/enums/RespEnum'; +import { useI18n } from 'vue-i18n'; -const userStore = useUserStore() +const { proxy } = getCurrentInstance() as ComponentInternalInstance; + +const userStore = useUserStore(); const router = useRouter(); -const { proxy } = getCurrentInstance(); +const { t } = useI18n(); -const loginForm = ref({ - tenantId: "000000", - username: "admin", - password: "admin123", +const loginForm = ref<LoginData>({ + tenantId: '000000', + username: '', + password: '', rememberMe: false, - code: "", - uuid: "" -}); + code: '', + uuid: '' +} as LoginData); -const loginRules = { - tenantId: [{ required: true, trigger: "blur", message: "璇疯緭鍏ユ偍鐨勭鎴风紪鍙�" }], - username: [{ required: true, trigger: "blur", message: "璇疯緭鍏ユ偍鐨勮处鍙�" }], - password: [{ required: true, trigger: "blur", message: "璇疯緭鍏ユ偍鐨勫瘑鐮�" }], - code: [{ required: true, trigger: "change", message: "璇疯緭鍏ラ獙璇佺爜" }] +const loginRules: ElFormRules = { + tenantId: [{ required: true, trigger: 'blur', message: t('login.rule.tenantId.required') }], + username: [{ required: true, trigger: 'blur', message: t('login.rule.username.required') }], + password: [{ required: true, trigger: 'blur', message: t('login.rule.password.required') }], + code: [{ required: true, trigger: 'change', message: t('login.rule.code.required') }] }; -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 redirect = ref('/'); +const loginRef = ref<ElFormInstance>(); // 绉熸埛鍒楄〃 -const tenantList = ref([]); +const tenantList = ref<TenantVO[]>([]); -function handleLogin() { - proxy.$refs.loginRef.validate(valid => { +watch( + () => router.currentRoute.value, + (newRoute: any) => { + redirect.value = newRoute.query && newRoute.query.redirect && decodeURIComponent(newRoute.query.redirect); + }, + { immediate: true } +); + +const handleLogin = () => { + loginRef.value?.validate(async (valid: boolean, fields: any) => { if (valid) { loading.value = true; - // 鍕鹃�変簡闇�瑕佽浣忓瘑鐮佽缃湪 cookie 涓缃浣忕敤鎴峰悕鍜屽瘑鐮� + // 鍕鹃�変簡闇�瑕佽浣忓瘑鐮佽缃湪 localStorage 涓缃浣忕敤鎴峰悕鍜屽瘑鐮� 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 }); + localStorage.setItem('tenantId', String(loginForm.value.tenantId)); + localStorage.setItem('username', String(loginForm.value.username)); + localStorage.setItem('password', String(loginForm.value.password)); + localStorage.setItem('rememberMe', String(loginForm.value.rememberMe)); } else { // 鍚﹀垯绉婚櫎 - Cookies.remove("tenantId"); - Cookies.remove("username"); - Cookies.remove("password"); - Cookies.remove("rememberMe"); + localStorage.removeItem('tenantId'); + localStorage.removeItem('username'); + localStorage.removeItem('password'); + localStorage.removeItem('rememberMe'); } // 璋冪敤action鐨勭櫥褰曟柟娉� - userStore.login(loginForm.value).then(() => { - router.push({ path: redirect.value || "/" }); - }).catch(() => { + const [err] = await to(userStore.login(loginForm.value)); + if (!err) { + const redirectUrl = redirect.value || '/'; + await router.push(redirectUrl); + loading.value = false; + } 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; +const getLoginData = () => { + const tenantId = localStorage.getItem('tenantId'); + const username = localStorage.getItem('username'); + const password = localStorage.getItem('password'); + const rememberMe = localStorage.getItem('rememberMe'); + loginForm.value = { + tenantId: tenantId === null ? String(loginForm.value.tenantId) : tenantId, + username: username === null ? String(loginForm.value.username) : username, + password: password === null ? String(loginForm.value.password) : String(password), + rememberMe: rememberMe === null ? false : Boolean(rememberMe) + } as LoginData; +}; + +/** + * 鑾峰彇绉熸埛鍒楄〃 + */ +const initTenantList = async () => { + const { data } = await getTenantList(false); + 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, loginForm.value.tenantId).then((res: any) => { + if (res.code === HttpStatus.SUCCESS) { + // 鑾峰彇鎺堟潈鍦板潃璺宠浆 + window.location.href = res.data; + } else { + ElMessage.error(res.msg); + } }); -} +}; -function getCookie() { - const tenantId = Cookies.get("tenantId"); - 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), - rememberMe: rememberMe === undefined ? false : Boolean(rememberMe) - }; -} - -getCode(); -initTenantList(); -getCookie(); +onMounted(() => { + doSocialLogin('keycloak') +}); </script> -<style lang='scss' scoped> +<style lang="scss" scoped> .login { display: flex; justify-content: center; align-items: center; height: 100%; - background-image: url("../assets/images/login-background.jpg"); + background-image: url('../assets/images/login-background.jpg'); background-size: cover; } -.title { - margin: 0px auto 30px auto; - text-align: center; - color: #707070; + +.title-box { + display: flex; + + .title { + margin: 0px auto 30px auto; + text-align: center; + color: #707070; + } + + :deep(.lang-select--style) { + line-height: 0; + color: #7483a3; + } } .login-form { @@ -199,32 +257,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 +298,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