add mobile and email filed for user

This commit is contained in:
RockYang 2024-08-13 18:40:50 +08:00
parent 5da879600a
commit 43843b92f2
15 changed files with 152 additions and 69 deletions

View File

@ -58,6 +58,8 @@ func (h *UserHandler) Register(c *gin.Context) {
var data struct { var data struct {
RegWay string `json:"reg_way"` RegWay string `json:"reg_way"`
Username string `json:"username"` Username string `json:"username"`
Mobile string `json:"mobile"`
Email string `json:"email"`
Password string `json:"password"` Password string `json:"password"`
Code string `json:"code"` Code string `json:"code"`
InviteCode string `json:"invite_code"` InviteCode string `json:"invite_code"`
@ -85,8 +87,15 @@ func (h *UserHandler) Register(c *gin.Context) {
// 检查验证码 // 检查验证码
var key string var key string
if data.RegWay == "email" || data.RegWay == "mobile" { if data.RegWay == "email" {
key = CodeStorePrefix + data.Username key = CodeStorePrefix + data.Email
code, err := h.redis.Get(c, key).Result()
if err != nil || code != data.Code {
resp.ERROR(c, "验证码错误")
return
}
} else if data.RegWay == "mobile" {
key = CodeStorePrefix + data.Mobile
code, err := h.redis.Get(c, key).Result() code, err := h.redis.Get(c, key).Result()
if err != nil || code != data.Code { if err != nil || code != data.Code {
resp.ERROR(c, "验证码错误") resp.ERROR(c, "验证码错误")
@ -106,7 +115,17 @@ func (h *UserHandler) Register(c *gin.Context) {
// check if the username is existing // check if the username is existing
var item model.User var item model.User
res := h.DB.Where("username = ?", data.Username).First(&item) session := h.DB.Session(&gorm.Session{})
if data.Mobile != "" {
session = session.Where("mobile = ?", data.Mobile)
data.Username = data.Mobile
} else if data.Email != "" {
session = session.Where("email = ?", data.Email)
data.Username = data.Email
} else if data.Username != "" {
session = session.Where("username = ?", data.Username)
}
session.First(&item)
if item.Id > 0 { if item.Id > 0 {
resp.ERROR(c, "该用户名已经被注册") resp.ERROR(c, "该用户名已经被注册")
return return
@ -115,6 +134,8 @@ func (h *UserHandler) Register(c *gin.Context) {
salt := utils.RandString(8) salt := utils.RandString(8)
user := model.User{ user := model.User{
Username: data.Username, Username: data.Username,
Mobile: data.Mobile,
Email: data.Email,
Password: utils.GenPassword(data.Password, salt), Password: utils.GenPassword(data.Password, salt),
Avatar: "/images/avatar/user.png", Avatar: "/images/avatar/user.png",
Salt: salt, Salt: salt,
@ -134,7 +155,7 @@ func (h *UserHandler) Register(c *gin.Context) {
user.Nickname = fmt.Sprintf("极客学长@%d", utils.RandomNumber(6)) user.Nickname = fmt.Sprintf("极客学长@%d", utils.RandomNumber(6))
} }
res = h.DB.Create(&user) res := h.DB.Create(&user)
if res.Error != nil { if res.Error != nil {
resp.ERROR(c, "保存数据失败") resp.ERROR(c, "保存数据失败")
logger.Error(res.Error) logger.Error(res.Error)

View File

@ -4,6 +4,8 @@ type User struct {
BaseModel BaseModel
Username string Username string
Nickname string Nickname string
Email string
Mobile string
Password string Password string
Avatar string Avatar string
Salt string // 密码盐 Salt string // 密码盐

View File

@ -4,6 +4,8 @@ type User struct {
BaseVo BaseVo
Username string `json:"username"` Username string `json:"username"`
Nickname string `json:"nickname"` Nickname string `json:"nickname"`
Mobile string `json:"mobile"`
Email string `json:"email"`
Avatar string `json:"avatar"` Avatar string `json:"avatar"`
Salt string `json:"salt"` // 密码盐 Salt string `json:"salt"` // 密码盐
Power int `json:"power"` // 剩余算力 Power int `json:"power"` // 剩余算力

View File

@ -96,7 +96,8 @@
font-size 20px font-size 20px
background: #E9F1F6; background: #E9F1F6;
padding: 8px; padding: 8px;
border-radius: 50%; border-radius: 50%
cursor pointer
} }
.iconfont.icon-wechat { .iconfont.icon-wechat {
color #0bc15f color #0bc15f

View File

@ -4,8 +4,6 @@
.el-dialog { .el-dialog {
.el-dialog__body { .el-dialog__body {
padding-top 10px
.pay-container { .pay-container {
.amount { .amount {
text-align center text-align center

View File

@ -6,14 +6,12 @@
:before-close="close" :before-close="close"
:title="title" :title="title"
> >
<div class="form" id="bind-mobile-form"> <div class="form">
<el-alert v-if="username !== ''" type="info" show-icon :closable="false" style="margin-bottom: 20px;"> <div class="text-center">当前已绑手机号{{ mobile }}</div>
<p>当前绑定账号{{ username }}只允许使绑定有效的手机号或者邮箱地址作为登录账号</p>
</el-alert>
<el-form :model="form" label-width="120px"> <el-form :model="form" label-width="120px">
<el-form-item label="新账号"> <el-form-item label="手机号">
<el-input v-model="form.username"/> <el-input v-model="form.mobile"/>
</el-form-item> </el-form-item>
<el-form-item label="验证码"> <el-form-item label="验证码">
<el-row :gutter="20"> <el-row :gutter="20">
@ -21,7 +19,7 @@
<el-input v-model="form.code" maxlength="6"/> <el-input v-model="form.code" maxlength="6"/>
</el-col> </el-col>
<el-col :span="8"> <el-col :span="8">
<send-msg size="" :receiver="form.username"/> <send-msg size="" :receiver="form.username" type="mobile"/>
</el-col> </el-col>
</el-row> </el-row>
</el-form-item> </el-form-item>
@ -44,26 +42,31 @@ import SendMsg from "@/components/SendMsg.vue";
import {ElMessage} from "element-plus"; import {ElMessage} from "element-plus";
import {httpPost} from "@/utils/http"; import {httpPost} from "@/utils/http";
import {validateEmail, validateMobile} from "@/utils/validate"; import {validateEmail, validateMobile} from "@/utils/validate";
import {checkSession} from "@/store/cache";
const props = defineProps({ const props = defineProps({
show: Boolean, show: Boolean,
username: String
}); });
const showDialog = computed(() => { const showDialog = computed(() => {
return props.show return props.show
}) })
const title = ref('重置登录账号') const title = ref('绑定手机')
const mobile = ref('')
const form = ref({ const form = ref({
username: '', mobile: '',
code: '' code: ''
}) })
checkSession().then(user => {
mobile.value = user.mobile
})
const emits = defineEmits(['hide']); const emits = defineEmits(['hide']);
const save = () => { const save = () => {
if (!validateMobile(form.value.username) && !validateEmail(form.value.username)) { if (!validateMobile(form.value.mobile) && !validateEmail(form.value.mobile)) {
return ElMessage.error("请输入合法的手机号/邮箱地址") return ElMessage.error("请输入合法的手机号/邮箱地址")
} }
if (form.value.code === '') { if (form.value.code === '') {
@ -87,7 +90,15 @@ const close = function () {
</script> </script>
<style lang="stylus" scoped> <style lang="stylus" scoped>
#bind-mobile-form { .form {
.text-center {
text-align center
padding-bottom 15px
font-size 14px
color #a1a1a1
font-weight 700
}
.el-form-item__content { .el-form-item__content {
.el-row { .el-row {
width 100% width 100%

View File

@ -47,25 +47,29 @@
</div> </div>
<el-row class="btn-row" :gutter="20"> <el-row class="btn-row" :gutter="20">
<el-col :span="12"> <el-col :span="24">
<el-button class="login-btn" type="primary" size="large" @click="submitLogin">登录</el-button> <el-button class="login-btn" type="primary" size="large" @click="submitLogin">登录</el-button>
</el-col> </el-col>
<el-col :span="12">
<span class="text">
还没有账号
<el-tag @click="login = false">注册</el-tag>
</span>
<el-button type="info" class="forget" size="small" @click="showResetPass = true">忘记密码</el-button>
</el-col>
</el-row> </el-row>
<el-row v-if="wechatLoginURL !== ''"> <el-row>
<div class="c-login"> <el-col :span="12">
<div class="text">其他登录方式</div> <div class="reg">
<div class="login-type"> 还没有账号
<a class="wechat-login" :href="wechatLoginURL"><i class="iconfont icon-wechat"></i></a> <el-button type="primary" class="forget" size="small" @click="login = false">注册</el-button>
<el-button type="info" class="forget" size="small" @click="showResetPass = true">忘记密码</el-button>
</div> </div>
</div> </el-col>
<el-col :span="12">
<div class="c-login" v-if="wechatLoginURL !== ''">
<div class="text">其他登录方式</div>
<div class="login-type">
<a class="wechat-login" :href="wechatLoginURL"><i class="iconfont icon-wechat"></i></a>
</div>
</div>
</el-col>
</el-row> </el-row>
</el-form> </el-form>
</div> </div>
@ -77,7 +81,7 @@
<div class="block"> <div class="block">
<el-input placeholder="手机号码" <el-input placeholder="手机号码"
size="large" size="large"
v-model="data.username" v-model="data.mobile"
maxlength="11" maxlength="11"
autocomplete="off"> autocomplete="off">
<template #prefix> <template #prefix>
@ -102,7 +106,7 @@
</el-input> </el-input>
</el-col> </el-col>
<el-col :span="12"> <el-col :span="12">
<send-msg size="large" :receiver="data.username"/> <send-msg size="large" :receiver="data.mobile" type="mobile"/>
</el-col> </el-col>
</el-row> </el-row>
</div> </div>
@ -111,7 +115,7 @@
<div class="block"> <div class="block">
<el-input placeholder="邮箱地址" <el-input placeholder="邮箱地址"
size="large" size="large"
v-model="data.username" v-model="data.email"
autocomplete="off"> autocomplete="off">
<template #prefix> <template #prefix>
<el-icon> <el-icon>
@ -135,7 +139,7 @@
</el-input> </el-input>
</el-col> </el-col>
<el-col :span="12"> <el-col :span="12">
<send-msg size="large" :receiver="data.username"/> <send-msg size="large" :receiver="data.email" type="email"/>
</el-col> </el-col>
</el-row> </el-row>
</div> </div>
@ -257,8 +261,10 @@ watch(() => props.show, (newValue) => {
const login = ref(true) const login = ref(true)
const data = ref({ const data = ref({
username: process.env.VUE_APP_USER, username: "",
password: process.env.VUE_APP_PASS, password: "",
mobile: "",
email: "",
repass: "", repass: "",
code: "", code: "",
invite_code: "" invite_code: ""
@ -353,15 +359,15 @@ const doLogin = (verifyData) => {
// //
const submitRegister = () => { const submitRegister = () => {
if (data.value.username === '') { if (activeName.value === 'username' && data.value.username === '') {
return ElMessage.error('请输入用户名'); return ElMessage.error('请输入用户名');
} }
if (activeName.value === 'mobile' && !validateMobile(data.value.username)) { if (activeName.value === 'mobile' && !validateMobile(data.value.mobile)) {
return ElMessage.error('请输入合法的手机号'); return ElMessage.error('请输入合法的手机号');
} }
if (activeName.value === 'email' && !validateEmail(data.value.username)) { if (activeName.value === 'email' && !validateEmail(data.value.email)) {
return ElMessage.error('请输入合法的邮箱地址'); return ElMessage.error('请输入合法的邮箱地址');
} }
@ -468,8 +474,6 @@ const close = function () {
.c-login { .c-login {
display flex display flex
padding-top 20px
.text { .text {
font-size 16px font-size 16px
color #a1a1a1 color #a1a1a1
@ -482,7 +486,7 @@ const close = function () {
justify-content center justify-content center
.iconfont { .iconfont {
font-size 20px font-size 18px
background: #E9F1F6; background: #E9F1F6;
padding: 8px; padding: 8px;
border-radius: 50%; border-radius: 50%;
@ -492,6 +496,16 @@ const close = function () {
} }
} }
} }
.reg {
height 50px
display flex
align-items center
.el-button {
margin-left 10px
}
}
} }
.register-box { .register-box {

View File

@ -10,13 +10,13 @@
<div class="form"> <div class="form">
<el-form :model="form" label-width="80px" label-position="left"> <el-form :model="form" label-width="80px" label-position="left">
<el-form-item label="用户名"> <el-form-item label="手机号">
<el-input v-model="form.username" placeholder="手机号/邮箱地址"/> <el-input v-model="form.username" placeholder="手机号"/>
</el-form-item> </el-form-item>
<el-form-item label="验证码"> <el-form-item label="验证码">
<div class="code-box"> <div class="code-box">
<el-input v-model="form.code" maxlength="6"/> <el-input v-model="form.code" maxlength="6"/>
<send-msg size="" :receiver="form.username" style="margin-left: 10px; min-width: 100px"/> <send-msg size="" :receiver="form.username" type="mobile"/>
</div> </div>
<el-row :gutter="20"> <el-row :gutter="20">
<el-col :span="12"> <el-col :span="12">

View File

@ -21,6 +21,10 @@ import {getSystemInfo} from "@/store/cache";
const props = defineProps({ const props = defineProps({
receiver: String, receiver: String,
size: String, size: String,
type: {
type: String,
default: 'mobile'
}
}); });
const btnText = ref('发送验证码') const btnText = ref('发送验证码')
const canSend = ref(true) const canSend = ref(true)
@ -32,9 +36,13 @@ getSystemInfo().then(res => {
}) })
const sendMsg = () => { const sendMsg = () => {
if (!validateMobile(props.receiver) && !validateEmail(props.receiver)) { if (!validateMobile(props.receiver) && props.type === 'mobile') {
return showMessageError("请输入合法的手机号/邮箱地址") return showMessageError("请输入合法的手机号")
} }
if (!validateEmail(props.receiver) && props.type === 'email') {
return showMessageError("请输入合法的邮箱地址")
}
if (enableVerify.value) { if (enableVerify.value) {
captchaRef.value.loadCaptcha() captchaRef.value.loadCaptcha()
} else { } else {

View File

@ -48,7 +48,7 @@
<el-divider class="divider">其他登录方式</el-divider> <el-divider class="divider">其他登录方式</el-divider>
<div class="clogin"> <div class="clogin">
<a class="wechat-login" :href="wechatLoginURL"><i class="iconfont icon-wechat"></i></a> <a :href="wechatLoginURL"><i class="iconfont icon-wechat"></i></a>
</div> </div>
</div> </div>
</div> </div>
@ -76,6 +76,7 @@ import {setUserToken} from "@/store/session";
import ResetPass from "@/components/ResetPass.vue"; import ResetPass from "@/components/ResetPass.vue";
import {showMessageError} from "@/utils/dialog"; import {showMessageError} from "@/utils/dialog";
import Captcha from "@/components/Captcha.vue"; import Captcha from "@/components/Captcha.vue";
import QRCode from "qrcode";
const router = useRouter(); const router = useRouter();
const title = ref('Geek-AI'); const title = ref('Geek-AI');
@ -161,7 +162,6 @@ const doLogin = (verifyData) => {
showMessageError('登录失败,' + e.message) showMessageError('登录失败,' + e.message)
}) })
} }
</script> </script>
<style lang="stylus" scoped> <style lang="stylus" scoped>

View File

@ -24,7 +24,8 @@
</div> </div>
</template> </template>
<template #extra> <template #extra>
<el-button type="primary" @click="finishLogin">我知道了</el-button> <el-button type="primary" class="copy-user-info" :data-clipboard-text="'用户名'+username+' 密码'+password">复制</el-button>
<el-button type="danger" @click="finishLogin">关闭</el-button>
</template> </template>
</el-result> </el-result>
</el-dialog> </el-dialog>
@ -33,12 +34,14 @@
</template> </template>
<script setup> <script setup>
import {ref} from "vue" import {onMounted, onUnmounted, ref} from "vue"
import {useRouter} from "vue-router" import {useRouter} from "vue-router"
import {ElMessage, ElMessageBox} from "element-plus"; import {ElMessage, ElMessageBox} from "element-plus";
import {httpGet} from "@/utils/http"; import {httpGet} from "@/utils/http";
import {setUserToken} from "@/store/session"; import {setUserToken} from "@/store/session";
import {isMobile} from "@/utils/libs"; import {isMobile} from "@/utils/libs";
import Clipboard from "clipboard";
import {showMessageError, showMessageOK} from "@/utils/dialog";
const winHeight = ref(window.innerHeight) const winHeight = ref(window.innerHeight)
const loading = ref(true) const loading = ref(true)
@ -74,7 +77,25 @@ if (code === "") {
}) })
}) })
} }
const clipboard = ref(null)
onMounted(() => {
clipboard.value = new Clipboard('.copy-user-info');
clipboard.value.on('success', () => {
showMessageOK('复制成功!');
})
clipboard.value.on('error', () => {
showMessageError('复制失败!');
})
})
onUnmounted(() => {
clipboard.value.destroy();
})
const finishLogin = () => { const finishLogin = () => {
show.value = false
if (isMobile()) { if (isMobile()) {
router.push('/mobile') router.push('/mobile')
} else { } else {

View File

@ -10,7 +10,7 @@
<el-button type="primary" @click="showPasswordDialog = true">修改密码</el-button> <el-button type="primary" @click="showPasswordDialog = true">修改密码</el-button>
</el-col> </el-col>
<el-col :span="12"> <el-col :span="12">
<el-button type="primary" @click="showBindMobileDialog = true">更改账号</el-button> <el-button type="primary" @click="showBindMobileDialog = true">绑定手机</el-button>
</el-col> </el-col>
<el-col :span="24"> <el-col :span="24">
<el-button type="success" @click="showRedeemVerifyDialog = true">兑换码核销 <el-button type="success" @click="showRedeemVerifyDialog = true">兑换码核销
@ -93,8 +93,7 @@
<password-dialog v-if="isLogin" :show="showPasswordDialog" @hide="showPasswordDialog = false" <password-dialog v-if="isLogin" :show="showPasswordDialog" @hide="showPasswordDialog = false"
@logout="logout"/> @logout="logout"/>
<bind-mobile v-if="isLogin" :show="showBindMobileDialog" :username="user.username" <bind-mobile v-if="isLogin" :show="showBindMobileDialog" @hide="showBindMobileDialog = false"/>
@hide="showBindMobileDialog = false"/>
<redeem-verify v-if="isLogin" :show="showRedeemVerifyDialog" @hide="redeemCallback"/> <redeem-verify v-if="isLogin" :show="showRedeemVerifyDialog" @hide="redeemCallback"/>
@ -143,7 +142,7 @@ import {InfoFilled, SuccessFilled} from "@element-plus/icons-vue";
import {checkSession, getSystemInfo} from "@/store/cache"; import {checkSession, getSystemInfo} from "@/store/cache";
import UserProfile from "@/components/UserProfile.vue"; import UserProfile from "@/components/UserProfile.vue";
import PasswordDialog from "@/components/PasswordDialog.vue"; import PasswordDialog from "@/components/PasswordDialog.vue";
import BindMobile from "@/components/ResetAccount.vue"; import BindMobile from "@/components/BindMobile.vue";
import RedeemVerify from "@/components/RedeemVerify.vue"; import RedeemVerify from "@/components/RedeemVerify.vue";
import {useRouter} from "vue-router"; import {useRouter} from "vue-router";
import {removeUserToken} from "@/store/session"; import {removeUserToken} from "@/store/session";

View File

@ -41,7 +41,7 @@
</el-input> </el-input>
</el-col> </el-col>
<el-col :span="12"> <el-col :span="12">
<send-msg size="large" :receiver="data.username"/> <send-msg size="large" :receiver="data.username" type="mobile"/>
</el-col> </el-col>
</el-row> </el-row>
</div> </div>
@ -50,7 +50,7 @@
<div class="block"> <div class="block">
<el-input placeholder="邮箱地址" <el-input placeholder="邮箱地址"
size="large" size="large"
v-model="data.username" v-model="data.email"
autocomplete="off"> autocomplete="off">
<template #prefix> <template #prefix>
<el-icon> <el-icon>
@ -74,7 +74,7 @@
</el-input> </el-input>
</el-col> </el-col>
<el-col :span="12"> <el-col :span="12">
<send-msg size="large" :receiver="data.username"/> <send-msg size="large" :receiver="data.email" type="email"/>
</el-col> </el-col>
</el-row> </el-row>
</div> </div>
@ -135,13 +135,17 @@
<el-row class="btn-row" :gutter="20"> <el-row class="btn-row" :gutter="20">
<el-col :span="24"> <el-col :span="24">
<el-button class="login-btn" type="primary" size="large" @click="submitRegister">注册</el-button> <el-button class="login-btn" type="primary" size="large" @click="submitRegister" >注册</el-button>
</el-col> </el-col>
</el-row> </el-row>
<el-row class="text-line"> <el-row class="text-line" :gutter="24">
已经有账号 <el-col :span="12">
<el-link type="primary" @click="router.push('/login')">登录</el-link> <el-link type="primary" @click="router.push('/login')">登录</el-link>
</el-col>
<el-col :span="12">
<el-link type="primary" @click="router.push('/')">首页</el-link>
</el-col>
</el-row> </el-row>
</el-form> </el-form>
@ -375,6 +379,10 @@ const doSubmitRegister = (verifyData) => {
justify-content center justify-content center
padding-top 10px; padding-top 10px;
font-size 14px; font-size 14px;
.el-col {
text-align center
}
} }
} }
} }

View File

@ -262,9 +262,7 @@ const removeUser = function (user) {
).then(() => { ).then(() => {
httpGet('/api/admin/user/remove', {id: user.id}).then(() => { httpGet('/api/admin/user/remove', {id: user.id}).then(() => {
ElMessage.success('操作成功!') ElMessage.success('操作成功!')
users.value.items = removeArrayItem(users.value.items, user, function (v1, v2) { fetchUserList(users.value.page, users.value.page_size)
return v1.id === v2.id
})
}).catch((e) => { }).catch((e) => {
ElMessage.error('操作失败,' + e.message) ElMessage.error('操作失败,' + e.message)
}) })

View File

@ -19,7 +19,7 @@ module.exports = defineConfig({
outputDir: 'dist', outputDir: 'dist',
crossorigin: "anonymous", crossorigin: "anonymous",
devServer: { devServer: {
allowedHosts: ['127.0.0.1:5678'], allowedHosts: "all",
port: 8888, port: 8888,
} }
}) })