123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376 |
- <template>
- <div class="flex direction-column full">
- <div class="flex items-center p-extra-small fill-blank header-view">
- <el-button class="m-r-auto" size="small" plain @click="back()">返回</el-button>
- <div class="m-r-base m-l-base">
- <lock-entry></lock-entry>
- </div>
- <message class="m-r-base m-l-base" :paper-path="current?.filePath"></message>
- <div class="m-r-base m-l-base">
- <user-info></user-info>
- </div>
- </div>
- <div class="flex fill-blank detail-info">
- <div class="flex p-base detail-info-chart" style="width: 64%">
- <div style="width: 50%">
- <vue-echarts class="full" :option="markerSubjectiveChartsOption"></vue-echarts>
- </div>
- <div class="flex-1 m-l-base">
- <vue-echarts class="full" :option="markerObjectiveChartsOption"></vue-echarts>
- </div>
- </div>
- <div class="flex direction-column p-extra-small" style="width: 36%">
- <div class="flex items-center justify-between detail-info-table-header m-b-mini">
- <el-button custom-1 size="small" class="detail-info-label">
- <span class="">{{ dataType }}试卷总数:</span>
- <span class="m-l-extra-small detail-info-label-num">{{ total }}</span>
- </el-button>
- <el-pagination
- v-bind="pagination"
- v-model:current-page="currentPage"
- size="small"
- class="m-t-unset"
- background
- right
- hide-on-single-page
- :pager-count="3"
- small
- ></el-pagination>
- </div>
- <div class="flex-1 scroll-auto">
- <base-table
- ref="tableRef"
- border
- stripe
- size="small"
- height="100%"
- :data="tableData"
- :columns="tableColumn"
- highlight-current-row
- @current-change="onCurrentChange"
- >
- <template #empty> 暂无数据 </template>
- </base-table>
- </div>
- </div>
- </div>
- <div class="flex-1 p-t-base flex overflow-hidden">
- <div class="flex direction-column flex-1 overflow-hidden radius-base fill-blank relative paper-view">
- <div class="flex-1 p-extra-small scroll-auto img-wrap">
- <img :src="current?.filePath" alt="" class="paper-img relative" />
- <p v-if="current" class="absolute mark-score">{{ current.headerScore ?? current.markerScore }}</p>
- </div>
- <span class="preview" @click="onPreview">
- <svg-icon name="preview"></svg-icon>
- </span>
- <scoring-panel-with-confirm
- :id="current?.taskId"
- v-model:visible="scoringPanelVisible"
- v-model:score="modelScore"
- :main-number="current?.mainNumber"
- @submit="onSubmit"
- ></scoring-panel-with-confirm>
- </div>
- <div class="radius-base p-extra-small fill-blank m-l-base overflow-auto paper-mark-record">
- <mark-history-list
- :id="current?.taskId"
- height="100%"
- :model-value="true"
- :modal="false"
- :task="current"
- ></mark-history-list>
- </div>
- </div>
- </div>
- <image-preview v-model="previewModalVisible" :url="current?.filePath"></image-preview>
- </template>
- <script setup lang="ts" name="AnalysisGroupDetail">
- /** 小组监控数据详情 */
- import { ref, computed, watch } from 'vue'
- import { useRoute, useRouter } from 'vue-router'
- import { ElButton, ElPagination } from 'element-plus'
- import useFetch from '@/hooks/useFetch'
- import useTable from '@/hooks/useTable'
- import useTableCheck from '@/hooks/useTableCheck'
- import VueEcharts from 'vue-echarts'
- import BaseTable from '@/components/element/BaseTable.vue'
- import Message from '@/components/shared/message/Message.vue'
- import UserInfo from '@/components/shared/UserInfo.vue'
- import ScoringPanelWithConfirm from '@/components/shared/ScoringPanelWithConfirm.vue'
- import ImagePreview from '@/components/shared/ImagePreview.vue'
- import MarkHistoryList from '@/components/shared/MarkHistoryList.vue'
- import SvgIcon from '@/components/common/SvgIcon.vue'
- import LockEntry from '@/components/common/LockEntry.vue'
- import type { EChartsOption } from 'echarts'
- import type { ExtractMultipleApiResponse, ExtractApiResponse } from '@/api/api'
- import type { EpTableColumn } from 'global-type'
- const { back } = useRouter()
- const { query } = useRoute()
- /** 给分板 */
- const scoringPanelVisible = ref<boolean>(true)
- /** 图片预览 */
- const previewModalVisible = ref<boolean>(false)
- /** 分数 */
- const modelScore = ref<number[]>([])
- /** 类型 */
- const dataType = computed(() => {
- return query.operateType === 'VIEW' ? '已浏览' : '已给分'
- })
- /** 预览试卷 */
- const onPreview = () => {
- previewModalVisible.value = true
- }
- /** 抽查详情列表 */
- const { pagination, currentPage, total, data, fetchTable } = useTable(
- 'getGroupMonitorDetail',
- {
- operateType: query.operateType as 'VIEW' | 'MARK',
- headerId: query.headerId as string,
- },
- { pageSize: 4 }
- )
- const { tableRef, current, onCurrentChange, tableData, next } = useTableCheck(data)
- /** 抽查详情表格配置 */
- const tableColumn: EpTableColumn<ExtractMultipleApiResponse<'getGroupMonitorDetail'>>[] = [
- { label: '密号', prop: 'secretNumber', width: 100, fixed: 'left' },
- { label: '评卷员', prop: 'markerName' },
- { label: '给分', prop: 'markerScore', width: 50 },
- { label: '组长给分', prop: 'headerScore', width: 70 },
- { label: '客观分', prop: 'objectiveScore', width: 60 },
- { label: '客主比', prop: 'headerRatio', width: 60 },
- { label: '分档', prop: 'scoreLevel' },
- { label: '评卷时间', prop: 'markTime', width: 130 },
- ]
- const { fetch: getStatisticObjectiveByMarker, result: objectiveByMarker } = useFetch('getStatisticObjectiveByMarker')
- const { fetch: getStatisticSubjectiveByMarker, result: subjectiveByMarker } = useFetch('getStatisticSubjectiveByMarker')
- watch(
- [current],
- () => {
- if (current.value?.markerId) {
- getStatisticObjectiveByMarker({
- markerId: current.value.markerId,
- startTime: '',
- endTime: '',
- })
- getStatisticSubjectiveByMarker({
- markerId: current.value.markerId,
- startTime: '',
- endTime: '',
- })
- }
- },
- { immediate: true, deep: true }
- )
- type StatisticObjectiveByMarker = ExtractApiResponse<'getStatisticObjectiveByMarker'>
- type StatisticObjectiveByMarkerValues = StatisticObjectiveByMarker['segmentsByAll']
- const getXAxisData = <K extends keyof ExtractArrayValue<StatisticObjectiveByMarkerValues>>(
- field: K,
- data?: StatisticObjectiveByMarkerValues
- ) => {
- if (!data) {
- return []
- }
- const getValue = (key: K, item: ExtractArrayValue<StatisticObjectiveByMarkerValues>) => {
- return item[key]
- }
- return data?.map((v) => getValue(field, v))
- }
- const markerSubjectiveChartsOption = computed<EChartsOption>(() => {
- return {
- title: { text: '主观分布' },
- grid: {
- bottom: 30,
- right: 0,
- },
- legend: {
- itemWidth: 14,
- data: ['评卷员', '小组', '题组'],
- right: 10,
- top: 2,
- },
- xAxis: {
- axisLine: { show: false },
- axisTick: { show: false },
- splitLine: { show: false },
- axisLabel: {
- align: 'right',
- },
- data: getXAxisData('scoreStart', subjectiveByMarker?.value?.segmentsByAll),
- },
- yAxis: [
- // {
- // type: 'value',
- // },
- {
- type: 'value',
- axisLabel: {
- formatter: `{value}%`,
- },
- splitLine: { show: true },
- },
- ],
- series: [
- {
- name: '评卷员',
- type: 'line',
- itemStyle: {
- color: '#3AD500',
- },
- data: getXAxisData('rate', subjectiveByMarker?.value?.segmentsByUser),
- },
- {
- name: '小组',
- type: 'line',
- itemStyle: {
- color: '#0064FF',
- },
- data: getXAxisData('rate', subjectiveByMarker?.value?.segmentsByGroup),
- },
- {
- name: '题组',
- type: 'line',
- itemStyle: {
- color: '#008000',
- },
- data: getXAxisData('rate', subjectiveByMarker?.value?.segmentsByAll),
- },
- ],
- }
- })
- const markerObjectiveChartsOption = computed<EChartsOption>(() => {
- return {
- title: { text: '客观分布' },
- grid: {
- bottom: 30,
- right: 0,
- },
- legend: {
- itemWidth: 14,
- data: ['评卷员', '小组', '题组'],
- right: 10,
- top: 2,
- },
- xAxis: {
- axisLine: { show: false },
- axisTick: { show: false },
- splitLine: { show: false },
- axisLabel: {
- align: 'right',
- },
- data: getXAxisData('scoreStart', objectiveByMarker?.value?.segmentsByAll),
- },
- yAxis: [
- // {
- // type: 'value',
- // },
- {
- type: 'value',
- axisLabel: {
- formatter: `{value}%`,
- },
- splitLine: { show: true },
- },
- ],
- series: [
- {
- name: '评卷员',
- type: 'line',
- itemStyle: {
- color: '#3AD500',
- },
- data: getXAxisData('rate', objectiveByMarker?.value?.segmentsByUser),
- },
- {
- name: '小组',
- type: 'line',
- itemStyle: {
- color: '#0064FF',
- },
- data: getXAxisData('rate', objectiveByMarker?.value?.segmentsByGroup),
- },
- {
- name: '题组',
- type: 'line',
- itemStyle: {
- color: '#008000',
- },
- data: getXAxisData('rate', objectiveByMarker?.value?.segmentsByAll),
- },
- ],
- }
- })
- const onSubmit = async () => {
- try {
- if (!current.value?.taskId) {
- return
- }
- await useFetch('markMonitorDetailTask').fetch({ taskId: current.value.taskId, scores: modelScore.value })
- fetchTable()
- } catch (error) {
- console.error(error)
- }
- }
- fetchTable()
- </script>
- <style scoped lang="scss">
- .mark-score {
- color: red;
- font-size: 50px;
- left: 30px;
- top: 20px;
- }
- .header-view {
- border-bottom: $OnePixelLine;
- }
- .detail-info {
- height: 280px;
- .detail-info-chart {
- border-right: $OnePixelLine;
- }
- .detail-info-label {
- .detail-info-label-num {
- min-width: 32px;
- height: 24px;
- line-height: 24px;
- background: #00987b;
- border-radius: 4px;
- }
- }
- }
- .paper-view {
- .paper-img {
- max-width: 100%;
- }
- .preview {
- position: absolute;
- cursor: pointer;
- top: 10px;
- right: 20px;
- font-size: 24px;
- }
- }
- .paper-mark-record {
- width: 390px;
- }
- </style>
|