index.vue 9.6 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329
  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. <splitpanes class="default-theme" style="height: 100%" @resize="setPaneSize">
  12. <pane
  13. max-size="80"
  14. :size="paneSize"
  15. class="flex flex-1 direction-column radius-base fill-blank mark-content"
  16. :class="{ 'text-center': center }"
  17. :style="{ 'background-color': backgroundColor }"
  18. >
  19. <span class="preview" @click="onPreview">
  20. <svg-icon name="preview"></svg-icon>
  21. </span>
  22. <p v-if="currentAssessPaper" class="absolute mark-score">{{ currentAssessPaper.score }}</p>
  23. <!-- <right-button class="next-button" @click="checkNext" /> -->
  24. <div class="flex-1 p-base scroll-auto mark-content-paper img-wrap relative">
  25. <img :src="dataUrl" alt="" class="paper-img" :style="{ 'background-color': frontColor }" />
  26. </div>
  27. <!-- <scoring-panel-with-confirm
  28. v-model:visible="editScoreVisible"
  29. v-model:score="modelScore"
  30. :main-number="currentAssessPaper?.mainNumber"
  31. modal
  32. :toggle-modal="false"
  33. :auto-visible="false"
  34. @submit="onSubmit"
  35. ></scoring-panel-with-confirm> -->
  36. <scoring-panel-with-confirm
  37. v-model:visible="editScoreVisible"
  38. v-model:score="modelScore"
  39. :main-number="currentAssessPaper?.mainNumber"
  40. :subject-code="formModel.subjectCode"
  41. modal
  42. :auto-visible="false"
  43. @submit="onSubmit"
  44. ></scoring-panel-with-confirm>
  45. </pane>
  46. <pane
  47. max-size="80"
  48. :size="100 - paneSize"
  49. class="flex direction-column p-base radius-base fill-blank m-l-base table-view"
  50. >
  51. <base-form size="small" :model="formModel" :items="formItems" :label-width="'52px'">
  52. <template #form-item-search>
  53. <el-button :loading="loading" type="primary" @click="onSearch">查询</el-button>
  54. </template>
  55. </base-form>
  56. <div>
  57. <el-button custom-1 size="small" class="detail-info-label">
  58. <span class="">考核卷: 共</span>
  59. <span class="m-l-extra-small detail-info-label-num">{{ tableData?.length }}</span>
  60. </el-button>
  61. </div>
  62. <div class="flex-1 scroll-auto m-t-mini">
  63. <base-table
  64. ref="tableRef"
  65. border
  66. stripe
  67. size="small"
  68. height="100%"
  69. :data="tableData"
  70. :columns="columns"
  71. highlight-current-row
  72. :cell-style="{ padding: '6px 0' }"
  73. @current-change="onCurrentChange"
  74. ></base-table>
  75. </div>
  76. </pane>
  77. </splitpanes>
  78. </div>
  79. </div>
  80. <image-preview v-model="previewModalVisible" :url="currentAssessPaper?.filePath"></image-preview>
  81. </template>
  82. <script setup lang="ts" name="ExpertAssess">
  83. /** 专家卷浏览-强制考核卷 */
  84. import { reactive, ref, computed, watch } from 'vue'
  85. import { ElButton, ElMessage } from 'element-plus'
  86. import { add } from '@/utils/common'
  87. import { useSetImgBg } from '@/hooks/useSetImgBg'
  88. import useFetch from '@/hooks/useFetch'
  89. import useVW from '@/hooks/useVW'
  90. import useForm from '@/hooks/useForm'
  91. import useOptions from '@/hooks/useOptions'
  92. import useMarkHeader from '@/hooks/useMarkHeader'
  93. import useTableCheck from '@/hooks/useTableCheck'
  94. import MarkHeader from '@/components/shared/MarkHeader.vue'
  95. import BaseForm from '@/components/element/BaseForm.vue'
  96. import BaseTable from '@/components/element/BaseTable.vue'
  97. import RightButton from '@/components/shared/RightButton.vue'
  98. import SvgIcon from '@/components/common/SvgIcon.vue'
  99. import ScoringPanelWithConfirm from '@/components/shared/ScoringPanelWithConfirm.vue'
  100. import ImagePreview from '@/components/shared/ImagePreview.vue'
  101. import type { SetImgBgOption } from '@/hooks/useSetImgBg'
  102. import type { ExtractMultipleApiResponse, ExtractApiParams, ExtractApiResponse } from '@/api/api'
  103. import type { MarkHeaderInstance, EpFormItem, EpTableColumn } from 'global-type'
  104. import { Splitpanes, Pane } from 'splitpanes'
  105. import { setPaneSize } from '@/utils/common'
  106. import useMainStore from '@/store/main'
  107. const mainStore = useMainStore()
  108. const paneSize = computed(() => {
  109. return mainStore.paneSizeConfig[location.pathname] || 60
  110. })
  111. type RowType = ExtractMultipleApiResponse<'getExpertAssessList'> & { index: number }
  112. /** 给分板 */
  113. const editScoreVisible = ref<boolean>(true)
  114. /** 图片预览 */
  115. const previewModalVisible = ref<boolean>(false)
  116. /** 分数 */
  117. const modelScore = ref<number[]>([])
  118. const {
  119. rotate,
  120. scale,
  121. center,
  122. frontColor,
  123. backgroundColor,
  124. onBack,
  125. onScaleChange,
  126. onCenter,
  127. onRotate,
  128. setBackgroundColor,
  129. setFrontColor,
  130. onViewStandard,
  131. } = useMarkHeader()
  132. /** 刷新 */
  133. const onRefresh = () => {
  134. onSearch()
  135. }
  136. /** 预览试卷 */
  137. const onPreview = () => {
  138. previewModalVisible.value = true
  139. }
  140. const onEditScore = () => {
  141. editScoreVisible.value = true
  142. }
  143. type OperationClick = MarkHeaderInstance['onClick']
  144. type OperationType = Parameters<Exclude<OperationClick, undefined>>[0]['type']
  145. const operationHandles: Partial<Record<OperationType, (...args: any) => void>> = {
  146. back: onBack,
  147. 'scale-change': onScaleChange,
  148. center: onCenter,
  149. rotate: onRotate,
  150. 'front-color': setFrontColor,
  151. 'background-color': setBackgroundColor,
  152. refresh: onRefresh,
  153. standard: onViewStandard,
  154. }
  155. const onOperationClick: OperationClick = ({ type, value }) => {
  156. operationHandles[type]?.(value)
  157. }
  158. const { subjectList, mainQuestionList, dataModel, changeModelValue, onOptionInit, isLeader, isExpert } = useOptions([
  159. 'subject',
  160. 'question',
  161. ])
  162. const formModel = reactive<ExtractApiParams<'getExpertAssessList'>>({
  163. mainNumber: dataModel.question,
  164. pageNumber: 1,
  165. pageSize: 9999999,
  166. subjectCode: dataModel.subject,
  167. })
  168. watch(dataModel, () => {
  169. formModel.mainNumber = dataModel.question
  170. formModel.subjectCode = dataModel.subject
  171. })
  172. const { fetch: getForceCheckGroupList, result: forceCheckGroup } = useFetch('getForceCheckGroupList')
  173. watch(
  174. dataModel,
  175. () => {
  176. if (dataModel.subject && dataModel.question) {
  177. getForceCheckGroupList({ subjectCode: dataModel.subject, mainNumber: dataModel.question })
  178. }
  179. },
  180. { immediate: true }
  181. )
  182. const { defineColumn, _ } = useForm()
  183. const span10 = defineColumn(_, '', { span: 10 })
  184. const formItems = computed<EpFormItem[]>(() => [
  185. span10({
  186. rowKey: 'row-1',
  187. label: '科目',
  188. prop: 'subjectCode',
  189. slotType: 'select',
  190. labelWidth: '46px',
  191. slot: { options: subjectList.value, onChange: changeModelValue('subject'), disabled: !isExpert.value },
  192. }),
  193. span10({
  194. rowKey: 'row-1',
  195. label: '大题',
  196. prop: 'mainNumber',
  197. slotType: 'select',
  198. labelWidth: '46px',
  199. slot: {
  200. options: mainQuestionList.value,
  201. onChange: changeModelValue('question'),
  202. disabled: !isExpert.value && !isLeader.value,
  203. },
  204. }),
  205. span10({
  206. rowKey: 'row-2',
  207. label: '组号',
  208. prop: 'forceGroupNumber',
  209. slotType: 'select',
  210. slot: {
  211. options: forceCheckGroup.value?.map((d) => ({ value: d.forceGroupNumber, label: `第${d.forceGroupNumber}组` })),
  212. },
  213. }),
  214. { rowKey: 'row-1', slotName: 'search', labelWidth: '10px', colProp: { span: 4 } },
  215. ])
  216. /** 强制考核卷 */
  217. const columns: EpTableColumn<RowType>[] = [
  218. { label: '序号', type: 'index', width: 60 },
  219. { label: '密号', prop: 'secretNumber', minWidth: 110 },
  220. { label: '标准分', prop: 'score', minWidth: 70 },
  221. { label: '分组', prop: 'forceGroupNumber', minWidth: 90 },
  222. { label: '提交时间', prop: 'createTime', minWidth: 130 },
  223. ]
  224. const { fetch: getExpertAssessList, result: rfSampleList, loading } = useFetch('getExpertAssessList')
  225. const {
  226. tableRef,
  227. tableData,
  228. current: currentAssessPaper,
  229. next: checkNext,
  230. onCurrentChange,
  231. nextRow,
  232. } = useTableCheck(rfSampleList)
  233. const onSearch = async () => {
  234. getExpertAssessList(formModel)
  235. }
  236. /** 确定给分 */
  237. const { fetch: updateMarkScore } = useFetch('updateMarkScore')
  238. const onSubmit = async () => {
  239. if (currentAssessPaper.value) {
  240. const scores = JSON.parse(JSON.stringify(modelScore.value))
  241. await updateMarkScore({ id: currentAssessPaper.value.id, scores: modelScore.value })
  242. currentAssessPaper.value.score = add(...scores)
  243. ElMessage.success('修改成功')
  244. // editScoreVisible.value = false
  245. // onSearch()
  246. nextRow()
  247. }
  248. }
  249. onOptionInit(onSearch)
  250. const imgOption = computed<SetImgBgOption>(() => {
  251. return {
  252. image: currentAssessPaper?.value?.filePath,
  253. rotate: rotate.value,
  254. scale: scale.value,
  255. }
  256. })
  257. const { drawing, dataUrl } = useSetImgBg(imgOption, frontColor, setFrontColor)
  258. </script>
  259. <style scoped lang="scss">
  260. .mark-container {
  261. .mark-content {
  262. position: relative;
  263. .preview {
  264. position: absolute;
  265. cursor: pointer;
  266. top: 20px;
  267. right: 25px;
  268. font-size: 38px;
  269. }
  270. .next-button {
  271. position: absolute;
  272. right: -20px;
  273. top: 300px;
  274. }
  275. .mark-content-paper {
  276. img {
  277. max-width: 100%;
  278. }
  279. }
  280. }
  281. .table-view {
  282. // width: 580px;
  283. .detail-info-label {
  284. .detail-info-label-num {
  285. min-width: 32px;
  286. height: 24px;
  287. line-height: 24px;
  288. background: #00987b;
  289. border-radius: 4px;
  290. }
  291. }
  292. }
  293. }
  294. </style>