huliu преди 1 година
родител
ревизия
c1bf2277c6

+ 10 - 0
package-lock.json

@@ -254,6 +254,11 @@
254 254
       "integrity": "sha512-F0SAmZ8iUtS//m8DmCTA0jlh6TDKkHQyK6xc6V4KDTyZKA9dnvX9/3sRTVQrWm79glUAZbnmmNcdYwUIHWVybw==",
255 255
       "dev": true
256 256
     },
257
+    "amfe-flexible": {
258
+      "version": "2.2.1",
259
+      "resolved": "https://registry.npmmirror.com/amfe-flexible/-/amfe-flexible-2.2.1.tgz",
260
+      "integrity": "sha512-L2VfvDzoETBjhRptg5u/IUuzHSuxm22JpSRb404p/TBGeRfwWmmNEbB+TFPIP/sS/+pbM18bCFH9QnMojLuPNw=="
261
+    },
257 262
     "anymatch": {
258 263
       "version": "3.1.3",
259 264
       "resolved": "https://registry.npmmirror.com/anymatch/-/anymatch-3.1.3.tgz",
@@ -860,6 +865,11 @@
860 865
         "source-map-js": "^1.0.2"
861 866
       }
862 867
     },
868
+    "postcss-pxtorem": {
869
+      "version": "6.0.0",
870
+      "resolved": "https://registry.npmmirror.com/postcss-pxtorem/-/postcss-pxtorem-6.0.0.tgz",
871
+      "integrity": "sha512-ZRXrD7MLLjLk2RNGV6UA4f5Y7gy+a/j1EqjAfp9NdcNYVjUMvg5HTYduTjSkKBkRkfqbg/iKrjMO70V4g1LZeg=="
872
+    },
863 873
     "proxy-from-env": {
864 874
       "version": "1.1.0",
865 875
       "resolved": "https://registry.npmmirror.com/proxy-from-env/-/proxy-from-env-1.1.0.tgz",

+ 2 - 0
package.json

@@ -8,9 +8,11 @@
8 8
     "preview": "vite preview"
9 9
   },
10 10
   "dependencies": {
11
+    "amfe-flexible": "^2.2.1",
11 12
     "axios": "^1.4.0",
12 13
     "echarts": "^5.4.3",
13 14
     "element-plus": "^2.3.9",
15
+    "postcss-pxtorem": "^6.0.0",
14 16
     "vue": "^3.2.25",
15 17
     "vue3-video-play": "^1.3.1-beta.6"
16 18
   },

+ 48 - 0
src/assets/css/home.scss

@@ -0,0 +1,48 @@
1
+.container {
2
+    margin: 0;
3
+    padding: 0;
4
+    overflow: hidden;
5
+
6
+    .header {
7
+        position: absolute;
8
+        z-index: 555;
9
+        margin-bottom: 20px;
10
+
11
+        width: 100%;
12
+        top: 0;
13
+
14
+        img {
15
+            width: 100%;
16
+            height: 100%;
17
+        }
18
+    }
19
+}
20
+
21
+.el-container {
22
+    height: 938px;
23
+    overflow: hidden;
24
+}
25
+
26
+// 点击模型弹出框
27
+.model-detail {
28
+    display: flex;
29
+    justify-content: space-between;
30
+    flex-wrap: wrap;
31
+    padding: 10px;
32
+
33
+    span {
34
+        font-size: 16px;
35
+
36
+        width: 160px;
37
+        font-family: Microsoft YaHei UI;
38
+
39
+        // border: 1px solid red;
40
+        margin: 6px;
41
+        font-weight: 400;
42
+
43
+        .title {
44
+            color: #000;
45
+            font-weight: 700;
46
+        }
47
+    }
48
+}

+ 222 - 0
src/assets/css/left.scss

