huliu 1 год назад
Родитель
Сommit
d2ae53432a

+ 81 - 0
package-lock.json

@@ -269,6 +269,21 @@
269 269
       "resolved": "https://registry.npmmirror.com/async-validator/-/async-validator-4.2.5.tgz",
270 270
       "integrity": "sha512-7HhHjtERjqlNbZtqNqy2rckN/SpOOlmDliet+lP7k+eKZEjPk3DgyeU9lIXLdeLz0uBbbVp+9Qdow9wJWgwwfg=="
271 271
     },
272
+    "asynckit": {
273
+      "version": "0.4.0",
274
+      "resolved": "https://registry.npmmirror.com/asynckit/-/asynckit-0.4.0.tgz",
275
+      "integrity": "sha512-Oei9OH4tRh0YqU3GxhX79dM/mwVgvbZJaSNaRk+bshkj0S5cfHcgYakreBjrHwatXKbz+IoIdYLxrKim2MjW0Q=="
276
+    },
277
+    "axios": {
278
+      "version": "1.4.0",
279
+      "resolved": "https://registry.npmmirror.com/axios/-/axios-1.4.0.tgz",
280
+      "integrity": "sha512-S4XCWMEmzvo64T9GfvQDOXgYRDJ/wsSZc7Jvdgx5u1sd0JwsuPLqb3SYmusag+edF6ziyMensPVqLTSc1PiSEA==",
281
+      "requires": {
282
+        "follow-redirects": "^1.15.0",
283
+        "form-data": "^4.0.0",
284
+        "proxy-from-env": "^1.1.0"
285
+      }
286
+    },
272 287
     "balanced-match": {
273 288
       "version": "1.0.2",
274 289
       "resolved": "https://registry.npmmirror.com/balanced-match/-/balanced-match-1.0.2.tgz",
@@ -315,6 +330,14 @@
315 330
         "readdirp": "~3.6.0"
316 331
       }
317 332
     },
333
+    "combined-stream": {
334
+      "version": "1.0.8",
335
+      "resolved": "https://registry.npmmirror.com/combined-stream/-/combined-stream-1.0.8.tgz",
336
+      "integrity": "sha512-FQN4MRfuJeHf7cBbBMJFXhKSDq+2kAArBlmRBvcvFE5BB1HZKXtSFASDhdlz9zOYwxh8lDdnvmMOe/+5cdoEdg==",
337
+      "requires": {
338
+        "delayed-stream": "~1.0.0"
339
+      }
340
+    },
318 341
     "csstype": {
319 342
       "version": "2.6.20",
320 343
       "resolved": "https://registry.npmmirror.com/csstype/-/csstype-2.6.20.tgz",
@@ -334,6 +357,11 @@
334 357
         "ms": "2.1.2"
335 358
       }
336 359
     },
360
+    "delayed-stream": {
361
+      "version": "1.0.0",
362
+      "resolved": "https://registry.npmmirror.com/delayed-stream/-/delayed-stream-1.0.0.tgz",
363
+      "integrity": "sha512-ZySD7Nf91aLB0RxL4KGrKHBXl7Eds1DAmEdcoVawXnLD7SDhpNgtuII2aAkg7a7QS41jxPSZ17p4VdGnMHk3MQ=="
364
+    },
337 365
     "echarts": {
338 366
       "version": "5.4.3",
339 367
       "resolved": "https://registry.npmmirror.com/echarts/-/echarts-5.4.3.tgz",
@@ -580,6 +608,21 @@
580 608
         "to-regex-range": "^5.0.1"
581 609
       }
582 610
     },
611
+    "follow-redirects": {
612
+      "version": "1.15.2",
613
+      "resolved": "https://registry.npmmirror.com/follow-redirects/-/follow-redirects-1.15.2.tgz",
614
+      "integrity": "sha512-VQLG33o04KaQ8uYi2tVNbdrWp1QWxNNea+nmIB4EVM28v0hmP17z7aG1+wAkNzVq4KeXTq3221ye5qTJP91JwA=="
615
+    },
616
+    "form-data": {
617
+      "version": "4.0.0",
618
+      "resolved": "https://registry.npmmirror.com/form-data/-/form-data-4.0.0.tgz",
619
+      "integrity": "sha512-ETEklSGi5t0QMZuiXoA/Q6vcnxcLQP5vdugSpuAyi6SVGi2clPPp+xgEhuMaHC+zGgn31Kd235W35f7Hykkaww==",
620
+      "requires": {
621
+        "asynckit": "^0.4.0",
622
+        "combined-stream": "^1.0.8",
623
+        "mime-types": "^2.1.12"
624
+      }
625
+    },
583 626
     "fsevents": {
584 627
       "version": "2.3.2",
585 628
       "resolved": "https://registry.npmmirror.com/fsevents/-/fsevents-2.3.2.tgz",
@@ -611,6 +654,11 @@
611 654
         "function-bind": "^1.1.1"
612 655
       }
613 656
     },
657
+    "hls.js": {
658
+      "version": "1.4.10",
659
+      "resolved": "https://registry.npmmirror.com/hls.js/-/hls.js-1.4.10.tgz",
660
+      "integrity": "sha512-wAVSj4Fm2MqOHy5+BlYnlKxXvJlv5IuZHjlzHu18QmjRzSDFQiUDWdHs5+NsFMQrgKEBwuWDcyvaMC9dUzJ5Uw=="
661
+    },
614 662
     "immutable": {
615 663
       "version": "4.3.2",
616 664
       "resolved": "https://registry.npmmirror.com/immutable/-/immutable-4.3.2.tgz",
@@ -712,6 +760,19 @@
712 760
         "picomatch": "^2.3.1"
713 761
       }
714 762
     },
763
+    "mime-db": {
764
+      "version": "1.52.0",
765
+      "resolved": "https://registry.npmmirror.com/mime-db/-/mime-db-1.52.0.tgz",
766
+      "integrity": "sha512-sPU4uV7dYlvtWJxwwxHD0PuihVNiE7TyAbQ5SWxDCB9mUYvOgroQOwYQQOKPJ8CIbE+1ETVlOoK1UC2nU3gYvg=="
767
+    },
768
+    "mime-types": {
769
+      "version": "2.1.35",
770
+      "resolved": "https://registry.npmmirror.com/mime-types/-/mime-types-2.1.35.tgz",
771
+      "integrity": "sha512-ZDY+bPm5zTTF+YpCrAU9nK0UgICYPT0QtT1NZWFv4s++TNkcgVaT0g6+4R2uI4MjQjzysHB1zxuWL50hzaeXiw==",
772
+      "requires": {
773
+        "mime-db": "1.52.0"
774
+      }
775
+    },
715 776
     "minimatch": {
716 777
       "version": "9.0.3",
717 778
       "resolved": "https://registry.npmmirror.com/minimatch/-/minimatch-9.0.3.tgz",
@@ -799,6 +860,11 @@
799 860
         "source-map-js": "^1.0.2"
800 861
       }
