index.vue 9.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332
  1. <template>
  2. <el-container class="app-wrapper">
  3. <el-aside :width="sidebarOpened ? '210px' : '64px'" class="sidebar-container">
  4. <div class="logo">
  5. <img src="@/assets/logo.svg" alt="logo" />
  6. <h1 v-show="sidebarOpened">摄像头管理</h1>
  7. </div>
  8. <el-menu
  9. :default-active="activeMenu"
  10. :collapse="!sidebarOpened"
  11. :collapse-transition="false"
  12. background-color="#304156"
  13. text-color="#bfcbd9"
  14. active-text-color="#409eff"
  15. router
  16. >
  17. <el-menu-item index="/machine">
  18. <el-icon><Monitor /></el-icon>
  19. <template #title>机器管理</template>
  20. </el-menu-item>
  21. <el-menu-item index="/camera">
  22. <el-icon><VideoCamera /></el-icon>
  23. <template #title>摄像头管理</template>
  24. </el-menu-item>
  25. <el-menu-item index="/user">
  26. <el-icon><UserFilled /></el-icon>
  27. <template #title>用户管理</template>
  28. </el-menu-item>
  29. <el-sub-menu index="/stream">
  30. <template #title>
  31. <el-icon><Film /></el-icon>
  32. <span>Cloudflare Stream</span>
  33. </template>
  34. <el-menu-item index="/stream/videos">
  35. <el-icon><Film /></el-icon>
  36. <template #title>视频管理</template>
  37. </el-menu-item>
  38. <el-menu-item index="/stream/live">
  39. <el-icon><VideoCameraFilled /></el-icon>
  40. <template #title>直播管理</template>
  41. </el-menu-item>
  42. <el-menu-item index="/stream/config">
  43. <el-icon><Setting /></el-icon>
  44. <template #title>Stream 配置</template>
  45. </el-menu-item>
  46. <el-menu-item index="/stream-test">
  47. <el-icon><Monitor /></el-icon>
  48. <template #title>快速测试</template>
  49. </el-menu-item>
  50. </el-sub-menu>
  51. </el-menu>
  52. </el-aside>
  53. <el-container class="main-container">
  54. <el-header class="app-header">
  55. <div class="header-left">
  56. <el-icon class="collapse-btn" @click="toggleSidebar">
  57. <Fold v-if="sidebarOpened" />
  58. <Expand v-else />
  59. </el-icon>
  60. <el-breadcrumb separator="/">
  61. <el-breadcrumb-item v-for="item in breadcrumbs" :key="item.path">
  62. {{ item.meta?.title }}
  63. </el-breadcrumb-item>
  64. </el-breadcrumb>
  65. </div>
  66. <div class="header-right">
  67. <el-dropdown @command="handleCommand">
  68. <span class="user-info">
  69. <el-avatar :size="32" :src="'https://cube.elemecdn.com/0/88/03b0d39583f4b3a790e9d12d41f23e23jps.jpg'" />
  70. <span class="username">{{ userInfo?.username }}</span>
  71. <el-icon><ArrowDown /></el-icon>
  72. </span>
  73. <template #dropdown>
  74. <el-dropdown-menu>
  75. <el-dropdown-item command="changePassword">修改密码</el-dropdown-item>
  76. <el-dropdown-item command="logout">退出登录</el-dropdown-item>
  77. </el-dropdown-menu>
  78. </template>
  79. </el-dropdown>
  80. </div>
  81. </el-header>
  82. <el-main class="app-main">
  83. <router-view v-slot="{ Component }">
  84. <transition name="fade" mode="out-in">
  85. <component :is="Component" />
  86. </transition>
  87. </router-view>
  88. </el-main>
  89. </el-container>
  90. <!-- 修改密码弹窗 -->
  91. <el-dialog v-model="passwordDialogVisible" title="修改密码" width="400px" :close-on-click-modal="false">
  92. <el-form ref="passwordFormRef" :model="passwordForm" :rules="passwordRules" label-width="80px">
  93. <el-form-item label="原密码" prop="oldPassword">
  94. <el-input v-model="passwordForm.oldPassword" type="password" show-password placeholder="请输入原密码" />
  95. </el-form-item>
  96. <el-form-item label="新密码" prop="newPassword">
  97. <el-input v-model="passwordForm.newPassword" type="password" show-password placeholder="请输入新密码" />
  98. </el-form-item>
  99. <el-form-item label="确认密码" prop="confirmPassword">
  100. <el-input v-model="passwordForm.confirmPassword" type="password" show-password placeholder="请再次输入新密码" />
  101. </el-form-item>
  102. </el-form>
  103. <template #footer>
  104. <el-button @click="passwordDialogVisible = false">取消</el-button>
  105. <el-button type="primary" :loading="passwordLoading" @click="handleChangePassword">确定</el-button>
  106. </template>
  107. </el-dialog>
  108. </el-container>
  109. </template>
  110. <script setup lang="ts">
  111. import { computed, onMounted, ref, reactive } from 'vue'
  112. import { useRoute, useRouter } from 'vue-router'
  113. import { ElMessage, type FormInstance, type FormRules } from 'element-plus'
  114. import {
  115. VideoCamera,
  116. Fold,
  117. Expand,
  118. ArrowDown,
  119. Monitor,
  120. Film,
  121. VideoCameraFilled,
  122. Setting,
  123. UserFilled
  124. } from '@element-plus/icons-vue'
  125. import { useAppStore } from '@/store/app'
  126. import { useUserStore } from '@/store/user'
  127. import { changePassword } from '@/api/login'
  128. const route = useRoute()
  129. const router = useRouter()
  130. const appStore = useAppStore()
  131. const userStore = useUserStore()
  132. const sidebarOpened = computed(() => appStore.sidebarOpened)
  133. const userInfo = computed(() => userStore.userInfo)
  134. const activeMenu = computed(() => {
  135. const { path } = route
  136. return path
  137. })
  138. const breadcrumbs = computed(() => {
  139. return route.matched.filter((item) => item.meta && item.meta.title && !item.meta.hidden)
  140. })
  141. function toggleSidebar() {
  142. appStore.toggleSidebar()
  143. }
  144. // 修改密码相关
  145. const passwordDialogVisible = ref(false)
  146. const passwordLoading = ref(false)
  147. const passwordFormRef = ref<FormInstance>()
  148. const passwordForm = reactive({
  149. oldPassword: '',
  150. newPassword: '',
  151. confirmPassword: ''
  152. })
  153. const validateConfirmPassword = (_rule: any, value: string, callback: any) => {
  154. if (value !== passwordForm.newPassword) {
  155. callback(new Error('两次输入的密码不一致'))
  156. } else {
  157. callback()
  158. }
  159. }
  160. const passwordRules: FormRules = {
  161. oldPassword: [{ required: true, message: '请输入原密码', trigger: 'blur' }],
  162. newPassword: [
  163. { required: true, message: '请输入新密码', trigger: 'blur' },
  164. { min: 6, message: '密码长度不能少于6位', trigger: 'blur' }
  165. ],
  166. confirmPassword: [
  167. { required: true, message: '请再次输入新密码', trigger: 'blur' },
  168. { validator: validateConfirmPassword, trigger: 'blur' }
  169. ]
  170. }
  171. function resetPasswordForm() {
  172. passwordForm.oldPassword = ''
  173. passwordForm.newPassword = ''
  174. passwordForm.confirmPassword = ''
  175. passwordFormRef.value?.clearValidate()
  176. }
  177. async function handleChangePassword() {
  178. if (!passwordFormRef.value) return
  179. await passwordFormRef.value.validate(async (valid) => {
  180. if (valid) {
  181. passwordLoading.value = true
  182. try {
  183. const res = await changePassword({
  184. oldPassword: passwordForm.oldPassword,
  185. newPassword: passwordForm.newPassword
  186. })
  187. if (res.code === 200) {
  188. ElMessage.success('密码修改成功,请重新登录')
  189. passwordDialogVisible.value = false
  190. resetPasswordForm()
  191. await userStore.logoutAction()
  192. router.push('/login')
  193. } else {
  194. ElMessage.error(res.message || '密码修改失败')
  195. }
  196. } catch (error: any) {
  197. ElMessage.error(error.message || '密码修改失败')
  198. } finally {
  199. passwordLoading.value = false
  200. }
  201. }
  202. })
  203. }
  204. async function handleCommand(command: string) {
  205. if (command === 'changePassword') {
  206. resetPasswordForm()
  207. passwordDialogVisible.value = true
  208. } else if (command === 'logout') {
  209. await userStore.logoutAction()
  210. router.push('/login')
  211. }
  212. }
  213. onMounted(() => {
  214. userStore.getUserInfo()
  215. })
  216. </script>
  217. <style lang="scss" scoped>
  218. .app-wrapper {
  219. height: 100%;
  220. }
  221. .sidebar-container {
  222. background-color: #304156;
  223. transition: width 0.3s;
  224. overflow: hidden;
  225. .logo {
  226. height: 60px;
  227. display: flex;
  228. align-items: center;
  229. padding: 0 15px;
  230. background-color: #2b3a4a;
  231. img {
  232. width: 32px;
  233. height: 32px;
  234. }
  235. h1 {
  236. margin-left: 12px;
  237. font-size: 16px;
  238. font-weight: 600;
  239. color: #fff;
  240. white-space: nowrap;
  241. }
  242. }
  243. .el-menu {
  244. border-right: none;
  245. }
  246. }
  247. .main-container {
  248. flex-direction: column;
  249. min-height: 100%;
  250. overflow: hidden;
  251. }
  252. .app-header {
  253. display: flex;
  254. align-items: center;
  255. justify-content: space-between;
  256. height: 60px;
  257. background-color: #fff;
  258. box-shadow: 0 1px 4px rgba(0, 21, 41, 0.08);
  259. .header-left {
  260. display: flex;
  261. align-items: center;
  262. .collapse-btn {
  263. font-size: 20px;
  264. cursor: pointer;
  265. margin-right: 15px;
  266. color: #5a5e66;
  267. &:hover {
  268. color: #409eff;
  269. }
  270. }
  271. }
  272. .header-right {
  273. .user-info {
  274. display: flex;
  275. align-items: center;
  276. cursor: pointer;
  277. .username {
  278. margin: 0 8px;
  279. color: #5a5e66;
  280. }
  281. }
  282. }
  283. }
  284. .app-main {
  285. background-color: #f5f7fa;
  286. overflow: auto;
  287. }
  288. .fade-enter-active,
  289. .fade-leave-active {
  290. transition: opacity 0.2s ease;
  291. }
  292. .fade-enter-from,
  293. .fade-leave-to {
  294. opacity: 0;
  295. }
  296. </style>