@@ -0,0 +1,222 @@
1
+ .left {
2
+     position: absolute;
3
+     width: 412px;
4
+     height: 900px;
5
+     top: 111px;
6
+     z-index: 300;
7
+     flex-shrink: 0;
8
+     border-radius: 2px;
9
+     background: rgba(125, 125, 125, 0.17);
10
+     backdrop-filter: blur(30.5px);
11
+     color: #FFF;
12
+     font-size: 14px;
13
+     overflow: hidden;
14
+     left: 30px;
15
+     display: flex;
16
+     flex-direction: column;
17
+     padding: 11px 10px 15px 10px; //教室分类统计
18
+
19
+     .left_top {
20
+         box-sizing: border-box;
21
+         width: 392px;
22
+         height: 261px;
23
+         //  border: 2px solid darkcyan;
24
+
25
+         .list {
26
+             display: flex;
27
+             flex-direction: column;
28
+             justify-content: center;
29
+             width: 180px;
30
+             margin-left: 23.2px;
31
+             margin-right: 28px;
32
+
33
+             .item {
34
+                 width: 100%;
35
+                 display: flex;
36
+                 align-items: center;
37
+                 padding-top: 16px;
38
+                 padding-bottom: 11px;
39
+                 border-bottom: 1px dotted rgba(255, 255, 255, 0.33);
40
+                 font-size: 12px;
41
+                 font-family: Microsoft YaHei;
42
+
43
+                 i {
44
+                     display: inline-block;
45
+                     width: 8px;
46
+                     height: 8px;
47
+                     border-radius: 4px;
48
+                     margin-right: 6px;
49
+                 }
50
+
51
+                 .descr {
52
+                     display: flex;
53
+                     justify-content: space-between;
54
+                     width: 100%;
55
+                 }
56
+
57
+             }
58
+         }
59
+     }
60
+
61
+     // 本周课程统计
62
+     .left-center {
63
+         // border: 1px solid sienna;
64
+         box-sizing: border-box;
65
+         height: 219px;
66
+         width: 392px;
67
+         height: 219px;
68
+         //  border: 2px solid #5F7BDC;
69
+
70
+         .count {
71
+             display: flex;
72
+             flex-direction: column;
73
+             align-items: center;
74
+             justify-content: space-around;
75
+             height: 108px;
76
+             // border: 1px solid sandybrown;
77
+             width: 93px;
78
+             box-sizing: border-box;
79
+             // margin-top: 42px;
80
+             margin: 40px 16px 39px 22px;
81
+
82
+             img {
83
+                 width: 22px;
84
+                 height: 22px;
85
+             }
86
+
87
+             .text {
88
+                 font-size: 12px;
89
+                 line-height: 20px;
90
+             }
91
+
92
+             // width: 70px;
93
+
94
+             // text-align: center;
95
+             // justify-content: space-around;
96
+             // margin: 24px 0;
97
+             // padding: 20px;
98
+
99
+             // .el-progress-circle {
100
+             //     width: 56px !important;
101
+             //     height: 56px !important;
102
+
103
+             //     .el-progress-circle__track {
104
+             //         stroke: rgba(255, 255, 255, 0.2);
105
+             //     }
106
+
107
+             //     .el-progress__text {
108
+             //         margin-top: 5px;
109
+             //     }
110
+             // }
111
+
112
+             // .number {
113
+             //     margin-top: 10px;
114
+             //     margin-bottom: 5px;
115
+             //     font-size: 12px;
116
+
117
+             //     span {
118
+             //         font-size: 14px;
119
+             //         font-weight: bold;
120
+             //     }
121
+             // }
122
+
123
+             // .text {
124
+             //     font-size: 12px;
125
+             //     line-height: 20px;
126
+             //     width: 60px;
127
+             // }
128
+         }
129
+     }
130
+
131
+     // 智慧教室使用明细
132
+     .left-bottom {
133
+         flex: 1;
134
+         overflow: hidden;
135
+         box-sizing: border-box;
136
+         width: 392px;
137
+         //  height: 219px;
138
+         //  border: 2px solid #5F7BDC;
139
+
140
+         .content {
141
+             box-sizing: border-box;
142
+             padding: 10px;
143
+             padding-top: 0;
144
+             height: 100%;
145
+             overflow: hidden;
146
+         }
147
+
148
+         .title {
149
+             margin-bottom: 20px;
150
+         }
151
+
152
+         .table {
153
+             width: 100%;
154
+
155
+             .th {
156
+                 font-weight: 700;
157
+                 display: flex;
158
+                 width: 100%;
159
+
160
+                 .long {
161
+                     width: 500px;
162
+                 }
163
+             }
164
+
165
+             ul {
166
+                 display: flex;
167
+                 list-style: none;
168
+                 height: 40px;
169
+                 line-height: 40px;
170
+                 font-size: 12px;
171
+                 border-bottom: 1px dotted #fff;
172
+                 padding: 0;
173
+
174
+                 &:last-child {
175
+                     margin-bottom: 10px;
176
+                 }
177
+
178
+                 li {
179
+                     display: flex;
180
+                     width: 100%;
181
+                     align-items: center;
182
+                     justify-content: space-around
183
+                 }
184
+             }
185
+
186
+             .el-button {
187
+                 background-color: #5F7BDC;
188
+                 color: #fff;
189
+                 border: none;
190
+             }
191
+         }
192
+     }
193
+
194
+ }
195
+
196
+ // 通用标题
197
+ .title {
198
+     width: 380px;
199
+     height: 32px;
200
+     flex-shrink: 0;
201
+     background: linear-gradient(90deg, rgba(255, 255, 255, 0.17) 2.49%, rgba(255, 255, 255, 0.00) 98.69%);
202
+     ;
203
+     line-height: 32px;
204
+     margin: 11px 22px 0 10px;
205
+
206
+     .text {
207
+         color: #FFF;
208
+         font-family: PingFang SC;
209
+         font-size: 14px;
210
+         padding-left: 18px;
211
+         letter-spacing: 1px;
212
+     }
213
+ }
214
+
215
+ // 通用content
216
+ .content {
217
+     display: flex;
218
+     width: 100%;
219
+     overflow: hidden;
220
+     justify-content: space-evenly;
221
+
222
+ }

+ 269 - 0
src/assets/css/right.scss