801 862
     },
863
+    "proxy-from-env": {
864
+      "version": "1.1.0",
865
+      "resolved": "https://registry.npmmirror.com/proxy-from-env/-/proxy-from-env-1.1.0.tgz",
866
+      "integrity": "sha512-D+zkORCbA9f1tdWRK0RaCR3GPv50cMxcrz4X8k5LTSUD1Dkw47mKJEZQNunItRTkWwgtaUSo1RVFRIG9ZXiFYg=="
867
+    },
802 868
     "queue-microtask": {
803 869
       "version": "1.2.3",
804 870
       "resolved": "https://registry.npmmirror.com/queue-microtask/-/queue-microtask-1.2.3.tgz",
@@ -896,6 +962,11 @@
896 962
       "integrity": "sha512-ot0WnXS9fgdkgIcePe6RHNk1WA8+muPa6cSjeR3V8K27q9BB1rTE3R1p7Hv0z1ZyAc8s6Vvv8DIyWf681MAt0w==",
897 963
       "dev": true
898 964
     },
965
+    "throttle-debounce": {
966
+      "version": "3.0.1",
967
+      "resolved": "https://registry.npmmirror.com/throttle-debounce/-/throttle-debounce-3.0.1.tgz",
968
+      "integrity": "sha512-dTEWWNu6JmeVXY0ZYoPuH5cRIwc0MeGbJwah9KUNYSJwommQpCzTySTpEe8Gs1J23aeWEuAobe4Ag7EHVt/LOg=="
969
+    },
899 970
     "to-regex-range": {
900 971
       "version": "5.0.1",
901 972
       "resolved": "https://registry.npmmirror.com/to-regex-range/-/to-regex-range-5.0.1.tgz",
@@ -1064,6 +1135,16 @@
1064 1135
       "resolved": "https://registry.npmmirror.com/vue-demi/-/vue-demi-0.14.5.tgz",
1065 1136
       "integrity": "sha512-o9NUVpl/YlsGJ7t+xuqJKx8EBGf1quRhCiT6D/J0pfwmk9zUwYkC7yrF4SZCe6fETvSM3UNL2edcbYrSyc4QHA=="
1066 1137
     },
1138
+    "vue3-video-play": {
1139
+      "version": "1.3.1-beta.6",
1140
+      "resolved": "https://registry.npmmirror.com/vue3-video-play/-/vue3-video-play-1.3.1-beta.6.tgz",
1141
+      "integrity": "sha512-Olrx2/LNAds7fuor/yX9ZKT9sOcwcfTt2g2YbbCrEaAmZ5Tb0hwBr5z+/CoLwELzzRzXCHPmWWoT0Wm5W/Nwpw==",
1142
+      "requires": {
1143
+        "hls.js": "^1.0.10",
1144
+        "throttle-debounce": "^3.0.1",
1145
+        "vue": "^3.2.2"
1146
+      }
1147
+    },
1067 1148
     "webpack-sources": {
1068 1149
       "version": "3.2.3",
1069 1150
       "resolved": "https://registry.npmmirror.com/webpack-sources/-/webpack-sources-3.2.3.tgz",

+ 3 - 1
package.json

@@ -8,9 +8,11 @@
8 8
     "preview": "vite preview"
9 9
   },
10 10
   "dependencies": {
11
+    "axios": "^1.4.0",
11 12
     "echarts": "^5.4.3",
12 13
     "element-plus": "^2.3.9",
13
-    "vue": "^3.2.25"
14
+    "vue": "^3.2.25",
15
+    "vue3-video-play": "^1.3.1-beta.6"
14 16
   },
15 17
   "devDependencies": {
16 18
     "@vitejs/plugin-vue": "^2.3.3",

BIN
src/assets/img/d1_1.png


BIN
src/assets/img/d1_2.png


BIN
src/assets/img/r3.png


+ 116 - 0
src/components/CircleProgress.vue

@@ -0,0 +1,116 @@
1
+<template>
2
+    <div>
3
+        <div class="number" v-if="textPosition == 'top'">
4
+            {{ showProgress }} <span> (节)</span>
5
+        </div>
6
+    </div>
7
+    <div class="progress" :style="{ width, height }">
8
+        <svg viewBox="0 0 96 96" class="svg-circle-progress">
9
+            <circle r="38" cx="48" cy="48" fill="none" stroke-miterlimit="20" stroke-width="4" class="svg-progress"
10
+                style="stroke-dasharray: 275, 279.602; stroke: rgba(255, 255, 255, 0.19)"></circle>
11
+            <circle r="38" cx="48" cy="48" fill="none" stroke-miterlimit="20" stroke-width="4" class="svg-progress"
12
+                :style="`stroke-dasharray: ${progressValue}, 279.602;stroke:${color};`"></circle>
13
+        </svg>
14
+        <div class="mask">
15
+            <slot></slot>
16
+        </div>
17
+    </div>
18
+    <div class="bottomNumber" v-if="textPosition == 'bottom'">
19
+        {{ showProgress }} <span> (节)</span>
20
+    </div>
21
+</template>
22
+<script setup lang="ts">
23
+import { ref, toRefs, watch, onMounted } from 'vue';
24
+
25
+const props = defineProps({
26
+    targetValue: {
27
+        type: Number,
28
+        require: false,
29
+        default: -1,
30
+    },
31
+    color: {
32
+        type: String,
33
+        default: '#4c7cee',
34
+    },
35
+    width: {
36
+        type: String,
37
+        default: '60px',
38
+    },
39
+    height: {
40
+        type: String,
41
+        default: '60px',
42
+    },
43
+    textPosition: {
44
+        type: String,
45
+        require: false,
46
+        default: 'top',
47
+    }
48
+});
49
+const { height, width, color, targetValue, textPosition } = toRefs(props);
50
+
51
+const showProgress = ref<number>(0);
52
+const addData = () => {
53
+    if (targetValue.value === 0) return;
54
+    let timer = setInterval(() => {
55
+        if (targetValue.value === showProgress.value) {
56
+            clearInterval(timer), timer == null;
57
+            return;
58
+        }
59
+        ++showProgress.value;
60
+    }, 15);
61
+};
62
+onMounted(() => {
63
+    addData();
64
+});
65
+const progressValue = ref<number>((showProgress.value / 360) * 250);
66
+
67
+watch(showProgress, (newValue: number) => {
68
+    progressValue.value = (newValue / 360) * 250;
69
+});
70
+</script>
71
+<style lang="scss" scoped>
72
+.progress {
73
+
74
+    display: flex;
75
+    flex-direction: column;
76
+
77
+    align-items: center;
78
+    justify-content: center;
79
+    text-align: center;
80
+}
81
+
82
+.svg-circle-progress {
83
+    position: relative;
84
+    transform: rotate(-90deg);
85
+
86
+}
87
+
88
+.svg-progress {
89
+    stroke: #2196f3;
90
+    stroke-linecap: round;
91
+    transition: all 0.3s linear;
92
+}
93
+
94
+.mask {
95
+    position: absolute;
96
+    margin-top: 5px;
97
+}
98
+
99
+.number {
100
+    font-size: 14px;
101
+    font-weight: bold;
102
+
103
+    span {
104
+        font-size: 12px;
105
+    }
106
+}
107
+
108
+.bottomNumber {
109
+    font-size: 14px;
110
+    font-weight: bold;
111
+
112
+    span {
113
+        font-size: 12px;
114
+    }
115
+}
116
+</style>

+ 10 - 9
src/components/UeVideo.vue

@@ -35,20 +35,21 @@ export default {
35 35
         },
36 36
       });
