report.vue 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556
  1. <template>
  2. <div class="report">
  3. <div
  4. style="background-color: #fff;width: 100%;height: 100%;box-sizing: border-box; padding: 15px;"
  5. >
  6. <div class="top">
  7. <div style="width: 348px;height: 40px;flex-shrink: 0;">
  8. <el-select
  9. v-model="year"
  10. @change="getData"
  11. style="width: 100%;"
  12. placeholder="请选择"
  13. >
  14. <el-option
  15. v-for="item in termList"
  16. :key="item.id"
  17. :label="item.name"
  18. :value="item.id"
  19. >
  20. </el-option>
  21. </el-select>
  22. </div>
  23. <!-- <div class="topBtnS">
  24. <div class="btn">生成pdf</div>
  25. <div class="btn">分享报告</div>
  26. </div> -->
  27. </div>
  28. <div style="margin: 20px 10px;">学期综合表现</div>
  29. <div
  30. style="font-size: 20px;width: 100%;height: 200px;display: flex;justify-content: center;"
  31. v-if="isShow"
  32. >
  33. 暂无数据
  34. </div>
  35. <div class="content" v-if="!isShow">
  36. <!-- <div class="conTitle" >
  37. <div>综合评价:</div>
  38. <div class="txt">表现优异,再接再厉!</div>
  39. </div> -->
  40. <div style="display: flex;width: 100%;justify-content: space-between;">
  41. <div class="radarCss">
  42. <radar
  43. ref="radar"
  44. :chartData="chartData"
  45. :categories="categories"
  46. :key="chartData.toString()"
  47. ></radar>
  48. </div>
  49. <div class="radarCss">
  50. <croColumnar
  51. ref="cro"
  52. :chartData="croColumnarData"
  53. :categories="categories"
  54. :key="croColumnarData.toString()"
  55. ></croColumnar>
  56. </div>
  57. </div>
  58. </div>
  59. <div></div>
  60. </div>
  61. <div v-if="!isShow">
  62. <div v-for="(i, index) in VeidooList" :key="i.id">
  63. <fieldMap
  64. :tid="i.id"
  65. :tit="i.name"
  66. :userid="userid"
  67. :year="year"
  68. :radarData="radarData"
  69. :key="radarData.toString()"
  70. ></fieldMap>
  71. </div>
  72. </div>
  73. </div>
  74. </template>
  75. <script>
  76. import radar from "./component/radar.vue";
  77. import croColumnar from "./component/croColumnar";
  78. // import verColumnar from "./component/verColumnar";
  79. import popTable from "./component/popTable";
  80. import fieldMap from "./component/fieldMap";
  81. export default {
  82. components: {
  83. radar,
  84. croColumnar,
  85. // verColumnar,
  86. popTable,
  87. fieldMap
  88. },
  89. props: {
  90. userid: {
  91. type: String
  92. },
  93. oid: {
  94. type: String
  95. },
  96. org: {
  97. type: String
  98. }
  99. },
  100. data() {
  101. return {
  102. year: "",
  103. // 学期数组
  104. termList: [],
  105. // dialogVisible: false,
  106. // 平均分柱状图数据
  107. croColumnarData: [1],
  108. // 雷达图数据
  109. chartData: [], //数据
  110. categories: ["1"], //标签 雷达图与平均分共用
  111. value: "",
  112. VeidooList: [], //大分类
  113. VeidooJsonList: [], //小分类
  114. // 后端获取的数据
  115. radarData: {},
  116. // 判断无数据的显示
  117. isShow: false
  118. };
  119. },
  120. methods: {
  121. //获取分类
  122. getVeidooType() {
  123. let params = {
  124. org: this.org,
  125. oid: this.oid
  126. };
  127. this.ajax
  128. .get(this.$store.state.api + "selectSTEType", params)
  129. .then(res => {
  130. var ftype = res.data[0]; //公共父级分类
  131. var stype = res.data[1]; //公共子级分类
  132. var sctype = res.data[2]; //该学校子级分类
  133. var fctype = res.data[3]; //该学校父级分类
  134. var fotype = res.data[4]; //组织父级分类
  135. var sotype = res.data[5]; //组织子级分类
  136. var allfType = [];
  137. var allsType = [];
  138. if (fotype.length == 0 && sotype.length == 0) {
  139. if (fctype.length == 0 && sctype.length == 0) {
  140. for (var i = 0; i < ftype.length; i++) {
  141. allfType.push(ftype[i]);
  142. }
  143. for (var i = 0; i < stype.length; i++) {
  144. allsType.push(stype[[i]]);
  145. }
  146. } else {
  147. for (var i = 0; i < fctype.length; i++) {
  148. allfType.push(fctype[i]);
  149. }
  150. for (var i = 0; i < sctype.length; i++) {
  151. allsType.push(sctype[[i]]);
  152. }
  153. }
  154. } else {
  155. for (var i = 0; i < fotype.length; i++) {
  156. allfType.push(fotype[i]);
  157. }
  158. for (var i = 0; i < sotype.length; i++) {
  159. allsType.push(sctysotypepe[[i]]);
  160. }
  161. }
  162. var VeidooJson = {};
  163. for (var i = 0; i < allfType.length; i++) {
  164. this.ftypeId = allfType[0].id;
  165. if (!VeidooJson[allfType[i].id]) {
  166. VeidooJson[allfType[i].id] = [];
  167. }
  168. for (var j = 0; j < allsType.length; j++) {
  169. if (allfType[i].id == allsType[j].pid) {
  170. VeidooJson[allfType[i].id].push(allsType[j]); // 去除公共分类
  171. }
  172. }
  173. }
  174. this.VeidooList = allfType;
  175. this.VeidooJsonList = VeidooJson;
  176. console.log(" allfType", allfType);
  177. // console.log(" allsType", allsType);
  178. // this.$forceUpdate();
  179. let aaa = [];
  180. // 大分类标签
  181. allfType.forEach(e => {
  182. // console.log(e.name);
  183. aaa.push(e.name);
  184. });
  185. // 数据分类底部标签
  186. this.categories = aaa;
  187. // 雷达图数据
  188. this.countRadar();
  189. // 柱状图
  190. this.countCro();
  191. // 健康横向柱状图
  192. // this.healthCro();
  193. // 健康纵向柱状图
  194. // this.healthVer(allfType);
  195. })
  196. .catch(err => {
  197. this.isLoading = false;
  198. console.error(err);
  199. });
  200. },
  201. // 获取学期
  202. getYear() {
  203. this.isLoading = true;
  204. this.ajax
  205. .get(this.$store.state.api + "selectTerm")
  206. .then(res => {
  207. this.isLoading = false;
  208. var yearJuri = res.data[0];
  209. for (var i = 0; i < yearJuri.length; i++) {
  210. if (yearJuri[i].defaultC == 1) {
  211. this.year = yearJuri[i].id;
  212. }
  213. }
  214. this.termList = yearJuri;
  215. this.getData();
  216. this.getEvidence();
  217. })
  218. .catch(err => {
  219. this.isLoading = false;
  220. console.error(err);
  221. });
  222. },
  223. // 获取关键证据数据
  224. getEvidence() {
  225. // this.isLoading = true;
  226. let params = {
  227. uid: this.userid,
  228. year: this.year,
  229. txt: "c0f39c82-b34b-11ee-b534-005056b86db5",
  230. page: 1
  231. };
  232. this.ajax
  233. .get(this.$store.state.api + "selectHealthRecord", params)
  234. .then(res => {
  235. // console.log("获取关键证据数据", res);
  236. this.healthEvidence = res.data[0];
  237. }); //
  238. },
  239. // 获取打分数据
  240. getData() {
  241. // this.isLoading = true;
  242. let params = {
  243. uid: this.userid,
  244. year: this.year
  245. };
  246. // console.log("params", params);
  247. this.ajax
  248. .get(this.$store.state.api + "selectMapStuScore", params)
  249. .then(res => {
  250. // console.log("获取打分数据", res);
  251. this.isLoading = false;
  252. if (!res.data[0].length) {
  253. // this.clearRadarData();
  254. this.isShow = true;
  255. return;
  256. }
  257. this.isShow = false;
  258. let data = res.data[0][0];
  259. // console.log(JSON.parse(data.json));
  260. this.radarData = JSON.parse(data.json);
  261. console.log("this.radarData", this.radarData);
  262. this.getVeidooType();
  263. // this.getEvidence();
  264. // console.log(JSON.s);
  265. // this.$emit("selectData");
  266. });
  267. },
  268. // // 查询不到数据,清除内容
  269. // clearRadarData() {
  270. // this.isShow = true;
  271. // // 平均分柱状图数据
  272. // this.croColumnarData = [];
  273. // // 雷达图数据
  274. // this.chartData = []; //数据
  275. // this.categories = []; //标签 雷达图与平均分共用
  276. // // categories: ["Series"],
  277. // // chartTitle: "My Radar Chart",
  278. // this.VeidooList = []; //大分类
  279. // this.VeidooJsonList = []; //小分类
  280. // // 健康横向柱状图数据
  281. // this.healthCroData = [];
  282. // this.healthCategories = [];
  283. // // 关键证据数据
  284. // this.healthEvidence = [];
  285. // // 后端获取的数据
  286. // this.radarData = {};
  287. // },
  288. // 雷达图数据
  289. countRadar() {
  290. // console.log("数据?", this.radarData);
  291. let raData = this.radarData;
  292. // 将数据复制在大分类里面用child来存储
  293. this.VeidooList.forEach(e => {
  294. // console.log(e);
  295. let fid = e.id;
  296. e.child = raData[fid];
  297. // console.log( raData[fid]);
  298. });
  299. // 将每个大分类下的小分类数据提取出来,在大分类下创建一个child2来存储
  300. this.VeidooList.forEach((e, k) => {
  301. e.child2 = [];
  302. e.child.forEach((i, index) => {
  303. e.child2.push(i[i.id]);
  304. });
  305. });
  306. // 第三次提取,将每个小分类的数据进行相加再除以小分类的长度得到平均值,再将平均值存储在大分类的num中
  307. let a = 0;
  308. let b = 0;
  309. let c = 0;
  310. this.VeidooList.forEach((e, k) => {
  311. e.num = [];
  312. e.child2.forEach((i, index) => {
  313. a = a + i[0];
  314. b = b + i[1];
  315. c = c + i[2];
  316. });
  317. a = (a / e.child2.length).toFixed(1);
  318. b = (b / e.child2.length).toFixed(1);
  319. c = (c / e.child2.length).toFixed(1);
  320. // 综合的柱状图,使用的是雷达图的数据再加工的数据。
  321. //toFixed会把数字转换成字符串,所以要转换成数字,不然柱状没法用
  322. e.num.push(a * 1, b * 1, c * 1);
  323. a = 0;
  324. b = 0;
  325. c = 0;
  326. });
  327. // 雷达图数据---- 最后把每个大分类下的num按照,第一次第二次第三次,存储到raDataCopy中,为
  328. let raDataCopy = [[], [], []];
  329. this.VeidooList.forEach((e, k) => {
  330. raDataCopy[0].push(e.num[0]);
  331. raDataCopy[1].push(e.num[1]);
  332. raDataCopy[2].push(e.num[2]);
  333. });
  334. // console.log("旧raDataCopy", raDataCopy);
  335. this.chartData = raDataCopy;
  336. },
  337. // 柱状图
  338. countCro() {
  339. // 平均分数柱状图数据----------
  340. let croMapData = [];
  341. this.chartData.forEach(e => {
  342. console.log(e);
  343. croMapData = this.addArrays(croMapData, e);
  344. });
  345. this.croColumnarData = croMapData.map(function(item) {
  346. return (item / 3).toFixed(1);
  347. });
  348. },
  349. // 数据相同下标相加组成一个新数组
  350. combineArrays(arrays) {
  351. const result = [];
  352. // 获取数组中的最大长度
  353. const maxLength = Math.max(...arrays.map(arr => arr.length));
  354. // 遍历每个下标
  355. for (let i = 0; i < maxLength; i++) {
  356. const combinedItem = [];
  357. // 遍历每个数组
  358. for (let j = 0; j < arrays.length; j++) {
  359. const array = arrays[j];
  360. // 检查当前下标是否存在于数组中
  361. if (i < array.length) {
  362. const item = array[i];
  363. // 将当前下标的字段添加到组合数组中
  364. combinedItem.push(item);
  365. }
  366. }
  367. // 将组合数组添加到结果数组中
  368. result.push(combinedItem);
  369. }
  370. return result;
  371. },
  372. // 数组相加
  373. addArrays(array1, array2) {
  374. const result = [];
  375. for (let i = 0; i < array2.length; i++) {
  376. const sum = (array1[i] || 0) + array2[i];
  377. result.push(sum);
  378. }
  379. return result;
  380. }
  381. },
  382. mounted() {},
  383. created() {
  384. this.getYear();
  385. }
  386. };
  387. </script>
  388. <style scoped>
  389. /deep/ .el-dialog {
  390. width: 1000px;
  391. }
  392. /deep/ .el-dialog__title {
  393. color: rgba(0, 0, 0, 0.9);
  394. text-align: center;
  395. font-family: "Microsoft YaHei";
  396. font-size: 20px;
  397. font-style: normal;
  398. font-weight: bold;
  399. line-height: 22px; /* 137.5% */
  400. }
  401. .report {
  402. box-sizing: border-box;
  403. padding: 15px;
  404. height: 100%;
  405. overflow-y: scroll;
  406. }
  407. .top {
  408. display: flex;
  409. justify-content: space-between;
  410. align-items: center;
  411. height: 40px;
  412. margin-bottom: 10px;
  413. }
  414. .topBtnS {
  415. display: flex;
  416. justify-content: space-between;
  417. align-items: center;
  418. height: 40px;
  419. margin-bottom: 10px;
  420. }
  421. .btn {
  422. border-radius: 8px;
  423. border: 1px solid #3681fc;
  424. color: rgba(54, 129, 252, 1);
  425. display: inline-flex;
  426. padding: 5px 16px;
  427. justify-content: center;
  428. align-items: center;
  429. margin-left: 10px;
  430. gap: 10px;
  431. }
  432. .content {
  433. width: 100%;
  434. height: 553px;
  435. flex-shrink: 0;
  436. border-radius: 16px;
  437. border: 1px solid var(--bg3, #e7e7e7);
  438. background: #fff;
  439. display: flex;
  440. align-items: center;
  441. flex-wrap: wrap;
  442. }
  443. .conTitle {
  444. width: 70%;
  445. margin: 20px auto;
  446. display: flex;
  447. justify-content: flex-start;
  448. align-items: center;
  449. }
  450. .txt {
  451. flex: 1;
  452. width: 712px;
  453. height: 40px;
  454. line-height: 40px;
  455. flex-shrink: 0;
  456. border-radius: 4px;
  457. background: #f0f2f5;
  458. margin-left: 5px;
  459. box-sizing: border-box;
  460. padding: 0 15px;
  461. }
  462. .radarCss {
  463. width: 45%;
  464. height: 400px;
  465. }
  466. .proofCss {
  467. color: rgba(0, 0, 0, 0.9);
  468. font-family: "Microsoft YaHei";
  469. font-size: 14px;
  470. font-style: normal;
  471. font-weight: 700;
  472. line-height: normal;
  473. letter-spacing: 0.7px;
  474. margin-right: 20px;
  475. }
  476. .proofCss2 {
  477. width: 200px;
  478. text-align: left;
  479. box-sizing: border-box;
  480. padding: 4px 20px;
  481. color: rgba(0, 0, 0, 0.9);
  482. background-color: #f0f2f5;
  483. font-family: "Microsoft YaHei";
  484. border-radius: 3px;
  485. font-size: 14px;
  486. font-style: normal;
  487. line-height: normal;
  488. letter-spacing: 0.7px;
  489. margin-right: 20px;
  490. }
  491. .proofCells {
  492. width: 600px;
  493. /* background-color: aqua; */
  494. display: flex;
  495. flex-wrap: wrap;
  496. }
  497. .proofCells > .proofCell {
  498. width: 45%;
  499. margin-right: 20px;
  500. margin-bottom: 10px;
  501. box-sizing: border-box;
  502. padding: 10px 15px;
  503. border-radius: 8px;
  504. background: var(--bg, #f0f2f5);
  505. }
  506. </style>