StudyBase.vue 7.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313
  1. <template>
  2. <m-card title="示范基地全景" :width="398" :height="450">
  3. <div class="study-base">
  4. <div class="core-indicator">
  5. <div class="indicator-main">
  6. <span class="main-value">30000</span>
  7. <span class="main-unit">㎡</span>
  8. </div>
  9. <div class="indicator-label">基地占地面积</div>
  10. </div>
  11. <div class="stats-grid">
  12. <div class="stat-item">
  13. <div class="stat-value">50+</div>
  14. <div class="stat-label">研学课程</div>
  15. </div>
  16. <div class="stat-item">
  17. <div class="stat-value">20+</div>
  18. <div class="stat-label">合作学校</div>
  19. </div>
  20. <div class="stat-item">
  21. <div class="stat-value">1200+</div>
  22. <div class="stat-label">参与学生</div>
  23. </div>
  24. <div class="stat-item">
  25. <div class="stat-value">2023</div>
  26. <div class="stat-label">建设年份</div>
  27. </div>
  28. </div>
  29. <div class="architecture-flow">
  30. <div class="architecture-title">功能区域</div>
  31. <div class="architecture-tabs">
  32. <div class="tab-item" :class="{ active: activeTab === 0 }" @click="switchTab(0)">技术实践区</div>
  33. <div class="tab-item" :class="{ active: activeTab === 1 }" @click="switchTab(1)">技术研发区</div>
  34. <div class="tab-item" :class="{ active: activeTab === 2 }" @click="switchTab(2)">沉浸体验区</div>
  35. </div>
  36. <div class="tab-content">
  37. <div v-if="activeTab === 0">
  38. <p>配备先进的水文监测设备和实践设施,提供 hands-on 实践体验,培养学生的实际操作能力。</p>
  39. </div>
  40. <div v-else-if="activeTab === 1">
  41. <p>集合水文领域的前沿技术和设备,开展水文监测新技术、新方法的研发与测试。</p>
  42. </div>
  43. <div v-else-if="activeTab === 2">
  44. <p>通过 VR/AR 技术,沉浸式体验水文监测过程,增强学习趣味性和互动性。</p>
  45. </div>
  46. </div>
  47. <div class="architecture-title">产学研协同</div>
  48. <div class="collaboration-flow">
  49. <div class="arch-node">
  50. <div class="node-icon">🔍</div>
  51. <div class="node-label">需求调研</div>
  52. </div>
  53. <div class="arch-arrow">→</div>
  54. <div class="arch-node">
  55. <div class="node-icon">🔬</div>
  56. <div class="node-label">联合实验室</div>
  57. </div>
  58. <div class="arch-arrow">→</div>
  59. <div class="arch-node">
  60. <div class="node-icon">🚀</div>
  61. <div class="node-label">技术产品化</div>
  62. </div>
  63. <div class="arch-arrow">→</div>
  64. <div class="arch-node">
  65. <div class="node-icon">💼</div>
  66. <div class="node-label">企业应用</div>
  67. </div>
  68. </div>
  69. </div>
  70. <div class="action-buttons">
  71. <button class="action-btn primary">
  72. <span>🏫</span> 查看基地详情
  73. </button>
  74. </div>
  75. </div>
  76. </m-card>
  77. </template>
  78. <script setup>
  79. import { ref, inject } from 'vue'
  80. import mCard from "@/components/mCard/index.vue"
  81. import emitter from "@/utils/emitter"
  82. const activeTab = ref(1)
  83. function switchTab(index) {
  84. activeTab.value = index
  85. // 当点击技术实践区时,切换背景图
  86. if (index === 0) {
  87. emitter.$emit('switchStudyBgImage', 'practice')
  88. } else {
  89. emitter.$emit('switchStudyBgImage', 'default')
  90. }
  91. }
  92. </script>
  93. <style lang="scss" scoped>
  94. .study-base {
  95. padding: 12px;
  96. height: 100%;
  97. display: flex;
  98. flex-direction: column;
  99. gap: 12px;
  100. box-sizing: border-box;
  101. overflow-y: auto;
  102. &::-webkit-scrollbar {
  103. width: 6px;
  104. }
  105. &::-webkit-scrollbar-track {
  106. background: rgba(0, 40, 80, 0.6);
  107. border-radius: 3px;
  108. }
  109. &::-webkit-scrollbar-thumb {
  110. background: rgba(48, 220, 255, 0.5);
  111. border-radius: 3px;
  112. }
  113. &::-webkit-scrollbar-thumb:hover {
  114. background: rgba(48, 220, 255, 0.8);
  115. }
  116. }
  117. .core-indicator {
  118. text-align: center;
  119. padding: 12px;
  120. background: linear-gradient(135deg, rgba(48, 220, 255, 0.1) 0%, rgba(0, 191, 255, 0.05) 100%);
  121. border-radius: 8px;
  122. border: 1px solid rgba(48, 220, 255, 0.3);
  123. }
  124. .indicator-main {
  125. margin-bottom: 5px;
  126. }
  127. .main-value {
  128. font-size: 40px;
  129. font-weight: 700;
  130. color: #30DCFF;
  131. font-family: "D-DIN";
  132. text-shadow: 0 0 20px rgba(48, 220, 255, 0.5);
  133. }
  134. .main-unit {
  135. font-size: 20px;
  136. color: rgba(48, 220, 255, 0.8);
  137. font-weight: 600;
  138. }
  139. .indicator-label {
  140. color: rgba(255, 255, 255, 0.8);
  141. font-size: 12px;
  142. letter-spacing: 1px;
  143. }
  144. .stats-grid {
  145. display: grid;
  146. grid-template-columns: repeat(4, 1fr);
  147. gap: 6px;
  148. }
  149. .stat-item {
  150. background: rgba(0, 180, 255, 0.08);
  151. border: 1px solid rgba(0, 180, 255, 0.2);
  152. border-radius: 4px;
  153. padding: 6px 4px;
  154. text-align: center;
  155. transition: all 0.3s ease;
  156. &:hover {
  157. background: rgba(0, 180, 255, 0.15);
  158. border-color: rgba(48, 220, 255, 0.5);
  159. }
  160. }
  161. .stat-value {
  162. font-size: 16px;
  163. font-weight: 700;
  164. color: #30DCFF;
  165. font-family: "D-DIN";
  166. margin-bottom: 2px;
  167. }
  168. .stat-label {
  169. font-size: 9px;
  170. color: rgba(255, 255, 255, 0.6);
  171. line-height: 1.2;
  172. }
  173. .architecture-flow {
  174. flex: 1;
  175. background: rgba(0, 180, 255, 0.05);
  176. border-radius: 6px;
  177. padding: 8px;
  178. display: flex;
  179. flex-direction: column;
  180. gap: 6px;
  181. }
  182. .architecture-title {
  183. font-size: 11px;
  184. color: #30DCFF;
  185. font-weight: 600;
  186. padding-bottom: 4px;
  187. border-bottom: 1px solid rgba(48, 220, 255, 0.2);
  188. }
  189. .architecture-tabs {
  190. display: flex;
  191. gap: 4px;
  192. margin: 4px 0;
  193. .tab-item {
  194. flex: 1;
  195. padding: 6px 0;
  196. text-align: center;
  197. font-size: 10px;
  198. background: rgba(0, 40, 80, 0.6);
  199. border: 1px solid rgba(48, 220, 255, 0.2);
  200. border-radius: 4px;
  201. color: rgba(255, 255, 255, 0.7);
  202. cursor: pointer;
  203. transition: all 0.3s ease;
  204. &.active {
  205. background: linear-gradient(135deg, #30DCFF, #00BFFF);
  206. color: white;
  207. border-color: #30DCFF;
  208. }
  209. }
  210. }
  211. .tab-content {
  212. padding: 8px;
  213. background: rgba(0, 40, 80, 0.4);
  214. border-radius: 4px;
  215. border: 1px solid rgba(48, 220, 255, 0.2);
  216. p {
  217. font-size: 11px;
  218. color: rgba(255, 255, 255, 0.8);
  219. line-height: 1.4;
  220. margin: 0;
  221. text-align: center;
  222. }
  223. }
  224. .collaboration-flow {
  225. display: flex;
  226. align-items: center;
  227. justify-content: center;
  228. gap: 4px;
  229. padding: 6px 0;
  230. flex-wrap: wrap;
  231. }
  232. .arch-node {
  233. background: rgba(48, 220, 255, 0.15);
  234. border: 1px solid rgba(48, 220, 255, 0.4);
  235. border-radius: 6px;
  236. padding: 6px 4px;
  237. text-align: center;
  238. min-width: 60px;
  239. }
  240. .node-icon {
  241. font-size: 14px;
  242. margin-bottom: 2px;
  243. }
  244. .node-label {
  245. font-size: 8px;
  246. color: rgba(255, 255, 255, 0.9);
  247. }
  248. .arch-arrow {
  249. color: #30DCFF;
  250. font-size: 12px;
  251. font-weight: bold;
  252. }
  253. .action-buttons {
  254. display: flex;
  255. gap: 10px;
  256. }
  257. .action-btn {
  258. flex: 1;
  259. padding: 10px 12px;
  260. border-radius: 6px;
  261. border: none;
  262. cursor: pointer;
  263. font-size: 12px;
  264. display: flex;
  265. align-items: center;
  266. justify-content: center;
  267. gap: 5px;
  268. transition: all 0.3s ease;
  269. &.primary {
  270. background: linear-gradient(135deg, #30DCFF, #00BFFF);
  271. color: #fff;
  272. &:hover {
  273. box-shadow: 0 0 15px rgba(48, 220, 255, 0.5);
  274. transform: translateY(-2px);
  275. }
  276. }
  277. &.secondary {
  278. background: rgba(48, 220, 255, 0.15);
  279. border: 1px solid rgba(48, 220, 255, 0.4);
  280. color: #30DCFF;
  281. &:hover {
  282. background: rgba(48, 220, 255, 0.25);
  283. }
  284. }
  285. span {
  286. font-size: 14px;
  287. }
  288. }
  289. </style>