37 37
 
38
-      // addResponseEventListener("hello", async (data) => {
39
-      //   alert(data);
40
-      // });
41 38
     });
42
-    function hadleResponseFunction() {
43
-      console.log("监听一下数据---asdfadsf",)
39
+    // function handleResponseFunction(data) {
40
+    //   console.log("从UE返回过来的值", data)
41
+    // }
42
+    function toUE() {
43
+      // 向UE发送消息
44
+      // callUIInteraction("open");
45
+      // 同时接收从UE返回过来的值
46
+      // debugger
47
+      // addResponseEventListener("open", handleResponseFunction)
44 48
     }
45 49
 
46 50
     return {
47 51
       video,
48
-      toUE() {
49
-        console.log("点击了ToUE");
50
-        addResponseEventListener("hello", hadleResponseFunction)
51
-      },
52
+      toUE
52 53
     };
53 54
   },
54 55
 };

+ 580 - 105
src/components/home.vue

@@ -6,6 +6,7 @@
6 6
         <el-container>
7 7
             <el-aside width="412px" class="left">
8 8
                 <div class="title">
9
+
9 10
                     <div class="text">教室分类统计</div>
10 11
                 </div>
11 12
                 <div class="content">
@@ -21,18 +22,49 @@
21 22
                     <div class="text">本周课程统计</div>
22 23
                 </div>
23 24
                 <div class="content">
25
+
24 26
                     <div class="count" v-for="(item, index) in classCount" :key="index">
25
-                        <el-progress type="circle" :percentage="25" :color="item.color">
27
+                        <CircleProgress :target-value="item.number" :color="item.color" text-position="bottom">
26 28
                             <img :src="'src/assets/img/' + item.imgnumber + '.png'" alt="">
27
-                        </el-progress>
28
-                        <div class="number"><span> {{ item.number }}</span>(节)</div>
29
+                        </CircleProgress>
30
+                        <!-- <el-progress type="circle" :percentage="25" :color="item.color">
31
+                            <img :src="'src/assets/img/' + item.imgnumber + '.png'" alt="">
32
+                        </el-progress> -->
33
+                        <!-- <div class="number"><span> {{ item.number }}</span>(节)</div> -->
29 34
                         <div class="text">{{ item.text }}</div>
30 35
                     </div>
31 36
                 </div>
32 37
                 <div class="title">
33 38
                     <div class="text">智慧教室使用明细</div>
34 39
                 </div>
40
+                <!-- <div ref="testMain" style="height: 110px;overflow: auto;">
41
+
42
+
43
+                    <ul v-for="item in classRoom">
44
+                        <li>
45
+                            {{ item.class }}
46
+                        </li>
47
+                        <li>{{ item.status }}</li>
48
+                        <li>{{ item.people }}</li>
49
+                        <li>{{ item.percentage }}</li>
50
+                        <li>
51
+                            <el-button size="small" class="look" @click.prevent="handleRoom">操作</el-button>
52
+                        </li>
53
+                    </ul>
54
+                </div> -->
35 55
                 <div class="content">
56
+
57
+                    <!-- <div @mousemove="testMove" @mouseleave="testMend">
58
+                        <div ref="testMain" style="height: 110px;overflow: auto; margin:20px;">
59
+
60
+                            <div v-for="item in listData" :key="item.id">
61
+                                <span>{{ item.name }}</span>
62
+                            </div>
63
+
64
+                        </div>
65
+                    </div> -->
66
+
67
+
36 68
                     <div class="table">
37 69
                         <ul class="th">
38 70
                             <li>教室</li>
@@ -42,7 +74,8 @@
42 74
                             <li>操作</li>
43 75
                         </ul>
44 76
 
45
-                        <el-scrollbar height="400px">
77
+                        <div ref="testMain" style="height: 310px;overflow: hidden;">
78
+
46 79
                             <ul v-for="item in classRoom">
47 80
                                 <li>
48 81
                                     {{ item.class }}
@@ -51,10 +84,27 @@
51 84
                                 <li>{{ item.people }}</li>
52 85
                                 <li>{{ item.percentage }}</li>
53 86
                                 <li>
54
-                                    <el-button size="small" class="look" @click="handleRoom">操作</el-button>
87
+                                    <el-button size="small" class="look" @click.prevent="handleRoom">操作</el-button>
55 88
                                 </li>
89
+
56 90
                             </ul>
