index.vue 40 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086108710881089109010911092109310941095109610971098109911001101110211031104110511061107110811091110111111121113111411151116111711181119112011211122112311241125112611271128112911301131113211331134113511361137113811391140114111421143114411451146114711481149115011511152115311541155115611571158115911601161116211631164116511661167116811691170117111721173117411751176117711781179118011811182118311841185118611871188118911901191119211931194119511961197119811991200120112021203120412051206120712081209121012111212121312141215121612171218121912201221122212231224122512261227122812291230123112321233123412351236123712381239124012411242124312441245124612471248124912501251125212531254125512561257125812591260126112621263126412651266126712681269127012711272127312741275127612771278127912801281128212831284128512861287128812891290129112921293129412951296129712981299130013011302130313041305130613071308130913101311131213131314131513161317131813191320132113221323132413251326132713281329133013311332133313341335133613371338133913401341134213431344134513461347134813491350135113521353135413551356135713581359136013611362136313641365136613671368136913701371137213731374137513761377137813791380138113821383138413851386138713881389139013911392139313941395139613971398139914001401
  1. <template>
  2. <div class="body1" v-loading="isLoading">
  3. <!-- 学生数据 -->
  4. <div class="left">
  5. <div class="top">
  6. <div class="titleBox">
  7. <div class="title">基础概况</div>
  8. </div>
  9. <div class="dataBox">
  10. <div class="info_box">
  11. <div class="info blueBG">
  12. <span>学生总数</span>
  13. <span>{{ studentCount }}</span>
  14. </div>
  15. <div class="info blueBG">
  16. <span>本月登录学生环比</span>
  17. <span v-if="loginCountMonthArray.length">{{ (loginCountMonthArray[loginCountMonthArray.length - 1].user) > 0 ? (((loginCountMonthArray[loginCountMonthArray.length - 1].user) / studentCount) * 100).toFixed(0) + '%' : 0 }}</span>
  18. <!-- <span v-if="loginCountMonthArray.length">{{
  19. loginCountMonthArray[loginCountMonthArray.length - 1].user -
  20. loginCountMonthArray[loginCountMonthArray.length - 2].user <
  21. 0
  22. ? 0
  23. : (((loginCountMonthArray[loginCountMonthArray.length - 1]
  24. .user -
  25. loginCountMonthArray[loginCountMonthArray.length - 2]
  26. .user) /
  27. loginCountMonthArray[loginCountMonthArray.length - 2]
  28. .user) *
  29. 100).toFixed(0) +
  30. "%"
  31. }}</span> -->
  32. </div>
  33. <div class="info blueBG">
  34. <span>本月登录学生用户总数</span>
  35. <span v-if="loginCountMonthArray.length">{{
  36. loginCountMonthArray[loginCountMonthArray.length - 1].user
  37. }}</span>
  38. </div>
  39. <div class="info blueBG">
  40. <span>本月新增登录学生用户</span>
  41. <span v-if="loginCountMonthArray.length">{{
  42. loginCountMonthArray[loginCountMonthArray.length - 1].user -
  43. loginCountMonthArray[loginCountMonthArray.length - 2].user <
  44. 0
  45. ? 0
  46. : loginCountMonthArray[loginCountMonthArray.length - 1].user -
  47. loginCountMonthArray[loginCountMonthArray.length - 2].user
  48. }}</span>
  49. </div>
  50. </div>
  51. <loginCount
  52. style="height: calc(100% - 140px)"
  53. :monthArray="loginCountMonthArray"
  54. ></loginCount>
  55. </div>
  56. </div>
  57. <div class="bottom">
  58. <div class="titleBox">
  59. <div
  60. class="title"
  61. :class="{ isClick: skType == 0 }"
  62. @click="skType = 0"
  63. style="cursor: pointer; padding: 0 0 5px 0"
  64. >
  65. 在线时长
  66. </div>
  67. <div
  68. class="title"
  69. :class="{ isClick: skType == 1 }"
  70. @click="skType = 1"
  71. style="cursor: pointer; padding: 0 0 5px 0"
  72. >
  73. 登录频次
  74. </div>
  75. <!-- <el-select v-model="cType" class="selectBox" style="width: 110px">
  76. <el-option label="全部" value="全部"></el-option>
  77. <el-option label="一年级" value="一年级"></el-option>
  78. <el-option label="二年级" value="二年级"></el-option>
  79. <el-option label="三年级" value="三年级"></el-option>
  80. </el-select>
  81. <div class="timeDiv">
  82. <div @click="tType = 0" :class="{ isClick: tType == 0 }">周</div>
  83. <div @click="tType = 1" :class="{ isClick: tType == 1 }">月</div>
  84. <div @click="tType = 2" :class="{ isClick: tType == 2 }">学期</div>
  85. </div> -->
  86. </div>
  87. <div class="info_box" v-if="skType == 1">
  88. <div class="info blueBG">
  89. <span>登录频次</span>
  90. <!-- <span>{{ count }}</span> -->
  91. <span>{{ countLogin }}</span>
  92. </div>
  93. <div class="info blueBG">
  94. <span>人均频次</span>
  95. <!-- <span>{{ weekCount }}</span> -->
  96. <span>{{
  97. countLogin && studentCount
  98. ? (countLogin / studentCount).toFixed(0)
  99. : 0
  100. }}</span>
  101. </div>
  102. </div>
  103. <div class="info_box" v-if="skType == 0">
  104. <div class="info blueBG">
  105. <span>累计时长</span>
  106. <span>{{ loginTime.toFixed(0) }}</span>
  107. </div>
  108. <div class="info blueBG">
  109. <span>人均在线时长</span>
  110. <span>{{
  111. loginTime && studentCount
  112. ? (loginTime / studentCount).toFixed(0)
  113. : 0
  114. }}</span>
  115. </div>
  116. </div>
  117. <div class="dataBox" style="height: calc(100% - 110px)">
  118. <toolUser
  119. style="height: calc(100% - 30px)"
  120. v-if="skType == 1 && !oType"
  121. :yearArray="loginCountYearArray"
  122. ></toolUser>
  123. <bar2
  124. style="height: calc(100% - 30px)"
  125. v-if="skType == 1 && oType"
  126. :EloginTimeArray="EloginTimeArray2"
  127. ></bar2>
  128. <bar
  129. style="height: calc(100% - 30px)"
  130. v-if="skType == 0 && !oType2"
  131. :EloginTimeArray="EloginTimeArray"
  132. ></bar>
  133. <toolUser2
  134. style="height: calc(100% - 30px)"
  135. v-if="skType == 0 && oType2"
  136. :yearArray="loginCountYearArray2"
  137. ></toolUser2>
  138. <div class="otherCss" v-if="skType == 1">
  139. <div v-if="!oType">切换为柱状图</div>
  140. <div v-if="oType">切换为热力图</div>
  141. <div class="otherImg" @click="otherEchart">
  142. <img src="../../../../assets/icon/other.png" alt="" />
  143. </div>
  144. </div>
  145. <div class="otherCss" v-if="skType == 0">
  146. <div v-if="oType2">切换为柱状图</div>
  147. <div v-if="!oType2">切换为热力图</div>
  148. <div class="otherImg" @click="otherEchart2">
  149. <img src="../../../../assets/icon/other.png" alt="" />
  150. </div>
  151. </div>
  152. </div>
  153. </div>
  154. </div>
  155. <div class="center">
  156. <div class="top">
  157. <div class="titleBox" style="justify-content: space-between">
  158. <div
  159. style="
  160. display: flex;
  161. flex-direction: row;
  162. flex-wrap: nowrap;
  163. align-items: center;
  164. "
  165. >
  166. <div
  167. class="title"
  168. :class="{ isClick: courseType == 0 }"
  169. @click="courseType = 0;rankLoading = false"
  170. style="cursor: pointer; padding: 0 0 5px 0"
  171. >
  172. 学生行为分布
  173. </div>
  174. <!-- <div
  175. class="title"
  176. :class="{ isClick: courseType == 1 }"
  177. @click="courseType = 1"
  178. style="cursor: pointer; padding: 0 0 5px 0"
  179. >
  180. 学生协同情况
  181. </div> -->
  182. <div
  183. class="title"
  184. :class="{ isClick: courseType == 2 }"
  185. @click="checkRank()"
  186. style="cursor: pointer; padding: 0 0 5px 0"
  187. >
  188. 学生排行榜
  189. </div>
  190. </div>
  191. <div
  192. style="
  193. display: flex;
  194. flex-direction: row;
  195. flex-wrap: nowrap;
  196. align-items: center;
  197. "
  198. v-if="courseType == 0"
  199. >
  200. <el-select v-model="cType2" class="selectBox" style="width: 110px">
  201. <el-option label="在线时长" value="onlineTime"></el-option>
  202. <el-option label="作业数量" value="worksNum"></el-option>
  203. </el-select>
  204. </div>
  205. <div
  206. style="
  207. display: flex;
  208. flex-direction: row;
  209. flex-wrap: nowrap;
  210. align-items: center;
  211. "
  212. v-if="courseType == 2"
  213. >
  214. <el-select v-model="cTypeRank" class="selectBox" style="width: 130px" @change="checkRank">
  215. <el-option :label="item.name" :value="item.id" v-for="(item,index) in classList" :key="index"></el-option>
  216. </el-select>
  217. </div>
  218. </div>
  219. <div class="dataBox" v-loading="rankLoading">
  220. <studentInfo
  221. v-if="courseType == 0 && cType2 == 'onlineTime'"
  222. style="height: calc(100%)"
  223. :EloginTimeArray="EloginTimeArray"
  224. ></studentInfo>
  225. <studentInfo3
  226. v-if="courseType == 0 && cType2 == 'worksNum'"
  227. style="height: calc(100%)"
  228. :EworksNumArray="EworksNumArray"
  229. ></studentInfo3>
  230. <courseNum
  231. v-if="courseType == 1"
  232. style="height: calc(100%)"
  233. ></courseNum>
  234. <cateRank v-if="courseType == 2" :rankData="rankData"></cateRank>
  235. </div>
  236. </div>
  237. <div class="bottom">
  238. <div class="titleBox" style="justify-content: space-between">
  239. <div
  240. style="
  241. display: flex;
  242. flex-direction: row;
  243. flex-wrap: nowrap;
  244. align-items: center;
  245. "
  246. >
  247. <div
  248. class="title"
  249. :class="{ isClick: bType == 0 }"
  250. @click="bType = 0"
  251. style="cursor: pointer; padding: 0 0 5px 0"
  252. >
  253. 课程参与情况
  254. </div>
  255. <div
  256. class="title"
  257. :class="{ isClick: bType == 1 }"
  258. @click="bType = 1"
  259. style="cursor: pointer; padding: 0 0 5px 0"
  260. >
  261. 项目参与情况
  262. </div>
  263. </div>
  264. </div>
  265. <div class="dataBox">
  266. <div
  267. class="allBox"
  268. style="height: calc(100% - 30px)"
  269. v-if="bType == 0"
  270. >
  271. <div class="allBox_left">
  272. <stuAct :pusaDep="courseArray"></stuAct>
  273. </div>
  274. <div class="allBox_right">
  275. <div class="depth">
  276. <span>参与课程</span>
  277. <div>
  278. <el-progress
  279. :width="80"
  280. type="circle"
  281. :percentage="
  282. courseJson.courseNum && studentCount
  283. ? parseInt(
  284. (
  285. (courseJson.courseNum / studentCount) *
  286. 100
  287. ).toFixed(0)
  288. )
  289. : 0
  290. "
  291. :stroke-width="5"
  292. :format="format"
  293. color="#106BFF"
  294. ></el-progress>
  295. </div>
  296. </div>
  297. <div class="depth">
  298. <span>使用工具</span>
  299. <div>
  300. <el-progress
  301. :width="80"
  302. type="circle"
  303. :percentage="
  304. courseJson.toolNum && studentCount
  305. ? parseInt(
  306. ((courseJson.toolNum / studentCount) * 100).toFixed(
  307. 0
  308. )
  309. )
  310. : 0
  311. "
  312. :stroke-width="5"
  313. :format="format"
  314. color="#106BFF"
  315. ></el-progress>
  316. </div>
  317. </div>
  318. <div class="depth">
  319. <span>协同合作</span>
  320. <div>
  321. <el-progress
  322. :width="80"
  323. type="circle"
  324. :percentage="0"
  325. :stroke-width="5"
  326. :format="format"
  327. color="#106BFF"
  328. ></el-progress>
  329. </div>
  330. </div>
  331. <div class="depth">
  332. <span>互动交流</span>
  333. <div>
  334. <el-progress
  335. :width="80"
  336. type="circle"
  337. :percentage="
  338. courseJson.exchangeNum && studentCount
  339. ? parseInt(
  340. (
  341. (courseJson.exchangeNum / studentCount) *
  342. 100
  343. ).toFixed(0)
  344. )
  345. : 0
  346. "
  347. :stroke-width="5"
  348. :format="format"
  349. color="#106BFF"
  350. ></el-progress>
  351. </div>
  352. </div>
  353. </div>
  354. </div>
  355. <div
  356. class="allBox"
  357. style="height: calc(100% - 30px)"
  358. v-if="bType == 1"
  359. >
  360. <div class="allBox_left">
  361. <stuAct2 :pusaDep="scourseArray"></stuAct2>
  362. </div>
  363. <div class="allBox_right">
  364. <div class="depth">
  365. <span>参与项目</span>
  366. <div>
  367. <el-progress
  368. :width="80"
  369. type="circle"
  370. :percentage="
  371. scourseJson.courseNum && studentCount
  372. ? parseInt(
  373. (
  374. (scourseJson.courseNum / studentCount) *
  375. 100
  376. ).toFixed(0)
  377. )
  378. : 0
  379. "
  380. :stroke-width="5"
  381. :format="format"
  382. color="#106BFF"
  383. ></el-progress>
  384. </div>
  385. </div>
  386. <div class="depth">
  387. <span>使用工具</span>
  388. <div>
  389. <el-progress
  390. :width="80"
  391. type="circle"
  392. :percentage="
  393. scourseJson.toolNum && studentCount
  394. ? parseInt(
  395. (
  396. (scourseJson.toolNum / studentCount) *
  397. 100
  398. ).toFixed(0)
  399. )
  400. : 0
  401. "
  402. :stroke-width="5"
  403. :format="format"
  404. color="#106BFF"
  405. ></el-progress>
  406. </div>
  407. </div>
  408. <div class="depth">
  409. <span>协同合作</span>
  410. <div>
  411. <el-progress
  412. :width="80"
  413. type="circle"
  414. :percentage="
  415. scourseJson.gcourseNum && studentCount
  416. ? parseInt(
  417. (
  418. (scourseJson.gcourseNum / studentCount) *
  419. 100
  420. ).toFixed(0)
  421. )
  422. : 0
  423. "
  424. :stroke-width="5"
  425. :format="format"
  426. color="#106BFF"
  427. ></el-progress>
  428. </div>
  429. </div>
  430. <div class="depth">
  431. <span>互动交流</span>
  432. <div>
  433. <el-progress
  434. :width="80"
  435. type="circle"
  436. :percentage="0"
  437. :stroke-width="5"
  438. :format="format"
  439. color="#106BFF"
  440. ></el-progress>
  441. </div>
  442. </div>
  443. </div>
  444. </div>
  445. </div>
  446. </div>
  447. </div>
  448. <div class="right">
  449. <div class="top" style="border-radius: 5px">
  450. <div class="titleBox" style="justify-content: space-between">
  451. <div class="title">整体分布</div>
  452. <div
  453. style="
  454. display: flex;
  455. flex-direction: row;
  456. flex-wrap: nowrap;
  457. align-items: center;
  458. "
  459. >
  460. <el-select v-model="cType7" class="selectBox" style="width: 110px">
  461. <el-option label="在线时长" value="onlineTime"></el-option>
  462. <el-option label="作业数量" value="worksNum"></el-option>
  463. </el-select>
  464. </div>
  465. </div>
  466. <div class="dataBox">
  467. <studentInfo2
  468. style="height: calc(100% - 40px)"
  469. v-if="cType7 == 'onlineTime'"
  470. :EloginTimeArray="EloginTimeArray"
  471. ></studentInfo2>
  472. <studentInfo4
  473. style="height: calc(100% - 40px)"
  474. v-if="cType7 == 'worksNum'"
  475. :EworksNumArray="EworksNumArray"
  476. ></studentInfo4>
  477. </div>
  478. </div>
  479. <div class="bottom">
  480. <div class="titleBox" style="justify-content: space-between">
  481. <div class="title">活动参与</div>
  482. <div
  483. style="
  484. display: flex;
  485. flex-direction: row;
  486. flex-wrap: nowrap;
  487. align-items: center;
  488. "
  489. >
  490. <!-- <el-select v-model="cType8" class="selectBox" style="width: 110px">
  491. <el-option label="全部年级" value="全部年级"></el-option>
  492. </el-select> -->
  493. <el-select
  494. v-model="cType9"
  495. class="selectBox"
  496. style="width: 110px"
  497. @change="classChange"
  498. >
  499. <el-option label="全部班级" value=""></el-option>
  500. <el-option
  501. :label="item.name"
  502. :value="item.id"
  503. v-for="item in classList"
  504. :key="item.id"
  505. ></el-option>
  506. </el-select>
  507. </div>
  508. </div>
  509. <div class="dataBox">
  510. <div class="info_box">
  511. <div class="info blueBG">
  512. <span>人均参与课程数</span>
  513. <span>{{
  514. courseNum && uArray.length
  515. ? (courseNum / uArray.length).toFixed(0)
  516. : 0
  517. }}</span>
  518. </div>
  519. <div class="info blueBG">
  520. <span>人均参与项目数</span>
  521. <span>{{
  522. scourseNum && uArray.length
  523. ? (scourseNum / uArray.length).toFixed(0)
  524. : 0
  525. }}</span>
  526. </div>
  527. </div>
  528. <loginTime
  529. style="height: calc(100% - 72px)"
  530. :uArray="uArray"
  531. ></loginTime>
  532. </div>
  533. </div>
  534. </div>
  535. </div>
  536. </template>
  537. <script>
  538. import loginCount from "./loginCount";
  539. import studentInfo from "./studentInfo";
  540. import studentInfo2 from "./studentInfo2";
  541. import studentInfo3 from "./studentInfo3";
  542. import studentInfo4 from "./studentInfo4";
  543. import loginTime from "./loginTime";
  544. import stuAct from "./stuAct";
  545. import stuAct2 from "./stuAct2";
  546. import toolUser from "./toolUser";
  547. import toolUser2 from "./toolUser2";
  548. import bar from "./bar";
  549. import bar2 from "./bar2";
  550. import courseNum from "./courseNum";
  551. import cateRank from "./cateRank";
  552. export default {
  553. props: {
  554. oid: {
  555. type: String,
  556. },
  557. org: {
  558. type: String,
  559. },
  560. },
  561. components: {
  562. loginCount,
  563. studentInfo,
  564. studentInfo2,
  565. studentInfo3,
  566. studentInfo4,
  567. loginTime,
  568. stuAct,
  569. stuAct2,
  570. toolUser,
  571. toolUser2,
  572. bar,
  573. bar2,
  574. courseNum,
  575. cateRank
  576. },
  577. data() {
  578. return {
  579. isLoading: false,
  580. skType: 0,
  581. tType: 0,
  582. courseType: 0,
  583. tType1: 0,
  584. sType: 0,
  585. bType: 0,
  586. cType: "全部",
  587. cType1: "全部",
  588. cType2: "onlineTime",
  589. cType3: "全部",
  590. cType4: "师生",
  591. cType5: "全部年级",
  592. cType7: "onlineTime",
  593. cType8: "全部年级",
  594. cType9: "",
  595. oType: false,
  596. oType2: false,
  597. studentCount: 0,
  598. loginCountMonthArray: [],
  599. countLogin: 0,
  600. loginCountYearArray: [],
  601. loginCountYearArray2: [],
  602. loginTime: 0,
  603. EloginTimeArray: [],
  604. EloginTimeArray2: [],
  605. classList: [],
  606. courseNum: 0,
  607. scourseNum: 0,
  608. uArray: [],
  609. courseNumArray: [],
  610. scourseNumArray: [],
  611. userArray: [],
  612. courseJson: {
  613. courseNum: 0,
  614. toolNum: 0,
  615. exchangeNum: 0,
  616. },
  617. scourseJson: {
  618. courseNum: 0,
  619. toolNum: 0,
  620. gcourseNum: 0,
  621. },
  622. courseArray: [],
  623. scourseArray: [],
  624. EworksNumArray: [],
  625. rankLoading: false,
  626. cTypeRank: '',
  627. rankData:[],
  628. };
  629. },
  630. mounted() {
  631. this.getData();
  632. },
  633. methods: {
  634. format(percentage) {
  635. return percentage + "%";
  636. },
  637. otherEchart() {
  638. this.oType = !this.oType;
  639. },
  640. otherEchart2() {
  641. this.oType2 = !this.oType2;
  642. },
  643. getData() {
  644. this.isLoading = true;
  645. let params = [
  646. {
  647. oid: this.oid,
  648. org: this.org,
  649. },
  650. ];
  651. this.ajax
  652. .post(this.$store.state.api + "selectDataBoardStudentNew", params)
  653. .then((res) => {
  654. this.isLoading = false;
  655. let _grade = res.data[0]; //年级
  656. let _subject = res.data[1]; //学科
  657. let _course = res.data[2]; //课程
  658. this.studentCount = res.data[3][0].count; //学生总数
  659. let _loginCount = res.data[4]; //统计半年的登录学生
  660. let loginCountMonthArray = [];
  661. const date = new Date();
  662. var Month = date.getMonth() + 1;
  663. var Year = date.getFullYear();
  664. for (var i = Month; i > Month - 6; i--) {
  665. if (i <= 0) {
  666. loginCountMonthArray.push({
  667. Year: Year - 1,
  668. Month: 12 + i,
  669. user: 0,
  670. });
  671. } else {
  672. loginCountMonthArray.push({
  673. Month: i,
  674. Year: Year,
  675. user: 0,
  676. });
  677. }
  678. }
  679. loginCountMonthArray = loginCountMonthArray.reverse();
  680. for (var i = 0; i < _loginCount.length; i++) {
  681. let _date = new Date(_loginCount[i].create_at);
  682. var _month = _date.getMonth() + 1;
  683. var _year = _date.getFullYear();
  684. for (var j = 0; j < loginCountMonthArray.length; j++) {
  685. if (
  686. _month == loginCountMonthArray[j].Month &&
  687. _year == loginCountMonthArray[j].Year
  688. ) {
  689. loginCountMonthArray[j].user++;
  690. break;
  691. }
  692. }
  693. }
  694. this.loginCountMonthArray = loginCountMonthArray;
  695. this.countLogin = res.data[5][0].count; //登录频次
  696. let loginCountYear = res.data[6]; //一年的登录频次
  697. let loginCountYearArray = [];
  698. for (var i = Month; i > Month - 12; i--) {
  699. if (i <= 0) {
  700. loginCountYearArray.push({
  701. Year: Year - 1,
  702. Month: 12 + i,
  703. mon: 0,
  704. tue: 0,
  705. wed: 0,
  706. thur: 0,
  707. fri: 0,
  708. sat: 0,
  709. sun: 0,
  710. });
  711. } else {
  712. loginCountYearArray.push({
  713. Month: i,
  714. Year: Year,
  715. mon: 0,
  716. tue: 0,
  717. wed: 0,
  718. thur: 0,
  719. fri: 0,
  720. sat: 0,
  721. sun: 0,
  722. });
  723. }
  724. }
  725. loginCountYearArray = loginCountYearArray.reverse();
  726. for (var i = 0; i < loginCountYear.length; i++) {
  727. let _date = new Date(loginCountYear[i].create_at);
  728. var _month = _date.getMonth() + 1;
  729. var _year = _date.getFullYear();
  730. var _day = _date.getDay();
  731. let dayArray = ["sun", "mon", "tue", "wed", "thur", "fri", "sat"];
  732. for (var j = 0; j < loginCountYearArray.length; j++) {
  733. if (
  734. _month == loginCountYearArray[j].Month &&
  735. _year == loginCountYearArray[j].Year
  736. ) {
  737. loginCountYearArray[j][dayArray[_day]]++;
  738. break;
  739. }
  740. }
  741. }
  742. console.log(loginCountYearArray);
  743. this.loginCountYearArray = loginCountYearArray;
  744. this.loginTime = parseInt(res.data[7][0].time) / 60 / 60;
  745. let _loginTimeArray = res.data[9]; //在线时间
  746. let _classList = res.data[8]; //班级
  747. this.classList = _classList.filter(el => {return el.studentCount > 0});
  748. this.cTypeRank = _classList[0].id
  749. let _EloginTimeArray = [];
  750. for (let i = 0; i < _classList.length; i++) {
  751. _EloginTimeArray.push({
  752. id: _classList[i].id,
  753. name: _classList[i].name,
  754. loginTime: 0,
  755. });
  756. for (let j = 0; j < _loginTimeArray.length; j++) {
  757. const _user = _loginTimeArray[j];
  758. if (_user.classid.indexOf(_classList[i].id) != -1) {
  759. _EloginTimeArray[i].loginTime += Number(_user.time);
  760. }
  761. }
  762. _EloginTimeArray[i].loginTime = (
  763. _EloginTimeArray[i].loginTime /
  764. 60 /
  765. 60
  766. ).toFixed(0); //换算成小时
  767. }
  768. this.EloginTimeArray = _EloginTimeArray.filter((el)=>{
  769. return el.loginTime > 0;
  770. });
  771. this.courseNumArray = res.data[10]; //参与课程数
  772. this.scourseNumArray = res.data[11]; //参与项目数
  773. this.userArray = res.data[12]; //参与项目数
  774. this.classChange();
  775. this.courseJson.courseNum = res.data[13][0].count; //参与课程
  776. this.courseJson.toolNum = res.data[14][0].count; //使用工具
  777. this.courseJson.exchangeNum = res.data[15][0].count; //互动交流
  778. this.scourseJson.courseNum = res.data[16][0].count; //参与项目
  779. this.scourseJson.toolNum = res.data[17][0].count; //使用工具
  780. this.scourseJson.gcourseNum = res.data[18][0].count; //协同
  781. let loginUser = res.data[20].length; //登录
  782. let worksEva = res.data[21][0].count; //评价
  783. this.courseArray = [
  784. {
  785. value: loginUser ? ((loginUser / this.studentCount) * 100).toFixed(0) : 0,
  786. name: "登录",//登录
  787. },
  788. {
  789. value: this.courseJson.courseNum
  790. ? (
  791. (this.courseJson.courseNum / this.studentCount) *
  792. 100
  793. ).toFixed(0)
  794. : 0,
  795. name: "参与课程",
  796. },
  797. {
  798. value: this.courseJson.toolNum
  799. ? ((this.courseJson.toolNum / this.studentCount) * 100).toFixed(
  800. 0
  801. )
  802. : 0,
  803. name: "提交作业",//使用工具
  804. },
  805. {
  806. value: worksEva
  807. ? (
  808. (worksEva / this.studentCount) *
  809. 100
  810. ).toFixed(0)
  811. : 0,
  812. name: "评价",//互动交流
  813. },
  814. ];
  815. this.scourseArray = [
  816. {
  817. value: loginUser ? ((loginUser / this.studentCount) * 100).toFixed(0) : 0,
  818. name: "学生总数",
  819. },
  820. {
  821. value: this.scourseJson.courseNum
  822. ? (
  823. (this.scourseJson.courseNum / this.studentCount) *
  824. 100
  825. ).toFixed(0)
  826. : 0,
  827. name: "参与项目",
  828. },
  829. {
  830. value: this.scourseJson.toolNum
  831. ? (
  832. (this.scourseJson.toolNum / this.studentCount) *
  833. 100
  834. ).toFixed(0)
  835. : 0,
  836. name: "提交作业",//使用工具
  837. },
  838. {
  839. value: this.scourseJson.gcourseNum
  840. ? (
  841. (this.scourseJson.gcourseNum / this.studentCount) *
  842. 100
  843. ).toFixed(0)
  844. : 0,
  845. name: "协同",
  846. },
  847. ];
  848. let _studentWorksArray = res.data[19]; //学生班级作业数量
  849. let _EworksNumArray = [];
  850. let _courseIdArray = [];
  851. for (let i = 0; i < _classList.length; i++) {
  852. _EworksNumArray.push({
  853. id: _classList[i].id,
  854. name: _classList[i].name,
  855. works: 0,
  856. });
  857. for (let j = 0; j < _studentWorksArray.length; j++) {
  858. const _user = _studentWorksArray[j];
  859. if (_courseIdArray.length == 0) {
  860. if (_user.classid.indexOf(_classList[i].id) != -1) {
  861. _EworksNumArray[i].works++;
  862. _courseIdArray.push(_user.courseid);
  863. }
  864. } else {
  865. if (
  866. _user.classid.indexOf(_classList[i].id) != -1 &&
  867. _courseIdArray.indexOf(_user.courseid) == -1
  868. ) {
  869. _EworksNumArray[i].works++;
  870. }
  871. }
  872. }
  873. }
  874. this.EworksNumArray = _EworksNumArray.filter((el)=>{
  875. return el.works > 0;
  876. });;
  877. let loginCountYear2 = res.data[22]; //一年的在线时长
  878. let loginCountYearArray2 = [];
  879. for (var i = Month; i > Month - 12; i--) {
  880. if (i <= 0) {
  881. loginCountYearArray2.push({
  882. Year: Year - 1,
  883. Month: 12 + i,
  884. mon: 0,
  885. tue: 0,
  886. wed: 0,
  887. thur: 0,
  888. fri: 0,
  889. sat: 0,
  890. sun: 0,
  891. });
  892. } else {
  893. loginCountYearArray2.push({
  894. Month: i,
  895. Year: Year,
  896. mon: 0,
  897. tue: 0,
  898. wed: 0,
  899. thur: 0,
  900. fri: 0,
  901. sat: 0,
  902. sun: 0,
  903. });
  904. }
  905. }
  906. loginCountYearArray2 = loginCountYearArray2.reverse();
  907. for (var i = 0; i < loginCountYear2.length; i++) {
  908. let _date = new Date(loginCountYear2[i].create_at);
  909. var _month = _date.getMonth() + 1;
  910. var _year = _date.getFullYear();
  911. var _day = _date.getDay();
  912. let dayArray = ["sun", "mon", "tue", "wed", "thur", "fri", "sat"];
  913. for (var j = 0; j < loginCountYearArray2.length; j++) {
  914. let time = (loginCountYear2[i].text / 3600).toFixed(0)
  915. time = parseInt(time) ? parseInt(time) : 0
  916. if (
  917. _month == loginCountYearArray2[j].Month &&
  918. _year == loginCountYearArray2[j].Year
  919. ) {
  920. loginCountYearArray2[j][dayArray[_day]]+=time;
  921. break;
  922. }
  923. }
  924. }
  925. console.log(loginCountYearArray2);
  926. this.loginCountYearArray2 = loginCountYearArray2;
  927. let _loginTimeArray2 = res.data[23]; //一年的在线时长
  928. let _EloginTimeArray2 = [];
  929. for (let i = 0; i < _classList.length; i++) {
  930. _EloginTimeArray2.push({
  931. id: _classList[i].id,
  932. name: _classList[i].name,
  933. loginTime: 0,
  934. });
  935. for (let j = 0; j < _loginTimeArray2.length; j++) {
  936. const _user = _loginTimeArray2[j];
  937. if (_user.classid.indexOf(_classList[i].id) != -1) {
  938. _EloginTimeArray2[i].loginTime++;
  939. }
  940. }
  941. }
  942. this.EloginTimeArray2 = _EloginTimeArray2.filter((el)=>{
  943. return el.loginTime > 0;
  944. });
  945. this.$forceUpdate();
  946. })
  947. .catch((err) => {
  948. this.isLoading = false;
  949. console.error(err);
  950. });
  951. },
  952. checkRank(){
  953. this.rankLoading = true;
  954. let params = [
  955. {
  956. oid: this.oid,
  957. org: this.org,
  958. cid: this.cTypeRank
  959. },
  960. ];
  961. this.ajax
  962. .post(this.$store.state.api + "selectDataboardStudentRank", params)
  963. .then((res) => {
  964. this.rankLoading = false;
  965. this.courseType = 2
  966. let rankData = []
  967. let students = res.data[0]
  968. let courses = res.data[1]
  969. let scourses = res.data[2]
  970. let works = res.data[3]
  971. let exchanges = res.data[4]
  972. for (let i = 0; i < students.length; i++) {
  973. rankData.push({
  974. name: students[i].name,
  975. userid: students[i].userid,
  976. classid: students[i].classid,
  977. courseCount:0,
  978. scourseCount: 0,
  979. worksCount: 0,
  980. exchangeCount: 0,
  981. rateCount: 0,
  982. rate: 0,
  983. score:0,
  984. })
  985. }
  986. for(var j = 0; j < rankData.length; j++){
  987. for(var i = 0; i < courses.length; i++){
  988. if(courses[i].userid == rankData[j].userid){
  989. rankData[j].courseCount++;
  990. }
  991. }
  992. for(var i = 0; i < scourses.length; i++){
  993. if(scourses[i].userid == rankData[j].userid){
  994. rankData[j].scourseCount++;
  995. }
  996. }
  997. for(var i = 0; i < works.length; i++){
  998. if(works[i].userid == rankData[j].userid){
  999. rankData[j].worksCount++;
  1000. if(works[i].score){
  1001. try {
  1002. let rate = JSON.parse(works[i].score)
  1003. rankData[j].rateCount += rate.wScore
  1004. rankData[j].rate++
  1005. } catch (error) {
  1006. }
  1007. }
  1008. }
  1009. }
  1010. rankData[j].rateCount = rankData[j].rateCount > 0 ? (rankData[j].rateCount / rankData[j].rate).toFixed(0) : 0
  1011. for(var i = 0; i < exchanges.length; i++){
  1012. if(exchanges[i].likesId == rankData[j].userid){
  1013. rankData[j].exchangeCount++;
  1014. }
  1015. }
  1016. rankData[j].score = ((rankData[j].rateCount * 0.5) + (rankData[j].exchangeCount * 0.1) + (rankData[j].worksCount * 0.2) + (rankData[j].scourseCount * 0.1) + (rankData[j].courseCount * 0.1)).toFixed(0);
  1017. }
  1018. this.rankData = rankData;
  1019. this.$forceUpdate();
  1020. })
  1021. .catch((err) => {
  1022. this.rankLoading = false;
  1023. console.error(err);
  1024. });
  1025. },
  1026. classChange() {
  1027. let _courseNumArray = this.courseNumArray; //参与课程数
  1028. let _scourseNumArray = this.scourseNumArray; //参与项目数
  1029. let _userArray = this.userArray; //参与项目数
  1030. let _courseNum = 0;
  1031. let _scourseNum = 0;
  1032. let uArray = [];
  1033. for (var i = 0; i < _userArray.length; i++) {
  1034. if (_userArray[i].classid.indexOf(this.cType9) != -1 || !this.cType9) {
  1035. uArray.push({
  1036. userid: _userArray[i].userid,
  1037. name: _userArray[i].name,
  1038. courseNum: 0,
  1039. scourseNum: 0,
  1040. });
  1041. }
  1042. }
  1043. for (let i = 0; i < _courseNumArray.length; i++) {
  1044. for (let j = 0; j < uArray.length; j++) {
  1045. if (_courseNumArray[i].userid == uArray[j].userid) {
  1046. uArray[j].courseNum++;
  1047. _courseNum++;
  1048. break;
  1049. }
  1050. }
  1051. }
  1052. for (let i = 0; i < _scourseNumArray.length; i++) {
  1053. for (let j = 0; j < uArray.length; j++) {
  1054. if (_scourseNumArray[i].userid == uArray[j].userid) {
  1055. uArray[j].scourseNum++;
  1056. _scourseNum++;
  1057. break;
  1058. }
  1059. }
  1060. }
  1061. this.courseNum = _courseNum;
  1062. this.scourseNum = _scourseNum;
  1063. this.uArray = uArray;
  1064. },
  1065. },
  1066. };
  1067. </script>
  1068. <style scoped>
  1069. .body1 {
  1070. width: 100%;
  1071. height: 100%;
  1072. display: flex;
  1073. padding: 20px;
  1074. box-sizing: border-box;
  1075. overflow: hidden;
  1076. }
  1077. .left {
  1078. width: calc(100% / 4 * 1);
  1079. height: 100%;
  1080. }
  1081. .left > .top {
  1082. width: 100%;
  1083. height: calc(100% / 2 - 10px);
  1084. background: #fff;
  1085. border-radius: 5px;
  1086. margin: 0 0 20px 0;
  1087. }
  1088. .left > .bottom {
  1089. width: 100%;
  1090. height: calc(100% / 2 - 10px);
  1091. background: #fff;
  1092. border-radius: 5px;
  1093. }
  1094. .center {
  1095. width: calc(100% / 4 * 2 - 40px);
  1096. height: 100%;
  1097. margin: 0 20px;
  1098. }
  1099. .center > .top {
  1100. width: 100%;
  1101. height: calc(100% / 6 * 4 - 10px);
  1102. background: #fff;
  1103. border-radius: 5px;
  1104. margin: 0 0 20px 0;
  1105. }
  1106. .center > .bottom {
  1107. width: 100%;
  1108. height: calc(100% / 6 * 2 - 10px);
  1109. background: #fff;
  1110. border-radius: 5px;
  1111. }
  1112. .right {
  1113. width: calc(100% / 4 * 1);
  1114. height: 100%;
  1115. }
  1116. .right > .top {
  1117. width: 100%;
  1118. height: calc(100% / 2 - 10px);
  1119. background: #fff;
  1120. border-radius: 5px;
  1121. margin: 0 0 20px 0;
  1122. }
  1123. .right > .bottom {
  1124. width: 100%;
  1125. height: calc(100% / 2 - 10px);
  1126. background: #fff;
  1127. border-radius: 5px;
  1128. }
  1129. .dataBox {
  1130. height: calc(100% - 40px);
  1131. width: 100%;
  1132. }
  1133. .titleBox {
  1134. height: 40px;
  1135. display: flex;
  1136. align-items: center;
  1137. padding: 0 15px;
  1138. box-sizing: border-box;
  1139. width: 100%;
  1140. }
  1141. .title {
  1142. color: #060e17;
  1143. margin-right: 10px;
  1144. }
  1145. .dataBox {
  1146. height: calc(100% - 40px);
  1147. width: 100%;
  1148. }
  1149. .greenBG {
  1150. background: linear-gradient(
  1151. 180deg,
  1152. rgb(211, 246, 228, 0.2) 0%,
  1153. rgb(23, 196, 105, 0.3) 100%
  1154. );
  1155. }
  1156. .depth_box {
  1157. display: flex;
  1158. flex-wrap: wrap;
  1159. height: 100%;
  1160. width: 95%;
  1161. margin: 0 auto;
  1162. overflow: hidden;
  1163. justify-content: space-between;
  1164. }
  1165. .depth {
  1166. width: calc(100% / 5 - 10px);
  1167. display: flex;
  1168. flex-direction: column;
  1169. align-items: center;
  1170. justify-content: center;
  1171. }
  1172. .depth > span:nth-child(1) {
  1173. font-size: 14px;
  1174. font-weight: 700;
  1175. margin: 0 0 10px;
  1176. }
  1177. .depth > div:nth-child(1) {
  1178. }
  1179. .selectBox {
  1180. width: 80px;
  1181. margin-left: 10px;
  1182. }
  1183. .selectBox >>> .el-input__inner {
  1184. height: 30px;
  1185. line-height: 30px;
  1186. }
  1187. .selectBox >>> .el-input__icon {
  1188. line-height: 30px;
  1189. }
  1190. .teaMiddle {
  1191. width: calc(100% / 2 - 10px);
  1192. height: 60px;
  1193. border-radius: 8px;
  1194. /* border: 1px solid #e0eafb; */
  1195. display: flex;
  1196. flex-direction: column;
  1197. align-items: flex-start;
  1198. justify-content: center;
  1199. padding: 0 10px;
  1200. margin: 0 10px;
  1201. }
  1202. .teaMiddle {
  1203. width: calc(100% / 3 - 10px);
  1204. }
  1205. .timeDiv {
  1206. display: flex;
  1207. flex-direction: row;
  1208. flex-wrap: nowrap;
  1209. align-items: center;
  1210. margin: 0 0 0 15px;
  1211. }
  1212. .timeDiv > div {
  1213. margin-right: 10px;
  1214. cursor: pointer;
  1215. }
  1216. .isClick {
  1217. color: #1684fc;
  1218. border-bottom: 2px solid #1684fc;
  1219. box-sizing: border-box;
  1220. }
  1221. .otherCss {
  1222. display: flex;
  1223. flex-direction: row;
  1224. flex-wrap: nowrap;
  1225. align-items: center;
  1226. justify-content: flex-end;
  1227. }
  1228. .otherImg {
  1229. width: 20px;
  1230. height: 20px;
  1231. margin: 0 10px;
  1232. cursor: pointer;
  1233. }
  1234. .otherImg > img {
  1235. width: 100%;
  1236. height: 100%;
  1237. }
  1238. .allBox {
  1239. width: 100%;
  1240. height: 100%;
  1241. display: flex;
  1242. flex-direction: row;
  1243. flex-wrap: nowrap;
  1244. align-items: center;
  1245. }
  1246. .allBox_left {
  1247. width: 30%;
  1248. height: 100%;
  1249. }
  1250. .allBox_right {
  1251. display: flex;
  1252. flex-wrap: wrap;
  1253. height: 100%;
  1254. width: 70%;
  1255. margin: 0 auto;
  1256. overflow: hidden;
  1257. justify-content: space-between;
  1258. }
  1259. .depth {
  1260. width: calc(100% / 4 - 10px);
  1261. display: flex;
  1262. flex-direction: column;
  1263. align-items: center;
  1264. justify-content: center;
  1265. }
  1266. .depth > span:nth-child(1) {
  1267. font-size: 14px;
  1268. font-weight: 700;
  1269. margin: 0 0 10px;
  1270. }
  1271. .depth > div:nth-child(1) {
  1272. }
  1273. .info_box {
  1274. display: flex;
  1275. flex-wrap: wrap;
  1276. align-items: center;
  1277. justify-content: space-between;
  1278. width: 90%;
  1279. margin: 0 auto;
  1280. }
  1281. .info_box > .info2,
  1282. .info_box > .info3,
  1283. .info_box > .info {
  1284. width: calc(50% - 10px);
  1285. display: flex;
  1286. flex-direction: row;
  1287. flex-wrap: nowrap;
  1288. height: 60px;
  1289. justify-content: space-between;
  1290. align-items: center;
  1291. padding: 0 10px;
  1292. box-sizing: border-box;
  1293. margin-bottom: 10px;
  1294. border-radius: 5px;
  1295. }
  1296. .info_box > .info2 {
  1297. width: calc(100% / 4 - 10px);
  1298. /* align-items: flex-end; */
  1299. }
  1300. .info_box > .info3 {
  1301. width: 100%;
  1302. margin-bottom: 5px;
  1303. }
  1304. .info_box > .info2 > span:nth-child(1),
  1305. .info_box > .info3 > span:nth-child(1),
  1306. .info_box > .info > span:nth-child(1) {
  1307. font-size: 12px;
  1308. /* margin: 0 0 0 20px; */
  1309. color: #565e6a;
  1310. width: 60px;
  1311. white-space: pre-wrap;
  1312. word-break: break-all;
  1313. }
  1314. .info_box > .info2 > span:nth-child(2),
  1315. .info_box > .info3 > span:nth-child(2),
  1316. .info_box > .info > span:nth-child(2) {
  1317. font-size: 24px;
  1318. /* font-weight: 700; */
  1319. }
  1320. .blueBG {
  1321. background: rgb(243, 248, 253);
  1322. border: 2px solid rgb(234, 246, 255);
  1323. }
  1324. </style>