myReport.vue 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632
  1. <template>
  2. <div :class="{ isWorkCss: reportVisible }">
  3. <div class="sr_body">
  4. <div class="mrTitle">
  5. <div>我的学习报告</div>
  6. <div class="mrIcon">
  7. <img src="../../../assets/icon/myReport/tIcon.png" alt="" />
  8. </div>
  9. </div>
  10. <div class="sr_box">
  11. <div class="sr_first">
  12. <div class="first" style="height: 260px">
  13. <div class="sub_title">
  14. <div class="yuan">
  15. <img src="../../../assets/icon/myReport/yuan.png" alt="" />
  16. </div>
  17. <div>基础信息</div>
  18. </div>
  19. <MrBasicData class="r_box" :sMes="stuMessage"></MrBasicData>
  20. </div>
  21. <div class="second" style="height: 260px">
  22. <div class="sub_title">
  23. <div class="yuan">
  24. <img src="../../../assets/icon/myReport/yuan.png" alt="" />
  25. </div>
  26. <div>我的综合表现</div>
  27. </div>
  28. <MrOverPer
  29. class="r_box"
  30. :overPData="JSON.parse(JSON.stringify(overPer))"
  31. ></MrOverPer>
  32. </div>
  33. <div class="third">
  34. <div class="sub_title">
  35. <div class="yuan">
  36. <img src="../../../assets/icon/myReport/yuan.png" alt="" />
  37. </div>
  38. <div>我的学习状态</div>
  39. </div>
  40. <MrLearnStatus class="r_box"></MrLearnStatus>
  41. </div>
  42. </div>
  43. <div class="sr_second" style="height: 830px">
  44. <div class="first" style="margin: 0; justify-content: flex-start">
  45. <div class="sub_title">
  46. <div class="yuan">
  47. <img src="../../../assets/icon/myReport/yuan.png" alt="" />
  48. </div>
  49. <div>我的课程报告</div>
  50. </div>
  51. <MyCourseReport class="r_box"></MyCourseReport>
  52. </div>
  53. </div>
  54. <div class="sr_third">
  55. <div class="first">
  56. <div class="sub_title">
  57. <div class="yuan">
  58. <img src="../../../assets/icon/myReport/yuan.png" alt="" />
  59. </div>
  60. <div>课程评价得分</div>
  61. </div>
  62. <CourseEvaScore
  63. class="r_box"
  64. style="
  65. display: flex;
  66. flex-direction: row;
  67. flex-wrap: nowrap;
  68. align-items: center;
  69. justify-content: center;
  70. margin: 0;
  71. "
  72. :courseArray="cEvaScore"
  73. ></CourseEvaScore>
  74. </div>
  75. <div class="second">
  76. <div class="sub_title">
  77. <div class="yuan">
  78. <img src="../../../assets/icon/myReport/yuan.png" alt="" />
  79. </div>
  80. <div>课程类型分析</div>
  81. </div>
  82. <CourseTypeAna
  83. class="r_box"
  84. :courseTypeArray="cTypeAnaList"
  85. ></CourseTypeAna>
  86. </div>
  87. <div class="third" style="height: 270px">
  88. <div class="sub_title">
  89. <div class="yuan">
  90. <img src="../../../assets/icon/myReport/yuan.png" alt="" />
  91. </div>
  92. <div>工具类型分析</div>
  93. </div>
  94. <ToolTypeAna
  95. class="r_box"
  96. :toolTypeArray="tTypeArray"
  97. ></ToolTypeAna>
  98. </div>
  99. </div>
  100. </div>
  101. <div style="height: 100%; overflow-x: hidden">
  102. <div>
  103. <MrEva class="r_box" :mrEvaList="evaList" @getEvaCid="getCid"></MrEva>
  104. </div>
  105. </div>
  106. </div>
  107. <div class="report_box" v-if="reportVisible">
  108. <myReportLook
  109. :checkCourse="checkCourse"
  110. :checkStudent="checkStudent"
  111. :oid="oid"
  112. ></myReportLook>
  113. </div>
  114. <div class="cancelbox" v-if="reportVisible">
  115. <el-button @click="cancelR" type="primary" size="small">返回</el-button>
  116. </div>
  117. </div>
  118. </template>
  119. <script>
  120. import MrBasicData from "./components/mrBasicData.vue";
  121. import MrOverPer from "./components/mrOverPer.vue";
  122. import MyCourseReport from "./components/myCourseReport.vue";
  123. import MrLearnStatus from "./components/mrLearnStatus.vue";
  124. import CourseEvaScore from "./components/courseEvaScore.vue";
  125. import CourseTypeAna from "./components/courseTypeAna.vue";
  126. import ToolTypeAna from "./components/toolTypeAna.vue";
  127. import MrEva from "./components/mrEva.vue";
  128. import myReportLook from "./components/MyLookComponent/myReportLook";
  129. export default {
  130. components: {
  131. MrBasicData,
  132. MrOverPer,
  133. MyCourseReport,
  134. MrLearnStatus,
  135. CourseEvaScore,
  136. CourseTypeAna,
  137. ToolTypeAna,
  138. MrEva,
  139. myReportLook,
  140. },
  141. data() {
  142. return {
  143. userid: this.$route.query.userid,
  144. org: this.$route.query.org,
  145. oid: this.$route.query.oid,
  146. reportVisible: false,
  147. checkStudent: "",
  148. checkCourse: [],
  149. evaList: [], //目标体系
  150. stuMessage: {}, //基础信息
  151. cEvaScore: [], //课程评价得分
  152. cTypeAnaList: [], //课程类型分析
  153. tTypeArray: [], //工具类型分析
  154. overPer: {}, //我的综合表现
  155. };
  156. },
  157. methods: {
  158. getCid(item) {
  159. // var b = [
  160. // { isChoose: "4cf01620-08d9-11ed-8c78-005056b86db5" },
  161. // { isChoose: "02ea6cdc-e7e3-11ec-8c78-005056b86db5" },
  162. // ];
  163. var a = [];
  164. for (var i = 0; i < item.isChoose.length; i++) {
  165. a.push(item.isChoose[i]);
  166. }
  167. // for (var i = 0; i < b.length; i++) {
  168. // a.push(b[i].isChoose);
  169. // }
  170. this.checkCourse = a.join(",");
  171. this.checkStudent = this.userid;
  172. this.reportVisible = true;
  173. },
  174. cancelR() {
  175. this.checkCourse = "";
  176. this.checkStudent = "";
  177. this.reportVisible = false;
  178. },
  179. getMrEva() {
  180. let params = {
  181. uid: this.userid,
  182. oid: this.oid,
  183. };
  184. this.ajax
  185. .get(this.$store.state.api + "selectMrEva", params)
  186. .then((res) => {
  187. this.stuMessage = res.data[0][0];
  188. this.evaList = res.data[1];
  189. var courseList = res.data[2];
  190. var sWorksList = res.data[3];
  191. var studyAllTime = res.data[4];
  192. var loginAllTime = res.data[5];
  193. var courseRateList = res.data[6];
  194. var sWorksCList = [];
  195. var cList = [];
  196. var cTypeList = [];
  197. var tool = 0;
  198. var isFinish = 0;
  199. var scoreXarray = [];
  200. for (var i = 0; i < this.evaList.length; i++) {
  201. this.evaList[i].course = [];
  202. this.evaList[i].isChoose = [];
  203. for (var j = 0; j < courseList.length; j++) {
  204. if (this.evaList[i].id == courseList[j].evaId) {
  205. this.evaList[i].course.push(courseList[j]);
  206. }
  207. }
  208. }
  209. for (var i = 0; i < courseList.length; i++) {
  210. cList.push(JSON.parse(courseList[i].chapters));
  211. scoreXarray.push(courseList[i].title);
  212. if (courseList[i].typename != null) {
  213. cTypeList.push(courseList[i].typename.split("/"));
  214. }
  215. }
  216. for (var i = 0; i < courseRateList.length; i++) {
  217. sWorksCList.push({
  218. courseid: courseRateList[i].courseId,
  219. title: courseRateList[i].title,
  220. rate: courseRateList[i].rate,
  221. });
  222. }
  223. var count = 0;
  224. for (var i = 0; i < this.evaList.length; i++) {
  225. count += this.evaList[i].course.length;
  226. }
  227. this.stuMessage.allCourseNum = count;
  228. for (var i = 0; i < cList.length; i++) {
  229. for (var j = 0; j < cList[i].length; j++) {
  230. for (
  231. var k = 0;
  232. k < cList[i][j].chapterInfo[0].taskJson.length;
  233. k++
  234. ) {
  235. let _toolsAarry = [
  236. 1, 3, 6, 7, 16, 15, 4, 40, 41, 42, 49, 50, 52, 32, 57,
  237. ];
  238. for (
  239. var q = 0;
  240. q < cList[i][j].chapterInfo[0].taskJson[k].toolChoose.length;
  241. q++
  242. ) {
  243. if (
  244. _toolsAarry.indexOf(
  245. cList[i][j].chapterInfo[0].taskJson[k].toolChoose[q]
  246. .tool[0]
  247. ) != -1
  248. ) {
  249. tool++;
  250. }
  251. }
  252. }
  253. }
  254. }
  255. var toolList = [
  256. [10, 49], //互动类
  257. [7, 1, 52, 3], //思维类
  258. [4, 45, 15, 16, 50, 40, 41, 47, 48], //评价类
  259. [18, 21, 22, 23, 24, 32, 57], //编程类
  260. [28, 37, 38, 31, 39, 58, 59, 60], //学科类
  261. [26, 25], //其他
  262. ];
  263. var toolAllArray = [];
  264. for (var p = 0; p < toolList.length; p++) {
  265. toolAllArray[p] = [];
  266. for (var i = 0; i < cList.length; i++) {
  267. for (var j = 0; j < cList[i].length; j++) {
  268. for (
  269. var k = 0;
  270. k < cList[i][j].chapterInfo[0].taskJson.length;
  271. k++
  272. ) {
  273. for (
  274. var q = 0;
  275. q <
  276. cList[i][j].chapterInfo[0].taskJson[k].toolChoose.length;
  277. q++
  278. ) {
  279. if (
  280. toolList[p].indexOf(
  281. cList[i][j].chapterInfo[0].taskJson[k].toolChoose[q]
  282. .tool[0]
  283. ) != -1
  284. ) {
  285. toolAllArray[p].push(
  286. cList[i][j].chapterInfo[0].taskJson[k].toolChoose[q]
  287. .tool[0]
  288. );
  289. }
  290. }
  291. }
  292. }
  293. }
  294. }
  295. var tTypeAnaArray = [];
  296. for (var i = 0; i < toolAllArray.length; i++) {
  297. tTypeAnaArray.push(toolAllArray[i].length);
  298. }
  299. this.tTypeArray.push(tTypeAnaArray);
  300. for (var i = 0; i < sWorksList.length; i++) {
  301. for (var j = 0; j < courseList.length; j++) {
  302. if (sWorksList[i].courseid == courseList[j].courseId) {
  303. isFinish++;
  304. }
  305. }
  306. }
  307. var allToolWorks = tool;
  308. this.stuMessage.isFinishTaskWorks = isFinish;
  309. this.stuMessage.noFinishTaskWorks = allToolWorks - isFinish;
  310. let overPer = {};
  311. overPer.toolSubRate =
  312. (this.stuMessage.isFinishTaskWorks / allToolWorks).toFixed(2) * 100;
  313. var xArray = [];
  314. for (var i = 0; i < sWorksCList.length; i++) {
  315. if (xArray.length == 0) {
  316. xArray.push({ name: sWorksCList[i].title, value: 0 });
  317. } else {
  318. if (xArray.indexOf(sWorksCList[i].title) == -1) {
  319. xArray.push({ name: sWorksCList[i].title, value: 0 });
  320. }
  321. }
  322. }
  323. for (var i = 0; i < xArray.length; i++) {
  324. for (var j = 0; j < sWorksCList.length; j++) {
  325. if (xArray[i].name == sWorksCList[j].title) {
  326. var rateList = JSON.parse(sWorksCList[j].rate);
  327. var a = Object.keys(rateList);
  328. var courseCount = 0;
  329. for (var k = 0; k < a.length; k++) {
  330. if (a[k] == "content") {
  331. continue;
  332. } else {
  333. courseCount += rateList[a[k]];
  334. }
  335. }
  336. xArray[j].value = Math.round(courseCount / (a.length - 1));
  337. }
  338. }
  339. }
  340. var yzArray = [];
  341. for (var i = 0; i < scoreXarray.length; i++) {
  342. var yCount = 0;
  343. for (var j = 0; j < xArray.length; j++) {
  344. if (scoreXarray[i] == xArray[j].name) {
  345. yCount += xArray[j].value;
  346. }
  347. }
  348. yzArray.push({ name: scoreXarray[i], value: yCount });
  349. }
  350. yzArray.sort(function (a, b) {
  351. return b.value - a.value;
  352. });
  353. this.cEvaScore = yzArray;
  354. var ctArray = [];
  355. var cxtArray = [];
  356. for (var i = 0; i < cTypeList.length; i++) {
  357. for (var j = 0; j < cTypeList[i].length; j++) {
  358. if (ctArray.length == 0) {
  359. ctArray.push(cTypeList[i][j]);
  360. } else {
  361. if (ctArray.indexOf(cTypeList[i][j]) == -1) {
  362. ctArray.push(cTypeList[i][j]);
  363. }
  364. }
  365. cxtArray.push(cTypeList[i][j]);
  366. }
  367. }
  368. const result = ctArray
  369. .filter((item) => cxtArray.includes(item))
  370. .map((item) => ({
  371. count: cxtArray.filter((i) => i === item).length,
  372. }));
  373. var tArray = [];
  374. for (var i = 0; i < result.length; i++) {
  375. tArray.push(result[i].count);
  376. }
  377. var tuples1 = ctArray.map((label, index) => [label, tArray[index]]);
  378. // 根据数据降序排序二元组
  379. tuples1.sort((a, b) => b[1] - a[1]);
  380. // 提取已排序的标签
  381. var txArrayNew = tuples1.map((tuple) => tuple[0]);
  382. var tarrayNew = tuples1.map((tuple) => tuple[1]);
  383. var subjectList = [
  384. "语文",
  385. "数学",
  386. "英语",
  387. "科学",
  388. "信息技术",
  389. "物理",
  390. "化学",
  391. "生物",
  392. "历史",
  393. "地理",
  394. "通用技术",
  395. "政治",
  396. "STEM",
  397. "美术",
  398. "其他",
  399. ];
  400. for (var i = 0; i < txArrayNew.length; i++) {
  401. if (subjectList.indexOf(txArrayNew[i]) != -1) {
  402. this.cTypeAnaList.push({
  403. name: txArrayNew[i],
  404. value: tarrayNew[i],
  405. });
  406. }
  407. }
  408. var lTime = 0;
  409. for (var i = 0; i < loginAllTime.length; i++) {
  410. lTime += parseInt(loginAllTime[i].text);
  411. }
  412. var days = Math.floor(lTime / (3600 * 24));
  413. lTime %= 3600 * 24;
  414. var hours = Math.floor(lTime / 3600);
  415. lTime %= 3600;
  416. var minutes = Math.floor(lTime / 60);
  417. lTime %= 60;
  418. if (days > 0) {
  419. overPer.loginTime = `${days}天${hours}小时`;
  420. } else if (days < 0 && hours > 0) {
  421. overPer.loginTime = `${hours}小时${minutes}分钟`;
  422. } else if (days == 0 && hours == 0 && minutes > 0) {
  423. overPer.loginTime = `${minutes}分钟${loginAllTime}秒`;
  424. } else {
  425. overPer.loginTime = `${loginAllTime}秒`;
  426. }
  427. var sTime = 0;
  428. for (var i = 0; i < studyAllTime.length; i++) {
  429. sTime += parseInt(studyAllTime[i].text);
  430. }
  431. var days1 = Math.floor(sTime / (3600 * 24));
  432. sTime %= 3600 * 24;
  433. var hours1 = Math.floor(sTime / 3600);
  434. sTime %= 3600;
  435. var minutes1 = Math.floor(sTime / 60);
  436. sTime %= 60;
  437. if (days1 > 0) {
  438. overPer.studyTime = `${days1}天${hours1}小时`;
  439. } else if (days1 < 0 && hours1 > 0) {
  440. overPer.studyTime = `${hours1}小时${minutes1}分钟`;
  441. } else if (days1 == 0 && hours1 == 0 && minutes1 > 0) {
  442. overPer.studyTime = `${minutes1}分钟${sTime}秒`;
  443. } else {
  444. overPer.studyTime = `${sTime}秒`;
  445. }
  446. overPer.sName = this.stuMessage.name;
  447. var courseNum = 0;
  448. for (var i = 0; i < yzArray.length; i++) {
  449. courseNum += yzArray[i].value;
  450. }
  451. overPer.courseaScore = Math.floor(courseNum / yzArray.length);
  452. this.overPer = overPer;
  453. this.$forceUpdate();
  454. })
  455. .catch((err) => {
  456. console.error(err);
  457. });
  458. },
  459. },
  460. created() {
  461. this.getMrEva();
  462. },
  463. };
  464. </script>
  465. <style scoped>
  466. .sr_head {
  467. color: rgb(21, 80, 183);
  468. font-size: 30px;
  469. font-weight: bolder;
  470. text-align: center;
  471. margin-bottom: 15px;
  472. }
  473. .sr_body {
  474. width: 95%;
  475. margin: 0 auto;
  476. height: calc(100% - 55px);
  477. }
  478. .sr_box {
  479. width: 100%;
  480. height: 100%;
  481. display: flex;
  482. min-width: 1100px;
  483. min-height: 700px;
  484. }
  485. .sr_first {
  486. width: calc(100% / 4);
  487. height: 100%;
  488. }
  489. .sr_second {
  490. width: calc(100% / 3.75 * 2);
  491. height: 100%;
  492. }
  493. .sr_third {
  494. width: calc(100% / 3.75);
  495. height: 100%;
  496. }
  497. .sr_first,
  498. .sr_second,
  499. .sr_third {
  500. display: flex;
  501. flex-direction: column;
  502. align-items: center;
  503. }
  504. .sr_first .first,
  505. .sr_first .second,
  506. .sr_first .third,
  507. .sr_second .first,
  508. .sr_second .second,
  509. .sr_third .first,
  510. .sr_third .second,
  511. .sr_third .third {
  512. background: #fff;
  513. border-radius: 10px;
  514. width: 98%;
  515. overflow: hidden;
  516. box-shadow: 0 0 10px 3px #cad1d9;
  517. }
  518. .sr_first .first {
  519. height: calc(100% / 3);
  520. margin-bottom: 20px;
  521. }
  522. .sr_first .second {
  523. height: calc(100% / 3);
  524. margin-bottom: 20px;
  525. }
  526. .sr_first .third {
  527. height: calc(100% / 3);
  528. }
  529. .sr_second .first {
  530. /* height: calc(100% / 3.5 * 2); */
  531. height: calc(100%);
  532. /* margin-bottom: 20px; */
  533. }
  534. .sr_second .second {
  535. /* height: calc(100% / 3.5 * 1.5); */
  536. }
  537. .sr_third .first {
  538. height: 260px;
  539. margin-bottom: 20px;
  540. }
  541. .sr_third .second {
  542. height: 260px;
  543. margin-bottom: 20px;
  544. }
  545. .sr_third .third {
  546. height: 260px;
  547. }
  548. .sub_title {
  549. display: flex;
  550. color: #3050c2;
  551. font-weight: bold;
  552. align-items: center;
  553. justify-content: center;
  554. height: 40px;
  555. width: 200px;
  556. margin: 0 auto;
  557. flex-direction: row;
  558. flex-wrap: nowrap;
  559. margin-top: 5px;
  560. }
  561. .r_box {
  562. height: calc(100% - 40px);
  563. width: 100%;
  564. }
  565. .mrTitle {
  566. display: flex;
  567. flex-direction: row;
  568. flex-wrap: nowrap;
  569. width: 100%;
  570. justify-content: center;
  571. align-items: flex-start;
  572. margin: 10px 0;
  573. }
  574. .mrTitle > div:nth-child(1) {
  575. font-size: 22px;
  576. }
  577. .mrIcon {
  578. width: 30px;
  579. margin-left: 5px;
  580. }
  581. .yuan {
  582. width: 30px;
  583. height: 30px;
  584. }
  585. .mrIcon > img,
  586. .yuan > img {
  587. width: 100%;
  588. height: 100%;
  589. }
  590. .cancelbox {
  591. position: absolute;
  592. z-index: 2;
  593. left: 50%;
  594. top: 20px;
  595. width: 95%;
  596. transform: translateX(-50%);
  597. display: flex;
  598. justify-content: flex-end;
  599. padding: 0 90px 0px 0px;
  600. box-sizing: border-box;
  601. }
  602. .report_box {
  603. height: 100%;
  604. position: absolute;
  605. top: 0;
  606. /* background: #fff; */
  607. background: rgb(231, 242, 252);
  608. overflow: auto;
  609. z-index: 1;
  610. width: 100%;
  611. left: 50%;
  612. transform: translateX(-50%);
  613. padding: 20px;
  614. box-sizing: border-box;
  615. }
  616. .isWorkCss {
  617. width: 100%;
  618. height: 100%;
  619. }
  620. </style>