account.vue 5.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178
  1. <template>
  2. <el-form size="large" class="login-content-form">
  3. <el-form-item class="login-animation1">
  4. <el-input text :placeholder="$t('message.account.accountPlaceholder1')" v-model="state.ruleForm.userName" clearable autocomplete="off">
  5. <template #prefix>
  6. <el-icon class="el-input__icon"><ele-User /></el-icon>
  7. </template>
  8. </el-input>
  9. </el-form-item>
  10. <el-form-item class="login-animation2">
  11. <el-input
  12. :type="state.isShowPassword ? 'text' : 'password'"
  13. :placeholder="$t('message.account.accountPlaceholder2')"
  14. v-model="state.ruleForm.password"
  15. autocomplete="off"
  16. >
  17. <template #prefix>
  18. <el-icon class="el-input__icon"><ele-Unlock /></el-icon>
  19. </template>
  20. <template #suffix>
  21. <i
  22. class="iconfont el-input__icon login-content-password"
  23. :class="state.isShowPassword ? 'icon-yincangmima' : 'icon-xianshimima'"
  24. @click="state.isShowPassword = !state.isShowPassword"
  25. >
  26. </i>
  27. </template>
  28. </el-input>
  29. </el-form-item>
  30. <!-- <el-form-item class="login-animation3">
  31. <el-col :span="15">
  32. <el-input
  33. text
  34. maxlength="4"
  35. :placeholder="$t('message.account.accountPlaceholder3')"
  36. v-model="state.ruleForm.code"
  37. clearable
  38. autocomplete="off"
  39. >
  40. <template #prefix>
  41. <el-icon class="el-input__icon"><ele-Position /></el-icon>
  42. </template>
  43. </el-input>
  44. </el-col>
  45. <el-col :span="1"></el-col>
  46. <el-col :span="8">
  47. <el-button class="login-content-code" v-waves>1234</el-button>
  48. </el-col>
  49. </el-form-item> -->
  50. <el-form-item class="login-animation4">
  51. <el-button type="primary" class="login-content-submit" round v-waves @click="onSignIn" :loading="state.loading.signIn">
  52. <span>{{ $t('message.account.accountBtnText') }}</span>
  53. </el-button>
  54. </el-form-item>
  55. </el-form>
  56. </template>
  57. <script setup lang="ts" name="loginAccount">
  58. import { reactive, computed } from 'vue'
  59. import { useRoute, useRouter } from 'vue-router'
  60. import { ElMessage } from 'element-plus'
  61. import { useI18n } from 'vue-i18n'
  62. // import Cookies from 'js-cookie'
  63. // import { storeToRefs } from 'pinia'
  64. // import { useThemeConfig } from '/@/stores/themeConfig'
  65. // import { initFrontEndControlRoutes } from '/@/router/frontEnd'
  66. import { initBackEndControlRoutes } from '/@/router/backEnd'
  67. import { Local, Session } from '/@/utils/storage'
  68. import { formatAxis } from '/@/utils/formatTime'
  69. import { NextLoading } from '/@/utils/loading'
  70. import { AuthApi } from '/@/api/admin/Auth'
  71. import { AuthLoginInput } from '/@/api/admin/data-contracts'
  72. import { adminTokenKey } from '/@/api/admin/http-client'
  73. // 定义变量内容
  74. const { t } = useI18n()
  75. // const storesThemeConfig = useThemeConfig()
  76. // const { themeConfig } = storeToRefs(storesThemeConfig)
  77. const route = useRoute()
  78. const router = useRouter()
  79. const state = reactive({
  80. isShowPassword: false,
  81. ruleForm: {
  82. userName: 'admin',
  83. password: '111111',
  84. //code: '1234',
  85. } as AuthLoginInput,
  86. loading: {
  87. signIn: false,
  88. },
  89. })
  90. // 时间获取
  91. const currentTime = computed(() => {
  92. return formatAxis(new Date())
  93. })
  94. // 登录
  95. const onSignIn = async () => {
  96. state.loading.signIn = true
  97. const res = await new AuthApi().login(state.ruleForm).catch(() => {
  98. state.loading.signIn = false
  99. })
  100. if (!res?.success) {
  101. state.loading.signIn = false
  102. return
  103. }
  104. const token = res.data?.token
  105. Local.set(adminTokenKey, token)
  106. Session.set('token', token)
  107. // 添加完动态路由,再进行 router 跳转,否则可能报错 No match found for location with path "/"
  108. const isNoPower = await initBackEndControlRoutes()
  109. // 执行完 initBackEndControlRoutes,再执行 signInSuccess
  110. signInSuccess(isNoPower)
  111. }
  112. // 登录成功后的跳转
  113. const signInSuccess = (isNoPower: boolean | undefined) => {
  114. if (isNoPower) {
  115. ElMessage.warning('抱歉,您没有分配权限,请联系管理员')
  116. Local.remove(adminTokenKey)
  117. Session.clear()
  118. } else {
  119. // 初始化登录成功时间问候语
  120. let currentTimeInfo = currentTime.value
  121. // 登录成功,跳到转首页
  122. // 如果是复制粘贴的路径,非首页/登录页,那么登录成功后重定向到对应的路径中
  123. if (route.query?.redirect) {
  124. router.push({
  125. path: <string>route.query?.redirect,
  126. query: Object.keys(<string>route.query?.params).length > 0 ? JSON.parse(<string>route.query?.params) : '',
  127. })
  128. } else {
  129. router.push('/')
  130. }
  131. // 登录成功提示
  132. const signInText = t('message.signInText')
  133. ElMessage.success(`${currentTimeInfo},${signInText}`)
  134. // 添加 loading,防止第一次进入界面时出现短暂空白
  135. NextLoading.start()
  136. }
  137. state.loading.signIn = false
  138. }
  139. </script>
  140. <style scoped lang="scss">
  141. .login-content-form {
  142. margin-top: 20px;
  143. @for $i from 1 through 4 {
  144. .login-animation#{$i} {
  145. opacity: 0;
  146. animation-name: error-num;
  147. animation-duration: 0.5s;
  148. animation-fill-mode: forwards;
  149. animation-delay: calc($i/10) + s;
  150. }
  151. }
  152. .login-content-password {
  153. display: inline-block;
  154. width: 20px;
  155. cursor: pointer;
  156. &:hover {
  157. color: #909399;
  158. }
  159. }
  160. .login-content-code {
  161. width: 100%;
  162. padding: 0;
  163. font-weight: bold;
  164. letter-spacing: 5px;
  165. }
  166. .login-content-submit {
  167. width: 100%;
  168. letter-spacing: 2px;
  169. font-weight: 300;
  170. margin-top: 15px;
  171. }
  172. }
  173. </style>