@@ -0,0 +1,269 @@
1
+// 通用标题
2
+.title {
3
+    width: 380px;
4
+    height: 32px;
5
+    flex-shrink: 0;
6
+    background: linear-gradient(90deg, rgba(255, 255, 255, 0.17) 2.49%, rgba(255, 255, 255, 0.00) 98.69%);
7
+    ;
8
+    line-height: 32px;
9
+    margin: 11px 22px 0 10px;
10
+
11
+    .text {
12
+        color: #FFF;
13
+        font-family: PingFang SC;
14
+        padding-left: 18px;
15
+        letter-spacing: 1px;
16
+    }
17
+}
18
+
19
+// 通用content
20
+.content {
21
+    display: flex;
22
+    width: 100%;
23
+    overflow: hidden;
24
+    justify-content: space-evenly;
25
+
26
+}
27
+
28
+.right {
29
+    position: absolute;
30
+    width: 412px;
31
+    height: 900px;
32
+    top: 111px;
33
+    z-index: 300;
34
+    flex-shrink: 0;
35
+    border-radius: 2px;
36
+    background: rgba(125, 125, 125, 0.17);
37
+    backdrop-filter: blur(30.5px);
38
+    color: #FFF;
39
+    font-size: 14px;
40
+    // overflow: hidden;
41
+    right: 20px;
42
+    // border: 2px solid salmon;
43
+    box-sizing: border-box;
44
+    display: flex;
45
+    flex-direction: column;
46
+    height: 900px;
47
+    overflow: hidden;
48
+    padding: 11px 10px 15px 10px;
49
+
50
+}
51
+
52
+// 物联设备统计
53
+.right_top {
54
+    // border: 3px solid darkcyan;
55
+    height: 267px;
56
+    // border: 2px solid darkcyan;
57
+    box-sizing: border-box;
58
+
59
+    .logo {
60
+        // border: 2px solid firebrick;
61
+        display: flex;
62
+        flex-direction: column;
63
+        align-items: center;
64
+        width: 134px;
65
+        height: 175px;
66
+        margin: 30px 0 20px 10px;
67
+
68
+        img {
69
+            width: 87px;
70
+            height: 88px;
71
+        }
72
+
73
+        .animate {
74
+            width: 87px;
75
+            height: 88px;
76
+            margin-bottom: -55px;
77
+            animation: myfirst 1.5s infinite;
78
+        }
79
+
80
+        span {
81
+            color: rgba(255, 255, 255, 0.76);
82
+            font-family: PingFang SC;
83
+            font-weight: 400;
84
+            line-height: 21px;
85
+
86
+            &:first-child {
87
+                color: #fff;
88
+                font-size: 20px;
89
+                font-weight: 900;
90
+                line-height: 30px;
91
+            }
92
+
93
+            b {
94
+                color: #fff;
95
+                font-size: 16px;
96
+                font-weight: 400;
97
+            }
98
+        }
99
+
100
+    }
101
+
102
+    .list {
103
+        // border: 2px solid darkcyan;
104
+        display: flex;
105
+        flex-direction: column;
106
+        align-items: center;
107
+        justify-content: space-evenly;
108
+        padding-top: 20px;
109
+
110
+        .item_r {
111
+            display: flex;
112
+            align-items: center;
113
+            width: 100%;
114
+            margin-bottom: 10px;
115
+
116
+            // border: 1px solid forestgreen;
117
+            img {
118
+                width: 39px;
119
+                height: 39px;
120
+                flex-shrink: 0;
121
+                // border: 1px solid darkgoldenrod;
122
+                margin-right: 12px;
123
+            }
124
+
125
+            .percent {
126
+                width: 132px;
127
+                border-bottom: 1px solid rgba(255, 255, 255, 0.47);
128
+                // margin: 10px;
129
+                padding-bottom: 7px;
130
+                font-size: 16px;
131
+
132
+                span {
133
+                    font-size: 12px;
134
+                    color: #fff;
135
+                    opacity: .76;
136
+                }
137
+            }
138
+        }
139
+    }
140
+}
141
+
142
+// 物联网设备统计动画
143
+@keyframes myfirst {
144
+    0% {
145
+        transform: translate(0, 0);
146
+    }
147
+
148
+    50% {
149
+        transform: translate(0, -8px);
150
+    }
151
+
152
+    100% {
153
+        transform: translate(0, 0);
154
+    }
155
+}
156
+
157
+// 物联设备类型统计
158
+.right_center {
159
+    height: 314px;
160
+    // border: 3px solid darkcyan;
161
+    box-sizing: border-box;
162
+    overflow: hidden;
163
+
164
+    .contentwrap {
165
+        display: flex;
166
+        flex-wrap: wrap;
167
+        margin: 10px 0;
168
+        height: 222px;
169
+        overflow: hidden;
170
+        // border: 2px solid sandybrown;
171
+
172
+        .count {
173
+            display: flex;
174
+            flex-direction: column;
175
+            align-items: center;
176
+            padding: 6px;
177
+            text-align: center;
178
+            // border: 2px solid darkkhaki;
179
+            width: 110px;
180
+        }
181
+    }
182
+}
183
+
184
+
185
+// 智慧教室实时监控     
186
+.right_bottom {
187
+    flex: 1;
188
+    box-sizing: border-box;
189
+    // border: 3px solid darkcyan;
190
+    // height: 332px;
191
+
192
+    .monitorContent {
193
+
194
+        .monitor {
195
+            box-sizing: border-box;
196
+            display: flex;
197
+            flex-direction: column;
198
+            background: rgba(0, 0, 0, 0.2);
199
+            padding: 10px;
200
+            width: 364px;
201
+            height: 268px;
202
+            margin: 0 auto;
203
+            margin-top: 10px;
204
+            // padding-bottom: 0;
205
+
206
+            // border: 1px solid red;
207
+
208
+            .interactclass {
209
+                display: flex;
210
+                justify-content: space-around;
211
+
212
+                .room {
213
+                    position: relative;
214
+                    width: 170px;
215
+                    height: 94px;
216
+                    // background: saddlebrown;
217
+                    margin-top: 10px;
218
+                    cursor: pointer;
219
+
220
+                    &:first-child {
221
+                        margin-right: 10px;
222
+                    }
223
+
224
+                    span {
225
+                        position: absolute;
226
+                        top: 0;
227
+                        left: 0;
228
+                    }
229
+                }
230
+            }
231
+        }
232
+
233
+        .el-carousel__indicators--horizontal {
234
+            display: none;
235
+            bottom: 6%;
236
+        }
237
+
238
+        // .el-carousel__container {
239
+        //     height: 248px;
240
+        // }
241
+
242
+        // .el-carousel__indicators .el-carousel__indicators--horizontal {
243
+        //     display: none;
244
+        // }
245
+
246
+        .el-pagination {
247
+            justify-content: center;
248
+            --el-pagination-bg-color: 'tranparent';
249
+            --el-pagination-text-color: '#fff'
250
+        }
251
+
252
+        .el-pagination button {
253
+            color: #fff;
254
+        }
255
+
256
+        .el-pagination button:disabled {
257
+            background: none;
258
+            color: '#fff';
259
+        }
260
+
261
+        .el-pagination button:hover {
262
+            color: '#fff';
263
+        }
264
+
265
+        .el-pager li {
266
+            color: #fff;
267
+        }
268
+    }
269
+}

