groupAnalysis.vue 51 KB

1234567891011121314151617181920212223242526272829303132333435363738394041424344454647484950515253545556575859606162636465666768697071727374757677787980818283848586878889909192939495969798991001011021031041051061071081091101111121131141151161171181191201211221231241251261271281291301311321331341351361371381391401411421431441451461471481491501511521531541551561571581591601611621631641651661671681691701711721731741751761771781791801811821831841851861871881891901911921931941951961971981992002012022032042052062072082092102112122132142152162172182192202212222232242252262272282292302312322332342352362372382392402412422432442452462472482492502512522532542552562572582592602612622632642652662672682692702712722732742752762772782792802812822832842852862872882892902912922932942952962972982993003013023033043053063073083093103113123133143153163173183193203213223233243253263273283293303313323333343353363373383393403413423433443453463473483493503513523533543553563573583593603613623633643653663673683693703713723733743753763773783793803813823833843853863873883893903913923933943953963973983994004014024034044054064074084094104114124134144154164174184194204214224234244254264274284294304314324334344354364374384394404414424434444454464474484494504514524534544554564574584594604614624634644654664674684694704714724734744754764774784794804814824834844854864874884894904914924934944954964974984995005015025035045055065075085095105115125135145155165175185195205215225235245255265275285295305315325335345355365375385395405415425435445455465475485495505515525535545555565575585595605615625635645655665675685695705715725735745755765775785795805815825835845855865875885895905915925935945955965975985996006016026036046056066076086096106116126136146156166176186196206216226236246256266276286296306316326336346356366376386396406416426436446456466476486496506516526536546556566576586596606616626636646656666676686696706716726736746756766776786796806816826836846856866876886896906916926936946956966976986997007017027037047057067077087097107117127137147157167177187197207217227237247257267277287297307317327337347357367377387397407417427437447457467477487497507517527537547557567577587597607617627637647657667677687697707717727737747757767777787797807817827837847857867877887897907917927937947957967977987998008018028038048058068078088098108118128138148158168178188198208218228238248258268278288298308318328338348358368378388398408418428438448458468478488498508518528538548558568578588598608618628638648658668678688698708718728738748758768778788798808818828838848858868878888898908918928938948958968978988999009019029039049059069079089099109119129139149159169179189199209219229239249259269279289299309319329339349359369379389399409419429439449459469479489499509519529539549559569579589599609619629639649659669679689699709719729739749759769779789799809819829839849859869879889899909919929939949959969979989991000100110021003100410051006100710081009101010111012101310141015101610171018101910201021102210231024102510261027102810291030103110321033103410351036103710381039104010411042104310441045104610471048104910501051105210531054105510561057105810591060106110621063106410651066106710681069107010711072107310741075107610771078107910801081108210831084108510861087108810891090109110921093109410951096109710981099110011011102110311041105110611071108110911101111111211131114111511161117111811191120112111221123112411251126112711281129113011311132113311341135113611371138113911401141114211431144114511461147114811491150115111521153115411551156115711581159116011611162116311641165116611671168116911701171117211731174117511761177117811791180118111821183118411851186118711881189119011911192119311941195119611971198119912001201120212031204120512061207120812091210121112121213121412151216121712181219122012211222122312241225122612271228122912301231123212331234123512361237123812391240124112421243124412451246124712481249125012511252125312541255125612571258125912601261126212631264126512661267126812691270127112721273127412751276127712781279128012811282128312841285128612871288128912901291129212931294129512961297129812991300130113021303130413051306130713081309131013111312131313141315131613171318131913201321132213231324132513261327132813291330133113321333133413351336133713381339134013411342134313441345134613471348134913501351135213531354135513561357135813591360136113621363136413651366136713681369137013711372137313741375137613771378137913801381138213831384138513861387138813891390139113921393139413951396139713981399140014011402140314041405140614071408140914101411141214131414141514161417141814191420142114221423142414251426142714281429143014311432143314341435143614371438143914401441144214431444144514461447144814491450145114521453145414551456145714581459146014611462146314641465146614671468146914701471147214731474147514761477147814791480148114821483148414851486148714881489149014911492149314941495149614971498149915001501150215031504
  1. <template>
  2. <ReportModule
  3. :showTitle="true"
  4. :titleList="[state.groupTitle]"
  5. :showDescribe="true"
  6. tableOrChart="chart"
  7. :showPrintBtn="false"
  8. :showExportBtn="false"
  9. >
  10. <template #title_right>
  11. <EchartType
  12. :chartTypeList="state.problemAnalysisData.chartTypeList"
  13. :current="state.problemAnalysisData.chartType"
  14. @ChangeEchartType="
  15. (val) => ChangeEchartType(val, 'problemAnalysisData')
  16. "
  17. />
  18. </template>
  19. <template #module_table_chart>
  20. <template v-if="state.problemAnalysisData.data.length > 0">
  21. <BarLineCharts
  22. v-if="state.problemAnalysisData.chartType == 'line_bar_chart'"
  23. :legendList="state.problemAnalysisData.legendList"
  24. :showBarLegendIndex="state.problemAnalysisData.showBarLegendIndex"
  25. title="得分率"
  26. :data="state.problemAnalysisData.data"
  27. @HandleChartClick="HandleChartClick"
  28. @ChangeChartOrder="
  29. (sortType, legendData, barIndex) =>
  30. ChangeChartOrder(sortType, legendData, barIndex, 1)
  31. "
  32. >
  33. </BarLineCharts>
  34. <BarsCharts
  35. v-if="state.problemAnalysisData.chartType == 'vertical_bar'"
  36. :key="state.chartKey"
  37. :data="state.problemAnalysisData.data"
  38. :legendList="state.problemAnalysisData.legendList"
  39. :showSortSelectbox="true"
  40. unit="%"
  41. title="得分率"
  42. :isClick="true"
  43. @HandleChartClick="HandleChartClick"
  44. @ChangeChartOrder="
  45. (sortType, legendData, barIndex) =>
  46. ChangeChartOrder(sortType, legendData, barIndex, 1)
  47. "
  48. >
  49. </BarsCharts>
  50. <RadarCharts
  51. v-if="state.problemAnalysisData.chartType == 'radar_chart'"
  52. :key="state.chartKey"
  53. :data="state.problemAnalysisData.data"
  54. :legendList="state.problemAnalysisData.legendList"
  55. :showCheckBox="true"
  56. :openShowAllLegend="true"
  57. :isClick="true"
  58. @HandleChartClick="HandleChartClick"
  59. >
  60. </RadarCharts>
  61. </template>
  62. <div
  63. v-else
  64. class="no_content_data"
  65. v-loading="state.dataLoading"
  66. :element-loading-text="state.loadingText"
  67. element-loading-spinner="el-icon-loading"
  68. element-loading-background="#ffffff"
  69. >
  70. <span>暂无数据</span>
  71. </div>
  72. </template>
  73. <template #module_describe>
  74. 展示每道试题的得分率图。得分率指实际得分/考核分的比值,换算成的百分数。可以用于分析每道试题的难易程度和质量,试题得分率高意味着试题难度低或者学生整体水平高;得分率低意味着试题难度较高或者考生整体水平较低。点击每道试题的柱或雷达图的题号可在下方查看该题每个班的得分率情况。
  75. </template>
  76. </ReportModule>
  77. <ReportModule
  78. :showTitle="true"
  79. :titleList="[state.groupTitle, state.groupName]"
  80. :showDescribe="true"
  81. tableOrChart="chart"
  82. :showPrintBtn="false"
  83. :showExportBtn="false"
  84. >
  85. <template #title_right>
  86. <EchartType
  87. :chartTypeList="state.majorQuestionData.chartTypeList"
  88. :current="state.majorQuestionData.chartType"
  89. @ChangeEchartType="(val) => ChangeEchartType(val, 'majorQuestionData')"
  90. />
  91. </template>
  92. <template #module_table_chart>
  93. <template v-if="state.majorQuestionData.data.length > 0">
  94. <BarLineCharts
  95. ref="majorQuestionCharts"
  96. v-if="state.majorQuestionData.chartType == 'line_bar_chart'"
  97. :legendList="state.majorQuestionData.legendList"
  98. :showBarLegendIndex="state.majorQuestionData.showBarLegendIndex"
  99. title="得分率"
  100. :data="state.majorQuestionData.data"
  101. @HandleChartClick="HandleQuestionChartClick"
  102. @ChangeChartOrder="
  103. (sortType, legendData, barIndex) =>
  104. ChangeChartOrder(sortType, legendData, barIndex, 2)
  105. "
  106. >
  107. </BarLineCharts>
  108. <BarsCharts
  109. ref="majorQuestionCharts"
  110. :key="state.chartKey"
  111. v-if="state.majorQuestionData.chartType == 'vertical_bar'"
  112. :data="state.majorQuestionData.data"
  113. :showSortSelectbox="true"
  114. :legendList="state.majorQuestionData.legendList"
  115. unit="%"
  116. title="得分率"
  117. :isClick="true"
  118. @HandleChartClick="HandleQuestionChartClick"
  119. @ChangeChartOrder="
  120. (sortType, legendData, barIndex) =>
  121. ChangeChartOrder(sortType, legendData, barIndex, 2)
  122. "
  123. ></BarsCharts>
  124. <RadarCharts
  125. v-if="state.majorQuestionData.chartType == 'radar_chart'"
  126. :key="state.chartKey"
  127. :data="state.majorQuestionData.data"
  128. :legendList="state.majorQuestionData.legendList"
  129. :showCheckBox="true"
  130. :openShowAllLegend="true"
  131. :isClick="true"
  132. @HandleChartClick="HandleQuestionChartClick"
  133. >
  134. </RadarCharts>
  135. </template>
  136. <div
  137. v-else
  138. class="no_content_data"
  139. v-loading="state.dataLoading"
  140. :element-loading-text="state.loadingText"
  141. element-loading-spinner="el-icon-loading"
  142. element-loading-background="#ffffff"
  143. >
  144. <span>暂无数据</span>
  145. </div>
  146. </template>
  147. <template #module_describe>
  148. 展示每道试题的得分率图。得分率指实际得分/考核分的比值,换算成的百分数。可以用于分析每道试题的难易程度和质量,试题得分率高意味着试题难度低或者学生整体水平高;得分率低意味着试题难度较高或者考生整体水平较低。点击每道试题的柱或雷达图的题号可在下方查看该题每个班的得分率情况。
  149. </template>
  150. </ReportModule>
  151. <ReportModule
  152. :showTitle="true"
  153. :titleList="[state.groupTitle, state.questionTitle]"
  154. :showDescribe="true"
  155. tableOrChart="chart"
  156. :showPrintBtn="false"
  157. :showExportBtn="false"
  158. >
  159. <template #title_right>
  160. <EchartType
  161. :chartTypeList="state.questionScoreStatsData.chartTypeList"
  162. :current="state.questionScoreStatsData.chartType"
  163. @ChangeEchartType="
  164. (val) => ChangeEchartType(val, 'questionScoreStatsData')
  165. "
  166. />
  167. </template>
  168. <template #module_table_chart>
  169. <template v-if="state.questionScoreStatsData.datax.length > 0">
  170. <BarScoringRateVertical
  171. v-if="state.questionScoreStatsData.chartType == 'vertical_bar'"
  172. :datax="state.questionScoreStatsData.datax"
  173. :datay="state.questionScoreStatsData.dataStackY"
  174. :tooltipData="state.questionScoreStatsData.tooltipData"
  175. :isShowMarkLine="true"
  176. :average="state.questionScoreStatsData.rate"
  177. :markLineData="state.questionScoreStatsData.markLineData"
  178. :color="['#3BA272', '#EE6666']"
  179. typeName="得分率"
  180. :isClick="true"
  181. @HandleChartClick="HandleQuestionScoreChartClick"
  182. ></BarScoringRateVertical>
  183. <DifferenceChart
  184. v-if="state.questionScoreStatsData.chartType == 'difference_chart'"
  185. :datax="state.questionScoreStatsData.datax"
  186. :datay="state.questionScoreStatsData.datay"
  187. unit="%"
  188. type="1"
  189. title="得分率"
  190. :rate="state.questionScoreStatsData.rate"
  191. @HandleChartClick="HandleQuestionScoreChartClick"
  192. >
  193. </DifferenceChart>
  194. </template>
  195. <div
  196. v-else
  197. class="no_content_data"
  198. v-loading="state.dataLoading"
  199. :element-loading-text="state.loadingText"
  200. element-loading-spinner="el-icon-loading"
  201. element-loading-background="#ffffff"
  202. >
  203. <span>暂无数据</span>
  204. </div>
  205. </template>
  206. <template #module_describe>
  207. 说明:通过上面图形,可查看该题每个班的得分率情况,由图中可以看出,得分率最高为<span
  208. style="color: #3ba272"
  209. >{{ state.questionScoreStatsData.maxClass }}</span
  210. >,最低为<span style="color: #ee6666">{{
  211. state.questionScoreStatsData.minClass
  212. }}</span
  213. >。点击每个班级的柱可在下方查看该班各个选项/得分的人数分布。
  214. </template>
  215. </ReportModule>
  216. <ReportModule
  217. :showTitle="true"
  218. :titleList="[state.groupTitle, state.questionTitle, state.classTitle]"
  219. :showDescribe="true"
  220. tableOrChart="qita"
  221. :showPrintBtn="false"
  222. :showExportBtn="false"
  223. >
  224. <template #module_qita>
  225. <div class="content_left answer">
  226. <BarChart
  227. v-if="state.questionAnswerData.datax.length"
  228. :datax="state.questionAnswerData.datax"
  229. :datay="state.questionAnswerData.datay"
  230. typeName="人数"
  231. :showNuitY="false"
  232. unit="人"
  233. :unitX="
  234. state.questionAnswerData.questionType == '单选题' ||
  235. state.questionAnswerData.questionType == '多选题' ||
  236. state.questionAnswerData.questionType == '判断题'
  237. ? ''
  238. : '分'
  239. "
  240. :showMarkPoint="false"
  241. :isShowMarkLine="false"
  242. :color="state.questionAnswerData.color"
  243. :answerValue="state.questionAnswerData.answerValue"
  244. :answerScore="state.questionAnswerData.answerScore"
  245. :fullMark="state.questionAnswerData.fullMark"
  246. :isClick="true"
  247. @HandleChartClick="HandleQuestionAnswerChartClick"
  248. style="height: 300px"
  249. ></BarChart>
  250. <div
  251. v-else
  252. class="module_chart no_content_data"
  253. v-loading="state.dataLoading"
  254. :element-loading-text="state.loadingText"
  255. element-loading-spinner="el-icon-loading"
  256. element-loading-background="#ffffff"
  257. >
  258. <span>暂无数据</span>
  259. </div>
  260. </div>
  261. <div class="content_right answer table_42">
  262. <el-table
  263. border
  264. :data="state.questionAnswerData.tableData"
  265. stripe
  266. align="left"
  267. :header-row-style="HeaderRowStyle"
  268. >
  269. <el-table-column :label="state.classTitle" align="center">
  270. <el-table-column
  271. prop="title"
  272. align="center"
  273. label="名称"
  274. ></el-table-column>
  275. <el-table-column
  276. prop="value"
  277. align="center"
  278. label="数值"
  279. ></el-table-column>
  280. </el-table-column>
  281. </el-table>
  282. </div>
  283. </template>
  284. <template #module_describe>
  285. 说明:通过柱图,可查看该题下该班所有学生的选项或试题得分情况。点击每个柱可查看该选项或分数的学生明细和答题卡答题情况。
  286. </template>
  287. </ReportModule>
  288. <ReportModule
  289. :showTitle="true"
  290. :titleList="[
  291. state.groupTitle,
  292. state.questionTitle,
  293. state.classTitle,
  294. state.optionTitle,
  295. ]"
  296. :showDescribe="true"
  297. tableOrChart="qita"
  298. :showPrintBtn="false"
  299. :showExportBtn="false"
  300. >
  301. <template #title_right>
  302. <el-button class="default_button" @click="VisibleQuestionCard">
  303. <img src="@/assets/icon/card_view.webp" />批量查看
  304. </el-button>
  305. </template>
  306. <template #module_qita>
  307. <div class="content_left paper_card"></div>
  308. <div class="content_right paper_card table_42">
  309. <el-table
  310. border
  311. :data="state.majorAnswerData.tableData"
  312. stripe
  313. highlight-current-row
  314. align="left"
  315. height="401px"
  316. row-key="studentRegistrationCode"
  317. :row-style="{ cursor: 'pointer' }"
  318. :row-class-name="TableRowClassName"
  319. @row-click="HandleRowClick"
  320. >
  321. <el-table-column
  322. prop="studentUserName"
  323. align="center"
  324. label="学生"
  325. show-overflow-tooltip
  326. ></el-table-column>
  327. <el-table-column
  328. label="班级"
  329. prop="className"
  330. align="center"
  331. min-width="70"
  332. show-overflow-tooltip
  333. ></el-table-column>
  334. <el-table-column
  335. prop="questionScore"
  336. align="center"
  337. label="得分"
  338. width="70"
  339. show-overflow-tooltip
  340. ></el-table-column>
  341. <el-table-column
  342. prop="paperScore"
  343. align="center"
  344. label="成绩"
  345. width="70"
  346. show-overflow-tooltip
  347. ></el-table-column>
  348. <el-table-column align="center" label="操作">
  349. <template #default="scope">
  350. <el-button type="text" @click.stop="handleClick(scope.row)"
  351. >查看答题卡</el-button
  352. >
  353. </template>
  354. </el-table-column>
  355. </el-table>
  356. </div>
  357. </template>
  358. <template #module_describe>
  359. 说明:点击左侧表格列可切换学生查看具体学生答题图片,点击学生姓名可在下方查看学生均衡分析,点击学生成绩可查看学生答题卡原卷。
  360. </template>
  361. </ReportModule>
  362. <ReportModule
  363. ref="reportModuleRef"
  364. :showTitle="true"
  365. :titleList="[state.groupTitle]"
  366. :showDescribe="true"
  367. tableOrChart="table"
  368. :showPrintBtn="false"
  369. :showExportBtn="true"
  370. :currentPage="state.majorTableData.currentPage"
  371. :pageSize="state.majorTableData.pageSize"
  372. :total="state.majorTableData.total"
  373. @ChangePageSize="ChangePageSize"
  374. @ChangeCurrentPage="ChangeCurrentPage"
  375. @ExportExcel="ExportExcel"
  376. >
  377. <template #module_table_chart>
  378. <el-table
  379. :data="state.majorTableData.tableData"
  380. ref="majorTable"
  381. border
  382. stripe
  383. align="left"
  384. :key="state.majorTableData.tableKey"
  385. >
  386. <template v-for="item in state.majorTableData.headerData">
  387. <el-table-column
  388. v-if="item.display"
  389. :key="item.prop"
  390. align="center"
  391. :prop="item.prop"
  392. :min-width="item.label.length > 4 ? 110 : 90"
  393. :label="item.label"
  394. fixed="left"
  395. show-overflow-tooltip
  396. >
  397. <template #default="scope">
  398. {{ scope.row[item.prop] || "-"
  399. }}{{
  400. item.prop == "estimatedScoreRate" && scope.row[item.prop]
  401. ? "%"
  402. : ""
  403. }}
  404. </template>
  405. </el-table-column>
  406. </template>
  407. <template v-for="parent in state.majorTableData.changeHeaderData">
  408. <el-table-column
  409. v-if="parent.display"
  410. align="center"
  411. :prop="parent.prop"
  412. :label="parent.label"
  413. :key="parent.prop"
  414. >
  415. <template v-for="item in state.majorTableData.childHeaderData">
  416. <el-table-column
  417. v-if="item.display"
  418. :prop="`${parent.prop}_${item.prop}`"
  419. :key="`${parent.prop}_${item.prop}`"
  420. align="center"
  421. :label="item.label"
  422. :min-width="item.label.length > 4 ? 100 : 80"
  423. show-overflow-tooltip
  424. >
  425. <template #default="scope">
  426. <span v-if="item.prop.indexOf('Rate') > -1">{{
  427. scope.row[`${parent.prop}_${item.prop}`]
  428. ? `${scope.row[`${parent.prop}_${item.prop}`]}%`
  429. : "-"
  430. }}</span>
  431. <span
  432. v-else
  433. @click="OpenStudentDialog(item, parent, scope.row)"
  434. :class="{
  435. table_row_blue:
  436. (item.prop.indexOf('Count') > -1 ||
  437. item.prop.indexOf('Number') > -1) &&
  438. scope.row?.[`${parent.prop}_${item.prop}`] > 0,
  439. }"
  440. >{{
  441. scope.row?.[`${parent.prop}_${item.prop}`] ?? "-"
  442. }}</span
  443. >
  444. </template>
  445. </el-table-column>
  446. </template>
  447. </el-table-column>
  448. </template>
  449. </el-table>
  450. </template>
  451. </ReportModule>
  452. <!-- 批量查看小题答题卡 -->
  453. <!-- <QuestionCard :subjectId="state.analysisStore.filterObject.subjectId" :questionId="state.cardQuestionId" :platformNumbers="cardRegistrationCodeList" :groupTitle="groupTitle" :groupName="groupName" :questionTitle="questionTitle" :classTitle="classTitle" :optionTitle="optionTitle" :tagActive="tagActive" :showDialog="showQuestionCardDialog" @CloseDialog="CloseQuestionCardDialog"></QuestionCard> -->
  454. </template>
  455. <script lang="ts" setup>
  456. import ReportModule from "@/components/ReportModule.vue";
  457. import EchartType from "@/components/EchartType.vue";
  458. import { useAnalysisStore } from "@/store/analysis";
  459. import {
  460. questionGroupAnalysis,
  461. queryAnswerListByAnswerAndScore,
  462. publicExport
  463. } from "@/api/analysis";
  464. import BarLineCharts from "@/components/echarts/barLineCharts.vue"; //柱状图折线图组合图组件
  465. import RadarCharts from "@/components/echarts/radarCharts.vue"; //雷达图
  466. import BarsCharts from "@/components/echarts/barsCharts.vue"; //多柱状图组件
  467. import BarScoringRateVertical from "@/components/echarts/barScoringRate_vertical.vue"; //得分率 纵向柱状图
  468. import DifferenceChart from "@/components/echarts/differenceChart.vue"; //率差图
  469. import BarChart from "@/components/echarts/barChart_answer.vue"; //单柱状图
  470. import { downloadExcel, GetExcelFileName } from "@/utils/exportExcel";
  471. import { onMounted, reactive, watch, ref, computed,nextTick } from "vue";
  472. import { cloneDeep } from "lodash-es";
  473. const analysisStore = useAnalysisStore();
  474. const reportModuleRef = ref<any>(null);
  475. const getExamName = computed(() => {
  476. return analysisStore.analysisExamInfo.examName || "";
  477. });
  478. const state = reactive({
  479. questionGroupDefault: [
  480. {
  481. name: "小题分析",
  482. code: "problem",
  483. },
  484. {
  485. name: "题型分析",
  486. code: 11,
  487. },
  488. {
  489. name: "错题分析",
  490. code: "errors",
  491. },
  492. {
  493. name: "客观题分析",
  494. code: "selectQuestion",
  495. },
  496. {
  497. name: "命题分析",
  498. code: "proposition",
  499. },
  500. ],
  501. questionGroupList: [], //试题分组标签 动态接口获取
  502. tagActive: "problem", //选择的试题分组标签
  503. groupTitle: "分组分析",
  504. groupPreviousTitle: "",
  505. knowledgeLayeredTitle: "", //知识点分层标题
  506. groupName: "", // 分组名称
  507. questionTitle: "", //题目名称
  508. classTitle: "", //班级名称
  509. optionTitle: "", //选项名称
  510. studentUserName: "", //学生名称
  511. studentRegistrationCode: "", //学生code
  512. questionTypesData: {
  513. chartKey: 0,
  514. refresh: false,
  515. data: [],
  516. tableData: [],
  517. },
  518. problemAnalysisData: {
  519. chartType: "line_bar_chart", //默认显示折线图柱状图line_bar_chart
  520. chartTypeList: [
  521. {
  522. label: "组合图",
  523. value: "line_bar_chart",
  524. },
  525. {
  526. label: "柱状图",
  527. value: "vertical_bar",
  528. },
  529. {
  530. label: "雷达图",
  531. value: "radar_chart",
  532. },
  533. ],
  534. legendList: [],
  535. defaultLegendList: [],
  536. showBarLegendIndex: 1,
  537. questionList: [], // 题目列表
  538. questionTableList: [], // 小题分析table
  539. groupList: [], // 分组题目列表
  540. groupTableList: [], // 分组table
  541. headerList: [],
  542. changeHeaderList: [],
  543. childHeaderList: [],
  544. questionListIndex: 0, //题目 索引
  545. data: [], //柱状图
  546. }, //小题分析数据
  547. majorQuestionData: {
  548. chartType: "line_bar_chart", //默认显示折线图柱状图line_bar_chart
  549. chartTypeList: [
  550. {
  551. label: "组合图",
  552. value: "line_bar_chart",
  553. },
  554. {
  555. label: "柱状图",
  556. value: "vertical_bar",
  557. },
  558. {
  559. label: "雷达图",
  560. value: "radar_chart",
  561. },
  562. ],
  563. legendList: [],
  564. defaultLegendList: [],
  565. showBarLegendIndex: 1,
  566. questionListIndex: 0, //题目 索引
  567. data: [], //柱状图
  568. }, //分组题目 对应的题目列表
  569. questionScoreStatsData: {
  570. checked: false,
  571. isIndeterminate: true,
  572. markLineData: [], //平均分
  573. chartTypeList: [
  574. {
  575. label: "柱状图",
  576. value: "vertical_bar",
  577. },
  578. {
  579. label: "率差图",
  580. value: "difference_chart",
  581. },
  582. ],
  583. chartType: "vertical_bar", //默认显示率差图vertical_bar
  584. datax: [], //x轴数据
  585. datay: [], //Y轴数据
  586. dataStackY: [], //Y轴数据
  587. tooltipData: [], //
  588. rate: 60, //中间x的刻度线的值 数字类型
  589. maxClass: "", // 得分率最高班
  590. minClass: "", // 得分率最低班
  591. }, //小题分析 /第N题
  592. questionAnswerData: {
  593. datax: [], //x轴数据
  594. datay: [], //Y轴数据
  595. color: "#5470C6",
  596. answerValue: "", //正确答案 柱状图显示绿色
  597. answerScore: [], //答案所能得的分数
  598. questionType: "", //类型
  599. fullMark: "", //满分
  600. // average:0,//平均分辅助线
  601. tableData: [],
  602. classListIndex: 0, // 索引
  603. }, //小题分析 /第N题 / N班
  604. majorAnswerData: {
  605. paperUrl: "", //试卷地址
  606. tableData: [], //表格数据
  607. rowIndex: 0,
  608. }, // 通过答案或者分数查询某题作答情况
  609. majorStudentData: {
  610. tableData: [],
  611. legendList: [],
  612. radarChartData: [],
  613. excellentCourseList: "", //优秀科目
  614. inferiorCourseList: "", //弱势科目
  615. }, //科目得分率
  616. majorTableData: {
  617. tableKey: 0,
  618. tableData: [], //小题分析表数据
  619. allTableData: [],
  620. headerData: [], //固定表头数据
  621. changeHeaderData: [], //动态表头数据
  622. childHeaderData: [], //子级表头数据
  623. pageSize: 10, //每页显示数据
  624. total: 0, //总数
  625. currentPage: 1, //当前页
  626. }, //小题分析表 + 题目分组分析
  627. errorQuestionData: {
  628. tableKey: 0,
  629. allData: [], //
  630. gradeValue: "", //选中的年级
  631. className: "", //选中的年级名称
  632. type: "",
  633. gradeData: [], //年级下拉选项
  634. tableData: [], //错题分析表数据
  635. }, //错题分析表
  636. studentPreviousExamData: {
  637. data: null,
  638. chartType: "line_chart",
  639. selectList: [
  640. {
  641. value: "standardScore",
  642. name: "标准分",
  643. },
  644. {
  645. value: "scoreRate",
  646. name: "得分率",
  647. },
  648. {
  649. value: "classRank",
  650. name: "班排",
  651. },
  652. {
  653. value: "schoolRank",
  654. name: "校排",
  655. },
  656. {
  657. value: "examRank",
  658. name: "联排",
  659. },
  660. ],
  661. selectVal: "standardScore",
  662. selectName: "标准分",
  663. lineChartData: {
  664. datax: [],
  665. datay: [],
  666. title: [],
  667. tooltipData: [],
  668. },
  669. barChartData: {
  670. //柱状图
  671. legendList: [],
  672. data: [],
  673. },
  674. }, //学生成绩历次考试分析数据
  675. groupPrevious: {
  676. loading: false,
  677. examNameList: [],
  678. examChartList: [],
  679. selectedLegendList: [],
  680. examChartIndex: [],
  681. headerList: [],
  682. dataList: [],
  683. }, //题目分组历次分析(图表数据)
  684. stuGroupPreviousChart: {
  685. examNameList: [],
  686. examChartList: [],
  687. examChartIndex: [],
  688. }, //学生组块历次图
  689. stuGroupPreviousChartStuInfo: {
  690. studentName: "", //学生姓名搜索条件
  691. studentCodeList: [],
  692. }, //学生历次分析图 搜索学生信息
  693. stuGroupPrevious: {
  694. total: 0,
  695. pageNum: 1,
  696. pageSize: 10,
  697. headerList: [],
  698. tableList: [],
  699. }, //题目分组历次分析学生数据表(分页)
  700. classNameList: [],
  701. showHeaderSet: false, //是否显示设置表头
  702. showBenchTaskSelect: false, //历次考试弹框
  703. exportLoading: false, // 题型导出loading
  704. exportErrorLoading: false, // 错题导出loading
  705. chartKey: 0,
  706. dataLoading: false,
  707. loadingText: "加载中,请稍后……",
  708. errorPdfLoading: false, //错题导出PDF loading
  709. paperInfo: {
  710. examPaperId: "", //考试科目id
  711. platformNumber: "", //学籍号平台号
  712. questionId: "", //题目id
  713. }, //学生试卷信息
  714. paperTitle: "", //学生试卷标题
  715. showStudentPaperDialog: false, //是否显示学生答题卡弹窗
  716. paperInfos: {
  717. examPaperId: "", //考试科目id
  718. platformNumber: "", //学籍号平台号
  719. questionId: "", //题目id
  720. },
  721. knowledgeInputDialog: false,
  722. dialogData: {
  723. apiName: "",
  724. showDialog: false,
  725. title: "",
  726. tableTitle: "",
  727. fiveRateName: "", //当前选择的五率的名称
  728. selectSubjectName: "", //当前选择的科目名称
  729. selectSchoolLevel: "", //当前选择的学校级别:0-联考 1-学校分组 2-具体学校
  730. selectSchoolName: "", //当前选择的学校名称:联校 组合学校 单校名称
  731. selectClassLevel: "", //当前选择的班级级别:0-年级 1-组合班级 2-具体班级
  732. selectClassName: "", //当前选择的班级名称:年级 组合班级 单班
  733. questionId: "", //试题ID
  734. isPaper: "", //是否为全卷0-非1-是全卷
  735. quesScoreType: "", //选择的类型0-满分人数 1-0分人数 3-优秀率、良好率等
  736. knowledgeId: "", //如果选择的是知识点,则需要返回知识点ID
  737. questionGroupName: "", //试题分组的名称
  738. groupTitle: "",
  739. },
  740. showQuestionCardDialog: false,
  741. cardQuestionId: "", // 批量查看答题卡试题id
  742. cardRegistrationCodeList: [], // 批量查看答题卡学生账号数组
  743. });
  744. const majorQuestionCharts = ref(null);
  745. const majorTable = ref(null);
  746. //排序
  747. const ChangeChartOrder = (sortType, legendData, barIndex, number) => {
  748. if (number == 1) {
  749. const fullVolume = state.problemAnalysisData.groupList.filter(
  750. (item) => item.groupCode == 999,
  751. ); //全卷
  752. const groupList = state.problemAnalysisData.groupList.filter(
  753. (item) => item.groupCode != 999,
  754. );
  755. //题目分组排序
  756. if (sortType == "2") {
  757. //从低到高
  758. groupList.sort(function (a, b) {
  759. return (
  760. (a?.classList?.[barIndex]?.questionStats?.scoreRate || 0) -
  761. (b?.classList?.[barIndex]?.questionStats?.scoreRate || 0)
  762. );
  763. });
  764. } else if (sortType == "3") {
  765. //从高到低
  766. groupList.sort(function (a, b) {
  767. return (
  768. (b?.classList?.[barIndex]?.questionStats?.scoreRate || 0) -
  769. (a?.classList?.[barIndex]?.questionStats?.scoreRate || 0)
  770. );
  771. });
  772. } else {
  773. //默认排序 按题号排序
  774. groupList.sort(function (a, b) {
  775. return (a?.groupCode || 0) - (b.groupCode || 0);
  776. });
  777. }
  778. state.problemAnalysisData.groupList = [...groupList, ...fullVolume];
  779. state.problemAnalysisData.data = [];
  780. state.problemAnalysisData.groupList.forEach((group) => {
  781. const scoreRateArr =
  782. group.classList?.map((item) => item?.questionStats?.scoreRate ?? 0) ||
  783. [];
  784. state.problemAnalysisData.data.push([group.groupName, ...scoreRateArr]);
  785. });
  786. let chartTitle = ["group"];
  787. state.problemAnalysisData.changeHeaderList.forEach((item) => {
  788. chartTitle.push(item.label);
  789. });
  790. state.problemAnalysisData.data.unshift(chartTitle); // 柱状图 图例标题
  791. state.problemAnalysisData.data.pop(); // 删除最后一行 全卷
  792. state.problemAnalysisData.legendList = legendData;
  793. //大题分析 /阅读表达
  794. GetClassQuestionData(0, state.problemAnalysisData.groupList[0].groupName);
  795. } else {
  796. const fullVolume = state.problemAnalysisData.questionList.filter(
  797. (item) => item.showCode == 999,
  798. ); //全卷
  799. const questionList = state.problemAnalysisData.questionList.filter(
  800. (item) => item.showCode != 999,
  801. );
  802. if (sortType == "2") {
  803. //从低到高
  804. questionList.sort(function (a, b) {
  805. return (
  806. (a?.classList?.[barIndex]?.questionStats?.scoreRate || 0) -
  807. (b?.classList?.[barIndex]?.questionStats?.scoreRate || 0)
  808. );
  809. });
  810. } else if (sortType == "3") {
  811. //从高到低
  812. questionList.sort(function (a, b) {
  813. return (
  814. (b?.classList?.[barIndex]?.questionStats?.scoreRate || 0) -
  815. (a?.classList?.[barIndex]?.questionStats?.scoreRate || 0)
  816. );
  817. });
  818. } else {
  819. //默认排序 按题号排序
  820. questionList.sort(function (a, b) {
  821. return (a?.showCode || 0) - (b.showCode || 0);
  822. });
  823. }
  824. state.problemAnalysisData.questionList = [...questionList, ...fullVolume];
  825. //Y轴数据
  826. state.majorQuestionData.data = [];
  827. state.problemAnalysisData.questionList.forEach((question) => {
  828. const scoreRateArr = question.classList.map(
  829. (item) => item?.questionStats?.scoreRate || 0,
  830. );
  831. state.majorQuestionData.data.push([
  832. question.questionName,
  833. ...scoreRateArr,
  834. ]);
  835. });
  836. let chartTitle = [];
  837. state.problemAnalysisData.changeHeaderList.forEach((item) => {
  838. chartTitle.push(item.label);
  839. });
  840. chartTitle.unshift("group");
  841. state.majorQuestionData.legendList = legendData;
  842. state.majorQuestionData.data.unshift(chartTitle); // 柱状图 图例标题
  843. //大题分析 /阅读表达 / 第32题
  844. GetQuestionStatsData(0);
  845. }
  846. };
  847. //试题图表切换公共方法
  848. const ChangeEchartType = (value, prop) => {
  849. if (prop == "problemAnalysisData") {
  850. ChangeChartOrder("1", state.problemAnalysisData.defaultLegendList, "", 1);
  851. } else if (prop == "majorQuestionData") {
  852. ChangeChartOrder("1", state.majorQuestionData.defaultLegendList, "", 2);
  853. }
  854. state[prop].chartType = value;
  855. };
  856. //题目分组分析
  857. const GetQuestionGroupAnalysisData = () => {
  858. state.problemAnalysisData.data = []; //柱状图折线图
  859. state.dataLoading = true;
  860. questionGroupAnalysis({
  861. ...analysisStore.filterObject,
  862. analysisType: 4,
  863. })
  864. .then((res) => {
  865. // console.log("打印题目分组分析接口返回数据",res)
  866. state.problemAnalysisData.data = []; //柱状图折线图
  867. if (res.code == 200 && res.data && res.data.groupList.length) {
  868. const { data } = res;
  869. const groupList = data.groupList || [];
  870. const changeHeaderList = data.changeHeaderList || []; //柱状图折线图 班级名称
  871. state.problemAnalysisData.groupList = groupList;
  872. state.problemAnalysisData.groupTableList = cloneDeep(groupList);
  873. state.problemAnalysisData.headerList = data.headerList || [];
  874. state.problemAnalysisData.changeHeaderList = changeHeaderList;
  875. state.problemAnalysisData.childHeaderList = data.childHeaderList || [];
  876. state.chartKey++;
  877. //Y轴数据
  878. state.problemAnalysisData.data = groupList.map((item) => {
  879. return [item.groupName];
  880. });
  881. groupList.forEach((group, key) => {
  882. const scoreRateArr =
  883. group.classList?.map(
  884. (item) => item?.questionStats?.scoreRate ?? 0,
  885. ) || [];
  886. state.problemAnalysisData.data[key].push(...scoreRateArr);
  887. });
  888. let chartTitle = [],
  889. titleType = [],
  890. classSelectLegend = [];
  891. changeHeaderList.forEach((item) => {
  892. chartTitle.push(item.label);
  893. titleType.push(item.type ? item.type : ""); //1柱状图 2折线
  894. if (item.prop != "0" && item.prop.indexOf("school_group") == -1) {
  895. classSelectLegend.push(item.label);
  896. }
  897. });
  898. chartTitle.unshift("group");
  899. if (analysisStore.filterObject.classLevel != 2) {
  900. //0 年级 1 组合班级 2具体班级
  901. let startIndex = 1,
  902. endIndex = 3;
  903. if (analysisStore.filterObject.schoolLevel == 2) {
  904. //0-联考 1-学校分组 2-具体学校
  905. startIndex =
  906. titleType.lastIndexOf(1) > -1 ? titleType.lastIndexOf(1) + 1 : 1;
  907. endIndex = titleType.indexOf(2) > -1 ? titleType.indexOf(2) + 2 : 3; //单校时显示联校、组合校、年级是柱子;班级组合、班级是折线。
  908. } else {
  909. //联校 组合学校 默认当前的为第一个选中的柱子
  910. const barFirtIndex =
  911. titleType.lastIndexOf(1) > -1 ? titleType.lastIndexOf(1) + 1 : 1;
  912. startIndex = barFirtIndex;
  913. endIndex = barFirtIndex + 2;
  914. }
  915. state.problemAnalysisData.legendList =
  916. chartTitle.length > 1 ? chartTitle.slice(startIndex, endIndex) : [];
  917. state.problemAnalysisData.showBarLegendIndex = endIndex - 2;
  918. } else {
  919. //单班 联校和组合学校不默认显示
  920. state.problemAnalysisData.legendList = classSelectLegend;
  921. state.problemAnalysisData.showBarLegendIndex =
  922. changeHeaderList.length - 1;
  923. }
  924. state.problemAnalysisData.defaultLegendList = cloneDeep(
  925. state.problemAnalysisData.legendList,
  926. );
  927. state.problemAnalysisData.data.unshift(chartTitle); // 柱状图 图例标题
  928. state.problemAnalysisData.data.pop(); // 删除最后一行 全卷
  929. //大题分析 /阅读表达
  930. GetClassQuestionData(0, groupList[0].groupName);
  931. // 大题分析表
  932. GetMajorTableData();
  933. } else {
  934. state.questionTypesData.data = [];
  935. state.questionTypesData.tableData = [];
  936. state.problemAnalysisData.showBarLegendIndex = 1;
  937. state.problemAnalysisData.groupList = [];
  938. state.problemAnalysisData.questionList = [];
  939. state.problemAnalysisData.headerList = [];
  940. state.problemAnalysisData.changeHeaderList = [];
  941. state.problemAnalysisData.childHeaderList = [];
  942. state.majorQuestionData.data = [];
  943. state.majorQuestionData.legendList = [];
  944. state.questionScoreStatsData.datax = [];
  945. state.questionScoreStatsData.datay = [];
  946. state.questionScoreStatsData.dataStackY = [];
  947. state.questionScoreStatsData.tooltipData = [];
  948. state.questionScoreStatsData.rate = 0;
  949. state.questionScoreStatsData.maxClass = "";
  950. state.questionScoreStatsData.minClass = "";
  951. state.questionAnswerData.datax = [];
  952. state.questionAnswerData.datay = [];
  953. state.questionAnswerData.answerValue = "";
  954. state.questionAnswerData.answerScore = [];
  955. state.questionAnswerData.questionType = "";
  956. state.questionAnswerData.fullMark = "";
  957. state.questionAnswerData.tableData = [];
  958. state.questionAnswerData.classListIndex = 0;
  959. state.majorAnswerData.tableData = [];
  960. state.majorAnswerData.paperUrl = "";
  961. state.majorAnswerData.rowIndex = 0;
  962. state.majorStudentData.tableData = [];
  963. state.majorStudentData.legendList = [];
  964. state.majorStudentData.radarChartData = [];
  965. state.majorStudentData.excellentCourseList = "";
  966. state.majorStudentData.inferiorCourseList = "";
  967. state.majorTableData.tableData = [];
  968. state.majorTableData.headerData = [];
  969. state.majorTableData.changeHeaderData = [];
  970. state.majorTableData.childHeaderData = [];
  971. state.majorTableData.pageSize = 10;
  972. state.majorTableData.total = 0;
  973. state.majorTableData.currentPage = 1;
  974. }
  975. })
  976. .finally(() => {
  977. state.dataLoading = false;
  978. });
  979. };
  980. // 点击柱状图折线图
  981. const HandleChartClick = (index, name) => {
  982. if (majorQuestionCharts.value) {
  983. majorQuestionCharts.value.ChangeSelectValue();
  984. }
  985. GetClassQuestionData(index, name); //题目分组分析
  986. };
  987. //大题分析 /阅读表达 点击柱状图
  988. const HandleQuestionChartClick = (index, name) => {
  989. GetQuestionStatsData(index); //小题分析
  990. };
  991. // 获取小题分析 /第N题
  992. const GetQuestionStatsData = (index) => {
  993. const questionData = cloneDeep(state.problemAnalysisData.questionList[index]);
  994. state.problemAnalysisData.questionListIndex = index;
  995. state.questionTitle = questionData?.questionName || "";
  996. if (questionData) {
  997. state.questionScoreStatsData.datax = []; // x轴数据
  998. state.questionScoreStatsData.datay = []; // Y轴数据
  999. state.questionScoreStatsData.dataStackY = []; // Y轴数据
  1000. state.questionScoreStatsData.tooltipData = []; // 提示框内容
  1001. let dataStackY = [];
  1002. const isHasEstimatedScore = state.problemAnalysisData.headerList.find(
  1003. (item) => item.prop == "estimatedScore",
  1004. ); //是否存在预估满分
  1005. if (questionData.classList.length > 1) {
  1006. const key = state.problemAnalysisData.showBarLegendIndex;
  1007. state.questionScoreStatsData.markLineData = []; //辅助线
  1008. const classList = questionData?.classList || [];
  1009. if (
  1010. analysisStore.filterObject.classLevel == 0 ||
  1011. analysisStore.filterObject.classLevel == 1
  1012. ) {
  1013. //0 年级 1 组合班级 2具体班级
  1014. state.questionScoreStatsData.rate = Number(
  1015. classList?.[key - 1]?.questionStats?.scoreRate || 0,
  1016. );
  1017. } else {
  1018. state.questionScoreStatsData.rate = Number(
  1019. classList?.[key]?.questionStats?.scoreRate || 0,
  1020. );
  1021. }
  1022. //辅助线
  1023. classList.forEach((item, index) => {
  1024. const keyIndex =
  1025. analysisStore.filterObject.classLevel == 0 ||
  1026. analysisStore.filterObject.classLevel == 1
  1027. ? key - (isHasEstimatedScore ? 2 : 1)
  1028. : key;
  1029. if (index <= keyIndex) {
  1030. const rate = Number(item?.questionStats?.scoreRate || 0);
  1031. state.questionScoreStatsData.markLineData.push({
  1032. legendName: item.groupName,
  1033. value: rate,
  1034. isShow: index == keyIndex ? true : false, //是否显示
  1035. });
  1036. }
  1037. });
  1038. questionData.classList = questionData.classList.slice(
  1039. isHasEstimatedScore ? key - 1 : key,
  1040. ); // 取折线图数据
  1041. } else {
  1042. state.questionScoreStatsData.rate = 60;
  1043. }
  1044. questionData.classList.forEach((item) => {
  1045. state.questionScoreStatsData.datax.push(item.groupName);
  1046. state.questionScoreStatsData.datay.push(item.questionStats.scoreRate);
  1047. dataStackY.push(item.questionStats.lossRate);
  1048. state.questionScoreStatsData.tooltipData.push({
  1049. list: [
  1050. {
  1051. name: "失分率",
  1052. value: `${item.questionStats.lossRate}%`,
  1053. },
  1054. ],
  1055. });
  1056. });
  1057. // 获取最大值
  1058. const max = Math.max(...state.questionScoreStatsData.datay);
  1059. // 获取最小值
  1060. const min = Math.min(...state.questionScoreStatsData.datay);
  1061. const maxValue = max.toFixed(2);
  1062. const minValue = min.toFixed(2);
  1063. let maxClass = [],
  1064. minClass = [];
  1065. state.questionScoreStatsData.datay.forEach((item, index) => {
  1066. if (Number(item) == Number(maxValue)) {
  1067. maxClass.push(state.questionScoreStatsData.datax[index]);
  1068. }
  1069. if (Number(item) == Number(minValue)) {
  1070. minClass.push(state.questionScoreStatsData.datax[index]);
  1071. }
  1072. });
  1073. state.questionScoreStatsData.minClass = minClass.join("、");
  1074. state.questionScoreStatsData.maxClass = maxClass.join("、");
  1075. state.questionScoreStatsData.dataStackY = [
  1076. state.questionScoreStatsData.datay,
  1077. dataStackY,
  1078. ];
  1079. }
  1080. // 获取小题分析 /第N题 / 第N班
  1081. GetQuestionAnswerData(state.problemAnalysisData.showBarLegendIndex);
  1082. };
  1083. //大题分析 /阅读表达
  1084. const GetClassQuestionData = (index, name) => {
  1085. state.majorQuestionData.data = [];
  1086. state.groupName = name;
  1087. const groupList = state.problemAnalysisData.groupList[index];
  1088. const questionList = groupList.questionList;
  1089. state.problemAnalysisData.questionList = questionList;
  1090. //Y轴数据
  1091. state.majorQuestionData.data = questionList.map((item) => {
  1092. return [item.questionName];
  1093. });
  1094. questionList.forEach((question, key) => {
  1095. const scoreRateArr = question.classList.map(
  1096. (item) => item?.questionStats?.scoreRate || 0,
  1097. );
  1098. state.majorQuestionData.data[key].push(...scoreRateArr);
  1099. });
  1100. let chartTitle = [],
  1101. titleType = [],
  1102. classSelectLegend = [];
  1103. state.problemAnalysisData.changeHeaderList.forEach((item) => {
  1104. chartTitle.push(item.label);
  1105. titleType.push(item.type ? item.type : ""); //1柱状图 2折线
  1106. if (item.prop != "0" && item.prop.indexOf("school_group") == -1) {
  1107. classSelectLegend.push(item.label);
  1108. }
  1109. });
  1110. chartTitle.unshift("group");
  1111. if (analysisStore.filterObject.classLevel != 2) {
  1112. //0 年级 1 组合班级 2具体班级
  1113. let startIndex = 1,
  1114. endIndex = 3;
  1115. if (analysisStore.filterObject.schoolLevel == 2) {
  1116. //0-联考 1-学校分组 2-具体学校
  1117. startIndex =
  1118. titleType.lastIndexOf(1) > -1 ? titleType.lastIndexOf(1) + 1 : 1;
  1119. endIndex = titleType.indexOf(2) > -1 ? titleType.indexOf(2) + 2 : 3; //单校时显示联校、组合校、年级是柱子;班级组合、班级是折线。
  1120. } else {
  1121. //联校 组合学校 默认当前的为第一个选中的柱子
  1122. const barFirtIndex =
  1123. titleType.lastIndexOf(1) > -1 ? titleType.lastIndexOf(1) + 1 : 1;
  1124. startIndex = barFirtIndex;
  1125. endIndex = barFirtIndex + 2;
  1126. }
  1127. state.majorQuestionData.legendList =
  1128. chartTitle.length > 1 ? chartTitle.slice(startIndex, endIndex) : [];
  1129. state.majorQuestionData.showBarLegendIndex = endIndex - 2;
  1130. } else {
  1131. //联校和组合学校不默认显示
  1132. state.majorQuestionData.legendList = classSelectLegend;
  1133. }
  1134. state.majorQuestionData.defaultLegendList = cloneDeep(
  1135. state.majorQuestionData.legendList,
  1136. );
  1137. state.majorQuestionData.data.unshift(chartTitle); // 柱状图 图例标题
  1138. //大题分析 /阅读表达 / 第32题
  1139. GetQuestionStatsData(0);
  1140. };
  1141. //切换 获取小题分析 /第N题 获取第N班答题列表
  1142. const HandleQuestionScoreChartClick = (index) => {
  1143. GetQuestionAnswerData(index + state.problemAnalysisData.showBarLegendIndex);
  1144. };
  1145. // 获取小题分析 /第N题 / 第N班 选项
  1146. const GetQuestionAnswerData = (index) => {
  1147. const isHasEstimatedScore = state.problemAnalysisData.headerList.find(
  1148. (item) => item.prop == "estimatedScore",
  1149. ); //是否存在预估满分
  1150. const newINdex = isHasEstimatedScore ? index - 1 : index;
  1151. const classList =
  1152. state.problemAnalysisData.questionList?.[
  1153. state.problemAnalysisData.questionListIndex
  1154. ]?.classList?.[newINdex];
  1155. // console.log("打印获取小题分析班级列表",classList)
  1156. state.questionAnswerData.answerValue =
  1157. state.problemAnalysisData.questionList[
  1158. state.problemAnalysisData.questionListIndex
  1159. ].answerValue; //正确答案
  1160. state.questionAnswerData.questionType =
  1161. state.problemAnalysisData.questionList[
  1162. state.problemAnalysisData.questionListIndex
  1163. ].questionType; //类型
  1164. state.questionAnswerData.fullMark =
  1165. state.problemAnalysisData.questionList[
  1166. state.problemAnalysisData.questionListIndex
  1167. ].fullMark; //满分
  1168. state.questionAnswerData.classListIndex = newINdex;
  1169. if (classList) {
  1170. const answerList = classList.questionStats.answerList || [];
  1171. state.classTitle = classList.groupName;
  1172. state.questionAnswerData.datax = []; // x轴数据
  1173. state.questionAnswerData.datay = []; // Y轴数据
  1174. let sum = 0;
  1175. answerList.forEach((item) => {
  1176. sum += item.studentNum;
  1177. state.questionAnswerData.datax.push(item.name);
  1178. state.questionAnswerData.datay.push(item.studentNum);
  1179. state.questionAnswerData.answerScore.push(item.score);
  1180. });
  1181. state.questionAnswerData.tableData = [
  1182. {
  1183. title: "得分率",
  1184. value: `${classList.questionStats.scoreRate}%`,
  1185. },
  1186. {
  1187. title: "平均分",
  1188. value: `${classList.questionStats.averageScore}`,
  1189. },
  1190. {
  1191. title: "最高分",
  1192. value: `${classList.questionStats.maxScore}`,
  1193. },
  1194. {
  1195. title: "最低分",
  1196. value: `${classList.questionStats.minScore}`,
  1197. },
  1198. {
  1199. title: "人数",
  1200. value: `${sum}人`,
  1201. },
  1202. ];
  1203. if (analysisStore.filterObject.schoolLevel == 2) {
  1204. //单校时展示
  1205. // 获取答题情况
  1206. GetAnswerListByAnswerAndScore(
  1207. 0,
  1208. answerList && answerList[0] ? answerList[0].name : "",
  1209. );
  1210. }
  1211. }
  1212. };
  1213. // 点击柱状图小题分析 /第N题 / 第N班 选项 获取答题情况
  1214. const HandleQuestionAnswerChartClick = (index, name) => {
  1215. if (analysisStore.filterObject.schoolLevel == 2) {
  1216. //单校时展示
  1217. GetAnswerListByAnswerAndScore(index, name);
  1218. }
  1219. };
  1220. //通过答案或者分数查询某题作答情况
  1221. const GetAnswerListByAnswerAndScore = (index, name) => {
  1222. state.optionTitle = name; //选项名称
  1223. const question =
  1224. state.problemAnalysisData.questionList[
  1225. state.problemAnalysisData.questionListIndex
  1226. ];
  1227. const classItem = question.classList[state.questionAnswerData.classListIndex];
  1228. const classItemKeys = Object.keys(classItem);
  1229. const reportParam = {
  1230. ...analysisStore.filterObject,
  1231. };
  1232. Object.keys(analysisStore.filterObject).forEach((item) => {
  1233. if (classItemKeys.indexOf(item) > -1) {
  1234. reportParam[item] = classItem[item];
  1235. }
  1236. });
  1237. const answer =
  1238. question.classList[state.questionAnswerData.classListIndex].questionStats
  1239. .answerList[index];
  1240. const params = {
  1241. ...reportParam,
  1242. questionId: question.questionId, // 试题id
  1243. registrationCodeList: answer?.registrationCodeList || [], // 学生账号数组
  1244. };
  1245. state.cardQuestionId = params.questionId; // 批量查看答题卡试题id
  1246. state.cardRegistrationCodeList = params.registrationCodeList; // 批量查看答题卡学生账号数组
  1247. queryAnswerListByAnswerAndScore(params).then((res) => {
  1248. if (res.code == 200) {
  1249. state.majorAnswerData.tableData = res.data || [];
  1250. } else {
  1251. state.majorAnswerData.tableData = [];
  1252. }
  1253. });
  1254. };
  1255. // 获取项目分析表
  1256. const GetMajorTableData = () => {
  1257. state.majorTableData.tableKey += 1;
  1258. state.majorTableData.headerData = state.problemAnalysisData.headerList;
  1259. state.majorTableData.changeHeaderData =
  1260. state.problemAnalysisData.changeHeaderList;
  1261. state.majorTableData.childHeaderData =
  1262. state.problemAnalysisData.childHeaderList;
  1263. const headerPropData = state.problemAnalysisData.headerList.map(
  1264. (item) => item.prop,
  1265. ); //表头字段名
  1266. const childHeaderPropData = state.problemAnalysisData.childHeaderList.map(
  1267. (item) => item.prop,
  1268. ); //动态表头字段名
  1269. let allTableData = [];
  1270. const allList = state.problemAnalysisData.groupTableList;
  1271. allList.forEach((item) => {
  1272. let itemObj = {
  1273. questionId: item?.questionId || "",
  1274. knowledgeId: item?.knowledgeId || "",
  1275. };
  1276. const classList = item.classList;
  1277. headerPropData.forEach((title) => {
  1278. itemObj[title] = Array.isArray(item[title])
  1279. ? item[title].join("、")
  1280. : item[title];
  1281. });
  1282. classList.forEach((el) => {
  1283. if (
  1284. el.questionStats?.headDataBOList &&
  1285. el.questionStats.headDataBOList.length > 0
  1286. ) {
  1287. el.questionStats.headDataBOList.forEach((bo) => {
  1288. el.questionStats[`${bo.name}Rate`] = bo.rate;
  1289. el.questionStats[`${bo.name}StudentNumber`] = bo.studentNumber;
  1290. });
  1291. } else {
  1292. el.questionStats = [];
  1293. }
  1294. childHeaderPropData.forEach((field) => {
  1295. itemObj[`${el.groupId}_${field}`] = el.questionStats[field];
  1296. });
  1297. });
  1298. allTableData.push(itemObj);
  1299. });
  1300. state.majorTableData.total = allTableData.length; //总条数
  1301. state.majorTableData.allTableData = allTableData;
  1302. GetPageMajorTableData();
  1303. //重置表格滚动条位置
  1304. ResetTableScroll(); //重置表格滚动条位置
  1305. };
  1306. const GetPageMajorTableData = () => {
  1307. const start =
  1308. (state.majorTableData.currentPage - 1) * state.majorTableData.pageSize;
  1309. const end = start + state.majorTableData.pageSize;
  1310. state.majorTableData.tableData = state.majorTableData.allTableData.slice(
  1311. start,
  1312. end,
  1313. );
  1314. };
  1315. //重置表格滚动条位置
  1316. const ResetTableScroll = () => {
  1317. nextTick(() => {
  1318. if (majorTable.value) {
  1319. const tableBody = majorTable.value.$el.querySelector(
  1320. ".el-table__body-wrapper",
  1321. );
  1322. if (tableBody) {
  1323. tableBody.scrollTop = 0; //清除纵向滚动条位置
  1324. tableBody.scrollLeft = 0; // 清除横向滚动条位置
  1325. }
  1326. }
  1327. });
  1328. };
  1329. const ChangeCurrentPage = (val) => {
  1330. state.majorTableData.currentPage = val;
  1331. GetMajorTableData(); //加载分析表格数据
  1332. };
  1333. const ChangePageSize = (val: number) => {
  1334. state.majorTableData.pageSize = val;
  1335. state.majorTableData.currentPage = 1;
  1336. GetMajorTableData(); //加载分析表格数据
  1337. };
  1338. // 导出Excel
  1339. const ExportExcel = () => {
  1340. // 1. 设置加载状态
  1341. reportModuleRef.value?.SetExportLoading?.(true);
  1342. // 2. 参数
  1343. const examName = getExamName.value;
  1344. let params = {
  1345. fileName: `${GetExcelFileName(examName, analysisStore.filterObject, state.groupTitle)}`,
  1346. examName: examName, //考试名称
  1347. sheetName: state.groupTitle, //sheet页名称
  1348. };
  1349. const staticHeaderData = state.problemAnalysisData.headerList.filter(
  1350. (item) => item.display,
  1351. );
  1352. const childHeaderData = state.problemAnalysisData.childHeaderList.filter(
  1353. (item) => item.display,
  1354. );
  1355. const dynamicsHeaderData = state.problemAnalysisData.changeHeaderList.filter(
  1356. (item) => item.display,
  1357. );
  1358. params.staticHeaderData = staticHeaderData;
  1359. params.childHeaderData = childHeaderData;
  1360. params.dynamicsHeaderData = dynamicsHeaderData;
  1361. let dataList = [];
  1362. state.majorTableData.allTableData.forEach((item) => {
  1363. const rowData = [];
  1364. staticHeaderData.forEach((header) => {
  1365. rowData.push(item?.[header.prop] || "-");
  1366. });
  1367. dynamicsHeaderData.forEach((parent) => {
  1368. childHeaderData.forEach((child) => {
  1369. const itemValue = item[`${parent.prop}_${child.prop}`];
  1370. if (child.prop.indexOf("Rate") > -1) {
  1371. const rate = itemValue ? `${itemValue}%` : "-";
  1372. rowData.push(rate);
  1373. } else {
  1374. rowData.push(itemValue ?? "-");
  1375. }
  1376. });
  1377. });
  1378. dataList.push(rowData);
  1379. });
  1380. params.dataList = dataList;
  1381. // 3. 调用通用下载方法,并在完成后重置加载状态
  1382. downloadExcel(publicExport, params).finally(() => {
  1383. reportModuleRef.value?.SetExportLoading?.(false);
  1384. });
  1385. };
  1386. //设置表头样式
  1387. const HeaderRowStyle = ({ row, rowIndex }) => {
  1388. if (rowIndex === 1) {
  1389. return {
  1390. display: "none",
  1391. };
  1392. }
  1393. };
  1394. const pageInit = () => {
  1395. const chartTypeLists = [
  1396. {
  1397. label: "组合图",
  1398. value: "line_bar_chart",
  1399. },
  1400. {
  1401. label: "柱状图",
  1402. value: "vertical_bar",
  1403. },
  1404. {
  1405. label: "雷达图",
  1406. value: "radar_chart",
  1407. },
  1408. ];
  1409. const chartTypeList = [
  1410. {
  1411. label: "柱状图",
  1412. value: "vertical_bar",
  1413. },
  1414. {
  1415. label: "雷达图",
  1416. value: "radar_chart",
  1417. },
  1418. ];
  1419. state.problemAnalysisData.chartTypeList =
  1420. analysisStore.filterObject.classLevel != 2 ? chartTypeLists : chartTypeList;
  1421. state.problemAnalysisData.chartType =
  1422. analysisStore.filterObject.classLevel != 2
  1423. ? "line_bar_chart"
  1424. : "vertical_bar";
  1425. state.majorQuestionData.chartTypeList =
  1426. analysisStore.filterObject.classLevel != 2 ? chartTypeLists : chartTypeList;
  1427. state.majorQuestionData.chartType =
  1428. analysisStore.filterObject.classLevel != 2
  1429. ? "line_bar_chart"
  1430. : "vertical_bar";
  1431. state.questionScoreStatsData.chartType = "vertical_bar";
  1432. state.majorTableData.currentPage = 1;
  1433. state.majorAnswerData.rowIndex = 0;
  1434. GetQuestionGroupAnalysisData(); //题目分组分析
  1435. };
  1436. // 监听筛选条件
  1437. watch(
  1438. () => analysisStore.filterObject,
  1439. async () => {
  1440. pageInit();
  1441. },
  1442. { deep: true },
  1443. );
  1444. onMounted(() => {
  1445. pageInit();
  1446. });
  1447. </script>
  1448. <style lang="scss" scoped>
  1449. .content_left {
  1450. &.answer {
  1451. width: calc(100% - 220px);
  1452. }
  1453. &.paper_card {
  1454. width: calc(100% - 480px);
  1455. }
  1456. }
  1457. .content_right {
  1458. height: 100%;
  1459. display: flex;
  1460. align-items: center;
  1461. margin: auto 0 auto auto;
  1462. padding-bottom: 0;
  1463. &.answer {
  1464. width: 200px;
  1465. }
  1466. &.paper_card {
  1467. width: 460px;
  1468. }
  1469. :deep(.el-table) {
  1470. border-left: 0px;
  1471. border-right: 0px;
  1472. .el-table__cell {
  1473. height: 44px;
  1474. }
  1475. }
  1476. }
  1477. </style>