57
-                        </el-scrollbar>
91
+
92
+                        </div>
93
+
94
+                        <!-- <el-scrollbar height="310px">
95
+                            <ul v-for="item in classRoom">
96
+                                <li>
97
+                                    {{ item.class }}
98
+                                </li>
99
+                                <li>{{ item.status }}</li>
100
+                                <li>{{ item.people }}</li>
101
+                                <li>{{ item.percentage }}</li>
102
+                                <li>
103
+                                    <el-button size="small" class="look" @click.prevent="handleRoom">操作</el-button>
104
+                                </li>
105
+
106
+                            </ul>
107
+                        </el-scrollbar> -->
58 108
 
59 109
                     </div>
60 110
                 </div>
@@ -66,7 +116,10 @@
66 116
                 <div class="content">
67 117
                     <div class="img">
68 118
                         <span>120</span>
69
-                        <img src="../assets/img/d1.png" alt="">
119
+                        <div class="animate">
120
+                            <img src="../assets/img/d1_1.png" alt="">
121
+                            <img src="../assets/img/d1_2.png" alt="">
122
+                        </div>
70 123
                         <span>总数(个)</span>
71 124
                     </div>
72 125
                     <div class="list">
@@ -90,47 +143,113 @@
90 143
                 <div class="title">
91 144
                     <div class="text">物联设备类型统计</div>
92 145
                 </div>
146
+
147
+
93 148
                 <div class="contentwrap">
94 149
 
150
+                    <!-- <CircleProgress :target-value="1">
151
+                        <img :src="'src/assets/img/' + 1 + '.png'" alt="">
152
+                    </CircleProgress>
153
+                    <CircleProgress :target-value="52" />
154
+                    <CircleProgress :target-value="54" /> -->
95 155
                     <div class="count" v-for="(item, index) in interDevice" :key="index">
156
+                        <CircleProgress :target-value="item.number" :color="item.color" text-position="top">
157
+                            <img :src="'src/assets/img/' + item.icon + '.png'" alt="">
158
+                        </CircleProgress>
159
+                        <div class="text">{{ item.text }}</div>
160
+                    </div>
161
+                    <!-- <div class="count" v-for="(item, index) in interDevice" :key="index">
96 162
                         <div class="number"><span> {{ item.number }}</span>(节)</div>
97
-                        <el-progress type="circle" :percentage="100" :color="item.color">
163
+                        <el-progress type="circle" :percentage="45" :color="item.color">
98 164
                             <img :src="'src/assets/img/' + item.icon + '.png'" alt="">
99 165
                         </el-progress>
100 166
                         <div class="text">{{ item.text }}</div>
101
-                    </div>
167
+                    </div> -->
102 168
                 </div>
103 169
                 <div class="title">
104 170
                     <div class="text">智慧教室实时监控</div>
105 171
                 </div>
106 172
                 <div class="content">
107
-                    <div class="monitor">
108
-                        <p>多屏互动教室</p>
109
-                        <div class="interactclass">
110
-                            <div class="room"><span>互动教室1</span>
111
-                                <img src="../assets/img/rb1.png" alt="">
112
-                            </div>
113
-                            <div class="room"><span>互动教室2</span>
114
-                                <img src="../assets/img/rb2.png" alt="">
115
-                            </div>
116
-                        </div>
117
-                        <p>多屏智慧教室</p>
118
-                        <div class="interactclass">
119
-                            <div class="room"><span>智慧教室1</span>
120
-                                <img src="../assets/img/rb1.png" alt="">
121
-                            </div>
122
-                            <div class="room"><span>智慧教室2</span>
123
-                                <img src="../assets/img/rb2.png" alt="">
124
-                            </div>
125
-                        </div>
126
-                        <el-pagination layout="prev, pager, next" :total="50" />
173
+                    <div class="monitorContent">
174
+                        <el-carousel class="monitor">
175
+                            <el-carousel-item v-for="(item, index) in carouselData" :key="index">
176
+                                <p>{{ item.activeMonitor.title }}</p>
177
+                                <div class="interactclass">
178
+                                    <div class="room" v-for="(item, index) in item.activeMonitor.array" :key="index">
179
+                                        <span>{{ item.title }}</span>
180
+                                        <img :src="'src/assets/img/' + item.poster + '.png'" alt="" @click="lookVideo">
181
+                                    </div>
182
+                                </div>
183
+                                <p>{{ item.wisdomMonitor.title }}</p>
184
+                                <div class="interactclass">
185
+                                    <div class="room" v-for="(item, index) in item.wisdomMonitor.array" :key="index">
186
+                                        <span>{{ item.title }}</span>
187
+                                        <img :src="'src/assets/img/' + item.poster + '.png'" alt="" @click="lookVideo">
188
+                                    </div>
189
+                                </div>
190
+                                <!-- <p>{{ item.wisdomMonitor.title }}</p>
191
+                                <div class="interactclass">
192
+                                    <div class="room" v-for="(item, index) in item.wisdomMonitor.array" :key="index">
193
+                                        <span>{{ item.title }}</span>
194
+                                        <img :src="'src/assets/img/' + item.poster + '.png'" alt="" @click="lookVideo">
195
+                                    </div>
196
+                                </div> -->
197
+                                <!-- <p>多屏互动教室</p>
198
+                                <div class="interactclass">
199
+                                    <div class="room" v-for="(item, index) in activeMonitor.array" :key="index">
200
+                                        <span>{{ item.title }}</span>
201
+                                        <img :src="'src/assets/img/' + item.poster + '.png'" alt="" @click="lookVideo">
202
+                                    </div>
203
+                                </div>
204
+                                <p>多屏智慧教室</p>
205
+                                <div class="interactclass">
206
+                                    <div class="room" v-for="(item, index) in wisdomMonitor" :key="index">
207
+                                        <span>{{ item.title }}</span>
208
+                                        <img :src="'src/assets/img/' + item.poster + '.png'" alt="">
209
+                                    </div>
210
+                                </div> -->
211
+                                <!-- <p>多屏互动教室</p>
212
+                                <div class="interactclass" v-for="(item, index) in activeMonitor" :key="index">
213
+                                    <div class="room" @click="lookVideo">
214
+                                        <span>{{ item.title }}</span>
215
+                                        <img :src="'src/assets/img/' + item.poster + '.png'" alt="">
216
+                                    </div>
217
+                                </div> -->
218
+
219
+                                <!-- <p>多屏智慧教室</p>
220
+                                <div class="interactclass">
221
+                                    <div class="room"><span>智慧教室1</span>
222
+                                        <img src="../assets/img/rb1.png" alt="">
223
+                                    </div>
224
+                                    <div class="room"><span>智慧教室2</span>
225
+                                        <img src="../assets/img/rb2.png" alt="">
226
+                                    </div>
227
+                                </div> -->
228
+                                <!-- <p>多屏智慧教室</p>
229
+                                <div class="interactclass">
230
+                                    <div class="room"><span>智慧教室1</span>
231
+                                        <img src="../assets/img/rb1.png" alt="">
232
+                                    </div>
233
+                                    <div class="room"><span>智慧教室2</span>
234
+                                        <img src="../assets/img/rb2.png" alt="">
235
+                                    </div>
236
+                                </div> -->
237
+                            </el-carousel-item>
238
+                        </el-carousel>
239
+
240
+                        <el-pagination layout="prev, pager, next" :total="30" />
127 241
 
