refactor: profile

pull/2/head
xingyu4j 2022-11-16 16:01:36 +08:00
parent fab2366d55
commit 770bf45f01
9 changed files with 298 additions and 267 deletions

View File

@ -0,0 +1,77 @@
import request from '@/config/axios'
export interface ProfileDept {
id: number
name: string
}
export interface ProfileRole {
id: number
name: string
}
export interface ProfilePost {
id: number
name: string
}
export interface SocialUser {
id: number
type: number
openid: string
token: string
rawTokenInfo: string
nickname: string
avatar: string
rawUserInfo: string
code: string
state: string
}
export interface ProfileVO {
id: number
username: string
nickname: string
dept: ProfileDept
roles: ProfileRole[]
posts: ProfilePost[]
socialUsers: SocialUser[]
email: string
mobile: string
sex: number
avatar: string
status: number
remark: string
loginIp: string
loginDate: Date
createTime: Date
}
export interface UserProfileUpdateReqVO {
nickname: string
email: string
mobile: string
sex: number
}
// 查询用户个人信息
export const getUserProfileApi = () => {
return request.get({ url: '/system/user/profile/get' })
}
// 修改用户个人信息
export const updateUserProfileApi = (data: UserProfileUpdateReqVO) => {
return request.put({ url: '/system/user/profile/update', data })
}
// 用户密码重置
export const updateUserPwdApi = (oldPassword: string, newPassword: string) => {
return request.put({
url: '/system/user/profile/update-password',
data: {
oldPassword: oldPassword,
newPassword: newPassword
}
})
}
// 用户头像上传
export const uploadAvatarApi = (data) => {
return request.upload({ url: '/system/user/profile/update-avatar', data: data })
}

View File

@ -1,27 +0,0 @@
import request from '@/config/axios'
// 查询用户个人信息
export const getUserProfileApi = () => {
return request.get({ url: '/system/user/profile/get' })
}
// 修改用户个人信息
export const updateUserProfileApi = (params) => {
return request.put({ url: '/system/user/profile/update', params })
}
// 用户密码重置
export const updateUserPwdApi = (oldPassword: string, newPassword: string) => {
return request.put({
url: '/system/user/profile/update-password',
data: {
oldPassword: oldPassword,
newPassword: newPassword
}
})
}
// 用户头像上传
export const uploadAvatarApi = (data) => {
return request.upload({ url: '/system/user/profile/update-avatar', data: data })
}

View File

@ -1,42 +0,0 @@
export type ProfileDept = {
id: number
name: string
}
export type ProfileRole = {
id: number
name: string
}
export type ProfilePost = {
id: number
name: string
}
export type SocialUser = {
id: number
type: number
openid: string
token: string
rawTokenInfo: string
nickname: string
avatar: string
rawUserInfo: string
code: string
state: string
}
export type ProfileVO = {
id: number
username: string
nickname: string
dept: ProfileDept
roles: ProfileRole[]
posts: ProfilePost[]
socialUsers: SocialUser[]
email: string
mobile: string
sex: number
avatar: string
status: number
remark: string
loginIp: string
loginDate: Date
createTime: Date
}

View File

@ -0,0 +1,31 @@
import request from '@/config/axios'
// 社交绑定,使用 code 授权码
export const socialBind = (type, code, state) => {
return request.post({
url: '/system/social-user/bind',
data: {
type,
code,
state
}
})
}
// 取消社交绑定
export const socialUnbind = (type, openid) => {
return request.delete({
url: '/system/social-user/unbind',
data: {
type,
openid
}
})
}
// 社交授权的跳转
export const socialAuthRedirect = (type, redirectUri) => {
return request.get({
url: '/system/auth/social-auth-redirect?type=' + type + '&redirectUri=' + redirectUri
})
}

View File

