home.vue 4.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165
  1. <template>
  2. <div class="device-home app-container">
  3. <div class="device-home-header">
  4. <h3 class="title">设备功能</h3>
  5. <div class="device-meta">
  6. <el-tag size="small" type="success">已连接设备</el-tag>
  7. <span>SN {{ deviceSn }}</span>
  8. <span v-if="deviceInfo.imei" class="imei">IMEI {{ deviceInfo.imei }}</span>
  9. </div>
  10. </div>
  11. <el-tabs v-model="activeTab" class="device-menu-tabs">
  12. <el-tab-pane
  13. v-for="(tabName, tabIndex) in tabNames"
  14. :key="tabName"
  15. :label="tabName"
  16. :name="String(tabIndex)"
  17. >
  18. <el-row :gutter="16" class="demo-grid">
  19. <el-col
  20. v-for="demo in demosByTab[tabIndex]"
  21. :key="demo.name"
  22. :xs="12"
  23. :sm="8"
  24. :md="6"
  25. :lg="4"
  26. >
  27. <div
  28. class="demo-card"
  29. :class="{ disabled: !demo.enabled }"
  30. :style="{ '--card-color': demo.color }"
  31. @click="openDemo(demo)"
  32. >
  33. <div class="demo-icon">
  34. <svg-icon :icon-class="demo.icon" />
  35. </div>
  36. <div class="demo-name">{{ demo.name }}</div>
  37. <div v-if="!demo.enabled" class="demo-badge">敬请期待</div>
  38. </div>
  39. </el-col>
  40. <el-col v-if="!demosByTab[tabIndex].length" :span="24">
  41. <el-empty description="暂无功能" :image-size="80" />
  42. </el-col>
  43. </el-row>
  44. </el-tab-pane>
  45. </el-tabs>
  46. </div>
  47. </template>
  48. <script>
  49. import { DEVICE_TAB_NAMES, DEVICE_DEMOS } from './deviceMenus'
  50. import { buildDeviceSession, saveTsbDeviceSession } from '@/utils/tsbDeviceSession'
  51. export default {
  52. name: 'TsbDeviceHome',
  53. data() {
  54. return {
  55. activeTab: '0',
  56. tabNames: DEVICE_TAB_NAMES,
  57. demosByTab: DEVICE_DEMOS
  58. }
  59. },
  60. computed: {
  61. deviceSn() {
  62. return this.$store.getters.tsbCurrentDeviceSn
  63. },
  64. deviceInfo() {
  65. const list = this.$store.getters.tsbOpenedDevices || []
  66. return list.find(d => String(d.deviceSn) === String(this.deviceSn)) || {}
  67. }
  68. },
  69. methods: {
  70. openDemo(demo) {
  71. if (!demo.enabled || !demo.route) {
  72. this.$message.info('功能开发中,敬请期待')
  73. return
  74. }
  75. const sn = this.deviceSn
  76. if (sn == null) {
  77. return
  78. }
  79. this.$store.commit('tsb/SET_DEVICE_PANEL', { deviceSn: sn, panel: demo.route })
  80. if (demo.route === 'opw') {
  81. this.$store.commit('tsb/SET_OPW_SUB_PAGE', { deviceSn: sn, subPage: 'main' })
  82. }
  83. saveTsbDeviceSession(buildDeviceSession(
  84. this.$store.getters.tsbCurrentDevice,
  85. this.$store.getters.tsbOpenedDevices,
  86. this.$store.state.tsb.devicePanelMap
  87. ))
  88. }
  89. }
  90. }
  91. </script>
  92. <style scoped lang="scss">
  93. .device-home-header {
  94. display: flex;
  95. justify-content: space-between;
  96. align-items: center;
  97. margin-bottom: 12px;
  98. flex-wrap: wrap;
  99. gap: 8px;
  100. }
  101. .title {
  102. margin: 0;
  103. font-size: 18px;
  104. font-weight: 600;
  105. }
  106. .device-meta {
  107. display: flex;
  108. align-items: center;
  109. gap: 10px;
  110. color: #606266;
  111. font-size: 13px;
  112. }
  113. .imei {
  114. color: #909399;
  115. }
  116. .device-menu-tabs {
  117. ::v-deep .el-tabs__header {
  118. margin-bottom: 16px;
  119. }
  120. }
  121. .demo-grid {
  122. min-height: 200px;
  123. }
  124. .demo-card {
  125. position: relative;
  126. margin-bottom: 16px;
  127. padding: 20px 12px 16px;
  128. border-radius: 8px;
  129. background: linear-gradient(135deg, var(--card-color), rgba(255, 255, 255, 0.15));
  130. color: #fff;
  131. text-align: center;
  132. cursor: pointer;
  133. transition: transform 0.2s, box-shadow 0.2s;
  134. box-shadow: 0 2px 8px rgba(0, 0, 0, 0.08);
  135. &:hover:not(.disabled) {
  136. transform: translateY(-2px);
  137. box-shadow: 0 4px 12px rgba(0, 0, 0, 0.12);
  138. }
  139. &.disabled {
  140. cursor: not-allowed;
  141. opacity: 0.55;
  142. }
  143. }
  144. .demo-icon {
  145. font-size: 28px;
  146. margin-bottom: 8px;
  147. }
  148. .demo-name {
  149. font-size: 14px;
  150. font-weight: 500;
  151. }
  152. .demo-badge {
  153. position: absolute;
  154. top: 8px;
  155. right: 8px;
  156. font-size: 11px;
  157. padding: 2px 6px;
  158. border-radius: 10px;
  159. background: rgba(0, 0, 0, 0.25);
  160. }
  161. </style>