index.vue 35 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847848849850851852853854855856857858859860861862863864865866867868869870871872873874875876877878879880881882883884885886887888889890891892893894895896897898899900901902903904905906907908909910911912913914915916917918919920921922923924925926927928929930931932933934935936937938939940941942943944945946947948949950951952953954955956957958959960961962963964965966967968969970971972973974975976977978979980981982983984985986987988989990991992993994995996997998999100010011002100310041005100610071008100910101011101210131014101510161017101810191020102110221023102410251026102710281029103010311032103310341035103610371038103910401041104210431044104510461047104810491050105110521053105410551056105710581059106010611062106310641065106610671068106910701071107210731074107510761077107810791080108110821083108410851086
  1. <template>
  2. <div class="body1" v-loading="isLoading">
  3. <!-- 综合数据 -->
  4. <div class="left">
  5. <div class="top">
  6. <div class="titleBox" style="justify-content: space-between">
  7. <div class="title">基础概况</div>
  8. <!-- <el-select v-model="cType" class="selectBox" style="width: 110px">
  9. <el-option label="全部" value="全部"></el-option>
  10. </el-select> -->
  11. </div>
  12. <div class="dataBox">
  13. <div class="info_box">
  14. <div class="info blueBG">
  15. <span>用户总数</span>
  16. <span>{{ allUser }}</span>
  17. </div>
  18. <div class="info blueBG">
  19. <span>本月登录用户环比</span>
  20. <span v-if="loginCountMonthArray.length">{{ (loginCountMonthArray[loginCountMonthArray.length - 1].user -
  21. loginCountMonthArray[loginCountMonthArray.length - 2].user) < 0 ? 0 :
  22. ((loginCountMonthArray[loginCountMonthArray.length - 1].user -
  23. loginCountMonthArray[loginCountMonthArray.length - 2].user) /
  24. loginCountMonthArray[loginCountMonthArray.length - 2].user * 100) + '%' }}</span>
  25. </div>
  26. <div class="info blueBG">
  27. <span>本月登录用户总数</span>
  28. <span v-if="loginCountMonthArray.length">{{ loginCountMonthArray[loginCountMonthArray.length - 1].user
  29. }}</span>
  30. </div>
  31. <div class="info blueBG">
  32. <span>本月新增登录用户</span>
  33. <span v-if="loginCountMonthArray.length">{{ (loginCountMonthArray[loginCountMonthArray.length - 1].user -
  34. loginCountMonthArray[loginCountMonthArray.length - 2].user) < 0 ? 0 :
  35. loginCountMonthArray[loginCountMonthArray.length - 1].user -
  36. loginCountMonthArray[loginCountMonthArray.length - 2].user }}</span>
  37. </div>
  38. </div>
  39. <loginCount style="height: calc(100% - 140px)" :monthArray="loginCountMonthArray"></loginCount>
  40. </div>
  41. </div>
  42. <div class="bottom">
  43. <div class="titleBox">
  44. <div class="title" :class="{ isClick: skType == 0 }" @click="skType = 0"
  45. style="cursor: pointer; padding: 0 0 5px 0">
  46. 在线时长
  47. </div>
  48. <div class="title" :class="{ isClick: skType == 1 }" @click="skType = 1"
  49. style="cursor: pointer; padding: 0 0 5px 0">
  50. 登录频次
  51. </div>
  52. <!-- <el-select v-model="cType1" class="selectBox" style="width: 110px;margin-left:auto;">
  53. <el-option label="全部年级" value="全部年级"></el-option>
  54. <el-option label="一年级" value="一年级"></el-option>
  55. <el-option label="二年级" value="二年级"></el-option>
  56. <el-option label="三年级" value="三年级"></el-option>
  57. </el-select> -->
  58. <!-- <div class="timeDiv">
  59. <div @click="tType = 0" :class="{ isClick: tType == 0 }">周</div>
  60. <div @click="tType = 1" :class="{ isClick: tType == 1 }">月</div>
  61. <div @click="tType = 2" :class="{ isClick: tType == 2 }">学期</div>
  62. </div> -->
  63. </div>
  64. <div class="info_box" v-if="skType == 1">
  65. <div class="info blueBG">
  66. <span>登录频次</span>
  67. <span>{{ countLogin }}</span>
  68. </div>
  69. <div class="info blueBG">
  70. <span>人均登录频次</span>
  71. <span>{{ countLogin > 0 ? (countLogin / allUser).toFixed(0) : 0 }}</span>
  72. </div>
  73. </div>
  74. <div class="info_box" v-if="skType == 0">
  75. <div class="info blueBG">
  76. <span>累计时长</span>
  77. <span>{{ userOnlineTime }}</span>
  78. </div>
  79. <div class="info blueBG">
  80. <span>人均使用时长</span>
  81. <span>{{ userOnlineTime > 0 ? (userOnlineTime / allUser).toFixed(0) : 0 }}</span>
  82. </div>
  83. </div>
  84. <div class="dataBox" style="height: calc(100% - 115px);">
  85. <toolUser style="height: calc(100%)" v-if="skType == 1" :yearArray="loginCountYearArray"></toolUser>
  86. <bar style="height: calc(100%)" v-if="skType == 0" :loginArray="[teacherOnlineTime, studentOnlineTime]"></bar>
  87. <!-- <div class="otherCss">
  88. <div v-if="!oType">切换为柱状图</div>
  89. <div v-if="oType">切换为热力图</div>
  90. <div class="otherImg" @click="otherEchart">
  91. <img src="../../../../assets/icon/other.png" alt="" />
  92. </div>
  93. </div> -->
  94. </div>
  95. </div>
  96. </div>
  97. <div class="center">
  98. <div class="top">
  99. <div class="titleBox" style="justify-content: space-between">
  100. <div style="
  101. display: flex;
  102. flex-direction: row;
  103. flex-wrap: nowrap;
  104. align-items: center;
  105. ">
  106. <div class="title" :class="{ isClick: courseType == 0 }" @click="courseType = 0"
  107. style="cursor: pointer; padding: 0 0 5px 0">
  108. 整体分布
  109. </div>
  110. <div class="title" :class="{ isClick: courseType == 1 }" @click="courseType = 1"
  111. style="cursor: pointer; padding: 0 0 5px 0">
  112. 总量增幅
  113. </div>
  114. <div class="title" :class="{ isClick: courseType == 2 }" @click="courseType = 2"
  115. style="cursor: pointer; padding: 0 0 5px 0">
  116. 实施情况
  117. </div>
  118. </div>
  119. <!-- <div style="
  120. display: flex;
  121. flex-direction: row;
  122. flex-wrap: nowrap;
  123. align-items: center;
  124. ">
  125. <el-select v-model="cType3" class="selectBox" style="width: 110px" v-if="courseType == 1">
  126. <el-option label="按年级" value="按年级"></el-option>
  127. </el-select>
  128. <el-select v-model="cType2" class="selectBox" style="width: 110px">
  129. <el-option label="全部活动" value="全部活动"></el-option>
  130. </el-select>
  131. </div> -->
  132. </div>
  133. <div class="dataBox">
  134. <teacherInfo v-if="courseType == 0" style="height: calc(100%)" :courseArray="courseArray"></teacherInfo>
  135. <courseNum v-if="courseType == 1" style="height: calc(100%)" :weekCourse2="weekCourse2"></courseNum>
  136. <div style="height: calc(100%)" v-if="courseType == 2">
  137. <div class="info_box" style="width: 96%;">
  138. <div class="info blueBG" style="width:calc(100% / 4 - 10px)">
  139. <span>平台实施课程总数</span>
  140. <span>{{ isCourseCount }}</span>
  141. </div>
  142. <div class="info blueBG" style="width:calc(100% / 4 - 10px)">
  143. <span>平台实施课程占比</span>
  144. <span>{{ isCourseCount ? (isCourseCount / allCourseCount * 100).toFixed(0) + '%' : '0%' }}</span>
  145. </div>
  146. <div class="info blueBG" style="width:calc(100% / 4 - 10px)">
  147. <span>课程平均任务数量</span>
  148. <span>{{ isCourseCount ? (taskCount / isCourseCount).toFixed(0) : 0 }}</span>
  149. </div>
  150. <div class="info blueBG" style="width:calc(100% / 4 - 10px)">
  151. <span>课程平均作业数量</span>
  152. <span>{{ isCourseCount ? (workCount / isCourseCount).toFixed(0) : 0 }}</span>
  153. </div>
  154. </div>
  155. <workNum style="height:calc(100% - 70px)" :workNumList="workNumList"></workNum>
  156. </div>
  157. </div>
  158. </div>
  159. <div class="bottom">
  160. <div class="titleBox" style="justify-content: space-between">
  161. <div class="title">平台使用概况</div>
  162. <div class="tCircleC">
  163. <div class="tCircleBox">
  164. <span class="tCircle tt"></span>
  165. <span class="tname">老师</span>
  166. </div>
  167. <div class="tCircleBox">
  168. <span class="tCircle ts"></span>
  169. <span class="tname">学生</span>
  170. </div>
  171. </div>
  172. <!-- <div class="timeDiv">
  173. <div @click="tType = 0" :class="{ isClick: tType == 0 }">周</div>
  174. <div @click="tType = 1" :class="{ isClick: tType == 1 }">月</div>
  175. <div @click="tType = 2" :class="{ isClick: tType == 2 }">学期</div>
  176. </div> -->
  177. </div>
  178. <div class="dataBox">
  179. <div class="depth_box" style="height: calc(100%)">
  180. <div class="depth">
  181. <span>上传课程</span>
  182. <div>
  183. <el-progress :width="80" type="circle"
  184. :percentage="lightJson.upCourseTeachers ? parseInt((lightJson.upCourseTeachers / lightJson.teachers * 100).toFixed(0)) : 0"
  185. :stroke-width="5" :format="format" color="rgb(64, 149, 229)"></el-progress>
  186. </div>
  187. </div>
  188. <div class="depth">
  189. <span>上传项目</span>
  190. <div>
  191. <el-progress :width="80" type="circle"
  192. :percentage="lightJson.upSCourseTeachers ? parseInt((lightJson.upSCourseTeachers / lightJson.teachers * 100).toFixed(0)) : 0"
  193. :stroke-width="5" :format="format" color="rgb(64, 149, 229)"></el-progress>
  194. </div>
  195. </div>
  196. <div class="depth">
  197. <span>使用工具</span>
  198. <div>
  199. <el-progress :width="80" type="circle"
  200. :percentage="lightJson.toolTeachers ? parseInt((lightJson.toolTeachers / lightJson.teachers * 100).toFixed(0)) : 0"
  201. :stroke-width="5" :format="format" color="rgb(64, 149, 229)"></el-progress>
  202. </div>
  203. </div>
  204. <div class="depth">
  205. <span>协同合作</span>
  206. <div>
  207. <el-progress :width="80" type="circle"
  208. :percentage="lightJson.gCourseTeachers ? parseInt((lightJson.gCourseTeachers / lightJson.teachers * 100).toFixed(0)) : 0"
  209. :stroke-width="5" :format="format" color="rgb(64, 149, 229)"></el-progress>
  210. </div>
  211. </div>
  212. <div class="depth">
  213. <span>互动交流</span>
  214. <div>
  215. <el-progress :width="80" type="circle"
  216. :percentage="lightJson.commentTeachers ? parseInt((lightJson.commentTeachers / lightJson.teachers * 100).toFixed(0)) : 0"
  217. :stroke-width="5" :format="format" color="rgb(64, 149, 229)"></el-progress>
  218. </div>
  219. </div>
  220. <div class="depth">
  221. <span>参与课程</span>
  222. <div>
  223. <el-progress :width="80" type="circle"
  224. :percentage="lightJson.courseStudents ? parseInt((lightJson.courseStudents / lightJson.students * 100).toFixed(0)) : 0"
  225. :stroke-width="5" :format="format" color="#106BFF"></el-progress>
  226. </div>
  227. </div>
  228. <div class="depth">
  229. <span>参与项目</span>
  230. <div>
  231. <el-progress :width="80" type="circle"
  232. :percentage="lightJson.scourseStudents ? parseInt((lightJson.scourseStudents / lightJson.students * 100).toFixed(0)) : 0"
  233. :stroke-width="5" :format="format" color="#106BFF"></el-progress>
  234. </div>
  235. </div>
  236. <div class="depth">
  237. <span>使用工具</span>
  238. <div>
  239. <el-progress :width="80" type="circle"
  240. :percentage="lightJson.toolStudents ? parseInt((lightJson.toolStudents / lightJson.students * 100).toFixed(0)) : 0"
  241. :stroke-width="5" :format="format" color="#106BFF"></el-progress>
  242. </div>
  243. </div>
  244. <div class="depth">
  245. <span>协同合作</span>
  246. <div>
  247. <el-progress :width="80" type="circle"
  248. :percentage="lightJson.gsCourseStudents ? parseInt((lightJson.gsCourseStudents / lightJson.students * 100).toFixed(0)) : 0"
  249. :stroke-width="5" :format="format" color="#106BFF"></el-progress>
  250. </div>
  251. </div>
  252. <div class="depth">
  253. <span>互动交流</span>
  254. <div>
  255. <el-progress :width="80" type="circle"
  256. :percentage="lightJson.commentStudents ? parseInt((lightJson.commentStudents / lightJson.students * 100).toFixed(0)) : 0"
  257. :stroke-width="5" :format="format" color="#106BFF"></el-progress>
  258. </div>
  259. </div>
  260. </div>
  261. <!-- <div class="otherCss">
  262. <div v-if="!shType">切换为散点图</div>
  263. <div v-if="shType">切换为进度条</div>
  264. <div class="otherImg" @click="shEchart">
  265. <img src="../../../../assets/icon/other.png" alt="" />
  266. </div>
  267. </div> -->
  268. </div>
  269. </div>
  270. </div>
  271. <div class="right">
  272. <div class="top" style="border-radius: 5px">
  273. <div class="titleBox" style="justify-content: space-between">
  274. <div class="title">整体分布</div>
  275. <div style="
  276. display: flex;
  277. flex-direction: row;
  278. flex-wrap: nowrap;
  279. align-items: center;
  280. ">
  281. <el-select v-model="cType" @change="typeChange" class="selectBox">
  282. <el-option label="全部" value=""></el-option>
  283. <el-option label="年级" value="grade"></el-option>
  284. <el-option label="主题" value="theme"></el-option>
  285. <el-option label="学科" value="subject"></el-option>
  286. </el-select>
  287. <!-- <el-select v-model="cType4" class="selectBox" style="width: 110px">
  288. <el-option label="全部" value="全部"></el-option>
  289. </el-select>
  290. <el-select v-model="cType5" class="selectBox" style="width: 110px">
  291. <el-option label="总量占比" value="总量占比"></el-option>
  292. </el-select> -->
  293. </div>
  294. </div>
  295. <div class="dataBox">
  296. <courseInfo style="height: calc(100% - 40px)" :courseNumberArray="courseNumberArray"></courseInfo>
  297. </div>
  298. </div>
  299. <div class="bottom">
  300. <div class="titleBox" style="justify-content: space-between">
  301. <div class="title">跨学科教学情况</div>
  302. <el-select v-model="cType6" class="selectBox" style="width: 110px">
  303. <el-option label="全部" value="全部"></el-option>
  304. </el-select>
  305. </div>
  306. <div class="dataBox">
  307. <cateRank style="height: calc(100%)"></cateRank>
  308. </div>
  309. </div>
  310. </div>
  311. </div>
  312. </template>
  313. <script>
  314. import loginCount from "./loginCount";
  315. import teacherInfo from "./teacherInfo";
  316. import courseInfo from "./courseInfo";
  317. import cateRank from "./cateRank";
  318. import toolUser from "./toolUser";
  319. import bar from "./bar";
  320. import courseNum from "./courseNum";
  321. import workNum from "./workNum";
  322. export default {
  323. components: {
  324. loginCount,
  325. teacherInfo,
  326. courseInfo,
  327. cateRank,
  328. toolUser,
  329. bar,
  330. courseNum,
  331. workNum,
  332. },
  333. props: {
  334. oid: {
  335. type: String,
  336. },
  337. org: {
  338. type: String,
  339. },
  340. },
  341. data() {
  342. return {
  343. isLoading: false,
  344. cType: "",
  345. cType1: "全部年级",
  346. cType2: "全部活动",
  347. cType3: "按年级",
  348. cType4: "全部",
  349. cType5: "总量占比",
  350. cType6: "全部",
  351. skType: 0,
  352. tType: 0,
  353. courseType: 0,
  354. oType: false,
  355. shType: false,
  356. allUser: 0,
  357. loginCountMonthArray: [],
  358. countLogin: 0,
  359. loginCountYearArray: [],
  360. userOnlineTime: [],
  361. teacherOnlineTime: [],
  362. studentOnlineTime: [],
  363. courseArray: [],
  364. workNumList: [],
  365. taskCount: 0,
  366. workCount: 0,
  367. isCourseCount: 0,
  368. allCourseCount: 0,
  369. allArray: [],
  370. gradeArray: [],
  371. themeArray: [],
  372. subjectArray: [],
  373. courseNumberArray: [],
  374. lightJson: {
  375. teachers: 0,
  376. students: 0,
  377. upCourseTeachers: 0,//上传课程
  378. upSCourseTeachers: 0,//上传项目
  379. gCourseTeachers: 0,//协同课程
  380. toolTeachers: 0,//使用工具
  381. commentTeachers: 0,//交流
  382. courseStudents: 0,//参与课程
  383. scourseStudents: 0,//参与项目
  384. toolStudents: 0,//使用工具
  385. gsCourseStudents: 0,//协同课程
  386. commentStudents: 0,//协同交流
  387. },
  388. weekCourse2: []
  389. };
  390. },
  391. mounted() {
  392. this.getData();
  393. },
  394. methods: {
  395. otherEchart() {
  396. this.oType = !this.oType;
  397. },
  398. shEchart() {
  399. this.shType = !this.shType;
  400. },
  401. format(percentage) {
  402. return percentage + '%';
  403. },
  404. getData() {
  405. this.isLoading = true;
  406. let params = [
  407. {
  408. oid: this.oid,
  409. org: this.org,
  410. },
  411. ];
  412. this.ajax
  413. .post(this.$store.state.api + "selectDataBoardSchoolNew", params)
  414. .then((res) => {
  415. this.isLoading = false;
  416. let _grade = res.data[0]; //年级
  417. let _subject = res.data[1]; //学科
  418. let _theme = res.data[2]; //主题
  419. this.allUser = res.data[3][0].count; //总人数
  420. let _loginCount = res.data[4]; //统计半年的登录用户
  421. let loginCountMonthArray = []
  422. const date = new Date()
  423. var Month = date.getMonth() + 1
  424. var Year = date.getFullYear()
  425. for (var i = Month; i > Month - 6; i--) {
  426. if (i <= 0) {
  427. loginCountMonthArray.push({
  428. Year: Year - 1,
  429. Month: 12 + i,
  430. user: 0,
  431. })
  432. } else {
  433. loginCountMonthArray.push({
  434. Month: i,
  435. Year: Year,
  436. user: 0,
  437. })
  438. }
  439. }
  440. loginCountMonthArray = loginCountMonthArray.reverse()
  441. for (var i = 0; i < _loginCount.length; i++) {
  442. let _date = new Date(_loginCount[i].create_at)
  443. var _month = _date.getMonth() + 1
  444. var _year = _date.getFullYear()
  445. for (var j = 0; j < loginCountMonthArray.length; j++) {
  446. if (_month == loginCountMonthArray[j].Month && _year == loginCountMonthArray[j].Year) {
  447. loginCountMonthArray[j].user++
  448. break;
  449. }
  450. }
  451. }
  452. this.loginCountMonthArray = loginCountMonthArray
  453. let countLogin = res.data[5][0].count; //总的登录频次
  454. this.countLogin = countLogin
  455. let loginCountYear = res.data[6];//一年的登录频次
  456. let loginCountYearArray = []
  457. for (var i = Month; i > Month - 12; i--) {
  458. if (i <= 0) {
  459. loginCountYearArray.push({
  460. Year: Year - 1,
  461. Month: 12 + i,
  462. mon: 0,
  463. tue: 0,
  464. wed: 0,
  465. thur: 0,
  466. fri: 0,
  467. sat: 0,
  468. sun: 0,
  469. })
  470. } else {
  471. loginCountYearArray.push({
  472. Month: i,
  473. Year: Year,
  474. mon: 0,
  475. tue: 0,
  476. wed: 0,
  477. thur: 0,
  478. fri: 0,
  479. sat: 0,
  480. sun: 0,
  481. })
  482. }
  483. }
  484. loginCountYearArray = loginCountYearArray.reverse()
  485. for (var i = 0; i < loginCountYear.length; i++) {
  486. let _date = new Date(loginCountYear[i].create_at)
  487. var _month = _date.getMonth() + 1
  488. var _year = _date.getFullYear()
  489. var _day = _date.getDay()
  490. let dayArray = ['sun', 'mon', 'tue', 'wed', 'thur', 'fri', 'sat']
  491. for (var j = 0; j < loginCountYearArray.length; j++) {
  492. if (_month == loginCountYearArray[j].Month && _year == loginCountYearArray[j].Year) {
  493. loginCountYearArray[j][dayArray[_day]]++
  494. break;
  495. }
  496. }
  497. }
  498. console.log(loginCountYearArray);
  499. this.loginCountYearArray = loginCountYearArray
  500. let _userOnlineTime = res.data[7][0].time;//在线时长 所有用户
  501. let _teacherOnlineTime = res.data[8][0].time;//在线时长 老师
  502. let _studentOnlineTime = res.data[9][0].time;//在线时长 学生
  503. _userOnlineTime = (_userOnlineTime / 60 / 60).toFixed(0)
  504. _teacherOnlineTime = (_teacherOnlineTime / 60 / 60).toFixed(0)
  505. _studentOnlineTime = (_studentOnlineTime / 60 / 60).toFixed(0)
  506. this.userOnlineTime = _userOnlineTime
  507. this.teacherOnlineTime = _teacherOnlineTime
  508. this.studentOnlineTime = _studentOnlineTime
  509. let _course = res.data[10] //课程
  510. let _gradeCourse = 0 //各年级上传课程
  511. let _subjectCourse = 0 //各学科上传课程
  512. let _courseArray = []
  513. _subject.push({ name: '其他' })
  514. for (var i = 0; i < _grade.length; i++) {
  515. _courseArray.push({
  516. name: _grade[i].name,
  517. id: _grade[i].id,
  518. courseid: [],
  519. subject: [],
  520. })
  521. for (var z = 0; z < _course.length; z++) {
  522. if (_course[z].typeid == _grade[i].id) {
  523. _gradeCourse++
  524. if (_courseArray[i].courseid.indexOf(_course[z].courseid) === -1) {
  525. _courseArray[i].courseid.push(_course[z].courseid)
  526. }
  527. }
  528. }
  529. for (var j = 0; j < _subject.length; j++) {
  530. _courseArray[i].subject.push({
  531. name: _subject[j].name,
  532. id: _subject[j].id,
  533. course: 0
  534. })
  535. for (var z = 0; z < _course.length; z++) {
  536. if (_course[z].typeid == _subject[j].id && _courseArray[i].courseid.indexOf(_course[z].courseid) !== -1) {
  537. _courseArray[i].subject[j].course++
  538. }
  539. }
  540. }
  541. let sum = 0
  542. for (var j = 0; j < _courseArray[i].subject.length - 1; j++) {
  543. sum += _courseArray[i].subject[j].course
  544. }
  545. _courseArray[i].subject[_courseArray[i].subject.length - 1].course = (_courseArray[i].courseid.length - sum) < 0 ? 0 : _courseArray[i].courseid.length - sum
  546. }
  547. for (var j = 0; j < _subject.length; j++) {
  548. for (var z = 0; z < _course.length; z++) {
  549. if (_course[z].typeid == _subject[j].id) {
  550. _subjectCourse++
  551. }
  552. }
  553. }
  554. console.log(_courseArray);
  555. // this.gradeCourse = _gradeCourse / _grade.length
  556. // this.subjectCourse = _subjectCourse / _subject.length
  557. this.courseArray = _courseArray
  558. let _workCourse = res.data[11]; //带作业的课程
  559. let _taskCount = 0 //任务数量
  560. var wList = [];
  561. for (var i = 0; i < _workCourse.length; i++) {
  562. if (!wList[_workCourse[i].courseId]) {
  563. wList[_workCourse[i].courseId] = {
  564. cid: _workCourse[i].courseId,
  565. title: _workCourse[i].title,
  566. task: 0,
  567. work: 0,
  568. };
  569. let chapters = JSON.parse(_workCourse[i].chapters);
  570. for (var j = 0; j < chapters.length; j++) {
  571. if (wList[_workCourse[i].courseId].task == 0) {
  572. wList[_workCourse[i].courseId].task =
  573. chapters[j].chapterInfo[0].taskJson.length;
  574. } else {
  575. wList[_workCourse[i].courseId].task +=
  576. chapters[j].chapterInfo[0].taskJson.length;
  577. }
  578. _taskCount += chapters[j].chapterInfo[0].taskJson.length;
  579. }
  580. }
  581. }
  582. for (var i = 0; i < _workCourse.length; i++) {
  583. let a = Object.keys(wList);
  584. for (var j = 0; j < Object.keys(wList).length; j++) {
  585. if (_workCourse[i].courseId == wList[a[j]].cid && _workCourse[i].id) {
  586. wList[a[j]].work++;
  587. }
  588. }
  589. }
  590. var workNumList = Object.values(wList).map((item) => [
  591. item.task,
  592. item.work,
  593. item.title,
  594. item.cid,
  595. ]);
  596. this.workNumList = workNumList;
  597. this.taskCount = _taskCount
  598. this.workCount = _workCourse.length
  599. this.isCourseCount = res.data[12][0].count; //课程开展总数
  600. this.allCourseCount = res.data[13][0].count; //课程总数
  601. let _gradeArray = []
  602. let _subjectArray = []
  603. let _themeArray = []
  604. for (var i = 0; i < _grade.length; i++) {
  605. _gradeArray.push({
  606. name: _grade[i].name,
  607. typeid: _grade[i].id,
  608. course: 0,
  609. array: []
  610. })
  611. for (var z = 0; z < _course.length; z++) {
  612. if (_course[z].typeid == _grade[i].id) {
  613. _gradeArray[i].course++
  614. _gradeArray[i].array.push(_course[z].courseid)
  615. }
  616. }
  617. }
  618. for (var i = 0; i < _subject.length; i++) {
  619. _subjectArray.push({
  620. name: _subject[i].name,
  621. typeid: _subject[i].id,
  622. course: 0,
  623. array: []
  624. })
  625. for (var z = 0; z < _course.length; z++) {
  626. if (_course[z].typeid == _subject[i].id) {
  627. _subjectArray[i].course++
  628. _subjectArray[i].array.push(_course[z].courseid)
  629. }
  630. }
  631. }
  632. for (var i = 0; i < _theme.length; i++) {
  633. _themeArray.push({
  634. name: _theme[i].name,
  635. typeid: _theme[i].id,
  636. course: 0,
  637. array: []
  638. })
  639. for (var z = 0; z < _course.length; z++) {
  640. if (_course[z].typeid == _theme[i].id) {
  641. _themeArray[i].course++
  642. _themeArray[i].array.push(_course[z].courseid)
  643. }
  644. }
  645. }
  646. this.gradeArray = _gradeArray
  647. this.subjectArray = _subjectArray
  648. this.themeArray = _themeArray
  649. this.allArray = [..._gradeArray, ..._subjectArray, ..._themeArray]
  650. this.typeChange();
  651. this.lightJson.teachers = res.data[14][0].count; //教师总数
  652. this.lightJson.students = res.data[15][0].count; //学生总数
  653. this.lightJson.upCourseTeachers = res.data[16][0].count; //上传课程老师数量
  654. this.lightJson.upSCourseTeachers = res.data[17][0].count; //上传项目老师数量
  655. this.lightJson.toolTeachers = res.data[18][0].count; //使用工具老师数量
  656. this.lightJson.gCourseTeachers = res.data[19][0].count; //协同课程老师数量
  657. this.lightJson.commentTeachers = res.data[20][0].count; //协同交流老师数量
  658. this.lightJson.courseStudents = res.data[21][0].count; //参与课程学生数量
  659. this.lightJson.scourseStudents = res.data[22][0].count; //参与项目学生数量
  660. this.lightJson.toolStudents = res.data[23][0].count; //使用工具数量
  661. this.lightJson.gsCourseStudents = res.data[24][0].count; //获取协同项目数量
  662. this.lightJson.commentStudents = res.data[25][0].count; //互动交流学生数量
  663. var today = new Date();
  664. var lastDayOfWeek = new Date(today.setDate(today.getDate() - today.getDay() + 7));;//本周周日
  665. let weekArray = {
  666. lastWeek: [],//上周
  667. toWeek: [] //本周
  668. }
  669. for (var i = 0; i < 14; i++) {
  670. let time = JSON.parse(JSON.stringify(lastDayOfWeek))
  671. let time2 = new Date(time)
  672. var a = new Date(time2.setDate(time2.getDate() - i));
  673. if (i > 6) {
  674. weekArray.lastWeek.push(a.getFullYear() + '-' + (a.getMonth() + 1) + '-' + a.getDate());
  675. } else {
  676. weekArray.toWeek.push(a.getFullYear() + '-' + (a.getMonth() + 1) + '-' + a.getDate());
  677. }
  678. }
  679. console.log(weekArray);
  680. let weekCourse = [];
  681. let weekCourse2 = [];
  682. let lastWeekCouseCount = 0
  683. let toWeekCouseCount = 0
  684. for (var z = 0; z < _course.length; z++) {
  685. let _date = new Date(weekArray.lastWeek[weekArray.lastWeek.length - 1])
  686. console.log(_date)
  687. if (new Date(_course[z].create_at) > _date && _course[z].pid == '34628934-d02f-11ec-8c78-005056b86db5') {
  688. weekCourse.push(_course[z])
  689. var a = new Date(_course[z].create_at)
  690. var string = a.getFullYear() + '-' + (a.getMonth() + 1) + '-' + a.getDate()
  691. if (weekArray.lastWeek.indexOf(string) != -1) {
  692. lastWeekCouseCount++
  693. } else {
  694. toWeekCouseCount++
  695. }
  696. }
  697. }
  698. let lastCourseidWeek = []
  699. let toCourseidWeek = []
  700. for (var i = 0; i < _grade.length; i++) {
  701. weekCourse2.push({
  702. name: _grade[i].name,
  703. id: _grade[i].id,
  704. lastCourse: 0,
  705. toCourse: 0,
  706. })
  707. for (var z = 0; z < weekCourse.length; z++) {
  708. if (weekCourse[z].typeid == _grade[i].id) {
  709. var a = new Date(weekCourse[z].create_at)
  710. var string = a.getFullYear() + '-' + (a.getMonth() + 1) + '-' + a.getDate()
  711. if (weekArray.lastWeek.indexOf(string) != -1) {
  712. weekCourse2[i].lastCourse++
  713. if (lastCourseidWeek.indexOf(weekCourse[z].courseid) === -1) {
  714. lastCourseidWeek.push(weekCourse[z].courseid)
  715. }
  716. } else {
  717. weekCourse2[i].toCourse++
  718. if (toCourseidWeek.indexOf(weekCourse[z].courseid) === -1) {
  719. toCourseidWeek.push(weekCourse[z].courseid)
  720. }
  721. }
  722. }
  723. }
  724. }
  725. weekCourse2.push({
  726. name: '其他',
  727. id: '',
  728. lastCourse: (lastWeekCouseCount - lastCourseidWeek.length) > 0 ? (lastWeekCouseCount - lastCourseidWeek.length) : 0,
  729. toCourse: (toWeekCouseCount - toCourseidWeek.length) > 0 ? (toWeekCouseCount - toCourseidWeek.length) : 0,
  730. })
  731. console.log(weekCourse2);
  732. this.weekCourse2 = weekCourse2
  733. this.$forceUpdate();
  734. })
  735. .catch((err) => {
  736. this.isLoading = false;
  737. console.error(err);
  738. });
  739. },
  740. typeChange() {
  741. if (this.cType === '') {
  742. this.courseNumberArray = this.allArray
  743. } else if (this.cType === 'grade') {
  744. this.courseNumberArray = this.gradeArray
  745. } else if (this.cType === 'theme') {
  746. this.courseNumberArray = this.themeArray
  747. } else if (this.cType === 'subject') {
  748. this.courseNumberArray = this.subjectArray
  749. }
  750. this.$forceUpdate();
  751. },
  752. },
  753. };
  754. </script>
  755. <style scoped>
  756. .body1 {
  757. width: 100%;
  758. height: 100%;
  759. display: flex;
  760. padding: 20px;
  761. box-sizing: border-box;
  762. overflow: hidden;
  763. }
  764. .left {
  765. width: calc(100% / 4 * 1);
  766. height: 100%;
  767. }
  768. .left>.top {
  769. width: 100%;
  770. height: calc(100% / 5 * 2.8 - 20px);
  771. background: #fff;
  772. border-radius: 5px;
  773. margin: 0 0 20px 0;
  774. }
  775. .left>.bottom {
  776. width: 100%;
  777. height: calc(100% / 5 * 2.2);
  778. background: #fff;
  779. border-radius: 5px;
  780. }
  781. .center {
  782. width: calc(100% / 4 * 2 - 40px);
  783. height: 100%;
  784. margin: 0 20px;
  785. }
  786. .center>.top {
  787. width: 100%;
  788. height: calc(100% / 5 * 2.8 - 20px);
  789. background: #fff;
  790. border-radius: 5px;
  791. margin: 0 0 20px 0;
  792. }
  793. .center>.bottom {
  794. width: 100%;
  795. height: calc(100% / 5 * 2.2);
  796. background: #fff;
  797. border-radius: 5px;
  798. }
  799. .right {
  800. width: calc(100% / 4 * 1);
  801. height: 100%;
  802. }
  803. .right>.top {
  804. width: 100%;
  805. height: calc(100% / 5 * 2.8 - 20px);
  806. background: #fff;
  807. border-radius: 5px;
  808. margin: 0 0 20px 0;
  809. }
  810. .right>.bottom {
  811. width: 100%;
  812. height: calc(100% / 5 * 2.2);
  813. background: #fff;
  814. border-radius: 5px;
  815. }
  816. .titleBox {
  817. height: 40px;
  818. display: flex;
  819. align-items: center;
  820. padding: 0 15px;
  821. box-sizing: border-box;
  822. width: 100%;
  823. }
  824. .title {
  825. color: #060e17;
  826. margin-right: 10px;
  827. }
  828. .dataBox {
  829. height: calc(100% - 40px);
  830. width: 100%;
  831. }
  832. .greenBG {
  833. background: linear-gradient(180deg,
  834. rgb(211, 246, 228, 0.2) 0%,
  835. rgb(23, 196, 105, 0.3) 100%);
  836. }
  837. .depth_box {
  838. display: flex;
  839. flex-wrap: wrap;
  840. height: 100%;
  841. width: 95%;
  842. margin: 0 auto;
  843. overflow: hidden;
  844. justify-content: space-between;
  845. }
  846. .depth {
  847. width: calc(100% / 5 - 10px);
  848. display: flex;
  849. flex-direction: column;
  850. align-items: center;
  851. justify-content: center;
  852. }
  853. .depth>span:nth-child(1) {
  854. font-size: 14px;
  855. font-weight: 700;
  856. margin: 0 0 10px;
  857. }
  858. .depth>div:nth-child(1) {}
  859. .course_box {
  860. display: flex;
  861. height: 50%;
  862. width: 95%;
  863. margin: 0 auto;
  864. }
  865. .course_box .info_box {
  866. height: 100%;
  867. width: 100px;
  868. flex-direction: column;
  869. margin: 0;
  870. flex-wrap: nowrap;
  871. }
  872. .course_box_p {
  873. width: calc(100% - 100px);
  874. }
  875. .selectBox {
  876. width: 80px;
  877. margin-left: 10px;
  878. }
  879. .selectBox>>>.el-input__inner {
  880. height: 30px;
  881. line-height: 30px;
  882. }
  883. .selectBox>>>.el-input__icon {
  884. line-height: 30px;
  885. }
  886. .teaMiddle {
  887. width: calc(100% / 2 - 10px);
  888. height: 60px;
  889. border-radius: 8px;
  890. /* border: 1px solid #e0eafb; */
  891. display: flex;
  892. flex-direction: column;
  893. align-items: flex-start;
  894. justify-content: center;
  895. padding: 0 10px;
  896. margin: 0 10px;
  897. }
  898. .teaMiddle {
  899. width: calc(100% / 3 - 10px);
  900. }
  901. .timeDiv {
  902. display: flex;
  903. flex-direction: row;
  904. flex-wrap: nowrap;
  905. align-items: center;
  906. margin: 0 0 0 15px;
  907. }
  908. .timeDiv>div {
  909. margin-right: 10px;
  910. cursor: pointer;
  911. }
  912. .isClick {
  913. color: #1684fc;
  914. border-bottom: 2px solid #1684fc;
  915. box-sizing: border-box;
  916. }
  917. .otherCss {
  918. display: flex;
  919. flex-direction: row;
  920. flex-wrap: nowrap;
  921. align-items: center;
  922. justify-content: flex-end;
  923. }
  924. .otherImg {
  925. width: 20px;
  926. height: 20px;
  927. margin: 0 10px;
  928. cursor: pointer;
  929. }
  930. .otherImg>img {
  931. width: 100%;
  932. height: 100%;
  933. }
  934. .info_box {
  935. display: flex;
  936. flex-wrap: wrap;
  937. align-items: center;
  938. justify-content: space-between;
  939. width: 90%;
  940. margin: 0 auto;
  941. }
  942. .info_box>.info2,
  943. .info_box>.info3,
  944. .info_box>.info {
  945. width: calc(50% - 10px);
  946. display: flex;
  947. flex-direction: row;
  948. flex-wrap: nowrap;
  949. height: 60px;
  950. justify-content: space-between;
  951. align-items: center;
  952. padding: 0 10px;
  953. box-sizing: border-box;
  954. margin-bottom: 10px;
  955. border-radius: 5px;
  956. }
  957. .info_box>.info2 {
  958. width: calc(100% / 4 - 10px);
  959. /* align-items: flex-end; */
  960. }
  961. .info_box>.info3 {
  962. width: 100%;
  963. margin-bottom: 5px;
  964. }
  965. .info_box>.info2>span:nth-child(1),
  966. .info_box>.info3>span:nth-child(1),
  967. .info_box>.info>span:nth-child(1) {
  968. font-size: 14px;
  969. /* margin: 0 0 0 20px; */
  970. color: #565e6a;
  971. width: 60px;
  972. white-space: pre-wrap;
  973. word-break: break-all;
  974. }
  975. .info_box>.info2>span:nth-child(2),
  976. .info_box>.info3>span:nth-child(2),
  977. .info_box>.info>span:nth-child(2) {
  978. font-size: 24px;
  979. /* font-weight: 700; */
  980. }
  981. .blueBG {
  982. background: rgb(243, 248, 253);
  983. border: 2px solid rgb(234, 246, 255);
  984. }
  985. .tCircleC {
  986. display: flex;
  987. align-items: center;
  988. }
  989. .tCircleBox {
  990. display: flex;
  991. align-items: center;
  992. }
  993. .tCircleBox>span+span {
  994. margin-left: 5px;
  995. }
  996. .tCircleBox+.tCircleBox {
  997. margin-left: 10px;
  998. }
  999. .tCircleBox .tCircle {
  1000. width: 10px;
  1001. height: 10px;
  1002. border-radius: 50%;
  1003. display: inline-block;
  1004. }
  1005. .tCircleBox .tt {
  1006. background: rgb(64, 149, 229);
  1007. }
  1008. .tCircleBox .ts {
  1009. background: #106BFF;
  1010. }
  1011. </style>