128 242
                     </div>
129 243
                 </div>
130 244
             </el-aside>
131 245
         </el-container>
246
+
247
+
248
+
132 249
         <el-dialog v-model="centerDialogVisible" title="Warning" width="30%" align-center>
133
-            <span>Open the dialog from the center from the screen</span>
250
+            <span>
251
+                {{ clickData.value && clickData.value.status }}
252
+            </span>
134 253
             <template #footer>
135 254
                 <span class="dialog-footer">
136 255
                     <el-button @click="centerDialogVisible = false">Cancel</el-button>
@@ -140,17 +259,83 @@
140 259
                 </span>
141 260
             </template>
142 261
         </el-dialog>
262
+
263
+
264
+        <el-popover ref="popoverRef" v-model:visible="visible" :width="400" :virtual-ref="triggerRef" trigger="click"
265
+            virtual-triggering>
266
+            <div class="model-detail">
267
+                <span><span class="title">教室状态: </span>{{ clickData.value && clickData.value.status }}</span>
268
+                <span><span class="title">班级: </span>{{ clickData.value && clickData.value.classes }}</span>
269
+                <span><span class="title">课程名称: </span>{{ clickData.value && clickData.value.course }}</span>
270
+                <span><span class="title">应到人数: </span>{{ clickData.value && clickData.value.theory }}</span>
271
+                <span><span class="title">教室: </span>{{ clickData.value && clickData.value.name }}</span>
272
+                <span><span class="title">实到人数: </span>{{ clickData.value && clickData.value.actual }}</span>
273
+                <span><span class="title">老师: </span>{{ clickData.value && clickData.value.teacher }}</span>
274
+            </div>
275
+        </el-popover>
276
+
277
+        <!-- 视频播放弹框 -->
278
+        <el-dialog v-model="videoDialog" title="互动教师1" width="60%" align="center" top="4%">
279
+            <videoPlay v-bind="videoData" @play="onPlay" />
280
+
281
+            <!-- <template #footer>
282
+                <span class="dialog-footer">
283
+                    <el-button @click="centerDialogVisible = false">Cancel</el-button>
284
+                    <el-button type="primary" @click="centerDialogVisible = false">
285
+                        Confirm
286
+                    </el-button>
287
+                </span>
288
+            </template> -->
289
+        </el-dialog>
143 290
     </div>
144 291
 </template>
145 292
 
146
-<script>
293
+<script >
147 294
 import * as echarts from "echarts";
