index.vue 5.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190
  1. <template>
  2. <div class="flex direction-column full">
  3. <mark-header
  4. :exclude-operations="['remark', 'problem', 'example', 'delete', 'bookmark']"
  5. :paper-path="current?.filePath"
  6. @click="onOperationClick"
  7. >
  8. </mark-header>
  9. <div class="flex flex-1 overflow-hidden p-base mark-container">
  10. <div
  11. class="flex flex-1 direction-column radius-base fill-blank mark-content"
  12. :class="{ 'text-center': center }"
  13. :style="{ 'background-color': backgroundColor }"
  14. >
  15. <span class="preview" @click="onPreview">
  16. <svg-icon name="preview"></svg-icon>
  17. </span>
  18. <right-button class="next-button" @click="checkNext" />
  19. <div class="flex-1 p-base scroll-auto mark-content-paper">
  20. <img :src="dataUrl" alt="" class="paper-img" :style="{ 'background-color': frontColor }" />
  21. </div>
  22. </div>
  23. <div class="p-base radius-base fill-blank scroll-auto m-l-base table-view">
  24. <div class="flex items-center justify-between detail-info-table-header">
  25. <el-button custom-1 size="small" class="detail-info-label">
  26. <span class="">{{ PaperType }}共:</span>
  27. <span class="m-l-extra-small detail-info-label-num">{{ monitorDetail?.length }}</span>
  28. </el-button>
  29. </div>
  30. <base-table
  31. ref="tableRef"
  32. size="small"
  33. :data="tableData"
  34. :columns="columns"
  35. highlight-current-row
  36. @current-change="onCurrentChange"
  37. @row-dblclick="onDbClick"
  38. ></base-table>
  39. </div>
  40. </div>
  41. </div>
  42. <image-preview v-model="previewModalVisible" :url="current?.filePath"></image-preview>
  43. <mark-history-list :id="currentViewHistory?.taskId" v-model="visibleHistory"></mark-history-list>
  44. </template>
  45. <script setup lang="ts" name="TrainingMonitoringDetail">
  46. /** 培训监控调卷详情 */
  47. import { reactive, ref, computed } from 'vue'
  48. import { useRoute } from 'vue-router'
  49. import { ElButton } from 'element-plus'
  50. import { useSetImgBg } from '@/hooks/useSetImgBg'
  51. import useFetch from '@/hooks/useFetch'
  52. import useMarkHeader from '@/hooks/useMarkHeader'
  53. import useTableCheck from '@/hooks/useTableCheck'
  54. import MarkHeader from '@/components/shared/MarkHeader.vue'
  55. import BaseTable from '@/components/element/BaseTable.vue'
  56. import ImagePreview from '@/components/shared/ImagePreview.vue'
  57. import RightButton from '@/components/shared/RightButton.vue'
  58. import SvgIcon from '@/components/common/SvgIcon.vue'
  59. import MarkHistoryList from '@/components/shared/MarkHistoryList.vue'
  60. import type { SetImgBgOption } from '@/hooks/useSetImgBg'
  61. import type { ExtractApiResponse } from '@/api/api'
  62. import type { MarkHeaderInstance, EpTableColumn } from 'global-type'
  63. type RowType = ExtractApiResponse<'getTrainingMonitorDetail'> & { index: number }
  64. const { query } = useRoute()
  65. const PaperType = computed(() => {
  66. return query.stage === 'FORCE' ? '强制考核卷' : '培训卷'
  67. })
  68. /** 图片预览 */
  69. const previewModalVisible = ref<boolean>(false)
  70. const {
  71. rotate,
  72. scale,
  73. center,
  74. frontColor,
  75. backgroundColor,
  76. onBack,
  77. onScaleChange,
  78. onCenter,
  79. onRotate,
  80. setBackgroundColor,
  81. setFrontColor,
  82. onViewStandard,
  83. } = useMarkHeader()
  84. /** 刷新 */
  85. const onRefresh = () => {
  86. fetchData()
  87. }
  88. /** 预览试卷 */
  89. const onPreview = () => {
  90. previewModalVisible.value = true
  91. }
  92. type OperationClick = MarkHeaderInstance['onClick']
  93. type OperationType = Parameters<Exclude<OperationClick, undefined>>[0]['type']
  94. const operationHandles: Partial<Record<OperationType, (...args: any) => void>> = {
  95. back: onBack,
  96. 'scale-change': onScaleChange,
  97. center: onCenter,
  98. rotate: onRotate,
  99. 'front-color': setFrontColor,
  100. 'background-color': setBackgroundColor,
  101. refresh: onRefresh,
  102. standard: onViewStandard,
  103. }
  104. const onOperationClick: OperationClick = ({ type, value }) => {
  105. operationHandles[type]?.(value)
  106. }
  107. const columns: EpTableColumn<RowType>[] = [
  108. { label: '密号', prop: 'secretNumber' },
  109. { label: '标准分', prop: 'score' },
  110. { label: '评卷给分', prop: 'markScore' },
  111. { label: '分组', prop: 'group' },
  112. { label: '评卷员', prop: 'markerName' },
  113. { label: '评卷时间', prop: 'markTime' },
  114. ]
  115. const { fetch: getTrainingMonitorDetail, result: trainingMonitorDetail } = useFetch('getTrainingMonitorDetail')
  116. const { fetch: getAssessMonitorDetail, result: assessMonitorDetail } = useFetch('getAssessMonitorDetail')
  117. const fetchData = () => {
  118. query.stage === 'FORCE'
  119. ? getAssessMonitorDetail({ forceGroupMarkerId: query.forceGroupMarkerId as string })
  120. : getTrainingMonitorDetail({ markerId: query.markerId as string, taskType: query.stage as SamplePaperType })
  121. }
  122. const monitorDetail = computed(() => {
  123. return query.stage === 'FORCE' ? assessMonitorDetail.value : trainingMonitorDetail.value
  124. })
  125. const {
  126. tableRef,
  127. tableData,
  128. current,
  129. currentView: currentViewHistory,
  130. next: checkNext,
  131. visibleHistory,
  132. onDbClick,
  133. onCurrentChange,
  134. } = useTableCheck(monitorDetail)
  135. const imgOption = computed<SetImgBgOption>(() => {
  136. return {
  137. image: current?.value?.filePath,
  138. immediate: true,
  139. rotate: rotate.value,
  140. scale: scale.value,
  141. }
  142. })
  143. const { drawing, dataUrl } = useSetImgBg(imgOption)
  144. fetchData()
  145. </script>
  146. <style scoped lang="scss">
  147. .mark-container {
  148. .mark-content {
  149. position: relative;
  150. .preview {
  151. position: absolute;
  152. cursor: pointer;
  153. top: 10px;
  154. right: 20px;
  155. font-size: 24px;
  156. }
  157. .next-button {
  158. position: absolute;
  159. right: -20px;
  160. top: 300px;
  161. }
  162. .mark-content-paper {
  163. img {
  164. max-width: 100%;
  165. }
  166. }
  167. }
  168. }
  169. </style>