@ -1,11 +1,3 @@
<script setup lang="ts">
import { ref } from 'vue'
import { useI18n } from '@/hooks/web/useI18n'
import { ElCard, ElTabs, ElTabPane } from 'element-plus'
import { BasicInfo, ProfileUser, ResetPwd, UserSocial } from './components/'
const { t } = useI18n()
const activeName = ref('basicInfo')
</script>
<template> <template>
<div class="flex"> <div class="flex">
<el-card class="w-1/3 user" shadow="hover"> <el-card class="w-1/3 user" shadow="hover">
@ -38,6 +30,15 @@ const activeName = ref('basicInfo')
</el-card> </el-card>
</div> </div>
</template> </template>
<script setup lang="ts">
import { ref } from 'vue'
import { useI18n } from '@/hooks/web/useI18n'
import { ElCard, ElTabs, ElTabPane } from 'element-plus'
import { BasicInfo, ProfileUser, ResetPwd, UserSocial } from './components/'
const { t } = useI18n()
const activeName = ref('basicInfo')
</script>
<style scoped> <style scoped>
.user { .user {
max-height: 960px; max-height: 960px;

View File

@ -1,30 +1,28 @@
<template>
<Form ref="formRef" :rules="rules" :schema="schema" :labelWidth="80">
<template #sex>
<el-radio-group v-model="sexVlue">
<el-radio :label="1">{{ t('profile.user.man') }}</el-radio>
<el-radio :label="2">{{ t('profile.user.woman') }}</el-radio>
</el-radio-group>
</template>
</Form>
<el-button type="primary" @click="submit()">{{ t('common.save') }}</el-button>
<el-button type="danger" @click="init()">{{ t('common.reset') }}</el-button>
</template>
<script setup lang="ts"> <script setup lang="ts">
import { ref, reactive, onMounted } from 'vue' import { reactive, onMounted, unref, ref } from 'vue'
import type { FormRules, FormInstance } from 'element-plus' import type { FormRules } from 'element-plus'
import { ElForm, ElFormItem, ElInput, ElRadioGroup, ElRadio, ElMessage } from 'element-plus' import { ElMessage, ElRadioGroup, ElRadio } from 'element-plus'
import { useI18n } from '@/hooks/web/useI18n' import { useI18n } from '@/hooks/web/useI18n'
import { getUserProfileApi, updateUserProfileApi } from '@/api/system/user/profile' import {
getUserProfileApi,
updateUserProfileApi,
UserProfileUpdateReqVO
} from '@/api/system/user/profile'
import { FormSchema } from '@/types/form'
import { FormExpose } from '@/components/Form'
const { t } = useI18n() const { t } = useI18n()
const formRef = ref<FormInstance>()
interface BasicUserInfoVO {
id: number
nickname: string
email: string
mobile: string
sex: number
}
interface userInfoType {
basicUserInfo: BasicUserInfoVO
}
const user = reactive<userInfoType>({
basicUserInfo: {
id: 0,
nickname: '',
mobile: '',
email: '',
sex: 0
}
})
// //
const rules = reactive<FormRules>({ const rules = reactive<FormRules>({
nickname: [{ required: true, message: t('profile.rules.nickname'), trigger: 'blur' }], nickname: [{ required: true, message: t('profile.rules.nickname'), trigger: 'blur' }],
@ -45,47 +43,49 @@ const rules = reactive<FormRules>({
} }
] ]
}) })
const submit = (formEl: FormInstance | undefined) => { const schema = reactive<FormSchema[]>([
if (!formEl) return {
formEl.validate(async (valid) => { field: 'nickname',
label: t('profile.user.nickname'),
component: 'Input'
},
{
field: 'mobile',
label: t('profile.user.mobile'),
component: 'Input'
},
{
field: 'email',
label: t('profile.user.email'),
component: 'Input'
},
{
field: 'sex',
label: t('profile.user.sex'),
component: 'InputNumber'
}
])
const sexVlue = ref<number>()
const formRef = ref<FormExpose>() // Ref
const submit = () => {
const elForm = unref(formRef)?.getElFormRef()
if (!elForm) return
elForm.validate(async (valid) => {
if (valid) { if (valid) {
await updateUserProfileApi({ params: user.basicUserInfo }) const data = unref(formRef)?.formModel as UserProfileUpdateReqVO
data.sex = sexVlue.value as unknown as number
await updateUserProfileApi(data)
ElMessage.success(t('common.updateSuccess')) ElMessage.success(t('common.updateSuccess'))
await init()
} }
}) })
} }
const reset = async (formEl: FormInstance | undefined) => { const init = async () => {
if (!formEl) return const res = await getUserProfileApi()
await getUserInfo() sexVlue.value = res.sex
} unref(formRef)?.setValues(res)
const getUserInfo = async () => {
const users = await getUserProfileApi()
user.basicUserInfo = users
} }
onMounted(async () => { onMounted(async () => {
await getUserInfo() await init()
}) })
</script> </script>
<template>
<el-form ref="form" :model="user.basicUserInfo" :rules="rules" label-width="80px">
<el-form-item :label="t('profile.user.nickname')" prop="nickname">
<el-input v-model="user.basicUserInfo.nickname" />
</el-form-item>
<el-form-item :label="t('profile.user.mobile')" prop="mobile">
<el-input v-model="user.basicUserInfo.mobile" maxlength="11" />
</el-form-item>
<el-form-item :label="t('profile.user.email')" prop="email">
<el-input v-model="user.basicUserInfo.email" maxlength="50" />
</el-form-item>
<el-form-item :label="t('profile.user.sex')" prop="sex">
<el-radio-group v-model="user.basicUserInfo.sex">
<el-radio :label="1">{{ t('profile.user.man') }}</el-radio>
<el-radio :label="2">{{ t('profile.user.woman') }}</el-radio>
</el-radio-group>
</el-form-item>
<el-form-item>
<el-button type="primary" @click="submit(formRef)">{{ t('common.save') }}</el-button>
<el-button type="danger" @click="reset(formRef)">{{ t('common.reset') }}</el-button>
</el-form-item>
</el-form>
</template>

