index.vue 9.1 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321
  1. <template>
  2. <div class="flex direction-column full">
  3. <mark-header
  4. :exclude-operations="['remark', 'problem', 'example', 'delete', 'bookmark']"
  5. :paper-path="currentProblem?.filePath"
  6. @click="onOperationClick"
  7. >
  8. <el-button
  9. :disabled="
  10. !currentProblem?.id ||
  11. currentProblem.problemSolveType === 'MARK' ||
  12. currentProblem.problemSolveType === 'REJECT'
  13. "
  14. class="m-l-base m-r-auto"
  15. size="small"
  16. type="primary"
  17. @click="onSendBack"
  18. >打回</el-button
  19. >
  20. </mark-header>
  21. <div class="flex flex-1 overflow-hidden p-base mark-container">
  22. <div
  23. class="flex flex-1 direction-column radius-base fill-blank mark-content"
  24. :class="{ 'text-center': center }"
  25. :style="{ 'background-color': backgroundColor }"
  26. >
  27. <span class="preview" @click="onPreview">
  28. <svg-icon name="preview"></svg-icon>
  29. </span>
  30. <right-button class="next-button" @click="checkNext" />
  31. <div class="flex-1 p-base scroll-auto mark-content-paper">
  32. <img :src="dataUrl" alt="" class="paper-img" :style="{ 'background-color': frontColor }" />
  33. </div>
  34. <scoring-panel-with-confirm
  35. :id="currentProblem?.taskId"
  36. v-model:visible="scoringPanelVisible"
  37. v-model:score="modelScore"
  38. :main-number="currentProblem?.mainNumber"
  39. @submit="onSubmit"
  40. ></scoring-panel-with-confirm>
  41. </div>
  42. <div class="flex direction-column p-base radius-base fill-blank m-l-base table-view">
  43. <base-form size="small" :model="formModel" :items="formItems" :label-width="useVW(78)">
  44. <template #form-item-search>
  45. <el-button :loading="loading" type="primary" @click="onSearch">查询</el-button>
  46. </template>
  47. </base-form>
  48. <div>
  49. <el-button custom-1 size="small" class="detail-info-label">
  50. <span class="">问题卷: 共</span>
  51. <span class="m-l-extra-small detail-info-label-num">{{ tableData.length }}</span>
  52. </el-button>
  53. </div>
  54. <div class="flex-1 scroll-auto m-t-mini">
  55. <base-table
  56. ref="tableRef"
  57. border
  58. stripe
  59. size="small"
  60. height="100%"
  61. :data="tableData"
  62. :columns="columns"
  63. highlight-current-row
  64. @current-change="onCurrentChange"
  65. @row-dblclick="onDbClick"
  66. ></base-table>
  67. </div>
  68. </div>
  69. </div>
  70. </div>
  71. <image-preview v-model="previewModalVisible" :url="currentProblem?.filePath"></image-preview>
  72. <mark-history-list :id="currentViewHistory?.taskId" v-model="visibleHistory"></mark-history-list>
  73. <send-back-mark :id="currentProblem?.id" v-model="sendBackVisible" @rejected="onRejected"></send-back-mark>
  74. </template>
  75. <script setup lang="ts" name="MarkingProblem">
  76. /** 问题卷查看 */
  77. import { reactive, ref, computed, watch } from 'vue'
  78. import { ElButton, ElMessage } from 'element-plus'
  79. import { add } from '@/utils/common'
  80. import { useSetImgBg } from '@/hooks/useSetImgBg'
  81. import useVW from '@/hooks/useVW'
  82. import useFetch from '@/hooks/useFetch'
  83. import useForm from '@/hooks/useForm'
  84. import useOptions from '@/hooks/useOptions'
  85. import useMarkHeader from '@/hooks/useMarkHeader'
  86. import useTableCheck from '@/hooks/useTableCheck'
  87. import MarkHeader from '@/components/shared/MarkHeader.vue'
  88. import ScoringPanelWithConfirm from '@/components/shared/ScoringPanelWithConfirm.vue'
  89. import SvgIcon from '@/components/common/SvgIcon.vue'
  90. import BaseForm from '@/components/element/BaseForm.vue'
  91. import BaseTable from '@/components/element/BaseTable.vue'
  92. import MarkHistoryList from '@/components/shared/MarkHistoryList.vue'
  93. import SendBackMark from '@/components/shared/SendBackMark.vue'
  94. import RightButton from '@/components/shared/RightButton.vue'
  95. import ImagePreview from '@/components/shared/ImagePreview.vue'
  96. import type { SetImgBgOption } from '@/hooks/useSetImgBg'
  97. import type { ExtractApiParams, ExtractApiResponse, ExtractMultipleApiResponse } from '@/api/api'
  98. import type { MarkHeaderInstance, EpFormItem, EpTableColumn } from 'global-type'
  99. type RowType = ExtractMultipleApiResponse<'getProblemHistory'> & { index: number }
  100. const {
  101. rotate,
  102. scale,
  103. center,
  104. frontColor,
  105. backgroundColor,
  106. onBack,
  107. onScaleChange,
  108. onCenter,
  109. onRotate,
  110. setBackgroundColor,
  111. setFrontColor,
  112. onViewStandard,
  113. } = useMarkHeader()
  114. /** 给分板 */
  115. const scoringPanelVisible = ref<boolean>(false)
  116. /** 图片预览 */
  117. const previewModalVisible = ref<boolean>(false)
  118. /** 打回弹窗 */
  119. const sendBackVisible = ref<boolean>(false)
  120. const modelScore = ref<number[]>([])
  121. /** 刷新 */
  122. const onRefresh = () => {
  123. onSearch()
  124. }
  125. /** 预览试卷 */
  126. const onPreview = () => {
  127. previewModalVisible.value = true
  128. }
  129. /** 打回 */
  130. const onSendBack = () => {
  131. sendBackVisible.value = true
  132. }
  133. /** 打回成功 */
  134. const onRejected = () => {
  135. // checkNext()
  136. onSearch()
  137. }
  138. type OperationClick = MarkHeaderInstance['onClick']
  139. type OperationType = Parameters<Exclude<OperationClick, undefined>>[0]['type']
  140. const operationHandles: Partial<Record<OperationType, (...args: any) => void>> = {
  141. back: onBack,
  142. 'scale-change': onScaleChange,
  143. center: onCenter,
  144. rotate: onRotate,
  145. 'front-color': setFrontColor,
  146. 'background-color': setBackgroundColor,
  147. refresh: onRefresh,
  148. standard: onViewStandard,
  149. }
  150. const onOperationClick: OperationClick = ({ type, value }) => {
  151. operationHandles[type]?.(value)
  152. }
  153. const { mainQuestionList, dataModel, changeModelValue, onOptionInit } = useOptions(['question'])
  154. const formModel = reactive<ExtractApiParams<'getProblemHistory'>>({
  155. mainNumber: dataModel.question,
  156. problemType: '',
  157. status: 'INITIAL',
  158. pageNumber: 1,
  159. pageSize: 9999999,
  160. })
  161. watch(dataModel, () => {
  162. formModel.mainNumber = dataModel.question
  163. })
  164. const { defineColumn, _ } = useForm()
  165. const span10 = defineColumn(_, '', { span: 10 })
  166. const formItems = computed<EpFormItem[]>(() => [
  167. span10({
  168. rowKey: 'row-1',
  169. label: '大题',
  170. prop: 'mainNumber',
  171. slotType: 'select',
  172. slot: { options: mainQuestionList.value, onChange: changeModelValue('question'), disabled: true },
  173. }),
  174. span10({
  175. rowKey: 'row-1',
  176. label: '状态',
  177. prop: 'status',
  178. slotType: 'select',
  179. slot: {
  180. options: [
  181. { label: '未处理', value: 'INITIAL' },
  182. { label: '已处理', value: 'PROCESSED' },
  183. { label: '全部', value: '' },
  184. ],
  185. },
  186. }),
  187. { rowKey: 'row-1', slotName: 'search', labelWidth: '10px', colProp: { span: 4 } },
  188. span10({
  189. rowKey: 'row-2',
  190. label: '问题类型',
  191. prop: 'problemType',
  192. slotType: 'select',
  193. slot: {
  194. options: [
  195. { label: '跨套', value: 'OVERSTEP' },
  196. { label: '扫描不清晰', value: 'DIM' },
  197. { label: '全部', value: '' },
  198. ],
  199. },
  200. }),
  201. ])
  202. /** 查询问题卷列表 */
  203. const columns: EpTableColumn[] = [
  204. { label: '密号', prop: 'secretNumber', width: 100, fixed: 'left' },
  205. { label: '提交人', prop: 'markerName', width: 75 },
  206. { label: '问题类型', prop: 'problemTypeName', width: 90 },
  207. { label: '客观分', prop: 'objectiveScore', width: 55 },
  208. { label: '主观分', prop: 'markScore', width: 55 },
  209. { label: '客主比', prop: 'markRatio', width: 55 },
  210. { label: '评卷时间', prop: 'markTime', width: 160 },
  211. { label: '处理人', prop: 'headerName', width: 75 },
  212. { label: '处理时间', prop: 'solveTime', width: 160 },
  213. ]
  214. const { fetch: getProblemHistory, result: problemHistory, loading } = useFetch('getProblemHistory')
  215. const {
  216. tableRef,
  217. tableData,
  218. current: currentProblem,
  219. currentView: currentViewHistory,
  220. next: checkNext,
  221. visibleHistory,
  222. onDbClick,
  223. onCurrentChange,
  224. } = useTableCheck(problemHistory)
  225. watch(currentProblem, () => {
  226. if (currentProblem.value) {
  227. useFetch('viewProblemPaper').fetch({ id: currentProblem.value.id })
  228. }
  229. })
  230. const onSearch = async () => {
  231. getProblemHistory(formModel)
  232. }
  233. /** 问题卷打分 */
  234. const { fetch: markProblemPaper } = useFetch('markProblemPaper')
  235. const onSubmit = async () => {
  236. if (currentProblem.value) {
  237. await markProblemPaper({ id: currentProblem.value.id, scores: modelScore.value })
  238. currentProblem.value.markScore = add(...modelScore.value)
  239. ElMessage.success('修改成功')
  240. onSearch()
  241. }
  242. }
  243. onOptionInit(onSearch)
  244. const imgOption = computed<SetImgBgOption>(() => {
  245. return {
  246. image: currentProblem?.value?.filePath,
  247. rotate: rotate.value,
  248. scale: scale.value,
  249. }
  250. })
  251. const { drawing, dataUrl } = useSetImgBg(imgOption, frontColor, setFrontColor)
  252. </script>
  253. <style scoped lang="scss">
  254. .mark-container {
  255. .mark-content {
  256. position: relative;
  257. .preview {
  258. position: absolute;
  259. cursor: pointer;
  260. top: 10px;
  261. right: 20px;
  262. font-size: 24px;
  263. }
  264. .next-button {
  265. position: absolute;
  266. right: -20px;
  267. top: 300px;
  268. }
  269. .mark-content-paper {
  270. img {
  271. max-width: 100%;
  272. }
  273. }
  274. }
  275. .table-view {
  276. width: 480px;
  277. .detail-info-label {
  278. .detail-info-label-num {
  279. min-width: 32px;
  280. height: 24px;
  281. line-height: 24px;
  282. background: #00987b;
  283. border-radius: 4px;
  284. }
  285. }
  286. }
  287. }
  288. </style>