|
|
|
<template>
|
|
|
|
<div class="index" v-loading="loading || submiting">
|
|
|
|
<div class="top">
|
|
|
|
<div class="item">
|
|
|
|
<p class="names">{{ curStage.stageName }}:{{ curStage.paperName }}</p>
|
|
|
|
|
|
|
|
<div class="count m-l-30">
|
|
|
|
阶段剩余时间
|
|
|
|
<span>{{ counterVal.day }}</span>天
|
|
|
|
<span>{{ counterVal.hour }}</span>小时
|
|
|
|
<span>{{ counterVal.minutes }}</span>分
|
|
|
|
<span>{{ counterVal.seconds }}</span>秒
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
|
|
|
|
<div class="item">
|
|
|
|
<div class="count m-r-30">
|
|
|
|
用时
|
|
|
|
<span>{{ timeSum.day }}</span>天
|
|
|
|
<span>{{ timeSum.hour }}</span>小时
|
|
|
|
<span>{{ timeSum.minutes }}</span>分
|
|
|
|
<span>{{ timeSum.seconds }}</span>秒
|
|
|
|
</div>
|
|
|
|
|
|
|
|
<el-button class="submit" :loading="submiting" :disabled="submited" @click="confirmSubmit">提交</el-button>
|
|
|
|
<img class="exit" src="@/assets/img/exit.svg" alt="" @click="close">
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
|
|
|
|
<div class="wrap">
|
|
|
|
<div class="left">
|
|
|
|
<h6 class="title">答题卡</h6>
|
|
|
|
<div class="progress">
|
|
|
|
<p class="fs-14">答题进度</p>
|
|
|
|
<el-progress class="m-t-5 m-b-5" :percentage="progress" :format="progressFormat"></el-progress>
|
|
|
|
<p>共{{ form.questionCount }}题,满分{{ form.score }}分</p>
|
|
|
|
</div>
|
|
|
|
<div v-if="form.paperOutline" class="type-wrap">
|
|
|
|
<template v-for="(item, i) in form.paperOutline">
|
|
|
|
<div
|
|
|
|
v-if="item.examQuestions.length && (!sheetStatus || item.examQuestions.some(e => (sheetStatus === 1 && !e.answered) || (sheetStatus === 2 && e.answered) || (sheetStatus === 3 && e.partAnswer) || (sheetStatus === 4 && e.sign)))"
|
|
|
|
:key="i" class="type">
|
|
|
|
<h6 class="stem">{{ arabicToChinese(i + 1) }}、{{ item.outlineName }}(本题共{{ item.questionNum }}小题,共{{
|
|
|
|
item.targetScore }}分)</h6>
|
|
|
|
<ul class="serials">
|
|
|
|
<template v-for="(ques, j) in item.examQuestions">
|
|
|
|
<li
|
|
|
|
v-if="!sheetStatus || (sheetStatus === 1 && !ques.answered) || (sheetStatus === 2 && ques.answered) || (sheetStatus === 3 && ques.partAnswer) || (sheetStatus === 4 && ques.sign)"
|
|
|
|
:key="j" :class="{ answered: ques.answered, partAnswer: ques.partAnswer }"
|
|
|
|
@click="scrollToQues(ques, item)">
|
|
|
|
<img v-if="ques.sign" class="tag" src="@/assets/img/tag-active.svg" alt="">
|
|
|
|
{{ j + 1 }}
|
|
|
|
</li>
|
|
|
|
</template>
|
|
|
|
</ul>
|
|
|
|
</div>
|
|
|
|
</template>
|
|
|
|
</div>
|
|
|
|
<ul class="status-filter">
|
|
|
|
<li :class="{ active: sheetStatus === 1 }" @click="filterStatus(1)">未作答</li>
|
|
|
|
<li :class="{ active: sheetStatus === 2 }" @click="filterStatus(2)">已作答</li>
|
|
|
|
<li :class="{ active: sheetStatus === 3 }" @click="filterStatus(3)">部分已作答</li>
|
|
|
|
<li :class="{ active: sheetStatus === 4 }" @click="filterStatus(4)"><img class="tag"
|
|
|
|
src="@/assets/img/tag-active.svg" alt=""> 已标记</li>
|
|
|
|
</ul>
|
|
|
|
</div>
|
|
|
|
|
|
|
|
<ul v-if="form.paperOutline" class="ques-wrap" id="quesWrap">
|
|
|
|
<li v-for="(item, i) in form.paperOutline" :key="i">
|
|
|
|
<div class="outline">
|
|
|
|
{{ arabicToChinese(i + 1) }}、{{ item.outlineName }}(本题共{{ item.questionNum }}小题,共{{
|
|
|
|
item.targetScore }}分)
|
|
|
|
<img :class="['shrink', { active: item.shrink }]" src="@/assets/img/shrink.svg" alt=""
|
|
|
|
@click="item.shrink = !item.shrink">
|
|
|
|
</div>
|
|
|
|
<div :class="['ques', { hide: item.shrink }]">
|
|
|
|
<div v-for="(ques, j) in item.examQuestions" :key="j" class="item" :id="'ques' + ques.id">
|
|
|
|
<div class="stem-wrap">
|
|
|
|
<div class="labels">
|
|
|
|
<span class="label">{{ j + 1 }} / {{ item.questionNum }}</span>
|
|
|
|
<span class="label">{{ item.questionTypeName }}</span>
|
|
|
|
</div>
|
|
|
|
<div class="stem html-parse" :id="'stem' + ques.id" v-html="ques.stem"></div>
|
|
|
|
<p>({{ ques.score }}分)</p>
|
|
|
|
<img class="tag" :src="require('@/assets/img/' + (ques.sign ? 'tag-active' : 'tag') + '.svg')" alt=""
|
|
|
|
@click="ques.sign = ques.sign ? 0 : 1">
|
|
|
|
</div>
|
|
|
|
|
|
|
|
<!-- 单选、多选、判断的选项 -->
|
|
|
|
<template
|
|
|
|
v-if="item.questionType !== 'fill_blank' && item.questionType !== 'essay' && ques.questionAnswerVersionsList">
|
|
|
|
<div v-for="(opt, j) in ques.questionAnswerVersionsList" :key="j" class="opt">
|
|
|
|
<el-checkbox v-if="item.questionType === 'multiple_choice'" v-model="opt.answer" :true-label="1"
|
|
|
|
@change="mulChange(ques)"></el-checkbox>
|
|
|
|
<el-radio v-else v-model="opt.answer" tabindex="-1" :true-label="1" :label="1"
|
|
|
|
@change="singleChange(ques, j)"></el-radio>
|
|
|
|
|
|
|
|
<span>{{ numToLetter(j) }}. </span>
|
|
|
|
<div class="text html-parse" v-html="opt.optionText"></div>
|
|
|
|
</div>
|
|
|
|
</template>
|
|
|
|
<!-- 简答题需要展示题干文件及富文本 -->
|
|
|
|
<template v-if="item.questionType === 'essay'">
|
|
|
|
<div v-if="ques.stemAttachment" class="m-b-20">
|
|
|
|
<el-link class="m-r-10" type="primary" @click="preview(ques.stemAttachment)">{{ ques.fileName ||
|
|
|
|
ques.stemAttachment }}</el-link>
|
|
|
|
<el-button type="primary" size="mini" round
|
|
|
|
@click="download(ques.fileName || ques.stemAttachment, ques.stemAttachment)">下载</el-button>
|
|
|
|
</div>
|
|
|
|
<UeditorPlus v-if="ques.richTextStatus" :ref="'essayAnswer' + ques.id" v-model="ques.answer"
|
|
|
|
@ready="editor => essayAnswerReady(editor, ques)" />
|
|
|
|
<div v-if="ques.allowAttachment" class="m-t-20">
|
|
|
|
<el-upload style="max-width: 700px;" accept=".csv,.xlsx,.xls,.docx,.doc,.pdf,.jpg,.png,.zip,.rar,.7z"
|
|
|
|
:before-upload="beforeUpload" :on-remove="e => handleRemove(ques)" :on-error="uploadError"
|
|
|
|
:before-remove="beforeRemove" :on-preview="handlePreview" :limit="1" action="#"
|
|
|
|
:on-exceed="handleExceed" :file-list="ques.fileList" :http-request="e => handleRequest(e, ques)">
|
|
|
|
<el-button type="primary" plain>上传文件</el-button>
|
|
|
|
</el-upload>
|
|
|
|
<div v-if="ques.uploadInstructions" class="flex m-t-10 fs-12">
|
|
|
|
<span>上传要求说明:</span>
|
|
|
|
<div class="html-parse" v-html="ques.uploadInstructions"></div>
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
</template>
|
|
|
|
</div>
|
|
|
|
</div>
|
|
|
|
</li>
|
|
|
|
</ul>
|
|
|
|
</div>
|
|
|
|
|
|
|
|
<el-dialog title="图片预览" :visible.sync="previewImgVisible" width="800px" :close-on-click-modal="false">
|
|
|
|
<el-image style="max-width: 100px; max-height: 100px" :src="previewImg" :preview-src-list="[previewImg]">
|
|
|
|
</el-image>
|
|
|
|
</el-dialog>
|
|
|
|
<PdfDia :key="pdfVisible" :visible.sync="pdfVisible" :src.sync="pdfSrc" />
|
|
|
|
</div>
|
|
|
|
</template>
|
|
|
|
|
|
|
|
<script>
|
|
|
|
import Util from '@/libs/util'
|
|
|
|
import Setting from "@/setting"
|
|
|
|
import QuesConst from '@/const/ques'
|
|
|
|
import TestPaperConst from '@/const/testPaper'
|
|
|
|
import _ from 'lodash'
|
|
|
|
import Upload from '@/components/upload'
|
|
|
|
import UeditorPlus from '@/components/ueditorPlus'
|
|
|
|
import PdfDia from '@/components/pdf'
|
|
|
|
import Bus from '@/libs/bus'
|
|
|
|
import OSS from 'ali-oss'
|
|
|
|
import OssConfig from '@/components/upload/config.js'
|
|
|
|
import Oss from '@/components/upload/upload.js'
|
|
|
|
|
|
|
|
export default {
|
|
|
|
components: {
|
|
|
|
Upload, UeditorPlus, PdfDia
|
|
|
|
},
|
|
|
|
data () {
|
|
|
|
return {
|
|
|
|
loading: false,
|
|
|
|
per: 2, // 项目权限(0、练习 1、考核 2、竞赛)
|
|
|
|
questionTypes: QuesConst.questionTypes,
|
|
|
|
numToLetter: Util.numToLetter,
|
|
|
|
arabicToChinese: Util.arabicToChinese,
|
|
|
|
headers: {
|
|
|
|
token: Util.local.get(Setting.tokenKey)
|
|
|
|
},
|
|
|
|
token: Util.local.get(Setting.tokenKey),
|
|
|
|
id: +this.$route.query.id, // 赛事id
|
|
|
|
stageId: +this.$route.query.stageId,
|
|
|
|
teamId: this.$route.query.teamId || '',
|
|
|
|
entryTime: '',
|
|
|
|
curStage: {
|
|
|
|
stageName: '',
|
|
|
|
paperName: '',
|
|
|
|
},
|
|
|
|
timer: null,
|
|
|
|
counterTimer: null,
|
|
|
|
countVal: 0,
|
|
|
|
// 倒计时
|
|
|
|
counterVal: {
|
|
|
|
day: 0,
|
|
|
|
seconds: 0,
|
|
|
|
minutes: 0,
|
|
|
|
hour: 0,
|
|
|
|
},
|
|
|
|
timeSumVal: 0,
|
|
|
|
// 用时
|
|
|
|
timeSum: {
|
|
|
|
day: 0,
|
|
|
|
seconds: 0,
|
|
|
|
minutes: 0,
|
|
|
|
hour: 0,
|
|
|
|
},
|
|
|
|
totalAnswered: 0,
|
|
|
|
progress: 0,
|
|
|
|
sheetStatus: '',
|
|
|
|
form: {
|
|
|
|
questionCount: 0,
|
|
|
|
},
|
|
|
|
submiting: false,
|
|
|
|
submited: false,
|
|
|
|
warned: 0,
|
|
|
|
quesWrapWidth: 0,
|
|
|
|
previewImgVisible: false,
|
|
|
|
previewImg: '',
|
|
|
|
pdfVisible: false,
|
|
|
|
pdfSrc: '',
|
|
|
|
client: null,
|
|
|
|
};
|
|
|
|
},
|
|
|
|
mounted () {
|
|
|
|
this.$once('hook:beforeDestroy', function () {
|
|
|
|
clearInterval(this.counterTimer)
|
|
|
|
})
|
|
|
|
|
|
|
|
// websocket实时刷新
|
|
|
|
Bus.$on('matchSocket', () => {
|
|
|
|
this.getCompetition()
|
|
|
|
})
|
|
|
|
|
|
|
|
this.getCompetition(1)
|
|
|
|
this.initOss()
|
|
|
|
},
|
|
|
|
methods: {
|
|
|
|
// 获取竞赛信息
|
|
|
|
async getCompetition (load) {
|
|
|
|
if (load) this.loading = true
|
|
|
|
const { competition } = await this.$post(`${this.api.getCompetition}?competitionId=${this.id}`)
|
|
|
|
const stages = competition.contentList
|
|
|
|
if (stages) {
|
|
|
|
const stage = stages.find(e => e.stageId === this.stageId)
|
|
|
|
const item = competition.competitionStage.find(e => e.stageId === this.stageId)
|
|
|
|
if (item) {
|
|
|
|
stage.stageName = item.stageName
|
|
|
|
stage.resultAnnouncementTime = item.resultAnnouncementTime
|
|
|
|
}
|
|
|
|
this.curStage = stage
|
|
|
|
const endTime = new Date(stage.endTime)
|
|
|
|
const now = await Util.getNow()
|
|
|
|
// 如果已经结束
|
|
|
|
if (now >= endTime) {
|
|
|
|
this.loading = false
|
|
|
|
this.submit(1, 1)
|
|
|
|
} else { // 没结束,则显示倒计时
|
|
|
|
this.countVal = (endTime - now) / 1000
|
|
|
|
load && this.getPaper(now)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
},
|
|
|
|
// 获取试卷详情
|
|
|
|
async getPaper (now) {
|
|
|
|
try {
|
|
|
|
const { paperId, stageId } = this.curStage
|
|
|
|
if (paperId) {
|
|
|
|
// 试卷详情
|
|
|
|
const { examPaper } = await this.$get(this.api.examPaperDetails, { id: paperId })
|
|
|
|
|
|
|
|
// 缓存,如果有,则全部回显到页面上
|
|
|
|
const { examSubmitReq: cache } = await this.$post(this.api.getExamPaperCache, {
|
|
|
|
competitionId: this.id,
|
|
|
|
paperId,
|
|
|
|
stageId
|
|
|
|
})
|
|
|
|
let cacheQues = []
|
|
|
|
if (cache) {
|
|
|
|
this.entryTime = new Date(cache.startTime)
|
|
|
|
this.timeSumVal = (now - new Date(cache.startTime)) / 1000
|
|
|
|
cacheQues = cache.examSubmitJudgeList
|
|
|
|
} else {
|
|
|
|
this.entryTime = now
|
|
|
|
}
|
|
|
|
this.startCount()
|
|
|
|
|
|
|
|
const r = examPaper
|
|
|
|
const paper = r.paperOutline
|
|
|
|
const types = this.questionTypes
|
|
|
|
paper.map(e => {
|
|
|
|
const type = e.questionType
|
|
|
|
e.questionTypeName = types.find(n => n.id === type).name
|
|
|
|
e.shrink = false
|
|
|
|
e.examQuestions.map(n => {
|
|
|
|
Object.assign(n, n.question)
|
|
|
|
const curQues = cacheQues.find(m => m.outlineId === e.outlineId && m.questionVersionId === n.questionVersionId) // 缓存里的试题
|
|
|
|
n.sign = curQues ? curQues.sign : 0
|
|
|
|
const opts = n.questionAnswerVersionsList
|
|
|
|
if (type !== 'fill_blank' && type !== 'essay' && opts) { // 选择题
|
|
|
|
opts.map(m => {
|
|
|
|
m.answer = curQues && curQues.answer && curQues.answer.length && curQues.answer.some(n => n == m.optionNumber) ? 1 : 0
|
|
|
|
})
|
|
|
|
if (opts.some(m => m.answer)) n.answered = 1
|
|
|
|
} else if (type === 'fill_blank') { // 填空题
|
|
|
|
n.fills = curQues && curQues.answer && curQues.answer.length && curQues.answer.some(m => m) ? curQues.answer : ''
|
|
|
|
let { answerData } = opts[0]
|
|
|
|
if (answerData) answerData = JSON.parse(answerData)
|
|
|
|
n.answered = n.fills && answerData && answerData.length === n.fills.filter(m => m).length ? 1 : 0
|
|
|
|
n.partAnswer = n.fills && answerData && answerData.length !== n.fills.filter(m => m).length ? 1 : 0
|
|
|
|
n.stem = this.getQuesStem(n)
|
|
|
|
} else if (type === 'essay') { // 简答题
|
|
|
|
n.answer = type === 'essay' && curQues ? curQues.answerContent : ''
|
|
|
|
n.attachmentName = curQues ? curQues.attachmentName : ''
|
|
|
|
n.attachmentUrl = curQues ? curQues.attachmentUrl : ''
|
|
|
|
n.fileList = []
|
|
|
|
if (n.attachmentUrl) {
|
|
|
|
n.fileList = [{
|
|
|
|
name: n.attachmentName,
|
|
|
|
url: n.attachmentUrl
|
|
|
|
}]
|
|
|
|
}
|
|
|
|
// 附件
|
|
|
|
const url = n.attachmentUrl
|
|
|
|
if (url) {
|
|
|
|
n.uploadList = [{
|
|
|
|
name: n.attachmentName || url,
|
|
|
|
url
|
|
|
|
}]
|
|
|
|
}
|
|
|
|
if (n.answer || url) n.answered = 1
|
|
|
|
}
|
|
|
|
})
|
|
|
|
})
|
|
|
|
this.form = r
|
|
|
|
this.loading = false
|
|
|
|
|
|
|
|
// 给填空题的每个空监听input事件,用以显示已作答状态
|
|
|
|
this.$nextTick(() => {
|
|
|
|
cache || this.submit(0) // 如果没有缓存,则先缓存一次(只是为了缓存进入时间,因为只有在每个小题答题后才会调缓存接口,所以只能第一次进入就缓存一次进入时间)
|
|
|
|
paper.map(e => {
|
|
|
|
e.examQuestions.map(n => {
|
|
|
|
if (e.questionType === 'fill_blank') {
|
|
|
|
const stem = document.querySelector(`#stem` + n.id)
|
|
|
|
if (stem) {
|
|
|
|
const inputs = stem.querySelectorAll('.fill-input')
|
|
|
|
if (inputs) {
|
|
|
|
for (const e of inputs) {
|
|
|
|
e.addEventListener('input', () => {
|
|
|
|
const answers = []
|
|
|
|
let hasFillLen = 0
|
|
|
|
for (const e of inputs) {
|
|
|
|
const val = e.innerText
|
|
|
|
if (val) hasFillLen++
|
|
|
|
answers.push(val)
|
|
|
|
}
|
|
|
|
n.fills = answers
|
|
|
|
n.answered = hasFillLen === inputs.length ? 1 : 0
|
|
|
|
n.partAnswer = hasFillLen && hasFillLen !== inputs.length ? 1 : 0
|
|
|
|
|
|
|
|
clearTimeout(this.timer)
|
|
|
|
this.timer = setTimeout(() => {
|
|
|
|
this.submit(0)
|
|
|
|
this.calcProgress()
|
|
|
|
}, 500)
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
})
|
|
|
|
})
|
|
|
|
this.calcProgress()
|
|
|
|
})
|
|
|
|
}
|
|
|
|
} catch (e) {
|
|
|
|
this.loading = false
|
|
|
|
}
|
|
|
|
},
|
|
|
|
// 计时器
|
|
|
|
handleCounter (counterTime, isCount) {
|
|
|
|
let leave1 = counterTime % (24 * 3600) //计算天数后剩余的毫秒数
|
|
|
|
let leave2 = leave1 % 3600 //计算小时数后剩余的毫秒数
|
|
|
|
let leave3 = leave2 % 60 //计算分钟数后剩余的毫秒数
|
|
|
|
let day = Math.floor(counterTime / (24 * 3600)) //计算相差天数
|
|
|
|
let hour = Math.floor(leave1 / 3600) //计算相差小时
|
|
|
|
let minutes = Math.floor(leave2 / 60) //计算相差分钟
|
|
|
|
let seconds = Math.round(leave3) //计算相差秒
|
|
|
|
day = this.timeFormat(day)
|
|
|
|
hour = this.timeFormat(hour)
|
|
|
|
minutes = this.timeFormat(minutes)
|
|
|
|
seconds = this.timeFormat(seconds)
|
|
|
|
const count = this[isCount ? 'counterVal' : 'timeSum']
|
|
|
|
count.day = day
|
|
|
|
count.hour = hour
|
|
|
|
count.minutes = minutes
|
|
|
|
count.seconds = seconds
|
|
|
|
},
|
|
|
|
// 计时前的判断
|
|
|
|
counter (counterTime) {
|
|
|
|
if (counterTime <= 0) {
|
|
|
|
if (this.per) { // 竞赛/考核
|
|
|
|
clearInterval(this.counterTimer)
|
|
|
|
this.submit(1, 1)
|
|
|
|
} else {
|
|
|
|
this.handleCounter(counterTime, 1)
|
|
|
|
}
|
|
|
|
} else {
|
|
|
|
// 剩15分钟时要提示
|
|
|
|
if (counterTime === 900) {
|
|
|
|
this.warned || this.$alert(`请注意,${this.per == 2 ? '比赛' : '考核'}还剩15分钟,请尽快完成答题并提交试卷。`, '提示', {
|
|
|
|
confirmButtonText: '确定',
|
|
|
|
type: 'warning'
|
|
|
|
})
|
|
|
|
this.warned = 1
|
|
|
|
}
|
|
|
|
this.handleCounter(counterTime, 1)
|
|
|
|
}
|
|
|
|
},
|
|
|
|
// 启动倒计时
|
|
|
|
startCount () {
|
|
|
|
clearInterval(this.counterTimer)
|
|
|
|
this.counterTimer = setInterval(() => {
|
|
|
|
this.counter(this.per ? this.countVal-- : this.countVal++)
|
|
|
|
this.timeSumVal >= 0 && this.handleCounter(this.timeSumVal++)
|
|
|
|
}, 1000)
|
|
|
|
},
|
|
|
|
timeFormat (param) {
|
|
|
|
return param < 10 ? '0' + param : param
|
|
|
|
},
|
|
|
|
// 计算答题进度
|
|
|
|
calcProgress () {
|
|
|
|
let answered = 0
|
|
|
|
this.form.paperOutline.map(e => {
|
|
|
|
answered += e.examQuestions.filter(n => n.answered).length
|
|
|
|
})
|
|
|
|
this.totalAnswered = answered
|
|
|
|
this.progress = +(answered / this.form.questionCount * 100).toFixed(1)
|
|
|
|
},
|
|
|
|
// 答题进度格式
|
|
|
|
progressFormat (e) {
|
|
|
|
return this.totalAnswered + '/' + this.form.questionCount
|
|
|
|
},
|
|
|
|
scrollToSmooth (position, duration) {
|
|
|
|
let startTime = Date.now()
|
|
|
|
|
|
|
|
function scroll () {
|
|
|
|
let now = Date.now()
|
|
|
|
let progress = Math.min(1, (now - startTime) / duration)
|
|
|
|
document.querySelector('#quesWrap').scrollTo(0, position * progress)
|
|
|
|
|
|
|
|
if (progress < 1) {
|
|
|
|
window.requestAnimationFrame(scroll)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
window.requestAnimationFrame(scroll)
|
|
|
|
},
|
|
|
|
// 答题卡题目点击滚动
|
|
|
|
scrollToQues (e, item) {
|
|
|
|
item.shrink = false
|
|
|
|
const el = document.querySelector('#ques' + e.id)
|
|
|
|
this.$nextTick(() => {
|
|
|
|
el && this.scrollToSmooth(el.offsetTop - document.querySelector('#quesWrap').offsetTop, 200)
|
|
|
|
})
|
|
|
|
},
|
|
|
|
// 答题卡筛选
|
|
|
|
filterStatus (e) {
|
|
|
|
this.sheetStatus = this.sheetStatus === e ? '' : e
|
|
|
|
},
|
|
|
|
// 处理题干显示
|
|
|
|
getQuesStem (ques) {
|
|
|
|
let { stem } = ques
|
|
|
|
if (ques.questionType === 'fill_blank') { // 填空题
|
|
|
|
let { fills } = ques
|
|
|
|
|
|
|
|
const regex = /<span class="gapfilling-span" data-id="(.*?)">______<\/span>/g
|
|
|
|
let match
|
|
|
|
let index = 0 // 用于跟踪索引
|
|
|
|
let result = stem
|
|
|
|
|
|
|
|
while ((match = regex.exec(stem)) !== null) {
|
|
|
|
const newInput = `<span contenteditable placeholder="(${index + 1})" class="fill-input">${fills && fills.length ? fills[index] : ''}</span>`
|
|
|
|
result = result.replace(match[0], newInput)
|
|
|
|
index++
|
|
|
|
}
|
|
|
|
return result
|
|
|
|
} else {
|
|
|
|
return stem
|
|
|
|
}
|
|
|
|
},
|
|
|
|
preventAllKeyPress (event) {
|
|
|
|
// 阻止所有键盘事件
|
|
|
|
event.preventDefault();
|
|
|
|
},
|
|
|
|
// 单选题回调
|
|
|
|
singleChange (ques, j) {
|
|
|
|
ques.questionAnswerVersionsList.map(e => {
|
|
|
|
e.answer = 0
|
|
|
|
})
|
|
|
|
ques.questionAnswerVersionsList[j].answer = 1
|
|
|
|
|
|
|
|
ques.answered = 1
|
|
|
|
this.calcProgress()
|
|
|
|
this.submit(0)
|
|
|
|
},
|
|
|
|
// 多选题回调
|
|
|
|
mulChange (ques) {
|
|
|
|
ques.answered = ques.questionAnswerVersionsList.some(e => e.answer)
|
|
|
|
this.calcProgress()
|
|
|
|
this.submit(0)
|
|
|
|
},
|
|
|
|
|
|
|
|
// 简答题富文本加载完毕回调
|
|
|
|
essayAnswerReady (editor, ques) {
|
|
|
|
editor.ques = ques
|
|
|
|
editor.addListener('contentChange', () => {
|
|
|
|
const content = editor.getContent()
|
|
|
|
ques.answered = content ? 1 : 0
|
|
|
|
ques.answers = content
|
|
|
|
|
|
|
|
clearTimeout(this.timer)
|
|
|
|
this.timer = setTimeout(() => {
|
|
|
|
this.submit(0)
|
|
|
|
this.calcProgress()
|
|
|
|
}, 500)
|
|
|
|
})
|
|
|
|
ques.answer && editor.setContent(ques.answer)
|
|
|
|
},
|
|
|
|
// 预览附件
|
|
|
|
preview (url) {
|
|
|
|
const ext = url.split('.').pop()
|
|
|
|
if (Util.isDoc(ext)) {
|
|
|
|
window.open('https://view.officeapps.live.com/op/view.aspx?src=' + url)
|
|
|
|
} else if (Util.isImg(ext)) {
|
|
|
|
this.previewImgVisible = true
|
|
|
|
this.previewImg = url
|
|
|
|
} else if (ext === 'pdf') {
|
|
|
|
this.pdfVisible = true
|
|
|
|
this.pdfSrc = url
|
|
|
|
}
|
|
|
|
},
|
|
|
|
// 下载附件
|
|
|
|
download (name, url) {
|
|
|
|
Util.downloadFile(name, url)
|
|
|
|
},
|
|
|
|
|
|
|
|
|
|
|
|
async initOss () {
|
|
|
|
const o = await OssConfig()
|
|
|
|
this.client = new OSS(o.config)
|
|
|
|
},
|
|
|
|
// 附件上传前
|
|
|
|
beforeUpload (file) {
|
|
|
|
const oversize = file.size / 1024 / 1024 < 50
|
|
|
|
if (!oversize) Util.warningMsg('请上传小于50M的文件!')
|
|
|
|
if (oversize) {
|
|
|
|
return true
|
|
|
|
} else {
|
|
|
|
return false
|
|
|
|
}
|
|
|
|
},
|
|
|
|
// 自定义上传
|
|
|
|
async handleRequest (e, ques) {
|
|
|
|
const { file } = e
|
|
|
|
try {
|
|
|
|
ques.fileList = []
|
|
|
|
const { name } = await this.client.multipartUpload(Date.now() + '.' + Util.getFileExt(file.name), file)
|
|
|
|
const url = 'https://huoran.oss-cn-shenzhen.aliyuncs.com/' + name
|
|
|
|
ques.fileList = [{
|
|
|
|
name: file.name,
|
|
|
|
url
|
|
|
|
}]
|
|
|
|
ques.attachmentName = file.name
|
|
|
|
ques.attachmentUrl = url
|
|
|
|
this.submit(0)
|
|
|
|
} catch (e) { }
|
|
|
|
},
|
|
|
|
handlePreview ({ url }) {
|
|
|
|
this.preview(url)
|
|
|
|
},
|
|
|
|
uploadError () {
|
|
|
|
this.$message({
|
|
|
|
message: "上传出错,请重试!",
|
|
|
|
type: "error",
|
|
|
|
center: true
|
|
|
|
})
|
|
|
|
},
|
|
|
|
beforeRemove (file) {
|
|
|
|
return this.$confirm(`确定移除 ${file.name}?`);
|
|
|
|
},
|
|
|
|
handleExceed () {
|
|
|
|
Util.warningMsg(`当前限制选择 1 个文件,如需更换,请删除上一个文件再重新选择!`);
|
|
|
|
},
|
|
|
|
handleRemove (ques) {
|
|
|
|
Oss.del(ques.attachmentUrl)
|
|
|
|
ques.fileList = []
|
|
|
|
ques.attachmentName = ''
|
|
|
|
ques.attachmentUrl = ''
|
|
|
|
this.submit(0)
|
|
|
|
},
|
|
|
|
|
|
|
|
|
|
|
|
// 解析富文本加载完毕回调
|
|
|
|
answerAnalysisReady (editor) {
|
|
|
|
this.answerAnalysis && editor.setContent(this.answerAnalysis)
|
|
|
|
},
|
|
|
|
// 提交询问
|
|
|
|
async confirmSubmit () {
|
|
|
|
if (this.submiting) return false
|
|
|
|
let msg = '此操作将视为结束答题,确认要提交吗?'
|
|
|
|
|
|
|
|
for (const e of this.form.paperOutline) {
|
|
|
|
if (e.examQuestions.some(n => !n.answered)) {
|
|
|
|
msg = '还有试题未答完,确认要提交吗?'
|
|
|
|
break
|
|
|
|
}
|
|
|
|
}
|
|
|
|
try {
|
|
|
|
await this.$confirm(msg, '提示', {
|
|
|
|
// confirmButtonText: '确定',
|
|
|
|
// cancelButtonText: '取消',
|
|
|
|
type: 'warning',
|
|
|
|
closeOnClickModal: false
|
|
|
|
})
|
|
|
|
this.submit(1)
|
|
|
|
} catch (e) { }
|
|
|
|
},
|
|
|
|
// 提交
|
|
|
|
async submit (isSubmit, autoSubmit) {
|
|
|
|
if (isSubmit && this.submiting) return false
|
|
|
|
try {
|
|
|
|
if (isSubmit) this.submiting = true
|
|
|
|
const form = _.cloneDeep(this.form)
|
|
|
|
const { entryTime, curStage } = this
|
|
|
|
const ques = []
|
|
|
|
form.paperOutline.map(e => {
|
|
|
|
const type = e.questionType
|
|
|
|
e.examQuestions.map(n => {
|
|
|
|
let answer = []
|
|
|
|
let answerContent = ''
|
|
|
|
if (type !== 'fill_blank' && type !== 'essay') { // 选择题
|
|
|
|
answer = n.questionAnswerVersionsList.filter(m => m.answer).map(m => m.optionNumber)
|
|
|
|
} else if (type === 'essay') { // 简答题
|
|
|
|
if (n.answers) answerContent = n.answers
|
|
|
|
} else { // 填空题
|
|
|
|
answer = n.fills || []
|
|
|
|
}
|
|
|
|
ques.push({
|
|
|
|
answer,
|
|
|
|
answerContent,
|
|
|
|
attachmentName: n.attachmentName,
|
|
|
|
attachmentUrl: n.attachmentUrl,
|
|
|
|
outlineId: e.outlineId,
|
|
|
|
questionType: type,
|
|
|
|
questionVersionId: n.questionVersionId,
|
|
|
|
serialNumber: n.serialNumber,
|
|
|
|
setScore: n.score,
|
|
|
|
sign: n.sign,
|
|
|
|
})
|
|
|
|
})
|
|
|
|
})
|
|
|
|
|
|
|
|
const data = {
|
|
|
|
competitionId: this.id,
|
|
|
|
stageId: curStage.stageId,
|
|
|
|
teamId: this.teamId,
|
|
|
|
startTime: Util.formatDate('yyyy-MM-dd hh:mm:ss', entryTime), // 取页面进入的时间
|
|
|
|
totalScore: form.score,
|
|
|
|
paperId: curStage.paperId,
|
|
|
|
examSubmitJudgeList: ques,
|
|
|
|
}
|
|
|
|
const now = isSubmit ? await Util.getNow() : new Date()
|
|
|
|
const submitTime = Util.formatDate('yyyy-MM-dd hh:mm:ss', now)
|
|
|
|
data.endTime = this.per ? curStage.endTime : submitTime
|
|
|
|
data.submitTime = submitTime
|
|
|
|
data.timeSum = Math.ceil((now - entryTime) / 60000) // 计算实验用时(分钟),向上取整
|
|
|
|
|
|
|
|
// 缓存跟提交接口的参数一样
|
|
|
|
await this.$post(this.api[isSubmit ? 'submitTheExamPaper' : 'examPaperRecordCache'], data)
|
|
|
|
|
|
|
|
if (isSubmit) {
|
|
|
|
window.opener && window.opener.location.reload()
|
|
|
|
clearInterval(this.counterTimer)
|
|
|
|
this.submiting = false
|
|
|
|
this.submited = true
|
|
|
|
|
|
|
|
// 如果是时间到了自动提交,则提交完不弹框,直接关闭页面
|
|
|
|
if (autoSubmit) {
|
|
|
|
this.$alert(`${this.per == 2 ? '竞赛' : '考核'}时间已到,系统已自动交卷`, '提示', {
|
|
|
|
confirmButtonText: '确定',
|
|
|
|
type: 'warning',
|
|
|
|
callback: _ => {
|
|
|
|
this.close()
|
|
|
|
}
|
|
|
|
})
|
|
|
|
} else {
|
|
|
|
// 如果是竞赛,并且勾选了公布成绩详情的选项,则弹框提示
|
|
|
|
if (this.id) {
|
|
|
|
const time = curStage.resultAnnouncementTime
|
|
|
|
const msg =
|
|
|
|
time === 0 ? '提交成功!成绩将在比赛结束后公布,请前往参赛信息模块查看' : time > 0 ? `提交成功!成绩将在比赛结束后${time}小时公布,请前往参赛信息模块查看` : '提交成功';
|
|
|
|
this.$alert(msg, '提示', {
|
|
|
|
confirmButtonText: '确定',
|
|
|
|
type: 'success',
|
|
|
|
callback: () => {
|
|
|
|
this.close()
|
|
|
|
}
|
|
|
|
})
|
|
|
|
} else {
|
|
|
|
this.$message.success('提交成功!')
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
} catch (e) {
|
|
|
|
this.submiting = false
|
|
|
|
}
|
|
|
|
},
|
|
|
|
close () {
|
|
|
|
window.close()
|
|
|
|
},
|
|
|
|
}
|
|
|
|
};
|
|
|
|
</script>
|
|
|
|
|
|
|
|
<style lang="scss" scoped>
|
|
|
|
.top {
|
|
|
|
display: flex;
|
|
|
|
justify-content: space-between;
|
|
|
|
align-items: center;
|
|
|
|
padding: 10px 20px;
|
|
|
|
color: #fff;
|
|
|
|
background-color: #5786fc;
|
|
|
|
|
|
|
|
.item {
|
|
|
|
display: inline-flex;
|
|
|
|
align-items: center;
|
|
|
|
}
|
|
|
|
|
|
|
|
.names {
|
|
|
|
font-size: 18px;
|
|
|
|
font-weight: 600;
|
|
|
|
}
|
|
|
|
|
|
|
|
.submit {
|
|
|
|
width: 106px;
|
|
|
|
font-size: 15px;
|
|
|
|
}
|
|
|
|
|
|
|
|
.exit {
|
|
|
|
margin-left: 20px;
|
|
|
|
cursor: pointer;
|
|
|
|
|
|
|
|
&:hover {
|
|
|
|
opacity: .9;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/deep/.wrap {
|
|
|
|
display: flex;
|
|
|
|
padding: 15px;
|
|
|
|
|
|
|
|
.left {
|
|
|
|
width: 293px;
|
|
|
|
margin-right: 15px;
|
|
|
|
background-color: #fff;
|
|
|
|
|
|
|
|
.title {
|
|
|
|
padding: 10px 0;
|
|
|
|
font-size: 16px;
|
|
|
|
text-align: center;
|
|
|
|
color: #333;
|
|
|
|
}
|
|
|
|
|
|
|
|
.progress {
|
|
|
|
padding: 10px;
|
|
|
|
color: #5a5a5a;
|
|
|
|
background-color: #d4e9ff;
|
|
|
|
}
|
|
|
|
|
|
|
|
.type-wrap {
|
|
|
|
max-height: calc(100vh - 297px);
|
|
|
|
padding: 10px;
|
|
|
|
overflow: auto;
|
|
|
|
|
|
|
|
.type {
|
|
|
|
margin-bottom: 20px;
|
|
|
|
}
|
|
|
|
|
|
|
|
.stem {
|
|
|
|
font-size: 15px;
|
|
|
|
color: #333;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
.serials {
|
|
|
|
display: flex;
|
|
|
|
flex-wrap: wrap;
|
|
|
|
margin-top: 5px;
|
|
|
|
|
|
|
|
li {
|
|
|
|
position: relative;
|
|
|
|
width: 30px;
|
|
|
|
margin: 7px 9px;
|
|
|
|
font-size: 13px;
|
|
|
|
text-align: center;
|
|
|
|
line-height: 30px;
|
|
|
|
color: #505050;
|
|
|
|
border: 1px solid #d3d3d3;
|
|
|
|
cursor: pointer;
|
|
|
|
|
|
|
|
&:hover {
|
|
|
|
opacity: .9;
|
|
|
|
}
|
|
|
|
|
|
|
|
&.answered {
|
|
|
|
color: #fff;
|
|
|
|
background-color: #66b2ff;
|
|
|
|
border-color: #66b2ff;
|
|
|
|
}
|
|
|
|
|
|
|
|
&.partAnswer {
|
|
|
|
border-color: #66b2ff;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
.tag {
|
|
|
|
position: absolute;
|
|
|
|
top: -2px;
|
|
|
|
left: -4px;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
.status-filter {
|
|
|
|
display: flex;
|
|
|
|
justify-content: space-between;
|
|
|
|
padding: 10px;
|
|
|
|
border-top: 1px solid #e5e5e5;
|
|
|
|
|
|
|
|
li {
|
|
|
|
display: inline-flex;
|
|
|
|
align-items: center;
|
|
|
|
padding: 0 3px;
|
|
|
|
font-size: 12px;
|
|
|
|
color: #333;
|
|
|
|
cursor: pointer;
|
|
|
|
border: 1px solid transparent;
|
|
|
|
|
|
|
|
&:not(:last-child):before {
|
|
|
|
content: '';
|
|
|
|
width: 13px;
|
|
|
|
height: 13px;
|
|
|
|
margin-right: 5px;
|
|
|
|
border: 1px solid #ccc;
|
|
|
|
}
|
|
|
|
|
|
|
|
&:nth-child(2):before {
|
|
|
|
background-color: #56aaff;
|
|
|
|
border: 0;
|
|
|
|
}
|
|
|
|
|
|
|
|
&:nth-child(3):before {
|
|
|
|
border-color: #56aaff;
|
|
|
|
}
|
|
|
|
|
|
|
|
&.active {
|
|
|
|
font-weight: 600;
|
|
|
|
color: #007eff;
|
|
|
|
border-color: #007eff;
|
|
|
|
|
|
|
|
&:last-child {
|
|
|
|
color: #d81e06;
|
|
|
|
border-color: #d81e06;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
.ques-wrap {
|
|
|
|
width: calc(100% - 308px);
|
|
|
|
height: calc(100vh - 128px);
|
|
|
|
background-color: #fff;
|
|
|
|
overflow: auto;
|
|
|
|
outline: none;
|
|
|
|
|
|
|
|
&>li {
|
|
|
|
margin-bottom: 15px;
|
|
|
|
}
|
|
|
|
|
|
|
|
.outline {
|
|
|
|
display: flex;
|
|
|
|
justify-content: space-between;
|
|
|
|
align-items: center;
|
|
|
|
padding: 10px 15px;
|
|
|
|
font-size: 15px;
|
|
|
|
font-weight: 600;
|
|
|
|
color: #333;
|
|
|
|
background-color: #e8f0ff;
|
|
|
|
}
|
|
|
|
|
|
|
|
.shrink {
|
|
|
|
cursor: pointer;
|
|
|
|
transition: .5s;
|
|
|
|
|
|
|
|
&.active {
|
|
|
|
transform: rotate(180deg);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
.ques {
|
|
|
|
max-height: none;
|
|
|
|
padding: 15px;
|
|
|
|
margin-bottom: 15px;
|
|
|
|
transition: all 1.5s;
|
|
|
|
overflow: hidden;
|
|
|
|
|
|
|
|
&.hide {
|
|
|
|
max-height: 10px;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
.item:not(:last-child) {
|
|
|
|
padding-bottom: 15px;
|
|
|
|
margin-bottom: 15px;
|
|
|
|
border-bottom: 1px dashed #e1e1e1;
|
|
|
|
}
|
|
|
|
|
|
|
|
.stem-wrap {
|
|
|
|
display: flex;
|
|
|
|
align-items: baseline;
|
|
|
|
margin-bottom: 10px;
|
|
|
|
}
|
|
|
|
|
|
|
|
img {
|
|
|
|
max-width: 100%;
|
|
|
|
}
|
|
|
|
|
|
|
|
.tag {
|
|
|
|
margin-left: 10px;
|
|
|
|
cursor: pointer;
|
|
|
|
|
|
|
|
&:hover {
|
|
|
|
opacity: .9;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
.labels {
|
|
|
|
display: inline-flex;
|
|
|
|
align-items: center;
|
|
|
|
}
|
|
|
|
|
|
|
|
.label {
|
|
|
|
padding: 3px 5px;
|
|
|
|
margin-right: 10px;
|
|
|
|
font-size: 12px;
|
|
|
|
line-height: 1;
|
|
|
|
color: $main-color;
|
|
|
|
white-space: nowrap;
|
|
|
|
border: 1px solid;
|
|
|
|
border-radius: 2px;
|
|
|
|
}
|
|
|
|
|
|
|
|
.stem {
|
|
|
|
max-width: calc(100% - 197px);
|
|
|
|
}
|
|
|
|
|
|
|
|
.fill-input {
|
|
|
|
position: relative;
|
|
|
|
display: inline;
|
|
|
|
min-width: 50px;
|
|
|
|
height: 28px;
|
|
|
|
padding: 3px;
|
|
|
|
margin: 0 10px;
|
|
|
|
font-size: 14px;
|
|
|
|
line-height: 28px;
|
|
|
|
color: #333;
|
|
|
|
border: 0;
|
|
|
|
border-bottom: 1px solid #DCDEE0;
|
|
|
|
outline: none;
|
|
|
|
|
|
|
|
&:empty {
|
|
|
|
display: inline-block;
|
|
|
|
width: 3em;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
.opt {
|
|
|
|
display: flex;
|
|
|
|
flex-wrap: wrap;
|
|
|
|
align-items: baseline;
|
|
|
|
padding-left: 10px;
|
|
|
|
margin-bottom: 5px;
|
|
|
|
font-size: 14px;
|
|
|
|
color: #707070;
|
|
|
|
line-height: 1.6;
|
|
|
|
|
|
|
|
.el-radio,
|
|
|
|
.el-checkbox {
|
|
|
|
margin-right: 15px;
|
|
|
|
}
|
|
|
|
|
|
|
|
.el-radio__label {
|
|
|
|
display: none;
|
|
|
|
}
|
|
|
|
|
|
|
|
.text {
|
|
|
|
max-width: calc(100% - 48px);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
</style>
|