View File

@ -1,86 +1,61 @@
<template>
<div>
<div class="text-center">
<UserAvatar :img="userInfo?.avatar" />
</div>
<ul class="list-group list-group-striped">
<li class="list-group-item">
<Icon icon="ep:user" class="mr-5px" />{{ t('profile.user.username') }}
<div class="pull-right">{{ userInfo?.username }}</div>
</li>
<li class="list-group-item">
<Icon icon="ep:phone" class="mr-5px" />{{ t('profile.user.mobile') }}
<div class="pull-right">{{ userInfo?.mobile }}</div>
</li>
<li class="list-group-item">
<Icon icon="fontisto:email" class="mr-5px" />{{ t('profile.user.email') }}
<div class="pull-right">{{ userInfo?.email }}</div>
</li>
<li class="list-group-item">
<Icon icon="carbon:tree-view-alt" class="mr-5px" />{{ t('profile.user.dept') }}
<div class="pull-right" v-if="userInfo?.dept">{{ userInfo?.dept.name }}</div>
</li>
<li class="list-group-item">
<Icon icon="ep:suitcase" class="mr-5px" />{{ t('profile.user.posts') }}
<div class="pull-right" v-if="userInfo?.posts">
{{ userInfo?.posts.map((post) => post.name).join(',') }}
</div>
</li>
<li class="list-group-item">
<Icon icon="icon-park-outline:peoples" class="mr-5px" />{{ t('profile.user.roles') }}
<div class="pull-right" v-if="userInfo?.roles">
{{ userInfo?.roles.map((role) => role.name).join(',') }}
</div>
</li>
<li class="list-group-item">
<Icon icon="ep:calendar" class="mr-5px" />{{ t('profile.user.createTime') }}
<div class="pull-right">{{ dayjs(userInfo?.createTime).format('YYYY-MM-DD') }}</div>
</li>
</ul>
</div>
</template>
<script setup lang="ts"> <script setup lang="ts">
import { getUserProfileApi } from '@/api/system/user/profile' import { getUserProfileApi, ProfileVO } from '@/api/system/user/profile'
import { onMounted, reactive } from 'vue' import { onMounted, ref } from 'vue'
import dayjs from 'dayjs' import dayjs from 'dayjs'
import UserAvatar from './UserAvatar.vue' import UserAvatar from './UserAvatar.vue'
import { ProfileVO } from '@/api/system/user/profile/types'
import { useI18n } from '@/hooks/web/useI18n' import { useI18n } from '@/hooks/web/useI18n'
const { t } = useI18n() const { t } = useI18n()
interface userInfoType { const userInfo = ref<ProfileVO>()
user: ProfileVO
}
const userInfo = reactive<userInfoType>({
user: {
id: 0,
username: '',
nickname: '',
dept: {
id: 0,
name: ''
},
roles: [],
posts: [],
socialUsers: [],
email: '',
mobile: '',
sex: 0,
avatar: '',
status: 0,
remark: '',
loginIp: '',
loginDate: new Date(),
createTime: new Date()
}
})
const getUserInfo = async () => { const getUserInfo = async () => {
const users = await getUserProfileApi() const users = await getUserProfileApi()
userInfo.user = users userInfo.value = users
} }
onMounted(async () => { onMounted(async () => {
await getUserInfo() await getUserInfo()
}) })
</script> </script>
<template>
<div>
<div class="text-center">
<UserAvatar :img="userInfo.user.avatar" />
</div>
<ul class="list-group list-group-striped">
<li class="list-group-item">
<Icon icon="ep:user" class="mr-5px" />{{ t('profile.user.username') }}
<div class="pull-right">{{ userInfo.user.username }}</div>
</li>
<li class="list-group-item">
<Icon icon="ep:phone" class="mr-5px" />{{ t('profile.user.mobile') }}
<div class="pull-right">{{ userInfo.user.mobile }}</div>
</li>
<li class="list-group-item">
<Icon icon="fontisto:email" class="mr-5px" />{{ t('profile.user.email') }}
<div class="pull-right">{{ userInfo.user.email }}</div>
</li>
<li class="list-group-item">
<Icon icon="carbon:tree-view-alt" class="mr-5px" />{{ t('profile.user.dept') }}
<div class="pull-right" v-if="userInfo.user.dept">{{ userInfo.user.dept.name }}</div>
</li>
<li class="list-group-item">
<Icon icon="ep:suitcase" class="mr-5px" />{{ t('profile.user.posts') }}
<div class="pull-right" v-if="userInfo.user.posts">
{{ userInfo.user.posts.map((post) => post.name).join(',') }}
</div>
</li>
<li class="list-group-item">
<Icon icon="icon-park-outline:peoples" class="mr-5px" />{{ t('profile.user.roles') }}
<div class="pull-right" v-if="userInfo.user.roles">
{{ userInfo.user.roles.map((role) => role.name).join(',') }}
</div>
</li>
<li class="list-group-item">
<Icon icon="ep:calendar" class="mr-5px" />{{ t('profile.user.createTime') }}
<div class="pull-right">{{ dayjs(userInfo.user.createTime).format('YYYY-MM-DD') }}</div>
</li>
</ul>
</div>
</template>
<style scoped> <style scoped>
.text-center { .text-center {
text-align: center; text-align: center;

View File

@ -1,3 +1,20 @@
<template>
<el-form ref="formRef" :model="password" :rules="rules" label-width="80px">
<el-form-item :label="t('profile.password.oldPassword')">
<InputPassword v-model="password.oldPassword" />
</el-form-item>
<el-form-item :label="t('profile.password.newPassword')">
<InputPassword v-model="password.newPassword" strength />
</el-form-item>
<el-form-item :label="t('profile.password.confirmPassword')">
<InputPassword v-model="password.confirmPassword" strength />
</el-form-item>
<el-form-item>
<XButton type="primary" @click="submit(formRef)" :title="t('common.save')" />
<XButton type="danger" :title="t('common.reset')" @click="reset(formRef)" />
</el-form-item>
</el-form>
</template>
<script setup lang="ts"> <script setup lang="ts">
import { InputPassword } from '@/components/InputPassword' import { InputPassword } from '@/components/InputPassword'
import { ElForm, ElFormItem, ElMessage } from 'element-plus' import { ElForm, ElFormItem, ElMessage } from 'element-plus'
@ -49,20 +66,3 @@ const reset = (formEl: FormInstance | undefined) => {
formEl.resetFields() formEl.resetFields()
} }
</script> </script>
<template>
<el-form ref="formRef" :model="password" :rules="rules" label-width="80px">
<el-form-item :label="t('profile.password.oldPassword')">
<InputPassword v-model="password.oldPassword" />
</el-form-item>
<el-form-item :label="t('profile.password.newPassword')">
<InputPassword v-model="password.newPassword" strength />
</el-form-item>
<el-form-item :label="t('profile.password.confirmPassword')">
<InputPassword v-model="password.confirmPassword" strength />
</el-form-item>
<el-form-item>
<el-button type="primary" @click="submit(formRef)">{{ t('common.save') }}</el-button>
<el-button type="danger" @click="reset(formRef)">{{ t('common.reset') }}</el-button>
</el-form-item>
</el-form>
</template>

View File

@ -1,54 +1,70 @@
<script setup lang="ts">
import { ElTable, ElTableColumn } from 'element-plus'
import { onMounted, reactive } from 'vue'
interface sociaType {
title: string
type: string
source: string
img: string
}
interface socialUserType {
socialUser: {
socia: sociaType[]
}
}
const state = reactive<socialUserType>({
socialUser: {
socia: []
}
})
const initSocial = () => {
console.info(1)
}
const bind = () => {
console.info(1)
}
const unbind = () => {
console.info(1)
}
onMounted(async () => {
await initSocial()
})
</script>
<template> <template>
<el-table :data="state.socialUser.socia" :show-header="false"> <el-table :data="socialUsers" :show-header="false">
<el-table-column label="社交平台" align="left" width="120" prop="socia"> <el-table-column type="seq" title="序号" width="60" fixed="left" />
<template #socia="{ row }"> <el-table-column label="社交平台" align="left" width="120">
<template #default="{ row }">
<img style="height: 20px; vertical-align: middle" :src="row.img" alt="" /> <img style="height: 20px; vertical-align: middle" :src="row.img" alt="" />
{{ row.title }} {{ row.title }}
</template> </template>
</el-table-column> </el-table-column>
<el-table-column label="操作" align="left" prop="action"> <el-table-column label="操作" align="center">
<template #action="{ row }"> <template #default="{ row }">
<div v-if="row.openid"> <template v-if="row.openid">
已绑定 已绑定
<el-button link type="primary" @click="unbind()">()</el-button> <XTextButton type="primary" @click="unbind(row)" title="(解绑)" />
</div> </template>
<div v-else> <template v-else>
未绑定 未绑定
<el-button link type="primary" @click="bind()">()</el-button> <XTextButton type="primary" @click="bind(row)" title="(绑定)" />
</div> </template>
</template> </template>
</el-table-column> </el-table-column>
</el-table> </el-table>
</template> </template>
<script setup lang="ts">
import { onMounted, ref } from 'vue'
import { ElTable, ElTableColumn } from 'element-plus'
import { SystemUserSocialTypeEnum } from '@/utils/constants'
import { getUserProfileApi, ProfileVO } from '@/api/system/user/profile'
import { socialAuthRedirect, socialUnbind } from '@/api/system/user/socialUser'
import { ElMessage } from 'element-plus'
const socialUsers = ref<any[]>([])
const userInfo = ref<ProfileVO>()
const initSocial = async () => {
const res = await getUserProfileApi()
userInfo.value = res
for (const i in SystemUserSocialTypeEnum) {
const socialUser = { ...SystemUserSocialTypeEnum[i] }
socialUsers.value.push(socialUser)
if (userInfo.value?.socialUsers) {
for (const j in userInfo.value.socialUsers) {
if (socialUser.type === userInfo.value.socialUsers[j].type) {
socialUser.openid = userInfo.value.socialUsers[j].openid
break
}
}
}
}
console.info(socialUsers.value)
}
const bind = (row) => {
const redirectUri = location.origin + '/user/profile?type=' + row.type
//
socialAuthRedirect(row.type, encodeURIComponent(redirectUri)).then((res) => {
console.log(res.url)
window.location.href = res.data
})
}
const unbind = async (row) => {
const res = await socialUnbind(row.type, row.openid)
if (res) {
row.openid = undefined
}
ElMessage.success('解绑成功')
}
onMounted(async () => {
await initSocial()
})
</script>