feat:about account

This commit is contained in:
lqins 2024-11-23 15:40:05 +08:00
parent 9bf886fe98
commit d13fa1392f
21 changed files with 1580 additions and 1258 deletions

2
web/package-lock.json generated
View File

@ -8,7 +8,7 @@
"name": "geekai-web", "name": "geekai-web",
"version": "0.1.0", "version": "0.1.0",
"dependencies": { "dependencies": {
"@element-plus/icons-vue": "^2.1.0", "@element-plus/icons-vue": "^2.3.1",
"@openai/realtime-api-beta": "github:openai/openai-realtime-api-beta", "@openai/realtime-api-beta": "github:openai/openai-realtime-api-beta",
"animate.css": "^4.1.1", "animate.css": "^4.1.1",
"axios": "^0.27.2", "axios": "^0.27.2",

View File

@ -8,7 +8,7 @@
"lint": "vue-cli-service lint" "lint": "vue-cli-service lint"
}, },
"dependencies": { "dependencies": {
"@element-plus/icons-vue": "^2.1.0", "@element-plus/icons-vue": "^2.3.1",
"@openai/realtime-api-beta": "github:openai/openai-realtime-api-beta", "@openai/realtime-api-beta": "github:openai/openai-realtime-api-beta",
"animate.css": "^4.1.1", "animate.css": "^4.1.1",
"axios": "^0.27.2", "axios": "^0.27.2",

View File

@ -1,3 +1,17 @@
:root{
--sm-txt:rgba(163, 174, 208, 1);
--text-secondary: #8a939d;
--el-color-primary: rgba(67, 24, 255, 1);
--el-component-size: 48px;
--el-border-radius-base: 8px;
--el-color-primary-light-5:rgb(107, 85, 255);
--el-color-primary-light-3:rgb(78, 51, 254);
--theme-btn-color:rgba(117, 81, 255, 1)
--common-text-color:#6e4ef9
--text-fff:#fff
}
.btn-go{ .btn-go{
background: var(--btnColor); background: var(--btnColor);
color: #fff; color: #fff;
@ -7,3 +21,38 @@
color: #fff; color: #fff;
} }
} }
.btn-normal{
background: var(--theme-btn-color);
color: #fff;
border-radius: 5px;
padding: 5px 10px;
&:hover{
color: #fff;
}
}
.flex{
display: flex;
align-items: center;
}
.flex-center{
display: flex;
align-items: center;
justify-content: center;
}
.flex-between{
display: flex;
align-items: center;
justify-content: space-between;
}
.theme-color-primary{
color: var(--el-color-primary);
}
.text-color-primary{
color:var(--text-color-primary)
}
.w100{
width: 100%;
}
.el-input__wrapper{
background: var( --card-bg)
}

View File