BIN
src/assets/img/head.png


src/assets/img/头部.png → src/assets/img/head1.png


+ 0 - 0
src/assets/ue.css


+ 16 - 8
src/components/CircleProgress.vue

@@ -16,7 +16,9 @@
16 16
         </div>
17 17
     </div>
18 18
     <div class="bottomNumber" v-if="textPosition == 'bottom'">
19
-        {{ showProgress }} <span> (节)</span>
19
+        {{ showProgress }} <span v-if="showProgress
20
+            > 1000">(人)</span>
21
+        <span v-else>(节)</span>
20 22
     </div>
21 23
 </template>
22 24
 <script setup lang="ts">
@@ -34,11 +36,11 @@ const props = defineProps({
34 36
     },
35 37
     width: {
36 38
         type: String,
37
-        default: '60px',
39
+        default: '54.6px',
38 40
     },
39 41
     height: {
40 42
         type: String,
41
-        default: '60px',
43
+        default: '54.6px',
42 44
     },
43 45
     textPosition: {
44 46
         type: String,
@@ -48,7 +50,7 @@ const props = defineProps({
48 50
 });
49 51
 const { height, width, color, targetValue, textPosition } = toRefs(props);
50 52
 
51
-const showProgress = ref<number>(0);
53
+let showProgress = ref<number>(0);
52 54
 const addData = () => {
53 55
     if (targetValue.value === 0) return;
54 56
     let timer = setInterval(() => {
@@ -56,16 +58,22 @@ const addData = () => {
56 58
             clearInterval(timer), timer == null;
57 59
             return;
58 60
         }
61
+        // showProgress.value = showProgress.value + 200
62
+        // console.log(showProgress.value)
63
+        // console.log("targetValue.value", targetValue.value)
64
+        // showProgress.value = showProgress.value + 6
59 65
         ++showProgress.value;
60
-    }, 15);
66
+
67
+
68
+    }, 10);
61 69
 };
62 70
 onMounted(() => {
63 71
     addData();
64 72
 });
65
-const progressValue = ref<number>((showProgress.value / 360) * 250);
66
-
73
+// const progressValue = ref<number>((showProgress.value / 360) * 250);
74
+const progressValue = ref(0)
67 75
 watch(showProgress, (newValue: number) => {
68
-    progressValue.value = (newValue / 360) * 250;
76
+    progressValue.value = 360;
69 77
 });
70 78
 </script>
71 79
 <style lang="scss" scoped>

+ 2 - 4
src/components/UeVideo.vue

@@ -2,15 +2,13 @@
2 2
 <template>
3 3
   <div ref="video" id="player"></div>
4 4
   <home></home>
5
-  <div @click="toUE" style="position: absolute;top: 5%; left: 50%;background-color: darkcyan;z-index: 200">向UE发信息</div>
5
+  <!-- <div @click="toUE" style="position: absolute;top: 5%; left: 50%;background-color: darkcyan;z-index: 200">向UE发信息</div> -->
6 6
 </template>
7 7
 
8 8
 <script>
9 9
 import { onMounted, ref } from "vue";
10 10
 import {
11
-  initLoad,
12
-  callUIInteraction,
13
-  addResponseEventListener,
11
+  initLoad
14 12
 } from "../webrtcVideo.js";
15 13
 import { ElButton, ElInput } from "element-plus";
16 14
 import Home from './home.vue'

Файловите разлики са ограничени, защото са твърде много
+ 29 - 944
src/components/home.vue


+ 327 - 0
src/components/left.vue

