UserLayout.vue 3.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143
  1. <template>
  2. <div class="user-layout">
  3. <el-container>
  4. <el-header class="header">
  5. <div class="header-left" @click="router.push('/')">
  6. <img :src="metaStore.logo" alt="logo" class="logo">
  7. <h2 class="title">{{metaStore.title}}</h2>
  8. </div>
  9. <el-menu
  10. :default-active="activeIndex"
  11. mode="horizontal"
  12. router
  13. @select="handleSelect"
  14. class="nav-menu"
  15. >
  16. <el-menu-item :index="menu.path" v-for="menu in menus" :key="menu.path">
  17. <el-icon><component :is="menu.meta?.icon"/></el-icon>
  18. <template #title>{{ menu.meta?.title }}</template>
  19. </el-menu-item>
  20. <el-menu-item v-if="isAdmin" index="/admin">
  21. <el-icon><Tools/></el-icon>
  22. <template #title>后台管理</template>
  23. </el-menu-item>
  24. </el-menu>
  25. <div class="header-right">
  26. <template v-if="login">
  27. <el-dropdown placement="bottom">
  28. <el-button class="user-btn" text>
  29. <el-icon class="user-avatar"><UserFilled /></el-icon>
  30. <span class="username">{{ loginUserStore.loginUser?.user?.name }}</span>
  31. </el-button>
  32. <template #dropdown>
  33. <el-dropdown-menu>
  34. <el-dropdown-item>个人信息</el-dropdown-item>
  35. <el-dropdown-item divided @click="logoutHandler">退出登录</el-dropdown-item>
  36. </el-dropdown-menu>
  37. </template>
  38. </el-dropdown>
  39. </template>
  40. <template v-else>
  41. <el-button type="primary" size="small" @click="loginHandler">登录</el-button>
  42. <el-button size="small" @click="router.push('/login')">注册</el-button>
  43. </template>
  44. </div>
  45. </el-header>
  46. <el-main>
  47. <router-view/>
  48. </el-main>
  49. <el-footer>Footer</el-footer>
  50. </el-container>
  51. </div>
  52. </template>
  53. <script setup lang="ts">
  54. import {computed, ref} from 'vue'
  55. import {useLoginUserStore, useMetaStore} from "../store";
  56. import router, {userRoutes} from "../router";
  57. import {
  58. Tools,
  59. UserFilled
  60. } from '@element-plus/icons-vue'
  61. const metaStore = useMetaStore()
  62. const loginUserStore = useLoginUserStore()
  63. const login = computed(() => loginUserStore.loginUser.isLogin)
  64. const isAdmin = computed(() => loginUserStore.loginUser.user?.role === 'admin')
  65. const loginHandler = () => {
  66. router.push('/login')
  67. }
  68. let activeIndex = ref('/')
  69. const menus = computed(()=>{
  70. return userRoutes[0].children;
  71. })
  72. const handleSelect = () => {
  73. //todo: 处理菜单选择
  74. }
  75. const logoutHandler = () => {
  76. loginUserStore.logoutUser()
  77. router.push('/')
  78. }
  79. </script>
  80. <style scoped>
  81. .user-layout {
  82. display: flex;
  83. flex-direction: column;
  84. min-height: 100vh;
  85. }
  86. el-main{
  87. flex: 1;
  88. }
  89. .header {
  90. display: flex;
  91. align-items: center;
  92. justify-content: space-between;
  93. padding: 0 20px;
  94. }
  95. .header-left {
  96. display: flex;
  97. align-items: center;
  98. gap: 10px;
  99. }
  100. .logo {
  101. width: 40px;
  102. height: 40px;
  103. }
  104. .title {
  105. margin: 0;
  106. font-size: 20px;
  107. }
  108. .nav-menu {
  109. flex: 1;
  110. display: flex;
  111. justify-content: center;
  112. border-bottom: none;
  113. }
  114. .header-right {
  115. display: flex;
  116. align-items: center;
  117. gap: 10px;
  118. }
  119. .username {
  120. font-size: 14px;
  121. color: #606266;
  122. }
  123. .user-btn {
  124. display: flex;
  125. align-items: center;
  126. gap: 6px;
  127. cursor: pointer;
  128. transition: color 0.2s;
  129. }
  130. .user-btn:hover {
  131. color: #db2777;
  132. }
  133. .user-avatar {
  134. font-size: 20px;
  135. color: #be185d;
  136. }
  137. </style>