@ -1,4 +1,5 @@
.index-page { .index-page {
margin: 0 margin: 0
overflow hidden overflow hidden
@ -6,7 +7,11 @@
display flex display flex
justify-content center justify-content center
align-items baseline align-items baseline
padding-top 150px padding-top 158px
min-height: 75vh
background: var(--theme-bg) !important
.color-bg { .color-bg {
position absolute position absolute

View File

@ -1,117 +1,56 @@
.bg {
position fixed .loginPage{
left 0 background: var(--card-bg) !important
right 0 background-color: var(---card-bg) !important
top 0
bottom 0 .form-title{
background-color #313237 color:var( --text-theme-color)
background-image url("~@/assets/img/login-bg.jpg") }
background-size cover
background-position center
background-repeat repeat-y
//filter: blur(10px); /* */
} }
.main { .left{
.contain { width: 50%;
position fixed
left 50%
top 40%
width 90%
max-width 400px;
transform translate(-50%, -50%)
padding 20px 10px;
color #ffffff
border-radius 10px;
.logo { .login-box{
text-align center width: 410px;
margin: 0 auto;
min-height: calc(100vh - 48px);
}
.wechatLog{
width: 410px;
height: 50px;
line-height: 50px;
text-align: center
background: var( --sign-bg)
a{
color: var(--text-theme-color)
.el-image {
width 120px;
cursor pointer
border-radius 50%
}
} }
font-size: 14px;
.header { margin-bottom: 26px
width 100% border-radius: 16px;
margin-bottom 24px .icon-wechat{
font-size 24px color: #0bc15f
color $white_v1 margin-right: 9px
letter-space 2px font-size: 20px;
text-align center
padding-top 10px
}
.content {
width 100%
height: auto
border-radius 3px
.block {
margin-bottom 16px
.el-input__inner {
border 1px solid $gray-v6 !important
.el-icon-user, .el-icon-lock {
font-size 20px
}
}
}
.btn-row {
padding-top 10px;
.login-btn {
width 100%
font-size 16px
letter-spacing 2px
}
}
.text-line {
justify-content center
padding-top 10px;
font-size 14px;
}
.opt {
padding 15px
.el-col {
text-align center
}
}
.divider {
border-top: 2px solid #c1c1c1;
}
.clogin {
padding 15px
display flex
justify-content center
.iconfont {
font-size 20px
background: #E9F1F6;
padding: 8px;
border-radius: 50%
cursor pointer
}
.iconfont.icon-wechat {
color #0bc15f
}
}
} }
} }
.footer { .text-color-primary{
color #ffffff; cursor :pointer
.container {
padding 20px;
}
} }
} }
.login-btn {
width :100%
height: 40px;
border-radius: 16px;
}
.code-input{
width: 306px;
margin-right: 9px;
}

View File

@ -12,10 +12,19 @@
body, body,
#app, #app,
.wrapper { .wrapper {
background-color: rgb(13, 20, 53) !important; background: rgb(13, 20, 53)
background-color: rgb(13, 20, 53)
font-family: $font-regular; font-family: $font-regular;
} }
--btnColor: linear-gradient(88deg, #af61f0 1.44%, #5b62ce); --btnColor: linear-gradient(88deg, #af61f0 1.44%, #5b62ce);
--card-bg: rgba(17, 28, 68, 1); --card-bg: rgba(17, 28, 68, 1);
--theme-bg:rgb(13, 20, 53);
--sign-bg: rgba(27, 37, 75, 1);
--text-theme-color: #fff;
--text-color-primary: #d1c7ff;
--el-text-color-regular: rgba(163, 174, 208, 1)
--el-border-color:rgb(79, 80, 85)
--el-text-color-primary: #fff;
} }

View File

@ -13,13 +13,22 @@
body, body,
#app, #app,
.wrapper { .wrapper {
background: linear-gradient(88deg, #fff3f3 1.44%, #e7e8ff); // background: var(--theme-bg)
// background: linear-gradient(180deg, #fff, #fff4fa 40%, #fbd9fd 55%, #e2d5ff 70%); // background-color: var(--theme-bg)
// background: linear-gradient(88deg, #fff3f3 1.44%, #e7e8ff);
font-family: $font-regular; font-family: $font-regular;
} }//#6b61f6
--btnColor: linear-gradient(88deg, #af61f0 1.44%, #5b62ce); --btnColor: linear-gradient(88deg, #af61f0 1.44%, #5b62ce);
--code-btnColor: linear-gradient(88deg, #af61f0 1.44%, #5b62ce);
--card-bg:#fff; --card-bg:#fff;
--theme-bg:linear-gradient(88deg, #fff3f3 1.44%, #e7e8ff);
--sign-bg: rgba(244, 247, 254, 1);
--text-theme-color: rgba(43, 54, 116, 1)
--text-color-primary: rgba(67, 24, 255, 1);
} }

Binary file not shown.

After

Width:  |  Height:  |  Size: 824 KiB

View File

Before

Width:  |  Height:  |  Size: 74 KiB

After

Width:  |  Height:  |  Size: 74 KiB

BIN
web/src/assets/img/logo.png Normal file

Binary file not shown.

After

Width:  |  Height:  |  Size: 108 KiB

View File

@ -0,0 +1,60 @@
<template>
<div class="right flex-center">
<div class="logo">
<img src="@/assets/img/logo.png" alt="" />
</div>
<div>welcome</div>
<footer-bar />
</div>
</template>
<script setup>
import FooterBar from "@/components/FooterBar.vue";
</script>
<style lang="stylus" scoped>
.right{
font-size: 40px
font-weight: bold
color:#fff
flex-direction: column
background-image url("~@/assets/img/login-bg.png")
background-size cover
background-position center
width: 50%;
min-height: 100vh
max-height: 100vh
background-repeat: no-repeat;
position: relative;
overflow: hidden;
z-index: 1;
::v-deep(.foot-container){
position: absolute;
bottom: 20px;
width: 100%;
background: none;
color: var(--sm-txt);
font-size: 12px;
text-align: center;
.footer{
a,
span{
color: var(--text-fff)
}
}
}
}
.logo{
margin-bottom: 26px;
width: 200px
height: 200px
background: #fff
border-radius: 50%
img{
width: 100%;
object-fit: cover;
height: 100%;
}
}
</style>

View File

@ -0,0 +1,101 @@
<template>
<div>
<ThemeChange />
<div
@click="goBack"
class="flex back animate__animated animate__pulse animate__infinite"
>
<el-icon><ArrowLeftBold /></el-icon
>{{ title === "注册" ? "首页" : "返回" }}
</div>
<div class="title">{{ title }}</div>
<div class="smTitle" v-if="title !== '重置密码'">
{{ title === "登录" ? "没有账号?" : "已有账号?"
}}<span @click="goPageFun" class="text-color-primary sign"
>赶紧{{ title === "登录" ? "注册" : "登录" }}</span
>
</div>
<slot></slot>
<div class="flex orline" v-if="title !== '重置密码'">
<div class="lineor"></div>
<span></span>
<div class="lineor"></div>
</div>
</div>
</template>
<script setup>
import { ArrowLeftBold } from "@element-plus/icons-vue";
import ThemeChange from "@/components/ThemeChange.vue";
import { defineProps } from "vue";
import { useRouter } from "vue-router";
const props = defineProps({
title: {
type: String,
default: "登录"
},
smTitle: { type: String, default: "没有账号?" },
goPage: {
type: String,
default: "/register"
}
});
const router = useRouter();
const goBack = () => {
if (props.title === "注册") {
router.push("/");
} else {
router.go(-1);
}
};
const goPageFun = () => {
if (props.title === "登录") {
router.push("/register");
} else {
router.push("/login");
}
};
</script>
<style lang="stylus" scoped>
.back{
color:var(--sm-txt)
font-size: 14px;
margin-bottom: 140px
margin-top: 18px
cursor: pointer
.el-icon{
margin-right: 6px
}
}
.title{
font-size: 36px
margin-bottom: 16px
color: var(--text-color)
}
.smTitle{
color: var(--text-color)
font-size: 14px;
margin-bottom: 36px
}
.sign{
text-decoration: underline;
cursor :pointer
}
.orline{
color:var(--text-secondary)
span{
font-size: 14px;
margin: 0 10px
}
.lineor{
width: 182px; height: 1px;
background: var(--text-secondary)
}
}
</style>

View File

@ -63,6 +63,8 @@ getLicenseInfo()
width: 100%; width: 100%;
display flex; display flex;
justify-content center justify-content center
background: var(--theme-bg);
margin-top -4px
.footer { .footer {
max-width 400px; max-width 400px;

View File

@ -1,18 +1,18 @@
<template> <template>
<el-dialog <el-dialog
class="login-dialog" class="login-dialog"
v-model="showDialog" v-model="showDialog"
:close-on-click-modal="true" :close-on-click-modal="true"
:show-close="false" :show-close="false"
:before-close="close" :before-close="close"
> >
<template #header="{titleId, titleClass }"> <template #header="{ titleId, titleClass }">
<div class="header"> <div class="header">
<div class="title" v-if="login">用户登录</div> <div class="title" v-if="login">用户登录-</div>
<div class="title" v-else>用户注册</div> <div class="title" v-else>用户注册</div>
<div class="close-icon"> <div class="close-icon">
<el-icon @click="close"> <el-icon @click="close">
<Close/> <Close />
</el-icon> </el-icon>
</div> </div>
</div> </div>
@ -21,26 +21,32 @@
<div class="login-box" v-if="login"> <div class="login-box" v-if="login">
<el-form :model="data" label-width="120px" class="form"> <el-form :model="data" label-width="120px" class="form">
<div class="block"> <div class="block">
<el-input placeholder="账号" <el-input
size="large" placeholder="账号"
v-model="data.username" size="large"
autocomplete="off"> v-model="data.username"
autocomplete="off"
>
<template #prefix> <template #prefix>
<el-icon> <el-icon>
<Iphone/> <Iphone />
</el-icon> </el-icon>
</template> </template>
</el-input> </el-input>
</div> </div>
<div class="block"> <div class="block">
<el-input placeholder="请输入密码(8-16位)" <el-input
maxlength="16" size="large" placeholder="请输入密码(8-16位)"
v-model="data.password" show-password maxlength="16"
autocomplete="off"> size="large"
v-model="data.password"
show-password
autocomplete="off"
>
<template #prefix> <template #prefix>
<el-icon> <el-icon>
<Lock/> <Lock />
</el-icon> </el-icon>
</template> </template>
</el-input> </el-input>
@ -48,7 +54,13 @@
<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="submitLogin">登录</el-button> <el-button
class="login-btn"
type="primary"
size="large"
@click="submitLogin"
>登录</el-button
>
</el-col> </el-col>
</el-row> </el-row>
@ -56,9 +68,21 @@
<el-col :span="12"> <el-col :span="12">
<div class="reg"> <div class="reg">
还没有账号 还没有账号
<el-button type="primary" class="forget" size="small" @click="login = false">注册</el-button> <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> <el-button
type="info"
class="forget"
size="small"
@click="showResetPass = true"
>忘记密码</el-button
>
</div> </div>
</el-col> </el-col>
@ -66,7 +90,12 @@
<div class="c-login" v-if="wechatLoginURL !== ''"> <div class="c-login" v-if="wechatLoginURL !== ''">
<div class="text">其他登录方式</div> <div class="text">其他登录方式</div>
<div class="login-type"> <div class="login-type">
<a class="wechat-login" :href="wechatLoginURL" @click="setRoute(router.currentRoute.value.path)"><i class="iconfont icon-wechat"></i></a> <a
class="wechat-login"
:href="wechatLoginURL"
@click="setRoute(router.currentRoute.value.path)"
><i class="iconfont icon-wechat"></i
></a>
</div> </div>
</div> </div>
</el-col> </el-col>
@ -79,14 +108,16 @@
<el-tabs v-model="activeName" class="demo-tabs"> <el-tabs v-model="activeName" class="demo-tabs">
<el-tab-pane label="手机注册" name="mobile" v-if="enableMobile"> <el-tab-pane label="手机注册" name="mobile" v-if="enableMobile">
<div class="block"> <div class="block">
<el-input placeholder="手机号码" <el-input
size="large" placeholder="手机号码"
v-model="data.mobile" size="large"
maxlength="11" v-model="data.mobile"
autocomplete="off"> maxlength="11"
autocomplete="off"
>
<template #prefix> <template #prefix>
<el-icon> <el-icon>
<Iphone/> <Iphone />
</el-icon> </el-icon>
</template> </template>
</el-input> </el-input>
@ -94,32 +125,41 @@
<div class="block"> <div class="block">
<el-row :gutter="10"> <el-row :gutter="10">
<el-col :span="12"> <el-col :span="12">
<el-input placeholder="验证码" <el-input
size="large" maxlength="30" placeholder="验证码"
v-model="data.code" size="large"
autocomplete="off"> maxlength="30"
v-model="data.code"
autocomplete="off"
>
<template #prefix> <template #prefix>
<el-icon> <el-icon>
<Checked/> <Checked />
</el-icon> </el-icon>
</template> </template>
</el-input> </el-input>
</el-col> </el-col>
<el-col :span="12"> <el-col :span="12">
<send-msg size="large" :receiver="data.mobile" type="mobile"/> <send-msg
size="large"
:receiver="data.mobile"
type="mobile"
/>
</el-col> </el-col>
</el-row> </el-row>
</div> </div>
</el-tab-pane> </el-tab-pane>
<el-tab-pane label="邮箱注册" name="email" v-if="enableEmail"> <el-tab-pane label="邮箱注册" name="email" v-if="enableEmail">
<div class="block"> <div class="block">
<el-input placeholder="邮箱地址" <el-input
size="large" placeholder="邮箱地址"
v-model="data.email" size="large"
autocomplete="off"> v-model="data.email"
autocomplete="off"
>
<template #prefix> <template #prefix>
<el-icon> <el-icon>
<Message/> <Message />
</el-icon> </el-icon>
</template> </template>
</el-input> </el-input>
@ -127,32 +167,37 @@
<div class="block"> <div class="block">
<el-row :gutter="10"> <el-row :gutter="10">
<el-col :span="12"> <el-col :span="12">
<el-input placeholder="验证码" <el-input
size="large" maxlength="30" placeholder="验证码"
v-model="data.code" size="large"
autocomplete="off"> maxlength="30"
v-model="data.code"
autocomplete="off"
>
<template #prefix> <template #prefix>
<el-icon> <el-icon>
<Checked/> <Checked />
</el-icon> </el-icon>
</template> </template>
</el-input> </el-input>
</el-col> </el-col>
<el-col :span="12"> <el-col :span="12">
<send-msg size="large" :receiver="data.email" type="email"/> <send-msg size="large" :receiver="data.email" type="email" />
</el-col> </el-col>
</el-row> </el-row>
</div> </div>
</el-tab-pane> </el-tab-pane>
<el-tab-pane label="用户名注册" name="username" v-if="enableUser"> <el-tab-pane label="用户名注册" name="username" v-if="enableUser">
<div class="block"> <div class="block">
<el-input placeholder="用户名" <el-input
size="large" placeholder="用户名"
v-model="data.username" size="large"
autocomplete="off"> v-model="data.username"
autocomplete="off"
>
<template #prefix> <template #prefix>
<el-icon> <el-icon>
<Iphone/> <Iphone />
</el-icon> </el-icon>
</template> </template>
</el-input> </el-input>
@ -161,38 +206,49 @@
</el-tabs> </el-tabs>
<div class="block"> <div class="block">
<el-input placeholder="请输入密码(8-16位)" <el-input
maxlength="16" size="large" placeholder="请输入密码(8-16位)"
v-model="data.password" show-password maxlength="16"
autocomplete="off"> size="large"
v-model="data.password"
show-password
autocomplete="off"
>
<template #prefix> <template #prefix>
<el-icon> <el-icon>
<Lock/> <Lock />
</el-icon> </el-icon>
</template> </template>
</el-input> </el-input>
</div> </div>
<div class="block"> <div class="block">
<el-input placeholder="重复密码(8-16位)" <el-input
size="large" maxlength="16" v-model="data.repass" show-password placeholder="重复密码(8-16位)"
autocomplete="off"> size="large"
maxlength="16"
v-model="data.repass"
show-password
autocomplete="off"
>
<template #prefix> <template #prefix>
<el-icon> <el-icon>
<Lock/> <Lock />
</el-icon> </el-icon>
</template> </template>
</el-input> </el-input>
</div> </div>
<div class="block"> <div class="block">
<el-input placeholder="邀请码(可选)" <el-input
size="large" placeholder="邀请码(可选)"
v-model="data.invite_code" size="large"
autocomplete="off"> v-model="data.invite_code"
autocomplete="off"
>
<template #prefix> <template #prefix>
<el-icon> <el-icon>
<Message/> <Message />
</el-icon> </el-icon>
</template> </template>
</el-input> </el-input>
@ -200,7 +256,13 @@
<el-row class="btn-row" :gutter="20"> <el-row class="btn-row" :gutter="20">
<el-col :span="12"> <el-col :span="12">
<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-col :span="12"> <el-col :span="12">
<div class="text"> <div class="text">
@ -208,7 +270,6 @@
<el-tag @click="login = true">登录</el-tag> <el-tag @click="login = true">登录</el-tag>
</div> </div>
</el-col> </el-col>
</el-row> </el-row>
</el-form> </el-form>
@ -224,45 +285,48 @@
<el-col :span="12"> <el-col :span="12">
<div class="wechat-card"> <div class="wechat-card">
<el-image :src="wxImg"/> <el-image :src="wxImg" />
</div> </div>
</el-col> </el-col>
</el-row> </el-row>
</div> </div>
</div> </div>
<captcha v-if="enableVerify" @success="submit" ref="captchaRef"/> <captcha v-if="enableVerify" @success="submit" ref="captchaRef" />
<reset-pass @hide="showResetPass = false" :show="showResetPass"/> <reset-pass @hide="showResetPass = false" :show="showResetPass" />
</el-dialog> </el-dialog>
</template> </template>
<script setup> <script setup>
import {onMounted, ref, watch} from "vue" import { onMounted, ref, watch } from "vue";
import {httpGet, httpPost} from "@/utils/http"; import { httpGet, httpPost } from "@/utils/http";
import {ElMessage} from "element-plus"; import { ElMessage } from "element-plus";
import {setUserToken} from "@/store/session"; import { setUserToken } from "@/store/session";
import {validateEmail, validateMobile} from "@/utils/validate"; import { validateEmail, validateMobile } from "@/utils/validate";
import {Checked, Close, Iphone, Lock, Message} from "@element-plus/icons-vue"; import { Checked, Close, Iphone, Lock, Message } from "@element-plus/icons-vue";
import SendMsg from "@/components/SendMsg.vue"; import SendMsg from "@/components/SendMsg.vue";
import {arrayContains} from "@/utils/libs"; import { arrayContains } from "@/utils/libs";
import {getSystemInfo} from "@/store/cache"; import { getSystemInfo } from "@/store/cache";
import Captcha from "@/components/Captcha.vue"; import Captcha from "@/components/Captcha.vue";
import ResetPass from "@/components/ResetPass.vue"; import ResetPass from "@/components/ResetPass.vue";
import {setRoute} from "@/store/system"; import { setRoute } from "@/store/system";
import {useRouter} from "vue-router"; import { useRouter } from "vue-router";
import {useSharedStore} from "@/store/sharedata"; import { useSharedStore } from "@/store/sharedata";
// eslint-disable-next-line no-undef // eslint-disable-next-line no-undef
const props = defineProps({ const props = defineProps({
show: Boolean, show: Boolean
}); });
const showDialog = ref(false) const showDialog = ref(false);
watch(() => props.show, (newValue) => { watch(
showDialog.value = newValue () => props.show,
}) (newValue) => {
showDialog.value = newValue;
}
);
const login = ref(true) const login = ref(true);
const data = ref({ const data = ref({
username: process.env.VUE_APP_USER, username: process.env.VUE_APP_USER,
password: process.env.VUE_APP_PASS, password: process.env.VUE_APP_PASS,
@ -271,158 +335,169 @@ const data = ref({
repass: "", repass: "",
code: "", code: "",
invite_code: "" invite_code: ""
}) });
const enableMobile = ref(false) const enableMobile = ref(false);
const enableEmail = ref(false) const enableEmail = ref(false);
const enableUser = ref(false) const enableUser = ref(false);
const enableRegister = ref(true) const enableRegister = ref(true);
const wechatLoginURL = ref('') const wechatLoginURL = ref("");
const activeName = ref("") const activeName = ref("");
const wxImg = ref("/images/wx.png") const wxImg = ref("/images/wx.png");
const captchaRef = ref(null) const captchaRef = ref(null);
// eslint-disable-next-line no-undef // eslint-disable-next-line no-undef
const emits = defineEmits(['hide', 'success']); const emits = defineEmits(["hide", "success"]);
const action = ref("login") const action = ref("login");
const enableVerify = ref(false) const enableVerify = ref(false);
const showResetPass = ref(false) const showResetPass = ref(false);
const router = useRouter() const router = useRouter();
const store = useSharedStore() const store = useSharedStore();
// //
const needVerify = ref(false) const needVerify = ref(false);
onMounted(() => { onMounted(() => {
const returnURL = `${location.protocol}//${location.host}/login/callback?action=login` const returnURL = `${location.protocol}//${location.host}/login/callback?action=login`;
httpGet("/api/user/clogin?return_url="+returnURL).then(res => { httpGet("/api/user/clogin?return_url=" + returnURL)
wechatLoginURL.value = res.data.url .then((res) => {
}).catch(e => { wechatLoginURL.value = res.data.url;
console.log(e.message) })
}) .catch((e) => {
console.log(e.message);
});
getSystemInfo().then(res => { getSystemInfo()
if (res.data) { .then((res) => {
const registerWays = res.data['register_ways'] if (res.data) {
if (arrayContains(registerWays, "username")) { const registerWays = res.data["register_ways"];
enableUser.value = true if (arrayContains(registerWays, "username")) {
activeName.value = 'username' enableUser.value = true;
activeName.value = "username";
}
if (arrayContains(registerWays, "email")) {
enableEmail.value = true;
activeName.value = "email";
}
if (arrayContains(registerWays, "mobile")) {
enableMobile.value = true;
activeName.value = "mobile";
}
//
enableRegister.value = res.data["enabled_register"];
// 使
if (res.data["wechat_card_url"] !== "") {
wxImg.value = res.data["wechat_card_url"];
}
enableVerify.value = res.data["enabled_verify"];
} }
if (arrayContains(registerWays, "email")) { })
enableEmail.value = true .catch((e) => {
activeName.value = 'email' ElMessage.error("获取系统配置失败:" + e.message);
} });
if (arrayContains(registerWays, "mobile")) { });
enableMobile.value = true
activeName.value = 'mobile'
}
//
enableRegister.value = res.data['enabled_register']
// 使
if (res.data['wechat_card_url'] !== '') {
wxImg.value = res.data['wechat_card_url']
}
enableVerify.value = res.data['enabled_verify']
}
}).catch(e => {
ElMessage.error("获取系统配置失败:" + e.message)
})
})
const submit = (verifyData) => { const submit = (verifyData) => {
if (action.value === "login") { if (action.value === "login") {
doLogin(verifyData) doLogin(verifyData);
} else if (action.value === "register") { } else if (action.value === "register") {
doRegister(verifyData) doRegister(verifyData);
} }
} };
// //
const submitLogin = () => { const submitLogin = () => {
if (data.value.username === '') { if (data.value.username === "") {
return ElMessage.error('请输入用户名'); return ElMessage.error("请输入用户名");
} }
if (data.value.password === '') { if (data.value.password === "") {
return ElMessage.error('请输入密码'); return ElMessage.error("请输入密码");
} }
if (enableVerify.value && needVerify.value) { if (enableVerify.value && needVerify.value) {
captchaRef.value.loadCaptcha() captchaRef.value.loadCaptcha();
action.value = "login" action.value = "login";
} else { } else {
doLogin({}) doLogin({});
} }
} };
const doLogin = (verifyData) => { const doLogin = (verifyData) => {
data.value.key = verifyData.key data.value.key = verifyData.key;
data.value.dots = verifyData.dots data.value.dots = verifyData.dots;
data.value.x = verifyData.x data.value.x = verifyData.x;
httpPost('/api/user/login', data.value).then((res) => { httpPost("/api/user/login", data.value)
setUserToken(res.data.token) .then((res) => {
store.setIsLogin(true) setUserToken(res.data.token);
ElMessage.success("登录成功!") store.setIsLogin(true);
emits("hide") ElMessage.success("登录成功!");
emits('success') emits("hide");
needVerify.value = false emits("success");
}).catch((e) => { needVerify.value = false;
ElMessage.error('登录失败,' + e.message) })
needVerify.value = true .catch((e) => {
}) ElMessage.error("登录失败," + e.message);
} needVerify.value = true;
});
};
// //
const submitRegister = () => { const submitRegister = () => {
if (activeName.value === 'username' && data.value.username === '') { if (activeName.value === "username" && data.value.username === "") {
return ElMessage.error('请输入用户名'); return ElMessage.error("请输入用户名");
} }
if (activeName.value === 'mobile' && !validateMobile(data.value.mobile)) { if (activeName.value === "mobile" && !validateMobile(data.value.mobile)) {
return ElMessage.error('请输入合法的手机号'); return ElMessage.error("请输入合法的手机号");
} }
if (activeName.value === 'email' && !validateEmail(data.value.email)) { if (activeName.value === "email" && !validateEmail(data.value.email)) {
return ElMessage.error('请输入合法的邮箱地址'); return ElMessage.error("请输入合法的邮箱地址");
} }
if (data.value.password.length < 8) { if (data.value.password.length < 8) {
return ElMessage.error('密码的长度为8-16个字符'); return ElMessage.error("密码的长度为8-16个字符");
} }
if (data.value.repass !== data.value.password) { if (data.value.repass !== data.value.password) {
return ElMessage.error('两次输入密码不一致'); return ElMessage.error("两次输入密码不一致");
} }
if ((activeName.value === 'mobile' || activeName.value === 'email') && data.value.code === '') { if (
return ElMessage.error('请输入验证码'); (activeName.value === "mobile" || activeName.value === "email") &&
data.value.code === ""
) {
return ElMessage.error("请输入验证码");
} }
if (enableVerify.value && activeName.value === 'username') { if (enableVerify.value && activeName.value === "username") {
captchaRef.value.loadCaptcha() captchaRef.value.loadCaptcha();
action.value = "register" action.value = "register";
} else { } else {
doRegister({}) doRegister({});
} }
} };
const doRegister = (verifyData) => { const doRegister = (verifyData) => {
data.value.key = verifyData.key data.value.key = verifyData.key;
data.value.dots = verifyData.dots data.value.dots = verifyData.dots;
data.value.x = verifyData.x data.value.x = verifyData.x;
data.value.reg_way = activeName.value data.value.reg_way = activeName.value;
httpPost('/api/user/register', data.value).then((res) => { httpPost("/api/user/register", data.value)
setUserToken(res.data.token) .then((res) => {
ElMessage.success({ setUserToken(res.data.token);
"message": "注册成功!", ElMessage.success({
onClose: () => { message: "注册成功!",
emits("hide") onClose: () => {
emits('success') emits("hide");
}, emits("success");
duration: 1000 },
duration: 1000
});
}) })
}).catch((e) => { .catch((e) => {
ElMessage.error('注册失败,' + e.message) ElMessage.error("注册失败," + e.message);
}) });
} };
const close = function () { const close = function () {
emits('hide', false) emits("hide", false);
login.value = true login.value = true;
} };
</script> </script>
<style lang="stylus"> <style lang="stylus">

View File

@ -1,43 +1,42 @@
<template> <template>
<div class="reset-pass"> <div class="reset-pass">
<el-dialog <el-dialog
v-model="showDialog" v-model="showDialog"
:close-on-click-modal="true" :close-on-click-modal="true"
width="540px" width="540px"
:before-close="close" :before-close="close"
:title="title" :title="title"
class="reset-pass-dialog" class="reset-pass-dialog"
> >
<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-tabs v-model="form.type" class="demo-tabs"> <el-tabs v-model="form.type" class="demo-tabs">
<el-tab-pane label="手机号验证" name="mobile"> <el-tab-pane label="手机号验证" name="mobile">
<el-form-item label="手机号"> <el-form-item label="手机号">
<el-input v-model="form.mobile" placeholder="请输入手机号"/> <el-input v-model="form.mobile" placeholder="请输入手机号" />
</el-form-item> </el-form-item>
<el-form-item label="验证码"> <el-form-item label="验证码">
<el-row class="code-row"> <el-row class="code-row">
<el-col :span="16"> <el-col :span="16">
<el-input v-model="form.code" maxlength="6"/> <el-input v-model="form.code" maxlength="6" />
</el-col> </el-col>
<el-col :span="8" class="send-button"> <el-col :span="8" class="send-button">
<send-msg size="" :receiver="form.mobile" type="mobile"/> <send-msg size="" :receiver="form.mobile" type="mobile" />
</el-col> </el-col>
</el-row> </el-row>
</el-form-item> </el-form-item>
</el-tab-pane> </el-tab-pane>
<el-tab-pane label="邮箱验证" name="email"> <el-tab-pane label="邮箱验证" name="email">
<el-form-item label="邮箱地址"> <el-form-item label="邮箱地址">
<el-input v-model="form.email" placeholder="请输入邮箱地址"/> <el-input v-model="form.email" placeholder="请输入邮箱地址" />
</el-form-item> </el-form-item>
<el-form-item label="验证码"> <el-form-item label="验证码">
<el-row class="code-row"> <el-row class="code-row">
<el-col :span="16"> <el-col :span="16">
<el-input v-model="form.code" maxlength="6"/> <el-input v-model="form.code" maxlength="6" />
</el-col> </el-col>
<el-col :span="8" class="send-button"> <el-col :span="8" class="send-button">
<send-msg size="" :receiver="form.email" type="email"/> <send-msg size="" :receiver="form.email" type="email" />
</el-col> </el-col>
</el-row> </el-row>
</el-form-item> </el-form-item>
@ -45,19 +44,17 @@
</el-tabs> </el-tabs>
<el-form-item label="新密码"> <el-form-item label="新密码">
<el-input v-model="form.password" type="password"/> <el-input v-model="form.password" type="password" />
</el-form-item> </el-form-item>
<el-form-item label="重复密码"> <el-form-item label="重复密码">
<el-input v-model="form.repass" type="password"/> <el-input v-model="form.repass" type="password" />
</el-form-item> </el-form-item>
</el-form> </el-form>
</div> </div>
<template #footer> <template #footer>
<div class="dialog-footer"> <div class="dialog-footer">
<el-button type="primary" @click="save" round> <el-button type="primary" @click="save" round> 重置密码 </el-button>
重置密码
</el-button>
</div> </div>
</template> </template>
</el-dialog> </el-dialog>
@ -65,11 +62,11 @@
</template> </template>
<script setup> <script setup>
import {computed, ref} from "vue"; import { computed, ref } from "vue";
import SendMsg from "@/components/SendMsg.vue"; 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";
const props = defineProps({ const props = defineProps({
show: Boolean, show: Boolean,
@ -77,23 +74,23 @@ const props = defineProps({
}); });
const showDialog = computed(() => { const showDialog = computed(() => {
return props.show return props.show;
}) });
const title = ref('重置密码') const title = ref("重置密码");
const form = ref({ const form = ref({
mobile: '', mobile: "",
email: '', email: "",
type: 'mobile', type: "mobile",
code: '', code: "",
password: '', password: "",
repass: '' repass: ""
}) });
const emits = defineEmits(['hide']); const emits = defineEmits(["hide"]);
const save = () => { const save = () => {
if (form.value.code === '') { if (form.value.code === "") {
return ElMessage.error("请输入验证码"); return ElMessage.error("请输入验证码");
} }
if (form.value.password.length < 8) { if (form.value.password.length < 8) {
@ -103,18 +100,22 @@ const save = () => {
return ElMessage.error("两次输入密码不一致"); return ElMessage.error("两次输入密码不一致");
} }
httpPost('/api/user/resetPass', form.value).then(() => { httpPost("/api/user/resetPass", form.value)
ElMessage.success({ .then(() => {
message: '重置密码成功', duration: 1000, onClose: () => emits('hide', false) ElMessage.success({
message: "重置密码成功",
duration: 1000,
onClose: () => emits("hide", false)
});
}) })
}).catch(e => { .catch((e) => {
ElMessage.error("重置密码失败:" + e.message); ElMessage.error("重置密码失败:" + e.message);
}) });
} };
const close = function () { const close = function () {
emits('hide', false); emits("hide", false);
} };
</script> </script>
<style lang="stylus"> <style lang="stylus">
@ -140,5 +141,4 @@ const close = function () {
} }
} }
} }
</style> </style>

View File

@ -1,21 +1,28 @@
<template> <template>
<el-container class="send-verify-code"> <el-container class="send-verify-code">
<el-button type="primary" class="send-btn" :size="props.size" :disabled="!canSend" @click="sendMsg" plain> <el-button
type="primary"
class="btn-normal"
:size="props.size"
:disabled="!canSend"
@click="sendMsg"
plain
>
{{ btnText }} {{ btnText }}
</el-button> </el-button>
<captcha @success="doSendMsg" ref="captchaRef"/> <captcha @success="doSendMsg" ref="captchaRef" />
</el-container> </el-container>
</template> </template>
<script setup> <script setup>
// //
import {ref} from "vue"; import { ref } from "vue";
import {validateEmail, validateMobile} from "@/utils/validate"; import { validateEmail, validateMobile } from "@/utils/validate";
import {httpPost} from "@/utils/http"; import { httpPost } from "@/utils/http";
import {showMessageError, showMessageOK} from "@/utils/dialog"; import { showMessageError, showMessageOK } from "@/utils/dialog";
import Captcha from "@/components/Captcha.vue"; import Captcha from "@/components/Captcha.vue";
import {getSystemInfo} from "@/store/cache"; import { getSystemInfo } from "@/store/cache";
// eslint-disable-next-line no-undef // eslint-disable-next-line no-undef
const props = defineProps({ const props = defineProps({
@ -23,58 +30,65 @@ const props = defineProps({
size: String, size: String,
type: { type: {
type: String, type: String,
default: 'mobile' default: "mobile"
} }
}); });
const btnText = ref('发送验证码') const btnText = ref("发送验证码");
const canSend = ref(true) const canSend = ref(true);
const captchaRef = ref(null) const captchaRef = ref(null);
const enableVerify = ref(false) const enableVerify = ref(false);
getSystemInfo().then(res => { getSystemInfo().then((res) => {
enableVerify.value = res.data['enabled_verify'] enableVerify.value = res.data["enabled_verify"];
}) });
const sendMsg = () => { const sendMsg = () => {
if (!validateMobile(props.receiver) && props.type === 'mobile') { if (!validateMobile(props.receiver) && props.type === "mobile") {
return showMessageError("请输入合法的手机号") return showMessageError("请输入合法的手机号");
} }
if (!validateEmail(props.receiver) && props.type === 'email') { if (!validateEmail(props.receiver) && props.type === "email") {
return showMessageError("请输入合法的邮箱地址") return showMessageError("请输入合法的邮箱地址");
} }
if (enableVerify.value) { if (enableVerify.value) {
captchaRef.value.loadCaptcha() captchaRef.value.loadCaptcha();
} else { } else {
doSendMsg({}) doSendMsg({});
} }
} };
const doSendMsg = (data) => { const doSendMsg = (data) => {
if (!canSend.value) { if (!canSend.value) {
return return;
} }
canSend.value = false canSend.value = false;
httpPost('/api/sms/code', {receiver: props.receiver, key: data.key, dots: data.dots, x:data.x}).then(() => { httpPost("/api/sms/code", {
showMessageOK('验证码发送成功') receiver: props.receiver,
let time = 60 key: data.key,
btnText.value = time dots: data.dots,
const handler = setInterval(() => { x: data.x
time = time - 1
if (time <= 0) {
clearInterval(handler)
btnText.value = '重新发送'
canSend.value = true
} else {
btnText.value = time
}
}, 1000)
}).catch(e => {
canSend.value = true
showMessageError('验证码发送失败:' + e.message)
}) })
} .then(() => {
showMessageOK("验证码发送成功");
let time = 60;
btnText.value = time;
const handler = setInterval(() => {
time = time - 1;
if (time <= 0) {
clearInterval(handler);
btnText.value = "重新发送";
canSend.value = true;
} else {
btnText.value = time;
}
}, 1000);
})
.catch((e) => {
canSend.value = true;
showMessageError("验证码发送失败:" + e.message);
});
};
</script> </script>
<style lang="stylus" scoped> <style lang="stylus" scoped>

View File

@ -27,18 +27,19 @@ onMounted(() => {
<style lang="stylus" scoped> <style lang="stylus" scoped>
@import '@/assets/iconfont/iconfont.css' @import '@/assets/iconfont/iconfont.css'
.theme-box{ .theme-box{
z-index :111
position: fixed; position: fixed;
right: 40px; right: 40px;
bottom: 262px; bottom: 262px;
cursor: pointer; cursor: pointer;
background-color: #fff;
border: 1px solid #ccc; border: 1px solid #ccc;
border-radius: 50%; border-radius: 50%;
width 35px; width 35px;
height: 35px; height: 35px;
line-height: 35px; line-height: 35px;
text-align: center; text-align: center;
background-color: rgb(146, 147, 148); // background-color: rgb(146, 147, 148);
background: linear-gradient(135deg, rgba(134, 140, 255, 1) 0%, rgba(67, 24, 255, 1) 100%);
transition: all 0.3s ease; transition: all 0.3s ease;
&:hover{ &:hover{

View File

@ -5,352 +5,357 @@
// * @Author yangjian102621@163.com // * @Author yangjian102621@163.com
// * +++++++++++++++++++++++++++++++++++++++++++++++++++++++++++ // * +++++++++++++++++++++++++++++++++++++++++++++++++++++++++++
import {createRouter, createWebHistory} from "vue-router"; import { createRouter, createWebHistory } from "vue-router";
const routes = [ const routes = [
{ {
name: 'Index', name: "Index",
path: '/', path: "/",
meta: {title: "首页"}, meta: { title: "首页" },
component: () => import('@/views/Index.vue'), component: () => import("@/views/Index.vue")
}, },
{ {
name: 'home', name: "home",
path: '/home', path: "/home",
redirect: '/chat', redirect: "/chat",
component: () => import('@/views/Home.vue'), component: () => import("@/views/Home.vue"),
children: [ children: [
{ {
name: 'chat', name: "chat",
path: '/chat', path: "/chat",
meta: {title: '创作中心'}, meta: { title: "创作中心" },
component: () => import('@/views/ChatPlus.vue'), component: () => import("@/views/ChatPlus.vue")
}, },
{ {
name: 'chat-id', name: "chat-id",
path: '/chat/:id', path: "/chat/:id",
meta: {title: '创作中心'}, meta: { title: "创作中心" },
component: () => import('@/views/ChatPlus.vue'), component: () => import("@/views/ChatPlus.vue")
}, },
{ {
name: 'image-mj', name: "image-mj",
path: '/mj', path: "/mj",
meta: {title: 'MidJourney 绘画中心'}, meta: { title: "MidJourney 绘画中心" },
component: () => import('@/views/ImageMj.vue'), component: () => import("@/views/ImageMj.vue")
}, },
{ {
name: 'image-sd', name: "image-sd",
path: '/sd', path: "/sd",
meta: {title: 'stable diffusion 绘画中心'}, meta: { title: "stable diffusion 绘画中心" },
component: () => import('@/views/ImageSd.vue'), component: () => import("@/views/ImageSd.vue")
}, },
{ {
name: 'member', name: "member",
path: '/member', path: "/member",
meta: {title: '会员充值中心'}, meta: { title: "会员充值中心" },
component: () => import('@/views/Member.vue'), component: () => import("@/views/Member.vue")
}, },
{ {
name: 'chat-app', name: "chat-app",
path: '/apps', path: "/apps",
meta: {title: '应用中心'}, meta: { title: "应用中心" },
component: () => import('@/views/ChatApps.vue'), component: () => import("@/views/ChatApps.vue")
}, },
{ {
name: 'images', name: "images",
path: '/images-wall', path: "/images-wall",
meta: {title: '作品展示'}, meta: { title: "作品展示" },
component: () => import('@/views/ImagesWall.vue'), component: () => import("@/views/ImagesWall.vue")
}, },
{ {
name: 'user-invitation', name: "user-invitation",
path: '/invite', path: "/invite",
meta: {title: '推广计划'}, meta: { title: "推广计划" },
component: () => import('@/views/Invitation.vue'), component: () => import("@/views/Invitation.vue")
}, },
{ {
name: 'powerLog', name: "powerLog",
path: '/powerLog', path: "/powerLog",
meta: {title: '消费日志'}, meta: { title: "消费日志" },
component: () => import('@/views/PowerLog.vue'), component: () => import("@/views/PowerLog.vue")
}, },
{ {
name: 'xmind', name: "xmind",
path: '/xmind', path: "/xmind",
meta: {title: '思维导图'}, meta: { title: "思维导图" },
component: () => import('@/views/MarkMap.vue'), component: () => import("@/views/MarkMap.vue")
}, },
{ {
name: 'dalle', name: "dalle",
path: '/dalle', path: "/dalle",
meta: {title: 'DALLE-3'}, meta: { title: "DALLE-3" },
component: () => import('@/views/Dalle.vue'), component: () => import("@/views/Dalle.vue")
}, },
{ {
name: 'suno', name: "suno",
path: '/suno', path: "/suno",
meta: {title: 'Suno音乐创作'}, meta: { title: "Suno音乐创作" },
component: () => import('@/views/Suno.vue'), component: () => import("@/views/Suno.vue")
}, },
{ {
name: 'ExternalLink', name: "ExternalLink",
path: '/external', path: "/external",
component: () => import('@/views/ExternalPage.vue'), component: () => import("@/views/ExternalPage.vue")
}, },
{ {
name: 'song', name: "song",
path: '/song/:id', path: "/song/:id",
meta: {title: 'Suno音乐播放'}, meta: { title: "Suno音乐播放" },
component: () => import('@/views/Song.vue'), component: () => import("@/views/Song.vue")
}, },
{ {
name: 'luma', name: "luma",
path: '/luma', path: "/luma",
meta: {title: 'Luma视频创作'}, meta: { title: "Luma视频创作" },
component: () => import('@/views/Luma.vue'), component: () => import("@/views/Luma.vue")
}, }
] ]
}, },
{ {
name: 'chat-export', name: "chat-export",
path: '/chat/export', path: "/chat/export",
meta: {title: '导出会话记录'}, meta: { title: "导出会话记录" },
component: () => import('@/views/ChatExport.vue'), component: () => import("@/views/ChatExport.vue")
}, },
{ {
name: 'login', name: "login",
path: '/login', path: "/login",
meta: {title: '用户登录'}, meta: { title: "用户登录" },
component: () => import('@/views/Login.vue'), component: () => import("@/views/Login.vue")
}, },
{
name: 'login-callback',
path: '/login/callback',
meta: {title: '用户登录'},
component: () => import('@/views/LoginCallback.vue'),
},
{
name: 'register',
path: '/register',
meta: {title: '用户注册'}, {
component: () => import('@/views/Register.vue'), name: "login-callback",
}, path: "/login/callback",
{ meta: { title: "用户登录" },
path: '/admin/login', component: () => import("@/views/LoginCallback.vue")
name: 'admin-login', },
meta: {title: '控制台登录'}, {
component: () => import('@/views/admin/Login.vue'), name: "register",
}, path: "/register",
{
path: '/payReturn',
name: 'pay-return',
meta: {title: '支付回调'},
component: () => import('@/views/PayReturn.vue'),
},
{
name: 'admin',
path: '/admin',
redirect: '/admin/dashboard',
component: () => import("@/views/admin/Home.vue"),
meta: {title: 'Geek-AI 控制台'},
children: [
{
path: '/admin/dashboard',
name: 'admin-dashboard',
meta: {title: '仪表盘'},
component: () => import('@/views/admin/Dashboard.vue'),
},
{
path: '/admin/system',
name: 'admin-system',
meta: {title: '系统设置'},
component: () => import('@/views/admin/SysConfig.vue'),
},
{
path: '/admin/user',
name: 'admin-user',
meta: {title: '用户管理'},
component: () => import('@/views/admin/Users.vue'),
},
{
path: '/admin/app',
name: 'admin-app',
meta: {title: '应用列表'},
component: () => import('@/views/admin/Apps.vue'),
},
{
path: '/admin/app/type',
name: 'admin-app-type',
meta: {title: '应用分类'},
component: () => import('@/views/admin/AppType.vue'),
},
{
path: '/admin/apikey',
name: 'admin-apikey',
meta: {title: 'API-KEY 管理'},
component: () => import('@/views/admin/ApiKey.vue'),
},
{
path: '/admin/chat/model',
name: 'admin-chat-model',
meta: {title: '语言模型'},
component: () => import('@/views/admin/ChatModel.vue'),
},
{
path: '/admin/product',
name: 'admin-product',
meta: {title: '充值产品'},
component: () => import('@/views/admin/Product.vue'),
},
{
path: '/admin/order',
name: 'admin-order',
meta: {title: '充值订单'},
component: () => import('@/views/admin/Order.vue'),
},
{
path: '/admin/redeem',
name: 'admin-redeem',
meta: {title: '兑换码管理'},
component: () => import('@/views/admin/Redeem.vue'),
},
{
path: '/admin/loginLog',
name: 'admin-loginLog',
meta: {title: '登录日志'},
component: () => import('@/views/admin/LoginLog.vue'),
},
{
path: '/admin/functions',
name: 'admin-functions',
meta: {title: '函数管理'},
component: () => import('@/views/admin/Functions.vue'),
},
{
path: '/admin/chats',
name: 'admin-chats',
meta: {title: '对话管理'},
component: () => import('@/views/admin/ChatList.vue'),
},
{
path: '/admin/images',
name: 'admin-images',
meta: {title: '绘图管理'},
component: () => import('@/views/admin/ImageList.vue'),
},
{
path: '/admin/medias',
name: 'admin-medias',
meta: {title: '音视频管理'},
component: () => import('@/views/admin/Medias.vue'),
},
{
path: '/admin/powerLog',
name: 'admin-power-log',
meta: {title: '算力日志'},
component: () => import('@/views/admin/PowerLog.vue'),
},
{
path: '/admin/manger',
name: 'admin-manger',
meta: {title: '管理员'},
component: () => import('@/views/admin/Manager.vue'),
}
]
},
meta: { title: "用户注册" },
component: () => import("@/views/Register.vue")
},
{
name: "resetpassword",
path: "/resetpassword",
meta: { title: "重置密码" },
component: () => import("@/views/Resetpassword.vue")
},
{
path: "/admin/login",
name: "admin-login",
meta: { title: "控制台登录" },
component: () => import("@/views/admin/Login.vue")
},
{
path: "/payReturn",
name: "pay-return",
meta: { title: "支付回调" },
component: () => import("@/views/PayReturn.vue")
},
{
name: "admin",
path: "/admin",
redirect: "/admin/dashboard",
component: () => import("@/views/admin/Home.vue"),
meta: { title: "Geek-AI 控制台" },
children: [
{
path: "/admin/dashboard",
name: "admin-dashboard",
meta: { title: "仪表盘" },
component: () => import("@/views/admin/Dashboard.vue")
},
{
path: "/admin/system",
name: "admin-system",
meta: { title: "系统设置" },
component: () => import("@/views/admin/SysConfig.vue")
},
{
path: "/admin/user",
name: "admin-user",
meta: { title: "用户管理" },
component: () => import("@/views/admin/Users.vue")
},
{
path: "/admin/app",
name: "admin-app",
meta: { title: "应用列表" },
component: () => import("@/views/admin/Apps.vue")
},
{
path: "/admin/app/type",
name: "admin-app-type",
meta: { title: "应用分类" },
component: () => import("@/views/admin/AppType.vue")
},
{
path: "/admin/apikey",
name: "admin-apikey",
meta: { title: "API-KEY 管理" },
component: () => import("@/views/admin/ApiKey.vue")
},
{
path: "/admin/chat/model",
name: "admin-chat-model",
meta: { title: "语言模型" },
component: () => import("@/views/admin/ChatModel.vue")
},
{
path: "/admin/product",
name: "admin-product",
meta: { title: "充值产品" },
component: () => import("@/views/admin/Product.vue")
},
{
path: "/admin/order",
name: "admin-order",
meta: { title: "充值订单" },
component: () => import("@/views/admin/Order.vue")
},
{
path: "/admin/redeem",
name: "admin-redeem",
meta: { title: "兑换码管理" },
component: () => import("@/views/admin/Redeem.vue")
},
{
path: "/admin/loginLog",
name: "admin-loginLog",
meta: { title: "登录日志" },
component: () => import("@/views/admin/LoginLog.vue")
},
{
path: "/admin/functions",
name: "admin-functions",
meta: { title: "函数管理" },
component: () => import("@/views/admin/Functions.vue")
},
{
path: "/admin/chats",
name: "admin-chats",
meta: { title: "对话管理" },
component: () => import("@/views/admin/ChatList.vue")
},
{
path: "/admin/images",
name: "admin-images",
meta: { title: "绘图管理" },
component: () => import("@/views/admin/ImageList.vue")
},
{
path: "/admin/medias",
name: "admin-medias",
meta: { title: "音视频管理" },
component: () => import("@/views/admin/Medias.vue")
},
{
path: "/admin/powerLog",
name: "admin-power-log",
meta: { title: "算力日志" },
component: () => import("@/views/admin/PowerLog.vue")
},
{
path: "/admin/manger",
name: "admin-manger",
meta: { title: "管理员" },
component: () => import("@/views/admin/Manager.vue")
}
]
},
{ {
name: 'mobile-login', name: "mobile-login",
path: '/mobile/login', path: "/mobile/login",
meta: {title: '用户登录'}, meta: { title: "用户登录" },
component: () => import('@/views/Login.vue'), component: () => import("@/views/Login.vue")
}, },
{ {
name: 'mobile-register', name: "mobile-register",
path: '/mobile/register', path: "/mobile/register",
meta: {title: '用户注册'}, meta: { title: "用户注册" },
component: () => import('@/views/Register.vue'), component: () => import("@/views/Register.vue")
}, },
{ {
name: 'mobile', name: "mobile",
path: '/mobile', path: "/mobile",
meta: {title: '首页'}, meta: { title: "首页" },
component: () => import('@/views/mobile/Home.vue'), component: () => import("@/views/mobile/Home.vue"),
redirect: '/mobile/index', redirect: "/mobile/index",
children: [ children: [
{ {
path: '/mobile/index', path: "/mobile/index",
name: 'mobile-index', name: "mobile-index",
component: () => import('@/views/mobile/Index.vue'), component: () => import("@/views/mobile/Index.vue")
}, },
{ {
path: '/mobile/chat', path: "/mobile/chat",
name: 'mobile-chat', name: "mobile-chat",
component: () => import('@/views/mobile/ChatList.vue'), component: () => import("@/views/mobile/ChatList.vue")
}, },
{ {
path: '/mobile/image', path: "/mobile/image",
name: 'mobile-image', name: "mobile-image",
component: () => import('@/views/mobile/Image.vue'), component: () => import("@/views/mobile/Image.vue")
}, },
{ {
path: '/mobile/profile', path: "/mobile/profile",
name: 'mobile-profile', name: "mobile-profile",
component: () => import('@/views/mobile/Profile.vue'), component: () => import("@/views/mobile/Profile.vue")
}, },
{ {
path: '/mobile/imgWall', path: "/mobile/imgWall",
name: 'mobile-img-wall', name: "mobile-img-wall",
component: () => import('@/views/mobile/pages/ImgWall.vue'), component: () => import("@/views/mobile/pages/ImgWall.vue")
}, },
{ {
path: '/mobile/chat/session', path: "/mobile/chat/session",
name: 'mobile-chat-session', name: "mobile-chat-session",
component: () => import('@/views/mobile/ChatSession.vue'), component: () => import("@/views/mobile/ChatSession.vue")
}, },
{ {
path: '/mobile/chat/export', path: "/mobile/chat/export",
name: 'mobile-chat-export', name: "mobile-chat-export",
component: () => import('@/views/mobile/ChatExport.vue'), component: () => import("@/views/mobile/ChatExport.vue")
}, }
] ]
}, },
{ {
name: 'test', name: "test",
path: '/test', path: "/test",
meta: {title: '测试页面'}, meta: { title: "测试页面" },
component: () => import('@/views/Test.vue'), component: () => import("@/views/Test.vue")
}, },
{ {
name: 'test2', name: "test2",
path: '/test2', path: "/test2",
meta: {title: '测试页面'}, meta: { title: "测试页面" },
component: () => import('@/views/RealtimeTest.vue'), component: () => import("@/views/RealtimeTest.vue")
}, },
{ {
name: 'NotFound', name: "NotFound",
path: '/:all(.*)', path: "/:all(.*)",
meta: {title: '页面没有找到'}, meta: { title: "页面没有找到" },
component: () => import('@/views/404.vue'), component: () => import("@/views/404.vue")
}, }
] ];
// console.log(MY_VARIABLE) // console.log(MY_VARIABLE)
const router = createRouter({ const router = createRouter({
history: createWebHistory(), history: createWebHistory(),
routes: routes, routes: routes
}) });
let prevRoute = null;
let prevRoute = null
// dynamic change the title when router change // dynamic change the title when router change
router.beforeEach((to, from, next) => { router.beforeEach((to, from, next) => {
document.title = to.meta.title document.title = to.meta.title;
prevRoute = from prevRoute = from;
next() next();
}) });
export {router, prevRoute}; export { router, prevRoute };

View File

@ -1,174 +1,187 @@
<template> <template>
<div> <div class="flex-center loginPage">
<div class="bg"></div> <div class="left">
<div class="main"> <div class="login-box">
<div class="contain"> <AccountTop>
<div class="logo"> <template #default>
<el-image :src="logo" fit="cover" @click="router.push('/')"/> <div class="wechatLog flex-center" v-if="wechatLoginURL !== ''">
</div> <a
<div class="header">{{ title }}</div> :href="wechatLoginURL"
<div class="content"> @click="setRoute(router.currentRoute.value.path)"
<div class="block"> >
<el-input placeholder="账号" size="large" v-model="username" autocomplete="off" autofocus <i class="iconfont icon-wechat"></i>使用微信登录
@keyup="handleKeyup"> </a>
<template #prefix>
<el-icon>
<UserFilled/>
</el-icon>
</template>
</el-input>
</div>
<div class="block">
<el-input placeholder="密码" size="large" v-model="password" show-password autocomplete="off"
@keyup="handleKeyup">
<template #prefix>
<el-icon>
<Lock/>
</el-icon>
</template>
</el-input>
</div>
<el-row class="btn-row">
<el-button class="login-btn" size="large" type="primary" @click="login">登录</el-button>
</el-row>
<el-row class="opt" :gutter="24">
<el-col :span="8"><el-link type="primary" @click="router.push('/register')">注册</el-link></el-col>
<el-col :span="8">
<el-link type="info" @click="showResetPass = true">重置密码</el-link>
</el-col>
<el-col :span="8">
<el-link type="info" @click="router.push('/')">首页</el-link>
</el-col>
</el-row>
<div v-if="wechatLoginURL !== ''">
<el-divider class="divider">其他登录方式</el-divider>
<div class="clogin">
<a :href="wechatLoginURL" @click="setRoute(router.currentRoute.value.path)"><i class="iconfont icon-wechat"></i></a>
</div> </div>
</div> </template>
</AccountTop>
<div class="input-form">
<el-form ref="ruleFormRef" :model="ruleForm" :rules="rules">
<el-form-item label="" prop="username">
<div class="form-title">账号</div>
<el-input
v-model="ruleForm.username"
size="large"
clearable
placeholder="请输入账号"
@keyup="handleKeyup"
/>
</el-form-item>
<el-form-item label="" prop="password">
<div class="flex-between w100">
<div class="form-title">密码</div>
<div
class="form-forget text-color-primary"
@click="router.push('/resetpassword')"
>
忘记密码
</div>
</div>
<el-input
clearable
size="large"
v-model="ruleForm.password"
placeholder="请输入密码"
show-password
autocomplete="off"
@keyup="handleKeyup"
/>
</el-form-item>
<el-form-item>
<el-button
class="login-btn"
size="large"
type="primary"
@click="login"
>登录</el-button
>
</el-form-item>
</el-form>
</div> </div>
</div> </div>
<reset-pass @hide="showResetPass = false" :show="showResetPass"/>
<captcha v-if="enableVerify" @success="doLogin" ref="captchaRef"/>
<footer-bar/>
</div> </div>
<account-bg />
</div> </div>
</template> </template>
<script setup> <script setup>
import { onMounted, ref, reactive } from "vue";
import { httpGet, httpPost } from "@/utils/http";
import { useRouter } from "vue-router";
import AccountBg from "@/components/AccountBg.vue";
import { isMobile } from "@/utils/libs";
import { checkSession, getLicenseInfo, getSystemInfo } from "@/store/cache";
import { setUserToken } from "@/store/session";
import { showMessageError } from "@/utils/dialog";
import { setRoute } from "@/store/system";
import { useSharedStore } from "@/store/sharedata";
import {onMounted, ref} from "vue"; import AccountTop from "@/components/AccountTop.vue";
import {Lock, UserFilled} from "@element-plus/icons-vue";
import {httpGet, httpPost} from "@/utils/http";
import {useRouter} from "vue-router";
import FooterBar from "@/components/FooterBar.vue";
import {isMobile} from "@/utils/libs";
import {checkSession, getLicenseInfo, getSystemInfo} from "@/store/cache";
import {setUserToken} from "@/store/session";
import ResetPass from "@/components/ResetPass.vue";
import {showMessageError} from "@/utils/dialog";
import Captcha from "@/components/Captcha.vue";
import {setRoute} from "@/store/system";
import {useSharedStore} from "@/store/sharedata";
const router = useRouter(); const router = useRouter();
const title = ref('Geek-AI'); const title = ref("Geek-AI");
const username = ref(process.env.VUE_APP_USER); const username = ref(process.env.VUE_APP_USER);
const password = ref(process.env.VUE_APP_PASS); const password = ref(process.env.VUE_APP_PASS);
const showResetPass = ref(false)
const logo = ref("")
const licenseConfig = ref({})
const wechatLoginURL = ref('')
const enableVerify = ref(false)
const captchaRef = ref(null)
//
const needVerify = ref(false)
const logo = ref("");
const licenseConfig = ref({});
const wechatLoginURL = ref("");
const enableVerify = ref(false);
const captchaRef = ref(null);
//
const needVerify = ref(false);
const ruleFormRef = ref(null);
const ruleForm = reactive({
username: process.env.VUE_APP_USER,
password: process.env.VUE_APP_PASS
});
const rules = {
username: [{ required: true, trigger: "blur", message: "请输入账号" }],
password: [{ required: true, trigger: "blur", message: "请输入密码" }]
};
onMounted(() => { onMounted(() => {
// //
getSystemInfo().then(res => { getSystemInfo()
logo.value = res.data.logo .then((res) => {
title.value = res.data.title logo.value = res.data.logo;
enableVerify.value = res.data['enabled_verify'] title.value = res.data.title;
}).catch(e => { enableVerify.value = res.data["enabled_verify"];
showMessageError("获取系统配置失败:" + e.message) })
}) .catch((e) => {
showMessageError("获取系统配置失败:" + e.message);
});
getLicenseInfo().then(res => { getLicenseInfo()
licenseConfig.value = res.data .then((res) => {
}).catch(e => { licenseConfig.value = res.data;
showMessageError("获取 License 配置:" + e.message) })
}) .catch((e) => {
showMessageError("获取 License 配置:" + e.message);
});
checkSession().then(() => { checkSession()
if (isMobile()) { .then(() => {
router.push('/mobile') if (isMobile()) {
} else { router.push("/mobile");
router.push('/chat') } else {
} router.push("/chat");
}).catch(() => { }
}) })
.catch(() => {});
const returnURL = `${location.protocol}//${location.host}/login/callback?action=login` const returnURL = `${location.protocol}//${location.host}/login/callback?action=login`;
httpGet("/api/user/clogin?return_url="+returnURL).then(res => { httpGet("/api/user/clogin?return_url=" + returnURL)
wechatLoginURL.value = res.data.url .then((res) => {
}).catch(e => { wechatLoginURL.value = res.data.url;
console.error(e) })
}) .catch((e) => {
}) console.error(e);
});
});
const handleKeyup = (e) => { const handleKeyup = (e) => {
if (e.key === 'Enter') { if (e.key === "Enter") {
login(); login();
} }
}; };
const login = function () { const login = async function () {
if (username.value.trim() === '') { await ruleFormRef.value.validate(async (valid) => {
return showMessageError("请输入用户民") if (valid) {
} if (enableVerify.value && needVerify.value) {
if (password.value.trim() === '') { captchaRef.value.loadCaptcha();
return showMessageError('请输入密码'); } else {
} doLogin({});
}
}
});
};
if (enableVerify.value && needVerify.value) { const store = useSharedStore();
captchaRef.value.loadCaptcha()
} else {
doLogin({})
}
}
const store = useSharedStore()
const doLogin = (verifyData) => { const doLogin = (verifyData) => {
httpPost('/api/user/login', { httpPost("/api/user/login", {
username: username.value.trim(), username: username.value.trim(),
password: password.value.trim(), password: password.value.trim(),
key: verifyData.key, key: verifyData.key,
dots: verifyData.dots, dots: verifyData.dots,
x: verifyData.x x: verifyData.x
}).then((res) => {
setUserToken(res.data.token)
store.setIsLogin(true)
needVerify.value = false
if (isMobile()) {
router.push('/mobile')
} else {
router.push('/chat')
}
}).catch((e) => {
showMessageError('登录失败,' + e.message)
needVerify.value = true
}) })
} .then((res) => {
setUserToken(res.data.token);
store.setIsLogin(true);
needVerify.value = false;
if (isMobile()) {
router.push("/mobile");
} else {
router.push("/chat");
}
})
.catch((e) => {
showMessageError("登录失败," + e.message);
needVerify.value = true;
});
};
</script> </script>
<style lang="stylus" scoped> <style lang="stylus" scoped>

View File

@ -1,452 +1,330 @@
<template> <template>
<div> <div>
<div class="bg"></div> <div class="flex-center loginPage">
<div class="register-page"> <div class="left" v-if="enableRegister">
<div class="page-inner"> <div class="login-box">
<div class="contain" v-if="enableRegister"> <AccountTop title="注册" />
<div class="logo"> <div class="input-form">
<el-image :src="logo" fit="cover" @click="router.push('/')"/> <el-form :model="data" class="form">
</div> <el-tabs v-model="activeName">
<div class="header">{{ title }}</div>
<div class="content">
<el-form :model="data" class="form" v-if="enableRegister">
<el-tabs v-model="activeName" class="demo-tabs">
<el-tab-pane label="手机注册" name="mobile" v-if="enableMobile"> <el-tab-pane label="手机注册" name="mobile" v-if="enableMobile">
<div class="block"> <el-form-item>
<el-input placeholder="手机号码" <div class="form-title">手机号码</div>
size="large" <el-input
v-model="data.mobile" placeholder="请输入手机号码"
maxlength="11" size="large"
autocomplete="off"> v-model="data.mobile"
<template #prefix> maxlength="11"
<el-icon> autocomplete="off"
<Iphone/> >
</el-icon>
</template>
</el-input> </el-input>
</div> </el-form-item>
<div class="block"> <el-form-item>
<el-row :gutter="10"> <div class="form-title">验证码</div>
<el-col :span="12"> <div class="flex w100">
<el-input placeholder="验证码" <el-input
size="large" maxlength="30" placeholder="请输入验证码"
v-model="data.code" size="large"
autocomplete="off"> maxlength="30"
<template #prefix> class="code-input"
<el-icon> v-model="data.code"
<Checked/> autocomplete="off"
</el-icon> >
</template> </el-input>
</el-input>
</el-col> <send-msg
<el-col :span="12"> size="large"
<send-msg size="large" :receiver="data.mobile" type="mobile"/> :receiver="data.mobile"
</el-col> type="mobile"
</el-row> />
</div> </div>
</el-form-item>
</el-tab-pane> </el-tab-pane>
<el-tab-pane label="邮箱注册" name="email" v-if="enableEmail"> <el-tab-pane label="邮箱注册" name="email" v-if="enableEmail">
<div class="block"> <el-form-item class="block">
<el-input placeholder="邮箱地址" <div class="form-title">邮箱</div>
size="large" <el-input
v-model="data.email" placeholder="请输入邮箱地址"
autocomplete="off"> size="large"
<template #prefix> v-model="data.email"
<el-icon> autocomplete="off"
<Message/> >
</el-icon>
</template>
</el-input> </el-input>
</div> </el-form-item>
<div class="block"> <el-form-item class="block">
<el-row :gutter="10"> <div class="form-title">验证码</div>
<el-col :span="12"> <div class="flex w100">
<el-input placeholder="验证码" <el-input
size="large" maxlength="30" placeholder="请输入验证码"
v-model="data.code" size="large"
autocomplete="off"> maxlength="30"
<template #prefix> class="code-input"
<el-icon> v-model="data.code"
<Checked/> autocomplete="off"
</el-icon> >
</template> </el-input>
</el-input>
</el-col> <send-msg
<el-col :span="12"> size="large"
<send-msg size="large" :receiver="data.email" type="email"/> :receiver="data.email"
</el-col> type="email"
</el-row> />
</div> </div>
</el-form-item>
</el-tab-pane> </el-tab-pane>
<el-tab-pane label="用户名注册" name="username" v-if="enableUser"> <el-tab-pane
<div class="block"> label="用户名注册"
<el-input placeholder="用户名" name="username"
size="large" v-if="enableUser"
v-model="data.username" >
autocomplete="off"> <el-form-item class="block">
<template #prefix> <div class="form-title">用户名</div>
<el-icon>
<Iphone/> <el-input
</el-icon> placeholder="请输入用户名"
</template> size="large"
v-model="data.username"
autocomplete="off"
>
</el-input> </el-input>
</div> </el-form-item>
</el-tab-pane> </el-tab-pane>
</el-tabs> </el-tabs>
<div class="block"> <el-form-item class="block">
<el-input placeholder="请输入密码(8-16位)" <div class="form-title">密码</div>
maxlength="16" size="large"
v-model="data.password" show-password
autocomplete="off">
<template #prefix>
<el-icon>
<Lock/>
</el-icon>
</template>
</el-input>
</div>
<div class="block"> <el-input
<el-input placeholder="重复密码(8-16位)" placeholder="请输入密码(8-16位)"
size="large" maxlength="16" v-model="data.repass" show-password maxlength="16"
autocomplete="off"> size="large"
<template #prefix> v-model="data.password"
<el-icon> show-password
<Lock/> autocomplete="off"
</el-icon> >
</template>
</el-input> </el-input>
</div> </el-form-item>
<div class="block"> <el-form-item class="block">
<el-input placeholder="邀请码(可选)" <div class="form-title">重复密码</div>
size="large"
v-model="data.invite_code" <el-input
autocomplete="off"> placeholder="请再次输入密码(8-16位)"
<template #prefix> size="large"
<el-icon> maxlength="16"
<Message/> v-model="data.repass"
</el-icon> show-password
</template> autocomplete="off"
>
</el-input> </el-input>
</div> </el-form-item>
<el-form-item class="block">
<div class="form-title">邀请码</div>
<el-input
placeholder="请输入邀请码(可选)"
size="large"
v-model="data.invite_code"
autocomplete="off"
>
</el-input>
</el-form-item>
<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
</el-col> class="login-btn"
</el-row> type="primary"
size="large"
<el-row class="text-line" :gutter="24"> @click="submitRegister"
<el-col :span="12"> >注册</el-button
<el-link type="primary" class="text-link" @click="router.push('/login')">登录</el-link> >
</el-col>
<el-col :span="12">
<el-link type="primary" class="text-link" @click="router.push('/')">首页</el-link>
</el-col> </el-col>
</el-row> </el-row>
</el-form> </el-form>
</div> </div>
</div> </div>
<div class="tip-result" v-else>
<el-result icon="error" title="注册功能已关闭">
<template #sub-title>
<p>抱歉系统已关闭注册功能请联系管理员添加账号</p>
<div class="wechat-card">
<el-image :src="wxImg"/>
</div>
</template>
</el-result>
</div>
<captcha v-if="enableVerify" @success="doSubmitRegister" ref="captchaRef"/>
<footer class="footer" v-if="!licenseConfig.de_copy">
<footer-bar/>
</footer>
</div> </div>
<div class="tip-result left" v-else>
<el-result icon="error" title="注册功能已关闭">
<template #sub-title>
<p>抱歉系统已关闭注册功能请联系管理员添加账号</p>
<div class="wechat-card">
<el-image :src="wxImg" />
</div>
</template>
</el-result>
</div>
<captcha
v-if="enableVerify"
@success="doSubmitRegister"
ref="captchaRef"
/>
<account-bg />
</div> </div>
</div> </div>
</template> </template>
<script setup> <script setup>
import { ref } from "vue";
import AccountTop from "@/components/AccountTop.vue";
import AccountBg from "@/components/AccountBg.vue";
import { httpGet, httpPost } from "@/utils/http";
import { ElMessage } from "element-plus";
import { useRouter } from "vue-router";
import {ref} from "vue";
import {Checked, Iphone, Lock, Message} from "@element-plus/icons-vue";
import {httpGet, httpPost} from "@/utils/http";
import {ElMessage} from "element-plus";
import {useRouter} from "vue-router";
import FooterBar from "@/components/FooterBar.vue";
import SendMsg from "@/components/SendMsg.vue"; import SendMsg from "@/components/SendMsg.vue";
import {arrayContains, isMobile} from "@/utils/libs"; import { arrayContains, isMobile } from "@/utils/libs";
import {setUserToken} from "@/store/session"; import { setUserToken } from "@/store/session";
import {validateEmail, validateMobile} from "@/utils/validate"; import { validateEmail, validateMobile } from "@/utils/validate";
import {showMessageError, showMessageOK} from "@/utils/dialog"; import { showMessageError, showMessageOK } from "@/utils/dialog";
import {getLicenseInfo, getSystemInfo} from "@/store/cache"; import { getLicenseInfo, getSystemInfo } from "@/store/cache";
import Captcha from "@/components/Captcha.vue"; import Captcha from "@/components/Captcha.vue";
const router = useRouter(); const router = useRouter();
const title = ref(''); const title = ref("");
const logo = ref("") const logo = ref("");
const data = ref({ const data = ref({
username: '', username: "",
mobile: '', mobile: "",
email: '', email: "",
password: '', password: "",
code: '', code: "",
repass: '', repass: "",
invite_code: router.currentRoute.value.query['invite_code'], invite_code: router.currentRoute.value.query["invite_code"]
}) });
const enableMobile = ref(false) const enableMobile = ref(false);
const enableEmail = ref(false) const enableEmail = ref(false);
const enableUser = ref(false) const enableUser = ref(false);
const enableRegister = ref(true) const enableRegister = ref(true);
const activeName = ref("mobile") const activeName = ref("mobile");
const wxImg = ref("/images/wx.png") const wxImg = ref("/images/wx.png");
const licenseConfig = ref({}) const licenseConfig = ref({});
const enableVerify = ref(false) const enableVerify = ref(false);
const captchaRef = ref(null) const captchaRef = ref(null);
// //
if (data.value.invite_code) { if (data.value.invite_code) {
httpGet("/api/invite/hits",{code: data.value.invite_code}) httpGet("/api/invite/hits", { code: data.value.invite_code });
} }
getSystemInfo().then(res => { getSystemInfo()
if (res.data) { .then((res) => {
title.value = res.data.title if (res.data) {
logo.value = res.data.logo title.value = res.data.title;
const registerWays = res.data['register_ways'] logo.value = res.data.logo;
const registerWays = res.data["register_ways"];
if (arrayContains(registerWays, "username")) { if (arrayContains(registerWays, "username")) {
enableUser.value = true enableUser.value = true;
activeName.value = 'username' activeName.value = "username";
}
if (arrayContains(registerWays, "email")) {
enableEmail.value = true;
activeName.value = "email";
}
if (arrayContains(registerWays, "mobile")) {
enableMobile.value = true;
activeName.value = "mobile";
}
//
enableRegister.value = res.data["enabled_register"];
// 使
if (res.data["wechat_card_url"] !== "") {
wxImg.value = res.data["wechat_card_url"];
}
enableVerify.value = res.data["enabled_verify"];
} }
if (arrayContains(registerWays, "email")) { })
enableEmail.value = true .catch((e) => {
activeName.value = 'email' ElMessage.error("获取系统配置失败:" + e.message);
} });
if (arrayContains(registerWays, "mobile")) {
enableMobile.value = true
activeName.value = 'mobile'
}
//
enableRegister.value = res.data['enabled_register']
// 使
if (res.data['wechat_card_url'] !== '') {
wxImg.value = res.data['wechat_card_url']
}
enableVerify.value = res.data['enabled_verify']
}
}).catch(e => {
ElMessage.error("获取系统配置失败:" + e.message)
})
getLicenseInfo().then(res => { getLicenseInfo()
licenseConfig.value = res.data .then((res) => {
}).catch(e => { licenseConfig.value = res.data;
showMessageError("获取 License 配置:" + e.message) })
}) .catch((e) => {
showMessageError("获取 License 配置:" + e.message);
});
// //
const submitRegister = () => { const submitRegister = () => {
if (activeName.value === 'username' && data.value.username === '') { if (activeName.value === "username" && data.value.username === "") {
return showMessageError('请输入用户名'); return showMessageError("请输入用户名");
} }
if (activeName.value === 'mobile' && !validateMobile(data.value.mobile)) { if (activeName.value === "mobile" && !validateMobile(data.value.mobile)) {
return showMessageError('请输入合法的手机号'); return showMessageError("请输入合法的手机号");
} }
if (activeName.value === 'email' && !validateEmail(data.value.email)) { if (activeName.value === "email" && !validateEmail(data.value.email)) {
return showMessageError('请输入合法的邮箱地址'); return showMessageError("请输入合法的邮箱地址");
} }
if (data.value.password.length < 8) { if (data.value.password.length < 8) {
return showMessageError('密码的长度为8-16个字符'); return showMessageError("密码的长度为8-16个字符");
} }
if (data.value.repass !== data.value.password) { if (data.value.repass !== data.value.password) {
return showMessageError('两次输入密码不一致'); return showMessageError("两次输入密码不一致");
} }
if ((activeName.value === 'mobile' || activeName.value === 'email') && data.value.code === '') { if (
return showMessageError('请输入验证码'); (activeName.value === "mobile" || activeName.value === "email") &&
data.value.code === ""
) {
return showMessageError("请输入验证码");
} }
// //
if (enableVerify.value && activeName.value === 'username') { if (enableVerify.value && activeName.value === "username") {
captchaRef.value.loadCaptcha() captchaRef.value.loadCaptcha();
} else { } else {
doSubmitRegister({}) doSubmitRegister({});
} }
} };
const doSubmitRegister = (verifyData) => { const doSubmitRegister = (verifyData) => {
data.value.key = verifyData.key data.value.key = verifyData.key;
data.value.dots = verifyData.dots data.value.dots = verifyData.dots;
data.value.x = verifyData.x data.value.x = verifyData.x;
data.value.reg_way = activeName.value data.value.reg_way = activeName.value;
httpPost('/api/user/register', data.value).then((res) => { httpPost("/api/user/register", data.value)
setUserToken(res.data.token) .then((res) => {
showMessageOK("注册成功,即将跳转到对话主界面...") setUserToken(res.data.token);
if (isMobile()) { showMessageOK("注册成功,即将跳转到对话主界面...");
router.push('/mobile/index') if (isMobile()) {
} else { router.push("/mobile/index");
router.push('/chat') } else {
} router.push("/chat");
}).catch((e) => { }
showMessageError('注册失败,' + e.message) })
}) .catch((e) => {
} showMessageError("注册失败," + e.message);
});
};
</script> </script>
<style lang="stylus" scoped> <style lang="stylus" scoped>
.bg { @import "@/assets/css/login.styl"
position fixed ::v-deep(.back){
left 0 margin-bottom: 10px;
right 0
top 0
bottom 0
background-color #091519
background-image url("~@/assets/img/reg_bg.png")
background-size cover
background-position center
background-repeat no-repeat
filter: blur(10px); /* 调整模糊程度,可以根据需要修改值 */
}
.register-page {
display flex
justify-content center
.page-inner {
max-width 450px
width 100%
height 100vh
display flex
justify-content center
align-items center
.contain {
padding 20px 40px 20px 40px;
width 100%
color #ffffff
border-radius 10px;
z-index 10
background-color rgba(255, 255, 255, 0.2)
.logo {
text-align center
.el-image {
width 120px;
cursor pointer
border-radius 50%
}
}
.header {
width 100%
margin-bottom 24px
font-size 24px
color $white_v1
letter-space 2px
text-align center
padding-top 10px
}
.content {
width 100%
height: auto
border-radius 3px
.block {
margin-bottom 16px
.el-input__inner {
border 1px solid $gray-v6 !important
.el-icon-user, .el-icon-lock {
font-size 20px
}
}
}
.btn-row {
padding-top 10px;
.login-btn {
width 100%
font-size 16px
letter-spacing 2px
}
}
.text-link {
color #ffffff
}
.text-line {
justify-content center
padding-top 10px;
font-size 14px;
.el-col {
text-align center
}
}
}
}
.tip-result {
z-index 10
.wechat-card {
padding 20px
}
}
.footer {
color #ffffff;
.container {
padding 20px;
}
}
} }
} ::v-deep(.orline){
</style> margin-bottom: 10px;
}
<style lang="stylus"> .wechat-card {
.register-page { margin-top: 20px
.el-result {
}
border-radius 10px; ::v-deep(.el-tabs__item.is-active, .el-tabs__item:hover){
background-color rgba(14, 25, 30, 0.6) color: var(--common-text-color) !important;
border 1px solid #666 }
.el-tabs__item{
.el-result__title p { color:var( --text-theme-color)
color #ffffff }
}
.el-result__subtitle p {
color #c1c1c1
}
}
.el-tabs__item {
color #ffffff
}
}
</style> </style>

View File

@ -0,0 +1,162 @@
<template>
<div class="reset-pass"></div>
<div class="flex-center loginPage">
<div class="left">
<div class="login-box">
<AccountTop title="重置密码" />
<div class="input-form">
<el-form :model="form">
<el-tabs v-model="form.type">
<el-tab-pane label="手机号验证" name="mobile">
<el-form-item>
<div class="form-title">手机号码</div>
<el-input
v-model="form.mobile"
size="large"
placeholder="请输入手机号"
/>
</el-form-item>
<el-form-item>
<div class="form-title">验证码</div>
<div class="flex w100">
<el-input
v-model="form.code"
maxlength="6"
size="large"
class="code-input"
/>
<send-msg
size="large"
:receiver="form.mobile"
type="mobile"
/>
</div>
</el-form-item>
</el-tab-pane>
<el-tab-pane label="邮箱验证" name="email">
<el-form-item>
<div class="form-title">邮箱</div>
<el-input
v-model="form.email"
placeholder="请输入邮箱"
size="large"
/>
</el-form-item>
<el-form-item>
<div class="form-title">验证码</div>
<div class="flex w100">
<el-input v-model="form.code" maxlength="6" />
<send-msg size="" :receiver="form.email" type="email" />
</div>
</el-form-item>
</el-tab-pane>
</el-tabs>
<el-form-item>
<div class="form-title">新密码</div>
<el-input
v-model="form.password"
type="password"
placeholder="请输入新密码(8-16位)"
size="large"
/>
</el-form-item>
<el-form-item>
<div class="form-title">重复密码</div>
<el-input
v-model="form.repass"
type="password"
placeholder="请再次输入密码(8-16位)"
size="large"
/>
</el-form-item>
<el-form-item>
<el-button
class="login-btn"
size="large"
type="primary"
@click="save"
>
重置密码
</el-button>
</el-form-item>
</el-form>
</div>
</div>
</div>
<account-bg />
</div>
</template>
<script setup>
import { ref } from "vue";
import SendMsg from "@/components/SendMsg.vue";
import AccountTop from "@/components/AccountTop.vue";
import { ElMessage } from "element-plus";
import { httpPost } from "@/utils/http";
import AccountBg from "@/components/AccountBg.vue";
import { validateEmail, validateMobile } from "@/utils/validate";
const form = ref({
mobile: "",
email: "",
type: "mobile",
code: "",
password: "",
repass: ""
});
const save = () => {
if (form.value.code === "") {
return ElMessage.error("请输入验证码");
}
if (form.value.password.length < 8) {
return ElMessage.error("密码长度必须大于8位");
}
if (form.value.repass !== form.value.password) {
return ElMessage.error("两次输入密码不一致");
}
httpPost("/api/user/resetPass", form.value)
.then(() => {
ElMessage.success({
message: "重置密码成功",
duration: 1000
});
})
.catch((e) => {
ElMessage.error("重置密码失败:" + e.message);
});
};
</script>
<style lang="stylus">
@import "@/assets/css/login.styl"
.reset-pass {
.form {
padding 0 20px
}
.code-row {
width 100%
.send-button {
padding-left 10px
}
}
.reset-pass-dialog {
.el-dialog__footer {
text-align center
padding-top 0
}
.el-dialog__body {
padding 0
}
}
}
</style>