index.vue 7.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260
  1. <template>
  2. <div class="flex direction-column full">
  3. <mark-header
  4. :exclude-operations="['remark', 'problem', 'example', 'delete', 'bookmark']"
  5. :paper-path="currentAssessPaper?.filePath"
  6. @click="onOperationClick"
  7. >
  8. <el-button class="m-l-base m-r-auto" size="small" type="primary" @click="onEditScore">修改给分</el-button>
  9. </mark-header>
  10. <div class="flex flex-1 overflow-hidden p-base mark-container">
  11. <div
  12. class="flex flex-1 direction-column radius-base fill-blank mark-content"
  13. :class="{ 'text-center': center }"
  14. :style="{ 'background-color': backgroundColor }"
  15. >
  16. <span class="preview" @click="onPreview">
  17. <svg-icon name="preview"></svg-icon>
  18. </span>
  19. <right-button class="next-button" @click="checkNext" />
  20. <div class="flex-1 p-base scroll-auto mark-content-paper">
  21. <img :src="dataUrl" alt="" class="paper-img" :style="{ 'background-color': frontColor }" />
  22. </div>
  23. </div>
  24. <div class="p-base radius-base fill-blank scroll-auto m-l-base table-view">
  25. <base-form size="small" :model="formModel" :items="formItems" :label-width="useVW(40)">
  26. <template #form-item-search>
  27. <el-button type="primary" @click="onSearch">查询</el-button>
  28. </template>
  29. </base-form>
  30. <div class="flex items-center p-l-base">
  31. <span>考核卷</span>
  32. <span>: 共{{ tableData.length }}</span>
  33. </div>
  34. <base-table
  35. ref="tableRef"
  36. size="small"
  37. :data="tableData"
  38. :columns="columns"
  39. highlight-current-row
  40. @current-change="onCurrentChange"
  41. ></base-table>
  42. </div>
  43. </div>
  44. </div>
  45. <image-preview v-model="previewModalVisible" :url="currentAssessPaper?.filePath"></image-preview>
  46. <scoring-panel-with-confirm
  47. v-model:visible="editScoreVisible"
  48. v-model:score="modelScore"
  49. :main-number="currentAssessPaper?.mainNumber"
  50. modal
  51. :toggle-modal="false"
  52. @submit="onSubmit"
  53. ></scoring-panel-with-confirm>
  54. </template>
  55. <script setup lang="ts" name="ExpertAssess">
  56. /** 专家卷浏览-强制考核卷 */
  57. import { reactive, ref, computed, watch } from 'vue'
  58. import { ElButton } from 'element-plus'
  59. import { useSetImgBg } from '@/hooks/useSetImgBg'
  60. import useFetch from '@/hooks/useFetch'
  61. import useVW from '@/hooks/useVW'
  62. import useForm from '@/hooks/useForm'
  63. import useOptions from '@/hooks/useOptions'
  64. import useMarkHeader from '@/hooks/useMarkHeader'
  65. import useTableCheck from '@/hooks/useTableCheck'
  66. import MarkHeader from '@/components/shared/MarkHeader.vue'
  67. import BaseForm from '@/components/element/BaseForm.vue'
  68. import BaseTable from '@/components/element/BaseTable.vue'
  69. import RightButton from '@/components/shared/RightButton.vue'
  70. import SvgIcon from '@/components/common/SvgIcon.vue'
  71. import ScoringPanelWithConfirm from '@/components/shared/ScoringPanelWithConfirm.vue'
  72. import ImagePreview from '@/components/shared/ImagePreview.vue'
  73. import type { SetImgBgOption } from '@/hooks/useSetImgBg'
  74. import type { ExtractMultipleApiResponse, ExtractApiParams, ExtractApiResponse } from '@/api/api'
  75. import type { MarkHeaderInstance, EpFormItem, EpTableColumn } from 'global-type'
  76. type RowType = ExtractMultipleApiResponse<'getExpertAssessList'> & { index: number }
  77. /** 给分板 */
  78. const editScoreVisible = ref<boolean>(false)
  79. /** 图片预览 */
  80. const previewModalVisible = ref<boolean>(false)
  81. /** 分数 */
  82. const modelScore = ref<number[]>([])
  83. const {
  84. rotate,
  85. scale,
  86. center,
  87. frontColor,
  88. backgroundColor,
  89. onBack,
  90. onScaleChange,
  91. onCenter,
  92. onRotate,
  93. setBackgroundColor,
  94. setFrontColor,
  95. onViewStandard,
  96. } = useMarkHeader()
  97. /** 刷新 */
  98. const onRefresh = () => {
  99. onSearch()
  100. }
  101. /** 预览试卷 */
  102. const onPreview = () => {
  103. previewModalVisible.value = true
  104. }
  105. const onEditScore = () => {
  106. editScoreVisible.value = true
  107. }
  108. type OperationClick = MarkHeaderInstance['onClick']
  109. type OperationType = Parameters<Exclude<OperationClick, undefined>>[0]['type']
  110. const operationHandles: Partial<Record<OperationType, (...args: any) => void>> = {
  111. back: onBack,
  112. 'scale-change': onScaleChange,
  113. center: onCenter,
  114. rotate: onRotate,
  115. 'front-color': setFrontColor,
  116. 'background-color': setBackgroundColor,
  117. refresh: onRefresh,
  118. standard: onViewStandard,
  119. }
  120. const onOperationClick: OperationClick = ({ type, value }) => {
  121. operationHandles[type]?.(value)
  122. }
  123. const { mainQuestionList, dataModel, changeModelValue, onOptionInit } = useOptions(['question'])
  124. const formModel = reactive<ExtractApiParams<'getExpertAssessList'>>({
  125. mainNumber: dataModel.question,
  126. pageNumber: 1,
  127. pageSize: 9999999,
  128. })
  129. watch(dataModel, () => {
  130. formModel.mainNumber = dataModel.question
  131. })
  132. const { fetch: getForceCheckGroupList, result: forceCheckGroup } = useFetch('getForceCheckGroupList')
  133. watch(
  134. dataModel,
  135. () => {
  136. if (dataModel.subject && dataModel.question) {
  137. getForceCheckGroupList({ subjectCode: dataModel.subject, mainNumber: dataModel.question })
  138. }
  139. },
  140. { immediate: true }
  141. )
  142. const { defineColumn, _ } = useForm()
  143. const span10 = defineColumn(_, '', { span: 10 })
  144. const formItems = computed<EpFormItem[]>(() => [
  145. span10({
  146. rowKey: 'row-1',
  147. label: '大题',
  148. prop: 'mainNumber',
  149. slotType: 'select',
  150. slot: { options: mainQuestionList.value, onChange: changeModelValue('question'), disabled: true },
  151. }),
  152. span10({
  153. rowKey: 'row-1',
  154. label: '组号',
  155. prop: 'forceGroupNumber',
  156. slotType: 'select',
  157. slot: {
  158. options: forceCheckGroup.value?.map((d) => ({ value: d.forceGroupNumber, label: `第${d.forceGroupNumber}组` })),
  159. },
  160. }),
  161. { rowKey: 'row-1', slotName: 'search', labelWidth: '10px', colProp: { span: 4 } },
  162. ])
  163. /** 强制考核卷 */
  164. const columns: EpTableColumn<RowType>[] = [
  165. { label: '序号', type: 'index' },
  166. { label: '密号', prop: 'secretNumber' },
  167. { label: '标准分', prop: 'score' },
  168. { label: '分组', prop: 'forceGroupNumber' },
  169. { label: '提交时间', prop: 'createTime' },
  170. ]
  171. const { fetch: getExpertAssessList, result: rfSampleList } = useFetch('getExpertAssessList')
  172. const {
  173. tableRef,
  174. tableData,
  175. current: currentAssessPaper,
  176. next: checkNext,
  177. onCurrentChange,
  178. } = useTableCheck(rfSampleList)
  179. const onSearch = async () => {
  180. getExpertAssessList(formModel)
  181. }
  182. /** 确定给分 */
  183. const { fetch: updateMarkScore } = useFetch('updateMarkScore')
  184. const onSubmit = () => {
  185. if (currentAssessPaper.value) {
  186. updateMarkScore({ id: currentAssessPaper.value.id, scores: modelScore.value })
  187. }
  188. }
  189. onOptionInit(onSearch)
  190. const imgOption = computed<SetImgBgOption>(() => {
  191. return {
  192. image: currentAssessPaper?.value?.filePath,
  193. immediate: true,
  194. rotate: rotate.value,
  195. scale: scale.value,
  196. }
  197. })
  198. const { drawing, dataUrl } = useSetImgBg(imgOption)
  199. </script>
  200. <style scoped lang="scss">
  201. .mark-container {
  202. .mark-content {
  203. position: relative;
  204. .preview {
  205. position: absolute;
  206. cursor: pointer;
  207. top: 10px;
  208. right: 20px;
  209. font-size: 24px;
  210. }
  211. .next-button {
  212. position: absolute;
  213. right: -20px;
  214. top: 300px;
  215. }
  216. .mark-content-paper {
  217. img {
  218. max-width: 100%;
  219. }
  220. }
  221. }
  222. .table-view {
  223. width: 580px;
  224. }
  225. }
  226. </style>