|
@@ -7,9 +7,7 @@
|
|
</div>
|
|
</div>
|
|
|
|
|
|
<el-form-item prop="username">
|
|
<el-form-item prop="username">
|
|
- <span class="svg-container">
|
|
|
|
- <svg-icon icon-class="user" />
|
|
|
|
- </span>
|
|
|
|
|
|
+ <i class="iconfont icon-user form-left-container" />
|
|
<el-input
|
|
<el-input
|
|
ref="username"
|
|
ref="username"
|
|
v-model="loginForm.username"
|
|
v-model="loginForm.username"
|
|
@@ -22,9 +20,7 @@
|
|
</el-form-item>
|
|
</el-form-item>
|
|
|
|
|
|
<el-form-item prop="password">
|
|
<el-form-item prop="password">
|
|
- <span class="svg-container">
|
|
|
|
- <svg-icon icon-class="password" />
|
|
|
|
- </span>
|
|
|
|
|
|
+ <i class="iconfont icon-password form-left-container" />
|
|
<el-input
|
|
<el-input
|
|
:key="passwordType"
|
|
:key="passwordType"
|
|
ref="password"
|
|
ref="password"
|
|
@@ -36,10 +32,40 @@
|
|
auto-complete="on"
|
|
auto-complete="on"
|
|
@keyup.enter.native="handleLogin"
|
|
@keyup.enter.native="handleLogin"
|
|
/>
|
|
/>
|
|
- <span class="show-pwd" @click="showPwd">
|
|
|
|
- <svg-icon :icon-class="passwordType === 'password' ? 'eye' : 'eye-open'" />
|
|
|
|
- </span>
|
|
|
|
|
|
+ <i :class="`iconfont icon-${passwordType === 'password' ? 'eye' : 'eye-open'} form-right-container`" @click="showPwd" />
|
|
</el-form-item>
|
|
</el-form-item>
|
|
|
|
+ <template v-if="isAdmin">
|
|
|
|
+ <el-form-item prop="phone">
|
|
|
|
+ <i class="iconfont icon-phone form-left-container" />
|
|
|
|
+ <el-input
|
|
|
|
+ ref="phone"
|
|
|
|
+ v-model="loginForm.phone"
|
|
|
|
+ placeholder="手机号"
|
|
|
|
+ name="phone"
|
|
|
|
+ type="text"
|
|
|
|
+ tabindex="1"
|
|
|
|
+ auto-complete="on"
|
|
|
|
+ />
|
|
|
|
+ </el-form-item>
|
|
|
|
+
|
|
|
|
+ <div style="display: flex;align-items: center">
|
|
|
|
+ <el-form-item prop="code" style="flex: 1">
|
|
|
|
+ <i class="iconfont icon-captcha form-left-container" />
|
|
|
|
+ <el-input
|
|
|
|
+ ref="code"
|
|
|
|
+ v-model="loginForm.code"
|
|
|
|
+ placeholder="验证码"
|
|
|
|
+ name="code"
|
|
|
|
+ type="text"
|
|
|
|
+ tabindex="1"
|
|
|
|
+ auto-complete="on"
|
|
|
|
+ />
|
|
|
|
+ </el-form-item>
|
|
|
|
+ <el-button class="code_btn" type="primary" :disabled="isSending" :loading="submitLoading" @click="sendCode">
|
|
|
|
+ {{ isSending ? `${countdown}s后重试` : "发送验证码" }}
|
|
|
|
+ </el-button>
|
|
|
|
+ </div>
|
|
|
|
+ </template>
|
|
|
|
|
|
<el-button :loading="loading" type="primary" style="width:100%;margin-bottom:30px;" @click.native.prevent="handleLogin">登 录</el-button>
|
|
<el-button :loading="loading" type="primary" style="width:100%;margin-bottom:30px;" @click.native.prevent="handleLogin">登 录</el-button>
|
|
</el-form>
|
|
</el-form>
|
|
@@ -49,28 +75,33 @@
|
|
<script>
|
|
<script>
|
|
import { validUsername } from '@/utils/validate'
|
|
import { validUsername } from '@/utils/validate'
|
|
import store from '@/store'
|
|
import store from '@/store'
|
|
|
|
+import { getCode } from '@/api/user'
|
|
|
|
|
|
export default {
|
|
export default {
|
|
name: 'Login',
|
|
name: 'Login',
|
|
data() {
|
|
data() {
|
|
- const validateUsername = (rule, value, callback) => {
|
|
|
|
- if (!validUsername(value)) {
|
|
|
|
- callback(new Error('请输入你的账号'))
|
|
|
|
|
|
+ const validatePassword = (rule, value, callback) => {
|
|
|
|
+ if (!value || value.length < 6) {
|
|
|
|
+ callback(new Error('密码不能少于6位'))
|
|
} else {
|
|
} else {
|
|
callback()
|
|
callback()
|
|
}
|
|
}
|
|
}
|
|
}
|
|
- const validatePassword = (rule, value, callback) => {
|
|
|
|
- if (!value || value.length < 6) {
|
|
|
|
- callback(new Error('密码不能少于6位'))
|
|
|
|
|
|
+ const validatePhone = (rule, value, callback) => {
|
|
|
|
+ const phoneReg = /^1[3-9]\d{9}$/
|
|
|
|
+ if (!value || !phoneReg.test(value)) {
|
|
|
|
+ callback(new Error('请输入正确的手机号码'))
|
|
} else {
|
|
} else {
|
|
callback()
|
|
callback()
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
+
|
|
return {
|
|
return {
|
|
loginForm: {
|
|
loginForm: {
|
|
username: null,
|
|
username: null,
|
|
- password: null
|
|
|
|
|
|
+ password: null,
|
|
|
|
+ phone: null,
|
|
|
|
+ code: ''
|
|
},
|
|
},
|
|
loginRules: {
|
|
loginRules: {
|
|
username: [{
|
|
username: [{
|
|
@@ -78,11 +109,29 @@ export default {
|
|
whitespace: true,
|
|
whitespace: true,
|
|
message: '请输入你的账号',
|
|
message: '请输入你的账号',
|
|
trigger: ['change', 'blur'] }],
|
|
trigger: ['change', 'blur'] }],
|
|
- password: [{ required: true, trigger: 'blur', validator: validatePassword }]
|
|
|
|
|
|
+ password: [{ required: true, trigger: ['change', 'blur'], validator: validatePassword }],
|
|
|
|
+ phone: [{ required: true, trigger: ['change', 'blur'], validator: validatePhone }],
|
|
|
|
+ code: [{
|
|
|
|
+ required: true,
|
|
|
|
+ whitespace: true,
|
|
|
|
+ message: '请输入验证码',
|
|
|
|
+ trigger: ['change', 'blur'] }]
|
|
},
|
|
},
|
|
loading: false,
|
|
loading: false,
|
|
passwordType: 'password',
|
|
passwordType: 'password',
|
|
- redirect: undefined
|
|
|
|
|
|
+ redirect: undefined,
|
|
|
|
+
|
|
|
|
+ /** 验证码开始 **/
|
|
|
|
+ isSending: false,
|
|
|
|
+ submitLoading: false,
|
|
|
|
+ countdown: 60,
|
|
|
|
+ timer: null
|
|
|
|
+ /** 验证码结束 **/
|
|
|
|
+ }
|
|
|
|
+ },
|
|
|
|
+ computed: {
|
|
|
|
+ isAdmin() {
|
|
|
|
+ return this.loginForm.username == 'admin'
|
|
}
|
|
}
|
|
},
|
|
},
|
|
watch: {
|
|
watch: {
|
|
@@ -109,6 +158,10 @@ export default {
|
|
this.$refs.loginForm.validate(valid => {
|
|
this.$refs.loginForm.validate(valid => {
|
|
if (valid) {
|
|
if (valid) {
|
|
this.loading = true
|
|
this.loading = true
|
|
|
|
+ if (!this.isAdmin) {
|
|
|
|
+ this.loginForm.phone = ''
|
|
|
|
+ this.loginForm.code = ''
|
|
|
|
+ }
|
|
this.$store.dispatch('user/login', this.loginForm).then(async() => {
|
|
this.$store.dispatch('user/login', this.loginForm).then(async() => {
|
|
await this.$store.dispatch('user/getAuthAccountQuery')
|
|
await this.$store.dispatch('user/getAuthAccountQuery')
|
|
this.$router.push({ path: this.redirect || '/' })
|
|
this.$router.push({ path: this.redirect || '/' })
|
|
@@ -121,6 +174,43 @@ export default {
|
|
return false
|
|
return false
|
|
}
|
|
}
|
|
})
|
|
})
|
|
|
|
+ },
|
|
|
|
+ // 发送验证码
|
|
|
|
+ sendCode() {
|
|
|
|
+ this.$refs.loginForm.validateField(['phone'], (valid) => {
|
|
|
|
+ console.log('发送验证码校验', valid)
|
|
|
|
+ // 字段校验成功则返回空串
|
|
|
|
+ if (!valid) {
|
|
|
|
+ const _this = this
|
|
|
|
+ _this.submitLoading = true
|
|
|
|
+ getCode({
|
|
|
|
+ username: _this.loginForm.username,
|
|
|
|
+ phone: _this.loginForm.phone
|
|
|
|
+ }).then(res => {
|
|
|
|
+ this.$message({
|
|
|
|
+ type: 'success',
|
|
|
|
+ message: '发送验证码成功!'
|
|
|
|
+ })
|
|
|
|
+ _this.isSending = true
|
|
|
|
+ _this.countdown = 60
|
|
|
|
+ // 模拟请求后端发送验证码
|
|
|
|
+ console.log('验证码已发送至手机号:', _this.loginForm.phone)
|
|
|
|
+
|
|
|
|
+ _this.timer = setInterval(() => {
|
|
|
|
+ if (_this.countdown > 0) {
|
|
|
|
+ _this.countdown -= 1
|
|
|
|
+ } else {
|
|
|
|
+ clearInterval(_this.timer)
|
|
|
|
+ _this.isSending = false
|
|
|
|
+ }
|
|
|
|
+ }, 1000)
|
|
|
|
+ }).finally(() => {
|
|
|
|
+ _this.submitLoading = false
|
|
|
|
+ })
|
|
|
|
+ } else {
|
|
|
|
+ return
|
|
|
|
+ }
|
|
|
|
+ })
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
}
|
|
@@ -144,7 +234,7 @@ $cursor: #fff;
|
|
.login-container {
|
|
.login-container {
|
|
.el-input {
|
|
.el-input {
|
|
display: inline-block;
|
|
display: inline-block;
|
|
- height: 47px;
|
|
|
|
|
|
+ height: 46px;
|
|
width: 85%;
|
|
width: 85%;
|
|
|
|
|
|
input {
|
|
input {
|
|
@@ -154,7 +244,8 @@ $cursor: #fff;
|
|
border-radius: 0px;
|
|
border-radius: 0px;
|
|
padding: 12px 5px 12px 15px;
|
|
padding: 12px 5px 12px 15px;
|
|
color: $light_gray;
|
|
color: $light_gray;
|
|
- height: 47px;
|
|
|
|
|
|
+ height: 46px;
|
|
|
|
+ line-height: 46px;
|
|
caret-color: $cursor;
|
|
caret-color: $cursor;
|
|
|
|
|
|
&:-webkit-autofill {
|
|
&:-webkit-autofill {
|
|
@@ -169,6 +260,9 @@ $cursor: #fff;
|
|
background: rgba(0, 0, 0, 0.1);
|
|
background: rgba(0, 0, 0, 0.1);
|
|
border-radius: 5px;
|
|
border-radius: 5px;
|
|
color: #454545;
|
|
color: #454545;
|
|
|
|
+ height: 46px;
|
|
|
|
+ line-height: 46px;
|
|
|
|
+ //overflow: hidden;
|
|
}
|
|
}
|
|
}
|
|
}
|
|
</style>
|
|
</style>
|
|
@@ -193,12 +287,27 @@ $light_gray:#eee;
|
|
overflow: hidden;
|
|
overflow: hidden;
|
|
}
|
|
}
|
|
|
|
|
|
- .svg-container {
|
|
|
|
- padding: 6px 5px 6px 15px;
|
|
|
|
|
|
+ .form-left-container {
|
|
|
|
+ height: 46px;
|
|
|
|
+ width: 30px;
|
|
|
|
+ text-align: right;
|
|
color: $dark_gray;
|
|
color: $dark_gray;
|
|
vertical-align: middle;
|
|
vertical-align: middle;
|
|
|
|
+ display: inline-block;
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ .form-right-container {
|
|
|
|
+ height: 46px;
|
|
width: 30px;
|
|
width: 30px;
|
|
|
|
+ text-align: center;
|
|
|
|
+ color: $dark_gray;
|
|
|
|
+ vertical-align: middle;
|
|
display: inline-block;
|
|
display: inline-block;
|
|
|
|
+ cursor: pointer;
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ .icon-eye {
|
|
|
|
+ font-size: 14px;
|
|
}
|
|
}
|
|
|
|
|
|
.title-container {
|
|
.title-container {
|
|
@@ -222,5 +331,18 @@ $light_gray:#eee;
|
|
cursor: pointer;
|
|
cursor: pointer;
|
|
user-select: none;
|
|
user-select: none;
|
|
}
|
|
}
|
|
|
|
+
|
|
|
|
+ .code_btn {
|
|
|
|
+ margin-bottom: 22px;
|
|
|
|
+ width: 115px;
|
|
|
|
+ height: 46px;
|
|
|
|
+ margin-left: 10px;
|
|
|
|
+ font-size: 14px;
|
|
|
|
+ border-radius: 4px;
|
|
|
|
+ display: flex;
|
|
|
|
+ align-items: center;
|
|
|
|
+ justify-content: center;
|
|
|
|
+ cursor: pointer;
|
|
|
|
+ }
|
|
}
|
|
}
|
|
</style>
|
|
</style>
|