login.vue 4.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175
  1. <script setup lang="ts">
  2. import {
  3. VbenButton,
  4. VbenCheckbox,
  5. VbenInput,
  6. VbenInputPassword,
  7. } from '@vben-core/shadcn-ui';
  8. import { $t } from '@vben/locales';
  9. import { computed, reactive } from 'vue';
  10. import { useRouter } from 'vue-router';
  11. import Title from './auth-title.vue';
  12. import ThirdPartyLogin from './third-party-login.vue';
  13. import type { LoginEmits } from './typings';
  14. interface Props {
  15. /**
  16. * @zh_CN 是否处于加载处理状态
  17. */
  18. loading?: boolean;
  19. }
  20. defineOptions({
  21. name: 'AuthenticationLogin',
  22. });
  23. withDefaults(defineProps<Props>(), {
  24. loading: false,
  25. });
  26. const emit = defineEmits<{
  27. submit: LoginEmits['submit'];
  28. }>();
  29. const router = useRouter();
  30. const REMEMBER_ME_KEY = 'REMEMBER_ME_USERNAME';
  31. const localUsername = localStorage.getItem(REMEMBER_ME_KEY) || '';
  32. const formState = reactive({
  33. password: '',
  34. rememberMe: !!localUsername,
  35. submitted: false,
  36. username: localUsername,
  37. });
  38. const usernameStatus = computed(() => {
  39. return formState.submitted && !formState.username ? 'error' : 'default';
  40. });
  41. const passwordStatus = computed(() => {
  42. return formState.submitted && !formState.password ? 'error' : 'default';
  43. });
  44. function handleSubmit() {
  45. formState.submitted = true;
  46. if (
  47. usernameStatus.value !== 'default' ||
  48. passwordStatus.value !== 'default'
  49. ) {
  50. return;
  51. }
  52. localStorage.setItem(
  53. REMEMBER_ME_KEY,
  54. formState.rememberMe ? formState.username : '',
  55. );
  56. emit('submit', {
  57. password: formState.password,
  58. username: formState.username,
  59. });
  60. }
  61. function handleGo(path: string) {
  62. router.push(path);
  63. }
  64. </script>
  65. <template>
  66. <div @keypress.enter="handleSubmit">
  67. <Title>
  68. {{ $t('authentication.welcome-back') }} 👋🏻
  69. <template #desc>
  70. <span class="text-muted-foreground">
  71. {{ $t('authentication.login-subtitle') }}
  72. </span>
  73. </template>
  74. </Title>
  75. <VbenInput
  76. v-model="formState.username"
  77. :status="usernameStatus"
  78. :error-tip="$t('authentication.username-tip')"
  79. :label="$t('authentication.username')"
  80. name="username"
  81. :placeholder="$t('authentication.username')"
  82. type="text"
  83. required
  84. :autofocus="false"
  85. />
  86. <VbenInputPassword
  87. v-model="formState.password"
  88. :status="passwordStatus"
  89. :error-tip="$t('authentication.password-tip')"
  90. :label="$t('authentication.password')"
  91. name="password"
  92. :placeholder="$t('authentication.password')"
  93. required
  94. type="password"
  95. />
  96. <div class="mb-6 mt-4 flex justify-between">
  97. <div class="flex-center flex">
  98. <VbenCheckbox v-model:checked="formState.rememberMe" name="rememberMe">
  99. {{ $t('authentication.remember-me') }}
  100. </VbenCheckbox>
  101. </div>
  102. <span
  103. class="text-primary hover:text-primary/80 cursor-pointer text-sm font-normal"
  104. @click="handleGo('/auth/forget-password')"
  105. >
  106. {{ $t('authentication.forget-password') }}
  107. </span>
  108. <!-- <VbenButton variant="ghost" @click="handleGo('/auth/forget-password')">
  109. 忘记密码?
  110. </VbenButton> -->
  111. </div>
  112. <VbenButton :loading="loading" class="w-full" @click="handleSubmit">
  113. {{ $t('common.login') }}
  114. </VbenButton>
  115. <div class="mb-2 mt-4 flex items-center justify-between">
  116. <VbenButton
  117. variant="outline"
  118. class="w-1/2"
  119. @click="handleGo('/auth/code-login')"
  120. >
  121. {{ $t('authentication.mobile-login') }}
  122. </VbenButton>
  123. <VbenButton
  124. variant="outline"
  125. class="ml-4 w-1/2"
  126. @click="handleGo('/auth/qrcode-login')"
  127. >
  128. {{ $t('authentication.qrcode-login') }}
  129. </VbenButton>
  130. <!-- <VbenButton
  131. :loading="loading"
  132. variant="outline"
  133. class="w-1/3"
  134. @click="handleGo('/auth/register')"
  135. >
  136. 创建账号
  137. </VbenButton> -->
  138. </div>
  139. <!-- 第三方登录 -->
  140. <ThirdPartyLogin />
  141. <div class="text-center text-sm">
  142. {{ $t('authentication.account-tip') }}
  143. <span
  144. class="text-primary hover:text-primary/80 cursor-pointer text-sm font-normal"
  145. @click="handleGo('/auth/register')"
  146. >
  147. {{ $t('authentication.create-account') }}
  148. </span>
  149. </div>
  150. </div>
  151. </template>