@@ -0,0 +1,327 @@
1
+<template>
2
+    <el-aside class="left">
3
+
4
+        <div class="left_top">
5
+            <div class="title">
6
+                <div class="text">教室分类统计</div>
7
+            </div>
8
+            <div class="content">
9
+                <div id="myChart" style="width:140px; height: 140px;margin:48px 0 30px 18px">
10
+                </div>
11
+                <div class="list">
12
+                    <div class="item" v-for="(item, index) in classRoomCount">
13
+                        <i :style="{ backgroundColor: item.color }"></i>
14
+                        <div class="descr">
15
+                            <span>{{ item.name }}</span>
16
+                            <span>{{ item.value }}</span>
17
+                        </div>
18
+                    </div>
19
+                </div>
20
+            </div>
21
+        </div>
22
+        <div class="left-center">
23
+            <div class="title">
24
+                <div class="text">本周课程统计</div>
25
+            </div>
26
+            <div class="content">
27
+                <div class="count" v-for="(item, index) in classCount" :key="index">
28
+                    <CircleProgress :target-value="item.number" :color="item.color" text-position="bottom">
29
+                        <img :src="'src/assets/img/' + item.imgnumber + '.png'" alt="">
30
+                    </CircleProgress>
31
+                    <div class="text">{{ item.text }}</div>
32
+                </div>
33
+            </div>
34
+        </div>
35
+        <div class="left-bottom">
36
+
37
+            <div class="title">
38
+                <div class="text">智慧教室使用明细</div>
39
+            </div>
40
+
41
+            <div class="content" style="height: 100%;overflow: hidden;">
42
+                <div class="table">
43
+                    <ul class="th">
44
+                        <li>教室</li>
45
+                        <li>使用状态</li>
46
+                        <li class="long">实到/应到人数</li>
47
+                        <li>到课率</li>
48
+                        <li>操作</li>
49
+                    </ul>
50
+                    <div ref="testMain" style="height: 310px;overflow: hidden;">
51
+                        <ul v-for="item in classRoom" @mouseenter="testMove" @mouseleave="testMend">
52
+                            <li>
53
+                                {{ item.class }}
54
+                            </li>
55
+                            <li>{{ item.status }}</li>
56
+                            <li>{{ item.people }}</li>
57
+                            <li>{{ item.percentage }}</li>
58
+                            <li>
59
+                                <el-button size="small" class="look" @click.prevent="handleRoom">查看</el-button>
60
+                            </li>
61
+
62
+                        </ul>
63
+
64
+                    </div>
65
+
66
+                </div>
67
+            </div>
68
+        </div>
69
+
70
+    </el-aside>
71
+</template>
72
+
73
+<script >
74
+import * as echarts from "echarts";
75
+import { reactive, onMounted, ref, toRefs, onBeforeUnmount, onUnmounted, nextTick, getCurrentInstance } from 'vue';
76
+import { ElScrollbar, ElPagination, ElDialog, ElCarousel, ElCarouselItem, ElMessage } from "element-plus";
77
+import "vue3-video-play/dist/style.css";
78
+import {
79
+    callUIInteraction,
80
+    addResponseEventListener
81
+} from "../webrtcVideo.js";
82
+import { getClass } from '../request/api'
83
+import { videoPlay } from "vue3-video-play";
84
+import CircleProgress from './CircleProgress.vue';
85
+
86
+
87
+export default ({
88
+
89
+    name: 'Histogram',
90
+    components: { ElScrollbar, ElPagination, ElDialog, videoPlay, ElCarousel, ElCarouselItem, CircleProgress },
91
+    setup() {
92
+
93
+        const pieData = reactive([{
94
+            value: 1048,
95
+            name: '基础型'
96
+        },
97
+        {
98
+            value: 735,
99
+            name: '扩展型'
100
+        },
101
+        {
102
+            value: 580,
103
+            name: '其他'
104
+        }])
105
+        const classPie = reactive({
106
+            option: {
107
+                color: ['#4ED2E4', '#5E91F6', '#67F0A8'],
108
+                tooltip: {
109
+                    trigger: 'item',
110
+                    position: 'right'
111
+                },
112
+
113
+                series: [{
114
+                    type: 'pie',
115
+                    radius: ['60%', '90%'],
116
+                    avoidLabelOverlap: false,
117
+                    label: {
118
+                        show: true,
119
+                        position: 'center',
120
+                        color: '#fff',
121
+                        formatter: () => { // 格式化要展示的文本
122
+                            return `总计\n\n1430间`
123
+                        },
124
+                        // formatter: '{total|' + 200 + '}' + '\n\r' + '{active|共发布活动}',
125
+                        // 等着获取了数据在给样式
126
+                        rich: {
127
+                            total: {
128
+                                fontSize: 35,
129
+                                fontFamily: "微软雅黑",
130
+                                color: '#454c5c'
131
+                            },
132
+                            active: {
133
+                                fontFamily: "微软雅黑",
134
+                                fontSize: 16,
135
+                                color: '#6c7a89',
136
+                                lineHeight: 30,
137
+                            },
138
+                        }
139
+
140
+
141
+                    },
142
+                    labelLine: {
143
+                        show: false
144
+                    },
145
+                    data: pieData
146
+
147
+                }]
148
+            },
149
+        })
150
+        const initeCharts = () => {
151
+            let myChart = echarts.init(document.getElementById('myChart'))
152
+            // 绘制图表
153
+            console.log("绘制图表", classPie.option)
154
+            myChart.setOption(classPie.option)
155
+        }
156
+
157
+
158
+
159
+
160
+
161
+        // 物联设备类型统计
162
+        const interDevice = ref([
163
+            {
164
+                number: 98,
165
+                color: "#63ABFF",
166
+                icon: 'r1',
167
+                text: "显示系统"
168
+            },
169
+            {
170
+                number: 185,
171
+                color: "#63FFC7",
172
+                icon: 'r2',
173
+                text: "控制设备"
174
+            },
175
+            {
176
+                number: 58,
177
+                color: "#918EFF",
178
+                icon: 'r3',
179
+                text: "音频设备"
180
+            },
181
+            {
182
+                number: 68,
183
+                color: "#00C8E3",
184
+                icon: 'r4',
185
+                text: "录播设备"
186
+            },
187
+            {
188
+                number: 189,
189
+                color: "#FFBB54",
190
+                icon: 'r5',
191
+                text: "环境设备"
192
+            },
193
+            {
194
+                number: 98,
195
+                color: "#00CF78",
196
+                icon: 'r6',
197
+                text: "安全管理"
198
+            },
199
+
200
+        ])
201
+
202
+        const handleRoom = function () {
203
+            // 发送消息给UE
204
+            // console.log("查看当前教室使用情况", 'adsfasdfasdfa', ElMessage)
205
+            // ElMessage('this is a message.')
206
+
207
+            // ElMessage.success("xsxxx")
208
+            // callUIInteraction("open");
209
+            // addResponseEventListener("open", (data) => {
210
+            //     console.log("接收到的信息", data, typeof data,)
211
+            // })
212
+            // centerDialogVisible.value = true
213
+        }
214
+
215
+
216
+
217
+        // 教室分类统计
218
+        const classRoomCount = ref([
219
+            {
220
+                name: "基础性",
221
+                color: '#4ED2E4',
222
+                number: '40%'
223
+            },
224
+            {
225
+                name: "扩展型",
226
+                color: '#5E91F6',
227
+                number: '25%'
228
+            },
229
+            {
230
+                name: "其他",
231
+                color: '#67F0A8',
232
+                number: '45%'
233
+            }
234
+        ])
235
+        // 本周课程统计数据
236
+        const classCount = ref([])
237
+        // 智慧教室使用数据
238
+        const classRoom = ref([])
239
+
240
+
241
+        const getClassData = async () => {
242
+            let res = await getClass()
243
+            console.log("res----教室分类统计", res)
244
+            let { course, classroomDetail, category } = res.data;
245
+            classRoomCount.value = category
246
+            classCount.value = course
247
+            classRoom.value = classroomDetail
248
+
249
+        }
250
+
251
+        let timer = ref(null)
252
+        let testMain = ref(null)
253
+        onMounted(() => {
254
+            initeCharts()
255
+            getClassData()
256
+            start()
257
+        })
258
+
259
+
260
+        onBeforeUnmount(() => {
261
+            clearTimeout(timer.value)
262
+        })
263
+        onUnmounted(() => {
264
+            clearTimeout(timer.value)
265
+        })
266
+
267
+
268
+        function testMove() {
269
+            clearTimeout(timer.value)
270
+        }
271
+
272
+        function testMend() {
273
+            start()
274
+        }
275
+        //开启定时器方法
276
+        function start() {
277
+            //清除定时器
278
+            clearTimeout(timer.value)
279
+            //定时器触发周期
280
+            let speed = ref(75)
281
+            timer.value = setInterval(MarqueeTest, speed.value)
282
+        }
283
+        function MarqueeTest() {
284
+            let test1 = testMain.value
285
+
286
+            //判断组件是否渲染完成
287
+            if (test1.offsetHeight == 0) {
288
+                test1 = testMain.value
289
+            } else {
290
+                //如果列表数量过少不进行滚动
291
+                // console.log("test1", test1.scrollTop)
292
+                if (test1.childNodes.length < 6) {
293
+                    clearTimeout(timer.value)
294
+                    return;
295
+                }
296
+                //组件进行滚动
297
+                test1.scrollTop += 1
298
+
299
+                //判断滚动条是否滚动到底部
300
+                if (test1.scrollTop == (test1.scrollHeight - test1.clientHeight)) {
301
+                    //获取组件第一个节点
302
+                    let a = test1.childNodes[0]
303
+                    //删除节点
304
+                    test1.removeChild(a)
305
+                    //将该节点拼接到组件最后
306
+                    test1.append(a)
307
+                }
308
+            }
309
+        }
310
+        return {
311
+            classCount,
312
+            classRoom,
313
+            interDevice,
314
+            handleRoom,
315
+            testMend,
316
+            testMove,
317
+            testMain,
318
+            classRoomCount
319
+        }
320
+    },
321
+})
322
+
323
+</script>
324
+
325
+<style scoped lang="scss">
326
+@import '../assets/css/left.scss';
327
+</style>

