You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
618 lines
22 KiB
618 lines
22 KiB
<template> |
|
<div class="page"> |
|
<Breadcrumb style="margin-bottom: 0;" :data="crumbs" /> |
|
<div class="wrap"> |
|
<div class="side"> |
|
<div class="m-b-20"> |
|
<el-radio-group v-model="isNotJoin" @change="typeChange"> |
|
<div class="m-b-20"> |
|
<el-radio :label="0">所有试题</el-radio> |
|
</div> |
|
<div> |
|
<el-radio :label="1">未分配知识点的试题</el-radio> |
|
</div> |
|
</el-radio-group> |
|
</div> |
|
<el-divider></el-divider> |
|
<div> |
|
<div class="flex-between m-b-10"> |
|
<h6 class="page-name" style="margin-bottom: 0">知识点框架</h6> |
|
<el-button type="text" @click="toSet">设置</el-button> |
|
</div> |
|
<el-input class="m-b-10" placeholder="请输入知识点分类、知识点名称" prefix-icon="el-icon-search" size="small" |
|
v-model="keyword" clearable></el-input> |
|
<div style="height: 504px; max-height: 504px; overflow: auto"> |
|
<el-tree :data="types" default-expand-all ref="typeTree" node-key="id" highlight-current |
|
:expand-on-click-node="false" @node-click="handleNodeClick" :props="{ label: 'name', isLeaf: 'leaf' }"> |
|
<span class="custom-tree-node" slot-scope="{ node, data }"> |
|
<img v-if="data.type" class="m-r-5" src="@/assets/images/knowledge.svg" alt=""> |
|
<span class="org-name">{{ data.name }}</span> |
|
</span> |
|
</el-tree> |
|
</div> |
|
</div> |
|
</div> |
|
|
|
<div class="right"> |
|
<h6 class="page-name">筛选</h6> |
|
<div class="tool"> |
|
<ul class="filter"> |
|
<li> |
|
<label>题目类型</label> |
|
<el-select v-model="filter.questionTypes" clearable placeholder="请选择题目类型" multiple @change="initData"> |
|
<el-option v-for="(item, i) in questionTypes" :key="i" :label="item.name" :value="item.id"></el-option> |
|
</el-select> |
|
</li> |
|
<li> |
|
<label>专业</label> |
|
<el-select v-model="filter.specialtyIds" filterable clearable placeholder="请选择专业" multiple |
|
@change="initData"> |
|
<el-option v-for="(item, i) in professionals" :key="i" :label="item.professionalName" |
|
:value="item.professionalId"></el-option> |
|
</el-select> |
|
</li> |
|
<li> |
|
<label>年份</label> |
|
<el-date-picker v-model="givenYears" type="year" placeholder="请选择年份" format="yyyy" value-format="yyyy" |
|
multiple @change="initData"> |
|
</el-date-picker> |
|
</li> |
|
<li> |
|
<label>试题难度</label> |
|
<el-select v-model="filter.difficultys" clearable placeholder="请选择试题难度" multiple @change="initData"> |
|
<el-option v-for="(item, i) in difficults" :key="i" :label="item.name" :value="item.id"></el-option> |
|
</el-select> |
|
</li> |
|
<li> |
|
<label>知识点</label> |
|
<el-cascader :disabled="isNotJoin !== 0" filterable placeholder="请选择知识点" v-model="knowledgePointIds" |
|
:options="knowledges" :props="{ value: 'id', label: 'name', multiple: true }" clearable collapse-tags |
|
@change="initData"></el-cascader> |
|
</li> |
|
<li> |
|
<label>正确率</label> |
|
<el-input class="hide-spin" style="width: 90px;" placeholder="请输入" type="number" |
|
v-model.number="filter.correctRateStart" clearable /> |
|
<span style="margin: 0 10px;">% -- </span> |
|
<el-input class="hide-spin" style="width: 90px;" placeholder="请输入" type="number" |
|
v-model.number="filter.correctRateEnd" clearable /> |
|
<span style="margin-left: 10px;">%</span> |
|
</li> |
|
<li> |
|
<label>状态</label> |
|
<el-select v-model="filter.status" clearable placeholder="请选择状态" @change="initData"> |
|
<el-option v-for="(item, i) in status" :key="i" :label="item.name" :value="item.id"></el-option> |
|
</el-select> |
|
</li> |
|
<li> |
|
<label>搜索</label> |
|
<el-input style="width: 250px;" placeholder="请输入题干" prefix-icon="el-icon-search" v-model="filter.keyword" |
|
clearable /> |
|
</li> |
|
<div style="margin-bottom: 15px;"> |
|
<el-button type="primary" @click="add">新增试题</el-button> |
|
<el-button type="primary" @click="batchImport">批量导入</el-button> |
|
<!-- <el-button type="primary" @click="batchRemove">批量转移</el-button> --> |
|
<el-button type="primary" :disabled="!isKnowLedge" @click="batchRemove">批量移除 </el-button> |
|
<el-button type="primary" @click="batchDel">批量删除</el-button> |
|
</div> |
|
</ul> |
|
</div> |
|
|
|
<el-table :data="list" class="table" ref="table" stripe header-align="center" |
|
@selection-change="handleSelectionChange" row-key="id" @sort-change="sortChange"> |
|
<el-table-column type="selection" width="45" align="center" :reserve-selection="true"></el-table-column> |
|
<el-table-column type="index" width="50" label="序号" align="center"></el-table-column> |
|
<el-table-column prop="stem" label="题干" align="center" min-width="120" |
|
show-overflow-tooltip></el-table-column> |
|
<el-table-column prop="questionType" label="题型" align="center" width="70" sortable="custom"> |
|
<template slot-scope="scope">{{ questionTypes.find(e => e.id === scope.row.questionType) ? |
|
questionTypes.find(e => e.id === scope.row.questionType).name : '' }}</template> |
|
</el-table-column> |
|
<el-table-column prop="professionalName" label="专业" align="center" min-width="100" |
|
show-overflow-tooltip></el-table-column> |
|
<el-table-column prop="knowledgePointName" label="知识点" align="center" min-width="100" |
|
show-overflow-tooltip></el-table-column> |
|
<el-table-column prop="givenYear" label="年份" align="center" width="70" sortable="custom"></el-table-column> |
|
<el-table-column prop="difficulty" label="难度" align="center" width="70" sortable="custom"> |
|
<template slot-scope="scope">{{ difficults.find(e => e.id === scope.row.difficulty) ? difficults.find(e => |
|
e.id === scope.row.difficulty).name : '' }}</template> |
|
</el-table-column> |
|
<el-table-column prop="correctRate" label="正确率" align="center" width="80" sortable="custom"></el-table-column> |
|
<el-table-column prop="updateTime" label="更新时间" align="center" width="140" |
|
sortable="custom"></el-table-column> |
|
<el-table-column prop="lastEditor" label="最近编辑人" align="center" width="90"></el-table-column> |
|
<el-table-column prop="referenceCount" label="已引用试卷(套)" align="center" width="130" |
|
sortable="custom"></el-table-column> |
|
<el-table-column label="操作" align="center" width="300" fixed="right"> |
|
<template slot-scope="scope"> |
|
<el-button type="text" @click="toDetail(scope.row, 1)">复制</el-button> |
|
<el-button type="text" @click="toDetail(scope.row, 2)">查看</el-button> |
|
<el-button type="text" @click="toDetail(scope.row, 3)">编辑</el-button> |
|
<el-button type="text" :disabled="!isKnowLedge" @click="remove(scope.row)">移除</el-button> |
|
<el-button type="text" @click="del([scope.row])">删除</el-button> |
|
<el-switch v-model="scope.row.status" :active-value="1" :inactive-value="0" style="margin: 0 10px 0 5px" |
|
:active-text="scope.row.status ? '启用' : '禁用'" |
|
@change="switchOff($event, scope.row, scope.$index)"></el-switch> |
|
</template> |
|
</el-table-column> |
|
</el-table> |
|
<div class="pagination"> |
|
<el-pagination background @current-change="currentChange" :current-page="page" |
|
layout="total, prev, pager, next" :total="total"></el-pagination> |
|
</div> |
|
</div> |
|
</div> |
|
|
|
<el-dialog title="提示" :visible.sync="delVisible" width="400px" :close-on-click-modal="false" custom-class="del-dia"> |
|
<div class="del-wrap"> |
|
<div class="icon el-icon-warning"></div> |
|
<div class="flex-1"> |
|
<p>该试题已被以下试卷引用,确定删除?</p> |
|
<ul class="papers"> |
|
<li v-for="(item, i) in curUsePapers" :key="i"> |
|
<p>{{ item.name }}</p> |
|
<i class="el-icon-edit m-l-10" @click="toPaper(item)"></i> |
|
</li> |
|
</ul> |
|
</div> |
|
</div> |
|
<span slot="footer" class="dialog-footer"> |
|
<el-button @click="delVisible = false">取消</el-button> |
|
<el-button type="primary" @click="delSubmit">确定</el-button> |
|
</span> |
|
</el-dialog> |
|
|
|
<BatchImport :visible.sync="importVisible" :params.sync="$route.query" /> |
|
|
|
<Knowledge :visible.sync="kpVisible" @closed="getType" /> |
|
</div> |
|
</template> |
|
|
|
<script> |
|
import Breadcrumb from '@/components/breadcrumb' |
|
import BatchImport from './batchImport' |
|
import Knowledge from '@/pages/knowledge' |
|
import Util from '@/libs/util' |
|
import Setting from '@/setting' |
|
import Const from '@/const/ques' |
|
import Qs from 'qs' |
|
export default { |
|
components: { Breadcrumb, BatchImport, Knowledge }, |
|
data () { |
|
return { |
|
crumbs: [], |
|
questionBankId: this.$route.query.questionBankId, |
|
questionBankName: this.$route.query.questionBankName, |
|
questionBankCategory: this.$route.query.questionBankCategory, |
|
difficults: Const.difficults, |
|
questionTypes: Const.questionTypes, |
|
loading: false, |
|
isNotJoin: 0, |
|
createSource: 1, |
|
keyword: '', |
|
type: 1, |
|
types: [], |
|
professionals: [], |
|
status: [ |
|
{ |
|
id: 1, |
|
name: '启用' |
|
}, |
|
{ |
|
id: 0, |
|
name: '禁用' |
|
}, |
|
], |
|
knowledges: [], |
|
givenYears: '', |
|
knowledgePointIds: [], |
|
filter: { |
|
questionTypes: [], |
|
correctRateEnd: '', |
|
correctRateStart: '', |
|
difficultys: [], |
|
specialtyIds: [], |
|
status: '', |
|
keyword: '', |
|
questionTypeSort: '', |
|
givenYearSort: '', |
|
difficultySort: '', |
|
correctRateSort: '', |
|
updateTimeSort: '', |
|
referenceCountSort: '', |
|
}, |
|
list: [], |
|
page: +this.$route.query.page || 1, |
|
pageSize: 10, |
|
total: 0, |
|
multipleSelection: [], |
|
|
|
delVisible: false, |
|
curUsePapers: [], |
|
curRow: {}, |
|
|
|
importVisible: false, |
|
|
|
kpVisible: false, |
|
}; |
|
}, |
|
computed: { |
|
// 当前选择的是否是知识点 |
|
isKnowLedge () { |
|
if (this.isNotJoin === '') { |
|
let node = this.$refs.typeTree |
|
return node ? node.getCurrentNode().type === 1 : false |
|
} |
|
return false |
|
}, |
|
}, |
|
watch: { |
|
keyword: function () { |
|
clearTimeout(this.searchTimer) |
|
this.searchTimer = setTimeout(this.getType, 500) |
|
}, |
|
'filter.keyword': function () { |
|
clearTimeout(this.searchTimer) |
|
this.searchTimer = setTimeout(this.initData, 500) |
|
}, |
|
'filter.correctRateStart': function () { |
|
clearTimeout(this.searchTimer) |
|
this.searchTimer = setTimeout(this.initData, 500) |
|
}, |
|
'filter.correctRateEnd': function () { |
|
clearTimeout(this.searchTimer) |
|
this.searchTimer = setTimeout(this.initData, 500) |
|
}, |
|
}, |
|
mounted () { |
|
const { referrer1 } = this.$store.state.user |
|
this.crumbs = [ |
|
{ |
|
name: this.questionBankName || '题库管理', |
|
route: referrer1 || '/quesBank' |
|
}, |
|
{ |
|
name: '试题管理' |
|
}, |
|
] |
|
|
|
const { query } = this.$route |
|
if (query.page) { |
|
const { questionTypes, correctRateEnd, correctRateStart, difficultys, specialtyIds, status, keyword, questionTypeSort, givenYearSort, difficultySort, correctRateSort, updateTimeSort, referenceCountSort, givenYears, knowledgePointIds, questionBankId, questionBankName, questionBankCategory } = query |
|
this.filter = { |
|
questionTypes: questionTypes ? questionTypes.split(',') : [], |
|
correctRateEnd: correctRateEnd || '', |
|
correctRateStart: correctRateStart || '', |
|
difficultys: difficultys ? difficultys.split(',') : [], |
|
specialtyIds: specialtyIds ? specialtyIds.split(',').map(e => +e) : [], |
|
status: status ? +status : '', |
|
keyword: keyword || '', |
|
questionTypeSort: questionTypeSort || '', |
|
givenYearSort: givenYearSort || '', |
|
difficultySort: difficultySort || '', |
|
correctRateSort: correctRateSort || '', |
|
updateTimeSort: updateTimeSort || '', |
|
referenceCountSort: referenceCountSort || '', |
|
} |
|
this.givenYears = givenYears || '' |
|
this.knowledgePointIds = knowledgePointIds ? JSON.parse(knowledgePointIds) : [] |
|
|
|
this.$router.push(`/ques?questionBankId=${questionBankId}&questionBankName=${questionBankName}&questionBankCategory=${questionBankCategory}`).catch(() => { }) |
|
} |
|
|
|
this.getType() |
|
this.getProfessional() |
|
this.getKnowledge() |
|
}, |
|
methods: { |
|
// 获取知识点 |
|
async getType () { |
|
try { |
|
this.loading = true |
|
const { data } = await this.$post(this.api.TreeStructure, { |
|
createSource: 1, |
|
questionBankId: this.questionBankId, |
|
keyword: this.keyword, |
|
}) |
|
this.types = data |
|
this.getList() |
|
} finally { |
|
this.loading = false |
|
} |
|
}, |
|
// 前往知识点 |
|
toSet () { |
|
this.kpVisible = true |
|
}, |
|
// 缓存当前页面和参数,详情页返回到列表的时候直接取该url |
|
setReferrer () { |
|
const { filter } = this |
|
this.$store.commit('user/setReferrer', { |
|
i: 2, |
|
url: `${this.$route.path}?${Qs.stringify(filter)}&${Qs.stringify(this.$route.query)}&questionTypes=${filter.questionTypes}&difficultys=${filter.difficultys}&specialtyIds=${filter.specialtyIds}&givenYears=${this.givenYears}&knowledgePointIds=${JSON.stringify(this.knowledgePointIds)}&page=${this.page}` |
|
}) |
|
}, |
|
// 类型回调 |
|
typeChange () { |
|
this.$refs.typeTree.setCurrentKey(null) |
|
this.initData() |
|
}, |
|
// 点击树节点查询列表数据 |
|
handleNodeClick () { |
|
this.isNotJoin = '' |
|
this.initData() |
|
this.$refs.table.clearSelection() |
|
}, |
|
// 处理树形 |
|
handleType (list) { |
|
list.map(e => { |
|
if (e.children && e.children.length) { |
|
this.handleType(e.children) |
|
} else { |
|
delete e.children |
|
} |
|
}) |
|
}, |
|
// 获取知识点 |
|
async getKnowledge () { |
|
try { |
|
const { data } = await this.$post(this.api.TreeStructure, { |
|
createSource: 1, |
|
questionBankId: this.questionBankId, |
|
keyword: '' |
|
}) |
|
this.handleType(data) |
|
this.knowledges = data |
|
} catch (e) { } |
|
}, |
|
// 获取所属专业 |
|
async getProfessional () { |
|
try { |
|
const res = await this.$post(this.api.queryProfessional, { |
|
pageNum: 1, |
|
pageSize: 1000, |
|
}) |
|
this.professionals = res.pageList.records |
|
} catch (e) { } |
|
}, |
|
// 试题列表 |
|
async getList () { |
|
let type = this.$refs.typeTree.getCurrentKey() |
|
const k = this.knowledgePointIds |
|
if (k.length) { // 右边筛选里的知识点 |
|
type = k.map(e => { |
|
return e[e.length - 1] |
|
}) |
|
} else if (type) { |
|
type = [type] |
|
} |
|
const { message } = await this.$post(this.api.listQuestion, { |
|
...this.filter, |
|
givenYears: this.givenYears ? [this.givenYears] : [], |
|
isNotJoin: this.isNotJoin || '', |
|
pageNum: this.page, |
|
pageSize: this.pageSize, |
|
questionBankId: this.questionBankId, |
|
knowledgePointIds: type || [], |
|
}) |
|
this.list = Util.removeTag(message.records) |
|
this.total = message.total |
|
}, |
|
// 切换页码 |
|
currentChange (val) { |
|
this.page = val |
|
this.getList() |
|
}, |
|
handleSelectionChange (val) { |
|
this.multipleSelection = val |
|
}, |
|
initData () { |
|
this.$refs.table.clearSelection() |
|
this.page = 1 |
|
this.getList() |
|
}, |
|
getSort (o) { |
|
return o ? (o === 'ascending' ? 'asc' : 'desc') : '' |
|
}, |
|
// 排序回调 |
|
sortChange (column) { |
|
const { prop: p, order: o } = column |
|
const { filter } = this |
|
filter.questionTypeSort = p === 'questionType' ? this.getSort(o) : '' |
|
filter.givenYearSort = p === 'givenYear' ? this.getSort(o) : '' |
|
filter.difficultySort = p === 'difficulty' ? this.getSort(o) : '' |
|
filter.correctRateSort = p === 'correctRate' ? this.getSort(o) : '' |
|
filter.updateTimeSort = p === 'updateTime' ? this.getSort(o) : '' |
|
filter.referenceCountSort = p === 'referenceCount' ? this.getSort(o) : '' |
|
this.getList() |
|
}, |
|
|
|
// 添加 |
|
add () { |
|
this.setReferrer() |
|
const knowledgeCheck = this.$refs.typeTree.getCurrentNode() |
|
this.$router.push({ |
|
path: 'detail', |
|
query: { |
|
...this.$route.query, |
|
path: knowledgeCheck && knowledgeCheck.type ? knowledgeCheck.path : '' |
|
} |
|
}) |
|
}, |
|
// 编辑/查看 type: 1复制,2查看,3编辑 |
|
toDetail (row, type) { |
|
this.setReferrer() |
|
this.$router.push({ |
|
path: 'detail', |
|
query: { |
|
...this.$route.query, |
|
questionId: row.questionId, |
|
version: row.version, |
|
detailType: type |
|
} |
|
}) |
|
}, |
|
// 移除 |
|
async remove (row) { |
|
try { |
|
await this.$confirm(`确认要移除吗?`, '提示', { |
|
confirmButtonText: '确定', |
|
cancelButtonText: '取消', |
|
type: 'warning', |
|
closeOnClickModal: false, |
|
}) |
|
await this.$post(this.api.removeQuestionKnowledge, { |
|
knowledgePointId: this.$refs.typeTree.getCurrentKey() || '', |
|
questionVersionIds: [row.questionVersionId] |
|
}) |
|
Util.successMsg('移除成功') |
|
this.getList() |
|
} catch (e) { } |
|
}, |
|
// 删除 |
|
async del (rows) { |
|
const ids = rows.map(e => e.questionId) |
|
// 先查询是否有试卷引用了该试题 |
|
const { list } = await this.$post(this.api.checkQuestionIsUse, ids) |
|
if (list.length) { |
|
this.curRow = rows |
|
this.curUsePapers = list |
|
this.delVisible = true |
|
} else { |
|
try { |
|
await this.$confirm(`确认要删除吗?`, '提示', { |
|
confirmButtonText: '确定', |
|
cancelButtonText: '取消', |
|
type: 'warning', |
|
closeOnClickModal: false, |
|
}) |
|
await this.$post(this.api.delQuestion, ids) |
|
Util.successMsg('删除成功') |
|
this.getList() |
|
this.multipleSelection = [] |
|
this.$refs.table.clearSelection() |
|
} catch (e) { } |
|
} |
|
}, |
|
// 批量删除 |
|
async batchDel () { |
|
const list = this.multipleSelection |
|
list.length ? this.del(list) : Util.warningMsg('请选择数据') |
|
}, |
|
// 删除提交 |
|
async delSubmit () { |
|
await this.$post(this.api.delQuestion, this.curRow.map(e => e.questionId)) |
|
Util.successMsg('删除成功') |
|
this.multipleSelection = [] |
|
this.$refs.table.clearSelection() |
|
this.delVisible = false |
|
this.getList() |
|
}, |
|
// 删除提示里的跳转试卷 |
|
toPaper (row) { |
|
this.$router.push(`/testPaper/detail?paperId=${row.paperId}&libraryId=${row.libraryId}&classificationId=${row.classificationId}`) |
|
}, |
|
// 批量移除 |
|
async batchRemove () { |
|
const list = this.multipleSelection |
|
if (list.length) { |
|
try { |
|
await this.$confirm(`确认要移除吗?`, '提示', { |
|
confirmButtonText: '确定', |
|
cancelButtonText: '取消', |
|
type: 'warning', |
|
closeOnClickModal: false, |
|
}) |
|
await this.$post(this.api.removeQuestionKnowledge, { |
|
knowledgePointId: this.$refs.typeTree.getCurrentKey() || '', |
|
questionVersionIds: list.map(e => e.questionVersionId) |
|
}) |
|
Util.successMsg('移除成功') |
|
this.getList() |
|
this.multipleSelection = [] |
|
this.$refs.table.clearSelection() |
|
} catch (e) { } |
|
} else { |
|
Util.warningMsg('请选择数据') |
|
} |
|
}, |
|
// 禁启用 |
|
async switchOff (val, row) { |
|
try { |
|
row.status = val ? 0 : 1 // 调接口前先还原成点击前的状态,因为有几种原因会禁用启用失败 |
|
if (!val && row.referenceCount) { |
|
await this.$confirm(`<p>确认要禁用【${row.stem}】吗?</p><p style="color: #f56c6c;">禁用后已引用该试题的试卷将会变成草稿</p>`, '提示', { |
|
confirmButtonText: '确定', |
|
cancelButtonText: '取消', |
|
type: 'warning', |
|
closeOnClickModal: false, |
|
dangerouslyUseHTMLString: true, |
|
}) |
|
} |
|
await this.$post(`${this.api.disableOrEnableQuestion}?questionId=${row.questionId}&status=${val}`) |
|
row.status = val |
|
Util.successMsg(val ? '启用成功' : '禁用成功') |
|
this.getList() |
|
} catch (e) { |
|
row.status = val ? 0 : 1 |
|
} |
|
}, |
|
|
|
// 批量导入 |
|
batchImport () { |
|
this.importVisible = true |
|
}, |
|
} |
|
}; |
|
</script> |
|
|
|
<style lang="scss" scoped> |
|
.wrap { |
|
display: flex; |
|
|
|
.side { |
|
width: 300px; |
|
padding: 24px 10px 24px 0; |
|
margin-right: 24px; |
|
border-right: 1px solid rgba(0, 0, 0, 0.06); |
|
} |
|
|
|
.right { |
|
width: calc(100% - 324px); |
|
padding: 24px 0; |
|
} |
|
} |
|
|
|
.tool { |
|
margin-bottom: 0; |
|
|
|
.filter { |
|
flex-wrap: wrap; |
|
justify-content: space-between; |
|
|
|
li { |
|
margin-bottom: 15px; |
|
} |
|
} |
|
} |
|
|
|
|
|
|
|
.papers { |
|
margin-top: 10px; |
|
|
|
li { |
|
display: flex; |
|
justify-content: space-between; |
|
align-items: center; |
|
margin-bottom: 10px; |
|
} |
|
|
|
i { |
|
font-size: 14px; |
|
color: #7a7a7a; |
|
cursor: pointer; |
|
} |
|
} |
|
</style> |