IOTLeft.vue 11 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441
  1. <template>
  2. <div class="iot-data-left">
  3. <div class="left-box">
  4. <titleBgBox class="left-title-bg-box" title="基础情况">
  5. <div class="base-data">
  6. <div
  7. class="base-box base-data-device"
  8. :class="{ 'base-data-active': deviceStatus == 'all' }"
  9. @click="deviceStatusSelect('all')"
  10. >
  11. <span class="name">物联设备</span>
  12. <span class="num">{{ basicsInfo.totalDevices }}</span>
  13. </div>
  14. <div
  15. class="base-box base-data-online"
  16. :class="{ 'base-data-active': deviceStatus == 'online' }"
  17. @click="deviceStatusSelect('online')"
  18. >
  19. <span class="name">在线数</span>
  20. <span class="num">{{ basicsInfo.onlineDevices }}</span>
  21. </div>
  22. <div
  23. class="base-box base-data-unline"
  24. :class="{ 'base-data-active': deviceStatus == 'unline' }"
  25. @click="deviceStatusSelect('unline')"
  26. >
  27. <span class="name">离线数</span>
  28. <span class="num">{{ basicsInfo.offlineDevices }}</span>
  29. </div>
  30. <div class="base-box base-data-ratio">
  31. <span class="name">在线率</span>
  32. <span class="num">{{ basicsInfo.onlineRate }}</span>
  33. </div>
  34. </div>
  35. </titleBgBox>
  36. <titleBgBox class="left-title-bg-box" title="设备类型">
  37. <!-- 头部色彩比例块 -->
  38. <colorDiv
  39. class="color-box"
  40. :data="deviceTypeArr"
  41. :colorData="equipmentColor"
  42. ></colorDiv>
  43. <div class="device-type">
  44. <div
  45. class="device-type-item"
  46. v-for="(item, index) in deviceTypeArr"
  47. :key="index"
  48. @click="selectCurrentKey(item)"
  49. >
  50. <div
  51. class="icon-label"
  52. :style="{
  53. backgroundColor: equipmentColor[index % deviceTypeArr.length],
  54. }"
  55. ></div>
  56. <span
  57. class="name"
  58. :style="{
  59. color: selectDeviceType.includes(item.monitorScene)
  60. ? '#FFD041'
  61. : '',
  62. }"
  63. >{{ item.monitorScene }}</span
  64. >
  65. <span
  66. class="num"
  67. :style="{
  68. color: selectDeviceType.includes(item.monitorScene)
  69. ? '#FFC85A'
  70. : '',
  71. }"
  72. >{{ item.numRatio + '%' }}</span
  73. >
  74. </div>
  75. </div>
  76. </titleBgBox>
  77. <titleBgBox class="left-title-bg-box" title="设备分布">
  78. <div class="device-distribute">
  79. <div
  80. class="device-distribute-item"
  81. v-for="(item, index) in deviceDistrubuteArr"
  82. :key="index"
  83. >
  84. <span class="name">{{ item.name }}</span>
  85. <span class="num">{{ item.num }}</span
  86. ><span class="num1">个</span>
  87. <el-progress
  88. :percentage="getCount(deviceDistrubuteArr, item.num)"
  89. :show-text="false"
  90. />
  91. </div>
  92. </div>
  93. </titleBgBox>
  94. </div>
  95. </div>
  96. </template>
  97. <script setup>
  98. import { onMounted, reactive, ref, computed } from "vue";
  99. import titleBgBox from "./titleBgBox.vue";
  100. import colorDiv from "@/components/ColorDiv.vue";
  101. import {
  102. QueryBasicsInfo,
  103. QueryListEquipment,
  104. QueryTrendsOnline,
  105. } from "../../../service/iotService";
  106. //设备类型统计数据
  107. const equipmentData = ref([]);
  108. const equipmentColor = ref([
  109. "#549bf1",
  110. "#67d470",
  111. "#bcbf5c",
  112. "#0daee3",
  113. "#3182ea",
  114. "#aa8a6e",
  115. "#5a70bc",
  116. " #b2dfff",
  117. "#5cabbf",
  118. "#2cecbc",
  119. ]);
  120. //基本情况obj
  121. const basicsInfo = reactive({
  122. totalDevices: "",
  123. onlineDevices: "",
  124. offlineDevices: "",
  125. unregisterDevices: "",
  126. onlineRate: "",
  127. });
  128. //设备分类当前项选中
  129. const selectDeviceType = ref([]);
  130. //设备在线状态选中
  131. const deviceStatus = ref(null);
  132. const getCount = (totalArr, num) => {
  133. const total = totalArr
  134. .map((item) => item.num)
  135. .reduce(function (prev, curr) {
  136. return prev + curr;
  137. });
  138. console.log(total);
  139. return (num / total) * 100;
  140. };
  141. const getBasicsInfo = () => {
  142. QueryBasicsInfo().then((res) => {
  143. if (res.data.code == "200") {
  144. basicsInfo.totalDevices = res.data.data.totalDevices;
  145. basicsInfo.onlineDevices = res.data.data.onlineDevices;
  146. basicsInfo.offlineDevices = res.data.data.offlineDevices;
  147. basicsInfo.onlineRate = res.data.data.onlineRate;
  148. }
  149. });
  150. };
  151. //设备类型/设备分布数据
  152. const getListEquipment = () => {
  153. QueryListEquipment().then((res) => {
  154. if (res.data.code == "200") {
  155. const data = res.data.data;
  156. const typeObj = {};
  157. data.list.forEach((item) => {
  158. if (typeObj[item.monitorScene]) {
  159. typeObj[item.monitorScene].num++;
  160. //onlineState: 设备状态 0:未激活;1:在线;2:离线
  161. } else {
  162. typeObj[item.monitorScene] = {
  163. monitorScene: item.monitorScene, //监管场景
  164. serialNumber: item.serialNumber,
  165. district: item.district, //分布区域
  166. num: 1,
  167. onlineNum: 0,
  168. unlineNum: 0,
  169. total: data.total,
  170. };
  171. }
  172. item.onlineState === 1 && typeObj[item.monitorScene].onlineNum++;
  173. item.onlineState === 2 && typeObj[item.monitorScene].unlineNum++;
  174. });
  175. equipmentData.value = Object.values(typeObj);
  176. }
  177. });
  178. };
  179. //获取设备分布数据
  180. const deviceDistrubuteArr = computed(() => {
  181. const deviceDistruibute = {};
  182. //选择需要获取的字段
  183. const numName =
  184. deviceStatus.value == "online"
  185. ? "onlineNum"
  186. : deviceStatus.value == "unline"
  187. ? "unlineNum"
  188. : "num";
  189. equipmentData.value.forEach((item) => {
  190. if (
  191. selectDeviceType.value.length > 0 &&
  192. !selectDeviceType.value.includes(item.monitorScene)
  193. ) {
  194. return;
  195. }
  196. //统计数量
  197. if (deviceDistruibute[item.district]) {
  198. deviceDistruibute[item.district][numName] += item[numName];
  199. } else {
  200. deviceDistruibute[item.district] = {
  201. name: item.district,
  202. num: item[numName],
  203. };
  204. }
  205. });
  206. return Object.values(deviceDistruibute);
  207. });
  208. //获取设备类型数据
  209. const deviceTypeArr = computed(() => {
  210. const numName =
  211. deviceStatus.value == "online"
  212. ? "onlineNum"
  213. : deviceStatus.value == "unline"
  214. ? "unlineNum"
  215. : "num";
  216. const result = equipmentData.value.map((item) => {
  217. return {
  218. ...item,
  219. numRatio: ((item[numName] * 100) / item.total).toFixed(0),
  220. };
  221. });
  222. return result;
  223. });
  224. //设备类型选中
  225. const selectCurrentKey = (params) => {
  226. if (selectDeviceType.value.includes(params.monitorScene)) {
  227. selectDeviceType.value = selectDeviceType.value.filter(
  228. (item) => item != params.monitorScene
  229. );
  230. return;
  231. }
  232. //暂时只支持单选,如果后续可以多选,将下行代码注释
  233. selectDeviceType.value = [];
  234. selectDeviceType.value.push(params.monitorScene);
  235. };
  236. //基本情况设备状态选择
  237. const deviceStatusSelect = (status) => {
  238. if(deviceStatus.value == status){
  239. deviceStatus.value = null
  240. return
  241. }
  242. deviceStatus.value = status;
  243. };
  244. const init = () => {
  245. getBasicsInfo();
  246. getListEquipment();
  247. };
  248. onMounted(() => {
  249. init();
  250. });
  251. </script>
  252. <style scoped lang="scss">
  253. .iot-data-left {
  254. box-sizing: border-box;
  255. position: fixed;
  256. top: 0;
  257. left: 0;
  258. width: 450px;
  259. height: 100vh;
  260. font-size: 18px;
  261. z-index: 99;
  262. padding: 58px 0 44px 25px;
  263. background: linear-gradient(
  264. 90deg,
  265. rgba(0, 17, 50, 0.75),
  266. rgba(0, 17, 50, 0.5),
  267. rgba(0, 17, 51, 0)
  268. );
  269. .left-box {
  270. box-sizing: border-box;
  271. width: 100%;
  272. height: 100%;
  273. padding: 15px 25px 15px 25px;
  274. background-image: url("@/assets/imgs/IOTImage/left-bg.png");
  275. background-repeat: no-repeat;
  276. background-size: 100% 100%;
  277. // background-position-x: -220px;
  278. .color-box {
  279. width: calc(100% - 30px);
  280. padding-left: 3px;
  281. margin-top: 5px;
  282. }
  283. .base-data {
  284. position: relative;
  285. width: 100%;
  286. height: 151px;
  287. background-image: url("@/assets/imgs/IOTImage/1.png");
  288. background-size: 100% 100%;
  289. .base-box {
  290. display: flex;
  291. flex-direction: column;
  292. color: #ffffff;
  293. text-align: right;
  294. width: 60px;
  295. .name {
  296. font-size: 14px;
  297. color: #cce6ff;
  298. }
  299. .num {
  300. font-size: 18px;
  301. font-family: "heitao";
  302. padding-top: 2px;
  303. color: #dfecfb;
  304. }
  305. }
  306. .base-data-device {
  307. position: absolute;
  308. left: 30px;
  309. top: 22px;
  310. }
  311. .base-data-online {
  312. position: absolute;
  313. left: 30px;
  314. bottom: 22px;
  315. .num {
  316. // color: #ffd041;
  317. }
  318. }
  319. .base-data-active {
  320. .num,
  321. .name {
  322. color: #ffd041;
  323. }
  324. }
  325. .base-data-unline {
  326. position: absolute;
  327. right: 25px;
  328. top: 22px;
  329. text-align: left;
  330. }
  331. .base-data-ratio {
  332. position: absolute;
  333. right: 25px;
  334. bottom: 22px;
  335. text-align: left;
  336. }
  337. }
  338. .device-type {
  339. display: flex;
  340. justify-content: flex-start;
  341. flex-wrap: wrap;
  342. gap: 0px 19px;
  343. padding-bottom: 15px;
  344. .device-type-item {
  345. display: flex;
  346. flex-direction: column;
  347. text-align: left;
  348. position: relative;
  349. color: #ffffff;
  350. padding-top: 10px;
  351. padding-left: 20px;
  352. .icon-label {
  353. position: absolute;
  354. width: 8px;
  355. height: 8px;
  356. border-radius: 1px;
  357. top: 30%;
  358. left: 5px;
  359. }
  360. .name {
  361. font-size: 14px;
  362. color: #9bb9dd;
  363. min-width: 80px;
  364. }
  365. .num {
  366. color: #dfecfb;
  367. font-size: 18px;
  368. font-family: "heitao";
  369. }
  370. }
  371. }
  372. .device-distribute {
  373. padding-left: 8px;
  374. .device-distribute-item {
  375. display: flex;
  376. align-items: center;
  377. margin-bottom: 4px;
  378. .name {
  379. font-size: 14px;
  380. color: #bfd5e0;
  381. width: 50px;
  382. }
  383. .num {
  384. font-size: 18px;
  385. font-family: "heitao";
  386. padding-left: 15px;
  387. color: #c6daeb;
  388. width: 30px;
  389. }
  390. .num1 {
  391. font-size: 14px;
  392. padding-top: 2px;
  393. color: #c6daeb;
  394. }
  395. :deep(.el-progress) {
  396. width: calc(100% - 160px);
  397. margin-left: 13px;
  398. .el-progress-bar__outer {
  399. background-color: rgba(35, 64, 95, 0.51);
  400. height: 6px !important;
  401. }
  402. .el-progress-bar__inner {
  403. // background-color: rgba(60, 139, 219, 1);
  404. height: 6px;
  405. background: linear-gradient(
  406. to right,
  407. rgba(60, 139, 219, 1),
  408. rgba(60, 139, 219, 0.5)
  409. );
  410. }
  411. }
  412. }
  413. }
  414. .left-title-bg-box {
  415. :deep(.title-box) {
  416. width: calc(100% - 25px);
  417. }
  418. }
  419. }
  420. }
  421. </style>