+ 283 - 0
src/components/right.vue

@@ -0,0 +1,283 @@
1
+<template>
2
+    <el-aside class="right">
3
+        <div class="right_top">
4
+            <div class="title">
5
+                <div class="text">物联设备统计</div>
6
+            </div>
7
+            <div class="content">
8
+                <div class="logo">
9
+                    <span>{{ equipmentTotal.total }}</span>
10
+                    <img class="animate" src="../assets/img/d1_1.png" alt="">
11
+                    <img src="../assets/img/d1_2.png" alt="">
12
+                    <span><b>总数</b>(个)</span>
13
+                </div>
14
+                <div class="list">
15
+                    <div class="item_r">
16
+                        <img src="../assets/img/d2.png" alt="">
17
+                        <div class="percent">
18
+                            在线&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;{{ equipmentTotal.online }}
19
+                            <span>(个)</span>
20
+                        </div>
21
+                    </div>
22
+                    <div class="item_r">
23
+                        <img src="../assets/img/d3.png" alt="">
24
+                        <div class="percent">
25
+                            离线&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp;&nbsp; {{ equipmentTotal.underline }}
26
+                            <span>(个)</span>
27
+                        </div>
28
+                    </div>
29
+                </div>
30
+            </div>
31
+        </div>
32
+
33
+        <div class="right_center">
34
+            <div class="title">
35
+                <div class="text">物联设备类型统计</div>
36
+            </div>
37
+            <div class="contentwrap">
38
+                <div class="count" v-for="(item, index) in interDevice" :key="index">
39
+                    <CircleProgress :target-value="item.number" :color="item.color" text-position="top">
40
+                        <img :src="'src/assets/img/' + item.icon + '.png'" alt="">
41
+                    </CircleProgress>
42
+                    <div class="text">{{ item.text }}</div>
43
+                </div>
44
+            </div>
45
+        </div>
46
+
47
+        <div class="right_bottom">
48
+            <div class="title">
49
+                <div class="text">智慧教室实时监控</div>
50
+            </div>
51
+
52
+            <div class="monitorContent">
53
+                <el-carousel class="monitor">
54
+                    <el-carousel-item v-for="(item, index) in carouselData" :key="index">
55
+                        <div class="interactclass">
56
+                            <div class="room" v-for="(item, index) in item.activeMonitor.array" :key="index">
57
+                                <span>{{ item.title }}</span>
58
+                                <img :src="'src/assets/img/' + 'rb2' + '.png'" alt="" @click="lookVideo(item.src)">
59
+                            </div>
60
+                        </div>
61
+
62
+                        <p>{{ item.activeMonitor.title }}</p>
63
+                        <div class="interactclass">
64
+                            <div class="room" v-for="(item, index) in item.activeMonitor.array" :key="index">
65
+                                <span>{{ item.title }}</span>
66
+                                <img :src="'src/assets/img/' + 'rb2' + '.png'" alt="" @click="lookVideo(item.src)">
67
+                            </div>
68
+                        </div>
69
+                    </el-carousel-item>
70
+                </el-carousel>
71
+            </div>
72
+
73
+        </div>
74
+    </el-aside>
75
+
76
+    <!-- 视频播放弹框 -->
77
+    <el-dialog class="videoDialog" v-model="videoDialog" title="互动教师1" width="60%" align="center" top="4%">
78
+        <videoPlay v-bind="videoData" @play="onPlay" />
79
+    </el-dialog>
80
+</template>
81
+
82
+<script >
83
+import { reactive, onMounted, ref, } from 'vue';
84
+import { ElCarousel, ElCarouselItem } from "element-plus";
85
+import "vue3-video-play/dist/style.css";
86
+import { videoPlay } from "vue3-video-play";
87
+import CircleProgress from './CircleProgress.vue';
88
+import { getDevice } from '../request/api';
89
+
90
+export default ({
91
+    name: 'Histogram',
92
+    components: { videoPlay, ElCarousel, ElCarouselItem, CircleProgress },
93
+    setup() {
94
+        // 视频数据
95
+        const videoData = reactive({
96
+            width: "100%", //播放器高度
97
+            height: "100%", //播放器高度
98
+            color: "red", //主题色
99
+            title: "互动教室", //视频名称
100
+            src: "https://cdn.jsdelivr.net/gh/xdlumia/files/video-play/IronMan.mp4", //视频源
101
+            muted: false, //静音
102
+            webFullScreen: false,
103
+            // speedRate: ["0.75", "1.0", "1.25", "1.5", "2.0"], //播放倍速
104
+            autoPlay: false, //自动播放
105
+            loop: false, //循环播放
106
+            mirror: false, //镜像画面
107
+            ligthOff: false, //关灯模式
108
+            volume: 0.3, //默认音量大小
109
+            control: true, //是否显示控制
110
+            currentTime: 0,//跳转到固定播放时间
111
+            controlBtns: [
112
+                "audioTrack",
113
+                "quality",
114
+                "speedRate",
115
+                "volume",
116
+                "setting",
117
+                "pip",
118
+                "pageFullScreen",
119
+                "fullScreen",
120
+            ], //显示所有按钮,
121
+        });
122
+        //视频播放弹框
123
+        const videoDialog = ref(false)
124
+        // 播放视频
125
+        const lookVideo = function (src) {
126
+            videoDialog.value = true;
127
+            // console.log("点击了播放视频", 12312312, "播放路径", src)
128
+            videoData.src = src
129
+
130
+        }
131
+        // 播放视频
132
+        const onPlay = function () {
133
+            console.log("播放视频")
134
+        }
135
+        // 物联设备
136
+        let equipmentTotal = ref({})
137
+        //智慧教室实时监控---数据
138
+        // const carouselData = ref([
139
+        //     {
140
+        //         activeMonitor: {
141
+        //             title: "多屏互动教室",
142
+        //             array: [
143
+        //                 {
144
+        //                     title: "互动教室22",
145
+        //                     poster: 'rb1',
146
+        //                     src: '',
147
+        //                 },
148
+        //                 {
149
+        //                     title: "互动教室22",
150
+        //                     poster: 'rb2',
151
+        //                     src: '',
152
+        //                 }
153
+        //             ]
154
+        //         },
155
+        //         wisdomMonitor:
156
+        //         {
157
+        //             title: "多屏智慧教室",
158
+        //             array: [
159
+        //                 {
160
+        //                     title: "智慧教室11",
161
+        //                     poster: 'rb1',
162
+        //                     src: '',
163
+        //                 },
164
+        //                 {
165
+        //                     title: "智慧教室11",
166
+        //                     poster: 'rb2',
167
+        //                     src: '',
168
+        //                 }
169
+        //             ]
170
+        //         }
171
+        //     },
172
+        //     {
173
+        //         activeMonitor: {
174
+        //             title: "多屏互动教室",
175
+        //             array: [
176
+        //                 {
177
+        //                     title: "互动教室33",
178
+        //                     poster: 'rb1',
179
+        //                     src: '',
180
+        //                 },
181
+        //                 {
182
+        //                     title: "互动教室33",
183
+        //                     poster: 'rb2',
184
+        //                     src: '',
185
+        //                 }
186
+        //             ]
187
+        //         },
188
+
189
+        //         wisdomMonitor:
190
+        //         {
191
+        //             title: "多屏智慧教室",
192
+        //             array: [
193
+        //                 {
194
+        //                     title: "智慧教室22",
195
+        //                     poster: 'rb1',
196
+        //                     src: '',
197
+        //                 },
198
+        //                 {
199
+        //                     title: "智慧教室22",
200
+        //                     poster: 'rb2',
201
+        //                     src: '',
202
+        //                 }
203
+        //             ]
204
+        //         }
205
+        //     }
206
+
207
+        // ]);
208
+        // 物联设备类型统计
209
+        let interDevice = ref([])
210
+        // 智慧教室
211
+        let carouselData = ref([])
212
+
213
+        //获取页面数据
214
+        const getDeviceData = async () => {
215
+            let res = await getDevice()
216
+            if (res.code !== 200) {
217
+                ElMessage.error("数据请求出错");
218
+            }
219
+            let { category, monitor, status } = res.data
220
+            // 物联设备类型数据
221
+            interDevice.value = category;
222
+            // 互动教室
223
+            carouselData.value = monitor
224
+            //物联设备统计
225
+            equipmentTotal.value = status
226
+
227
+        }
228
+
229
+        onMounted(() => {
230
+            getDeviceData()
231
+
232
+        })
233
+        return {
234
+            interDevice,
235
+            videoData,
236
+            onPlay,
237
+            videoDialog,
238
+            lookVideo,
239
+            carouselData,
240
+            equipmentTotal
241
+        }
242
+    },
243
+})
244
+
245
+</script>
246
+
247
+<style scoped lang="scss">
248
+@import '../assets/css/right.scss';
249
+</style>
250
+<style lang="scss">
251
+.videoDialog {
252
+    .el-dialog__header {
253
+        height: 49px;
254
+        line-height: 49px;
255
+        padding: 0;
256
+        background-color: #1B67D9;
257
+        margin-right: 0;
258
+
259
+        .el-dialog__title {
260
+            color: #FFF;
261
+            font-family: Inter;
262
+        }
263
+
264
+        .el-dialog__headerbtn {
265
+            top: 0px;
266
+
267
+            i {
268
+                color: #fff;
269
+                font-size: 20px;
270
+                font-weight: bold;
271
+            }
272
+
273
+        }
274
+
275
+    }
276
+
277
+    .el-dialog__body {
278
+        padding: 0 !important;
279
+    }
280
+
281
+
282
+}
283
+</style>