148
-import { toRefs, reactive, onMounted, ref } from 'vue';
149
-import { ElScrollbar, ElPagination, ElDialog } from "element-plus";
295
+import { reactive, onMounted, ref, onBeforeUnmount, onUnmounted, nextTick } from 'vue';
296
+import { ElScrollbar, ElPagination, ElDialog, ElCarousel, ElCarouselItem } from "element-plus";
297
+import "vue3-video-play/dist/style.css";
298
+import {
299
+    callUIInteraction,
300
+    addResponseEventListener
301
+} from "../webrtcVideo.js";
302
+import { testApi } from '../request/api'
303
+import { videoPlay } from "vue3-video-play";
304
+import CircleProgress from './CircleProgress.vue';
305
+
306
+
150 307
 export default ({
151 308
     name: 'Histogram',
152
-    components: { ElScrollbar, ElPagination, ElDialog },
309
+    components: { ElScrollbar, ElPagination, ElDialog, videoPlay, ElCarousel, ElCarouselItem, CircleProgress },
153 310
     setup() {
311
+        // 视频数据
312
+        const videoData = reactive({
313
+            width: "100%", //播放器高度
314
+            height: "100%", //播放器高度
315
+            color: "red", //主题色
316
+            title: "互动教室", //视频名称
317
+            src: "https://cdn.jsdelivr.net/gh/xdlumia/files/video-play/IronMan.mp4", //视频源
318
+            muted: false, //静音
319
+            webFullScreen: false,
320
+            // speedRate: ["0.75", "1.0", "1.25", "1.5", "2.0"], //播放倍速
321
+            autoPlay: false, //自动播放
322
+            loop: false, //循环播放
323
+            mirror: false, //镜像画面
324
+            ligthOff: false, //关灯模式
325
+            volume: 0.3, //默认音量大小
326
+            control: true, //是否显示控制
327
+            currentTime: 0,//跳转到固定播放时间
328
+            controlBtns: [
329
+                "audioTrack",
330
+                "quality",
331
+                "speedRate",
332
+                "volume",
333
+                "setting",
334
+                "pip",
335
+                "pageFullScreen",
336
+                "fullScreen",
337
+            ], //显示所有按钮,
338
+        });
154 339
         const pieData = reactive([{
155 340
             value: 1048,
156 341
             name: '基础型'
@@ -235,21 +420,20 @@ export default ({
235 420
                 percentage: 20
236 421
             },
237 422
             {
238
-                number: 473,
423
+                number: 73,
239 424
                 text: '实上课程',
240 425
                 imgnumber: "2",
241 426
                 color: '#e6a23c'
242 427
             },
243 428
             {
244
-                number: 1352,
429
+                number: 352,
245 430
                 text: '应到学生数',
246 431
                 imgnumber: "3",
247 432
                 color: '#5cb87a',
248 433
             },
249 434
             {
250
-                number: 2532,
435
+                number: 53,
251 436
                 text: '实到学生数',
252
-
253 437
                 imgnumber: "4",
254 438
                 color: '#1989fa',
255 439
             }
@@ -373,21 +557,237 @@ export default ({
373 557
         ])
374 558
         // 位置弹框
375 559
         let centerDialogVisible = ref(false)
376
-
560
+        // 获取点击点数据
561
+        let clickData = reactive({})
562
+        const visible = ref(false)
563
+        const triggerRef = ref({
564
+            getBoundingClientRect() {
565
+                // console.log("positon----方法返回元素的大小及其相对于视口的位置", position.value)
566
+                return position.value
567
+            },
568
+        })
569
+        const position = ref({
570
+            top: 0,
571
+            left: 0,
572
+            bottom: 0,
573
+            right: 0,
574
+        })
575
+        // 点击某个模型跟随移动
576
+        const mousemoveHandler = (x, y) => {
577
+            position.value = DOMRect.fromRect({
578
+                width: 0,
579
+                height: 0,
580
+                x: x,
581
+                y: y,
582
+            })
583
+        }
377 584
         const handleRoom = function () {
585
+            // 发送消息给UE
378 586
             console.log("查看当前教室使用情况", 'adsfasdfasdfa')
587
+            callUIInteraction("open");
588
+            addResponseEventListener("open", (data) => {
589
+                console.log("接收到的信息", data, typeof data,)
590
+            })
379 591
             centerDialogVisible.value = true
380 592
         }
593
+        const testGetApi = async () => {
594
+            let res = await testApi("ymy")
595
+            console.log("请求到的", res);
596
+        }
597
+        //视频播放弹框
598
+        const videoDialog = ref(false)
599
+        // 播放视频
600
+        const lookVideo = function () {
601
+
602
+            videoDialog.value = true;
603
+            console.log("点击了播放视频", 12312312)
604
+        }
605
+        // 播放视频
606
+        const onPlay = function () {
607
+            console.log("播放视频")
608
+        }
609
+        //智慧教室实时监控---数据
610
+        const carouselData = ref([
611
+            {
612
+                activeMonitor: {
613
+                    title: "多屏互动教室",
614
+                    array: [
615
+                        {
616
+                            title: "互动教室22",
617
+                            poster: 'rb1',
618
+                            src: '',
619
+                        },
620
+                        {
621
+                            title: "互动教室22",
622
+                            poster: 'rb2',
623
+                            src: '',
624
+                        }
625
+                    ]
626
+                },
627
+
628
+                wisdomMonitor:
629
+                {
630
+                    title: "多屏智慧教室",
631
+                    array: [
632
+                        {
633
+                            title: "智慧教室11",
634
+                            poster: 'rb1',
635
+                            src: '',
636
+                        },
637
+                        {
638
+                            title: "智慧教室11",
639
+                            poster: 'rb2',
640
+                            src: '',
641
+                        }
642
+                    ]
643
+                }
644
+            },
645
+            {
646
+                activeMonitor: {
647
+                    title: "多屏互动教室",
648
+                    array: [
649
+                        {
650
+                            title: "互动教室33",
651
+                            poster: 'rb1',
652
+                            src: '',
653
+                        },
654
+                        {
655
+                            title: "互动教室33",
656
+                            poster: 'rb2',
657
+                            src: '',
658
+                        }
659
+                    ]
660
+                },
661
+
662
+                wisdomMonitor:
663
+                {
664
+                    title: "多屏智慧教室",
665
+                    array: [
666
+                        {
667
+                            title: "智慧教室22",
668
+                            poster: 'rb1',
669
+                            src: '',
670
+                        },
671
+                        {
672
+                            title: "智慧教室22",
673
+                            poster: 'rb2',
674
+                            src: '',
675
+                        }
676
+                    ]
677
+                }
678
+            }
679
+
680
+        ]
681
+        )
682
+
683
+        let timer = ref(null)
684
+        let testMain = ref(null)
381 685
         onMounted(() => {
382
-            initeCharts()
686
+            initeCharts();
687
+            testGetApi()
688
+            // 监听一下页面点击情况
689
+            addResponseEventListener("open", (data) => {
690
+                if (data) {
691
+                    let json = eval("(" + data + ")");//转对象
692
+                    clickData.value = json;
693
+                    console.log("clickData---点击的数据", clickData.value)
694
+                    visible.value = true;
695
+                    mousemoveHandler(json.x, json.y)
696
+                }
697
+            })
698
+
383 699
         })
700
+
701
+        const listData = reactive([
702
+            { name: '我是dom第一个' },
703
+            { name: '我是dom第二个' },
704
+            { name: '我是dom第三个' },
705
+            { name: '我是dom第四个' },
706
+            { name: '我是dom第五个' },
707
+            { name: '我是dom第六个' },
708
+            { name: '我是dom第七个' },
709
+            { name: '我是dom第八个' },
710
+            { name: '我是dom第九个' },
711
+            { name: '我是dom第十个' },
712
+        ])
713
+
714
+        onBeforeUnmount(() => {
715
+            clearTimeout(timer.value)
716
+        })
717
+        onUnmounted(() => {
718
+            clearTimeout(timer.value)
719
+        })
720
+
721
+
722
+        function testMove() {
723
+            clearTimeout(timer.value)
724
+        }
725
+
726
+        function testMend() {
727
+            start()
728
+        }
729
+        //开启定时器方法
730
+        function start() {
731
+            //清除定时器
732
+            clearTimeout(timer.value)
733
+            //定时器触发周期
734
+            let speed = ref(75)
735
+            timer.value = setInterval(MarqueeTest, speed.value)
736
+        }
737
+        function MarqueeTest() {
738
+            let test1 = testMain.value
739
+            //判断组件是否渲染完成
740
+            if (test1.offsetHeight == 0) {
741
+                test1 = testMain.value
742
+            } else {
743
+                //如果列表数量过少不进行滚动
744
+                console.log("test1", test1.scrollTop)
745
+                // if (test1.childNodes.length < 6) {
746
+                //     clearTimeout(timer.value)
747
+                //     return;
748
+                // }
749
+                console.log("test");
750
+                //组件进行滚动
751
+                test1.scrollTop += 1
752
+
753
+                //判断滚动条是否滚动到底部
754
+                if (test1.scrollTop == (test1.scrollHeight - test1.clientHeight)) {
755
+                    //获取组件第一个节点
756
+                    let a = test1.childNodes[0]
757
+                    //删除节点
758
+                    test1.removeChild(a)
759
+                    //将该节点拼接到组件最后
760
+                    test1.append(a)
761
+                }
762
+            }
763
+        }
764
+
765
+
766
+
767
+
768
+        nextTick(() => {
769
+            start()
770
+        })
771
+
384 772
         return {
385
-            ...toRefs(classPie),
386 773
             classCount,
387 774
             classRoom,
388 775
             centerDialogVisible,
389 776
             interDevice,
390
-            handleRoom
777
+            clickData,
778
+            handleRoom,
779
+            visible,
780
+            triggerRef,
781
+            videoData,
782
+            onPlay,
783
+            videoDialog,
784
+            lookVideo,
785
+            // ...toRefs(carouselData)
786
+            carouselData,
787
+            testMend,
788
+            testMove,
789
+            listData,
790
+            testMain
391 791
         }
392 792
     },
393 793
 })
@@ -416,30 +816,31 @@ export default ({
416 816
 }
417 817
 
418 818
 .el-container {
419
-    position: absolute;
420
-    top: 40px;
421
-    left: 0;
422
-    z-index: 300;
423
-    // border: 3px solid darkgreen;
424
-    width: calc(100% - 30px);
425
-    margin: 20px;
426
-    padding-left: 12px;
427
-    padding-top: 30px;
428 819
     height: 888px;
429
-
430 820
     overflow: hidden;
431
-    color: #FFF;
432
-    font-size: 14px;
433
-    display: flex;
434
-    justify-content: space-between;
435 821
 
436 822
     .left,
437 823
     .right {
824
+        position: absolute;
825
+        width: 410px;
826
+        top: 40px;
827
+        z-index: 300;
438 828
         flex-shrink: 0;
439 829
         border-radius: 2px;
440 830
         background: rgba(125, 125, 125, 0.17);
441 831
         backdrop-filter: blur(30.5px);
832
+        color: #FFF;
833
+        font-size: 14px;
442 834
         overflow: hidden;
835
+
836
+    }
837
+
838
+    .left {
839
+        left: 20px;
840
+    }
841
+
842
+    .right {
843
+        right: 20px;
443 844
     }
444 845
 
445 846
     .title {
@@ -512,10 +913,10 @@ export default ({
512 913
             flex-direction: column;
513 914
             align-items: center;
514 915
             width: 70px;
515
-            height: 107px;
916
+
516 917
             text-align: center;
517 918
             justify-content: space-around;
518
-            margin: 30px 0;
919
+            margin: 24px 0;
519 920
             padding: 20px;
520 921
 
521 922
             .el-progress-circle {
@@ -573,14 +974,9 @@ export default ({
573 974
                         border: 1px solid purple;
574 975
                     }
575 976
 
576
-                    // &:nth-child()
577
-                    // width: 130px;
578
-                    // 
579
-                    // text-align: center !important;
580 977
                 }
581 978
             }
582 979
 
583
-
584 980
             ul {
585 981
                 display: flex;
586 982
                 list-style: none;
@@ -619,6 +1015,23 @@ export default ({
619 1015
             align-items: center;
620 1016
             padding: 20px 10px;
621 1017
 
1018
+            .animate {
1019
+                display: flex;
1020
+                flex-direction: column;
1021
+
1022
+                img {
1023
+                    width: 120px;
1024
+                    vertical-align: top;
1025
+
1026
+                    &:first-child {
1027
+                        margin-bottom: -70px;
1028
+                        animation: myfirst 1.5s infinite;
1029
+
1030
+                    }
1031
+
1032
+                }
1033
+            }
1034
+
622 1035
             span {
623 1036
                 &:first-child {
624 1037
                     font-size: 18px;
@@ -662,11 +1075,26 @@ export default ({
662 1075
         }
663 1076
     }
664 1077
 
1078
+    // 物联网设备统计
1079
+    @keyframes myfirst {
1080
+        0% {
1081
+            transform: translate(0, 0);
1082
+        }
1083
+
1084
+        50% {
1085
+            transform: translate(0, -8px);
1086
+        }
1087
+
1088
+        100% {
1089
+            transform: translate(0, 0);
1090
+        }
1091
+    }
1092
+
665 1093
     // 物联设备类型统计
666 1094
     .contentwrap {
667
-
668 1095
         display: flex;
669 1096
         flex-wrap: wrap;
1097
+        margin: 10px 0;
670 1098
 
671 1099
         .count {
672 1100
             display: flex;
@@ -676,25 +1104,26 @@ export default ({
676 1104
             text-align: center;
677 1105
             width: 110px;
678 1106
 
679
-            .el-progress-circle {
680
-                width: 54.6px !important;
681
-                height: 54.6px !important;
1107
+            // .el-progress-circle {
1108
+            //     width: 54.6px !important;
1109
+            //     height: 54.6px !important;
682 1110
 
683
-                .el-progress-circle__track {
684
-                    stroke: rgba(255, 255, 255, 0.2);
685
-                }
1111
+            //     .el-progress-circle__track {
1112
+            //         stroke: rgba(255, 255, 255, 0.2);
1113
+            //     }
686 1114
 
687
-                margin: 5px 0;
688
-            }
1115
+            //     margin: 5px 0;
1116
+            // }
689 1117
 
690
-            .number {
691
-                font-size: 12px;
1118
+            // .number {
1119
+            //     border: 1px solid darkcyan;
1120
+            //     font-size: 12px;
692 1121
 
693
-                span {
694
-                    font-size: 14px;
695
-                    font-weight: bold;
696
-                }
697
-            }
1122
+            //     span {
1123
+            //         font-size: 14px;
1124
+            //         font-weight: bold;
1125
+            //     }
1126
+            // }
698 1127
 
699 1128
             .text {
700 1129
                 width: 60px;
@@ -702,46 +1131,63 @@ export default ({
702 1131
         }
703 1132
     }
704 1133
 
705
-    // 智慧教室实时监控
706
-    .monitor {
707
-        display: flex;
708
-        flex-direction: column;
1134
+    // 智慧教室实时监控     
1135
+    .monitorContent {
709 1136
         width: 100%;
710
-        height: 350px;
711
-        background: rgba(0, 0, 0, 0.2);
712
-        padding: 10px;
713
-        // border: 1px solid fuchsia;
714
-
715
-        p {
716
-            margin-left: 5px;
717
-            margin-top: 5px;
718
-        }
719 1137
 
720
-        .interactclass {
1138
+        // background-color: #000;
1139
+        // padding: 3px;
1140
+        .monitor {
1141
+            box-sizing: border-box;
721 1142
             display: flex;
722
-            justify-content: space-around;
1143
+            flex-direction: column;
1144
+            background: rgba(0, 0, 0, 0.2);
1145
+            padding: 10px;
1146
+            margin: 12px;
1147
+            margin-bottom: 0;
1148
+            // border: 1px solid red;
723 1149
 
724
-            .room {
725
-                position: relative;
726
-                width: 170px;
727
-                height: 94px;
728
-                // background: saddlebrown;
729
-                margin-top: 10px;
1150
+            .interactclass {
1151
+                display: flex;
1152
+                justify-content: space-around;
730 1153
 
731
-                span {
732
-                    position: absolute;
733
-                    top: 0;
734
-                    left: 0;
1154
+                .room {
1155
+                    position: relative;
1156
+                    width: 170px;
1157
+                    height: 94px;
1158
+                    // background: saddlebrown;
1159
+                    margin-top: 10px;
1160
+                    cursor: pointer;
1161
+
1162
+                    &:first-child {
1163
+                        margin-right: 10px;
1164
+                    }
1165
+
1166
+                    span {
1167
+                        position: absolute;
1168
+                        top: 0;
1169
+                        left: 0;
1170
+                    }
735 1171
                 }
736 1172
             }
737 1173
         }
738 1174
 
1175
+        .el-carousel__container {
1176
+            height: 248px;
1177
+        }
1178
+
1179
+        .el-carousel__indicators el-carousel__indicators--horizontal {
1180
+            display: none;
1181
+        }
1182
+
739 1183
         .el-pagination {
740 1184
             justify-content: center;
741 1185
             --el-pagination-bg-color: 'tranparent';
742 1186
             --el-pagination-text-color: '#fff'
1187
+        }
743 1188
 
744
-
1189
+        .el-pagination button {
1190
+            color: #fff;
745 1191
         }
746 1192
 
747 1193
         .el-pagination button:disabled {
@@ -758,4 +1204,33 @@ export default ({
758 1204
         }
759 1205
     }
760 1206
 }
1207
+
1208
+// 点击模型弹出框
1209
+.model-detail {
1210
+    display: flex;
1211
+    justify-content: space-between;
1212
+    flex-wrap: wrap;
1213
+    padding: 10px;
1214
+
1215
+    span {
1216
+        font-size: 16px;
1217
+
1218
+        width: 160px;
1219
+        font-family: Microsoft YaHei UI;
1220
+
1221
+        // border: 1px solid red;
1222
+        margin: 6px;
1223
+        font-weight: 400;
1224
+
1225
+        .title {
1226
+            color: #000;
1227
+            font-weight: 700;
1228
+        }
1229
+    }
1230
+}
1231
+
1232
+:deep(svg) {
1233
+    width: unset;
1234
+    height: unset;
1235
+}
761 1236
 </style>

+ 9 - 1
src/main.js

@@ -1,4 +1,12 @@
1 1
 import { createApp } from 'vue'
2 2
 import App from './App.vue'
3 3
 
4
-createApp(App).mount('#app')
4
+// 全局引入弹出框
5
+import { ElMessage } from 'element-plus'
6
+
7
+const app = createApp(App)
8
+console.log("全局的app", app)
9
+app.config.globalProperties.$message = ElMessage
10
+app.mount("#app")
11
+
12
+// createApp(App).mount('#app')

+ 4 - 0
src/request/api.js

@@ -0,0 +1,4 @@
1
+import { get, post } from './http'
2
+
3
+
4
+export const testApi = p => post(`https://test.lqkj.top/wsngt-server/system/sysuser/bind/ticket?userCode${p}`)

+ 75 - 0
src/request/http.js

@@ -0,0 +1,75 @@
1
+import axios from 'axios'
2
+
3
+
4
+// 请求超时时间
5
+axios.defaults.timeout = 10000
6
+
7
+// 请求拦截器
8
+axios.interceptors.request.use(
9
+    config => {
10
+        // const token = session.get('Token');
11
+        // token && (config.headers.Authorization = token);
12
+        if (config.method.toUpperCase() === "POST") {
13
+            config.headers['Content-Type'] = 'application/json;charset=utf-8';
14
+        }
15
+        return config
16
+    },
17
+    error => {
18
+        return Promise.error(error)
19
+    }
20
+)
21
+// 响应请求拦截器
22
+
23
+axios.interceptors.response.use(
24
+    response => {
25
+        console.log("response", response)
26
+        if (response.status === 200) {
27
+            return Promise.resolve(response)
28
+        } else {
29
+            return Promise.reject(response)
30
+        }
31
+    },
32
+    error => {
33
+        if (error.response.status) {
34
+            switch (error.response.status) {
35
+                case 500:
36
+                    console.error("网络错误,请稍后再试");
37
+                    break;
38
+                case 404:
39
+                    console.eroor("请求路径出错");
40
+                    break;
41
+
42
+                default:
43
+                    console.error(error.response.data.console)
44
+            }
45
+            return Promise.reject(error.response)
46
+        }
47
+    }
48
+)
49
+
50
+
51
+
52
+export function get(url, params) {
53
+    return new Promise((resolve, reject) => {
54
+        axios.get(url, {
55
+            params
56
+        }).then(res => {
57
+            resolve(res)
58
+        }).catch(err => {
59
+            reject(err)
60
+        })
61
+    })
62
+}
63
+
64
+export function post(url, params) {
65
+
66
+    return new Promise((resolve, reject) => {
67
+        axios.post(url, {
68
+            data: params
69
+        }).then(res => {
70
+            resolve(res)
71
+        }).catch(err => {
72
+            reject(err)
73
+        })
74
+    })
75
+}

+ 1 - 0
src/webrtcVideo.js

@@ -826,6 +826,7 @@ function setupWebRtcPlayer(htmlElement, config) {
826 826
         } else if (view[0] === ToClientMessageType.Response) {
827 827
             let response = new TextDecoder("utf-16").decode(data.slice(1));
828 828
             for (let listener of responseEventListeners.values()) {
829
+
829 830
                 listener(response);
830 831
             }
831 832
         } else if (view[0] === ToClientMessageType.Command) {