+ 5 - 3
src/main.js

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

+ 5 - 1
src/request/api.js

@@ -1,4 +1,8 @@
1 1
 import { get, post } from './http'
2 2
 
3 3
 
4
-export const testApi = p => post(`https://test.lqkj.top/wsngt-server/system/sysuser/bind/ticket?userCode${p}`)
4
+export const testApi = p => post(`https://test.lqkj.top/wsngt-server/system/sysuser/bind/ticket?userCode${p}`)
5
+
6
+export const getClass = () => get("http://192.168.4.219:12397/ioc-server/wisdomClass/classroomStatistic")
7
+
8
+export const getDevice = () => get("http://192.168.4.219:12397/ioc-server/wisdomClass/deviceStatistic")

+ 2 - 2
src/request/http.js

@@ -1,5 +1,5 @@
1 1
 import axios from 'axios'
2
-
2
+import { ElMessage } from 'element-plus';
3 3
 
4 4
 // 请求超时时间
5 5
 axios.defaults.timeout = 10000
@@ -24,7 +24,7 @@ axios.interceptors.response.use(
24 24
     response => {
25 25
         console.log("response", response)
26 26
         if (response.status === 200) {
27
-            return Promise.resolve(response)
27
+            return Promise.resolve(response.data)
28 28
         } else {
29 29
             return Promise.reject(response)
30 30
         }

+ 27 - 2
vite.config.js

@@ -3,8 +3,22 @@ import vue from '@vitejs/plugin-vue'
3 3
 import AutoImport from 'unplugin-auto-import/vite'
4 4
 import Components from 'unplugin-vue-components/vite'
5 5
 import { ElementPlusResolver } from 'unplugin-vue-components/resolvers'
6
+import postCssPxToRem from "postcss-pxtorem";
6 7
 
7 8
 // https://vitejs.dev/config/
9
+// export default defineConfig({
10
+//   plugins: [
11
+//     vue(),
12
+
13
+//     Components({
14
+//       resolvers: [ElementPlusResolver()]
15
+//     }),
16
+//     AutoImport({
17
+//       resolvers: [ElementPlusResolver()]
18
+//     })
19
+//   ]
20
+
21
+// })
8 22
 export default defineConfig({
9 23
   plugins: [
10 24
     vue(),
@@ -14,6 +28,17 @@ export default defineConfig({
14 28
     AutoImport({
15 29
       resolvers: [ElementPlusResolver()]
16 30
     })
17
-  ]
18
-
31
+  ],
32
+  css: {
33
+    postcss:
34
+    {
35
+      plugins:
36
+        [postCssPxToRem({
37
+          rootValue: 192,
38
+          // 1rem,根据 设计稿宽度/10 进行设置
39
+          propList: ['*']
40
+          // 需要转换的属性,这里选择全部都进行转换
41
+        })]
42
+    }
43
+  }
19 44
 })