考试平台前端
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.
 
 
 
 
 

489 lines
16 KiB

<template>
<div :class="['page', { inQues }]">
<Breadcrumb v-if="!inQues" style="margin-bottom: 0;" :data="crumbs" />
<div class="wrap">
<div class="side">
<div class="m-b-20">
<el-radio-group v-model="isNotJoin" @change="changeType">
<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="addType">添加</el-button>
</div>
<el-input class="m-b-10" placeholder="请输入知识点分类" prefix-icon="el-icon-search" size="small" clearable
v-model="keyword"></el-input>
<div style="overflow: auto">
<el-tree v-loading="loading" :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 }">
<span class="org-name">{{ data.name }}</span>
<span>
<el-button type="text" icon="el-icon-circle-plus-outline" @click="() => addType(node, data)">
</el-button>
<el-button type="text" icon="el-icon-edit-outline" @click="() => editType(node, data)">
</el-button>
<el-button type="text" icon="el-icon-delete" @click="() => delType(data)">
</el-button>
</span>
</span>
</el-tree>
</div>
</div>
<el-dialog :title="typeForm.id ? '编辑' : '新增' + '知识点分类'" :visible.sync="typeVisible"
:close-on-click-modal="false" append-to-body width="400px">
<el-form v-if="typeVisible" ref="typeForm" :model="typeForm" :rules="typeRules" label-width="130px">
<el-form-item label="知识点分类名称" prop="name">
<el-input v-model.trim="typeForm.name" placeholder="请输入知识点分类" maxlength="20"></el-input>
</el-form-item>
<el-form-item label="设置上一级">
<el-cascader :options="types" v-model="cascaderValue" :props="cascaderProps" clearable
style="width: 100%">
</el-cascader>
</el-form-item>
</el-form>
<span slot="footer" class="dialog-footer">
<el-button @click="closeType">取 消</el-button>
<el-button type="primary" @click="typeSubmit">确 定</el-button>
</span>
</el-dialog>
</div>
<div class="right">
<h6 class="page-name">筛选</h6>
<div class="tool">
<ul class="filter">
<li>
<el-input style="width: 250px;" placeholder="请输入知识点名称" prefix-icon="el-icon-search" v-model="filter.name"
clearable></el-input>
</li>
</ul>
<div>
<el-button type="primary" @click="add">新增知识点</el-button>
<el-button type="primary" @click="delAllSelection">批量删除</el-button>
</div>
</div>
<el-table :data="list" v-loading="listLoading" class="table" ref="table" stripe header-align="center"
@selection-change="handleSelectionChange" row-key="id" @sort-change="sortChange">
<el-table-column type="selection" width="55" align="center" :reserve-selection="true"></el-table-column>
<el-table-column type="index" width="60" label="序号" align="center"></el-table-column>
<el-table-column prop="name" label="知识点名称" align="center" min-width="120"></el-table-column>
<el-table-column prop="knowledgePointsClassification" label="知识点分类" align="center" min-width="120"
show-overflow-tooltip></el-table-column>
<el-table-column prop="relatedQuestionsNum" label="已关联试题" align="center" width="90"></el-table-column>
<el-table-column prop="createTime" label="创建时间" align="center" width="160"
sortable="custom"></el-table-column>
<el-table-column prop="createUserName" label="创建人" align="center" width="100"></el-table-column>
<el-table-column label="操作" align="center" width="100">
<template slot-scope="scope">
<el-button type="text" @click="edit(scope.row)">编辑</el-button>
<!-- <el-button type="text" v-auth="'/system:后台账号:重置密码'" @click="resetPassword(scope.row)">转移</el-button> -->
<el-button type="text" @click="del(scope.row)">删除</el-button>
</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>
<el-dialog :title="!form.id ? '添加知识点' : '编辑知识点'" :visible.sync="knowledgeVisible" width="400px" append-to-body
:close-on-click-modal="false">
<el-form :model="form" :rules="rules" label-width="100px">
<el-form-item prop="parentId" label="知识点分类">
<el-cascader placeholder="请选择题库分类" v-model="form.parentId" :options="types"
:props="{ value: 'id', label: 'name', checkStrictly: true }" clearable></el-cascader>
</el-form-item>
<el-form-item prop="name" label="知识点名称">
<el-input placeholder="请输入知识点名称" v-model="form.name" maxlength="20"></el-input>
</el-form-item>
</el-form>
<span slot="footer" class="dialog-footer">
<el-button @click="knowledgeVisible = false">取 消</el-button>
<el-button type="primary" @click="knowledgeSubmit">确 定</el-button>
</span>
</el-dialog>
</div>
</div>
<el-dialog title="提示" :visible.sync="delVisible" width="400px" :close-on-click-modal="false" append-to-body
custom-class="del-dia">
<div class="del-wrap">
<div class="icon el-icon-warning"></div>
<div>
<p>确认要删除【{{ curName }}】吗?</p>
<p class="tips">删除后,其子级分类与知识点也将被删除。</p>
</div>
</div>
<span slot="footer" class="dialog-footer">
<el-button @click="delVisible = false">取消</el-button>
<el-button type="primary" @click="delTypeSubmit">确定</el-button>
</span>
</el-dialog>
</div>
</template>
<script>
import Util from '@/libs/util'
import _ from 'lodash'
import Breadcrumb from '@/components/breadcrumb'
export default {
components: { Breadcrumb },
data () {
return {
crumbs: [],
questionBankId: this.$route.query.questionBankId,
inQues: false,
createSource: 1,
loading: false,
keyword: '',
types: [],
isNotJoin: 0,
typeVisible: false, // 员工组织架对话框
typeForm: {
id: '',
name: ''
},
cascaderValue: [],
cascaderProps: {
checkStrictly: true,
label: "name",
value: "id"
},
treeVisible: true,
isDetail: false,
name: '',
form: {
name: '',
parentId: []
},
typeRules: {
name: [
{ required: true, message: '请输入知识点分类名称', trigger: 'blur' }
]
},
rules: {
name: [
{ required: true, message: '请输入知识点名称', trigger: 'blur' }
],
},
filter: {
timeOrderBy: '',
name: '',
},
listLoading: false,
list: [],
page: 1,
pageSize: 10,
total: 0,
multipleSelection: [],
knowledgeVisible: false,
submiting: false, // 新增编辑防抖标识
delVisible: false,
deleteQuestions: false,
curName: '',
curId: [],
};
},
watch: {
keyword: function () {
clearTimeout(this.searchTimer)
this.searchTimer = setTimeout(this.getType, 500)
},
'filter.name': function () {
clearTimeout(this.searchTimer)
this.searchTimer = setTimeout(this.getList, 500)
},
},
mounted () {
const { query } = this.$route
this.crumbs = [
{
name: query.questionBankName || '题库管理',
route: '/quesBank'
},
{
name: '试题列表',
route: '/ques',
query: {
questionBankId: query.questionBankId,
questionBankName: query.questionBankName,
questionBankCategory: query.questionBankCategory
}
},
{
name: '知识点列表'
},
]
this.inQues = this.$route.path === '/ques/detail' // 当前页面是否在试题详情
this.getType()
},
methods: {
// 获取分类
async getType () {
try {
this.loading = true
const { data } = await this.$post(this.api.classificationTreeStructure, {
createSource: this.createSource,
keyword: this.keyword,
questionBankId: this.questionBankId,
})
this.handleType(data)
this.types = data
this.getList()
} finally {
this.loading = false
}
},
// 分类筛选回调
changeType () {
this.$refs.typeTree.setCurrentKey(null)
this.initData()
},
// 添加分类
addType (node, data) {
this.typeForm = {
id: '',
parentId: data ? data.id : '',
name: '',
}
this.typeVisible = true
this.cascaderValue = data.path.split('/').map(e => +e)
},
// 编辑分类
editType (node, data) {
this.typeForm = {
id: data.id,
name: data.name
}
this.typeVisible = true
const path = data.path.split('/').map(e => +e)
this.cascaderValue = path.length > 1 ? path.slice(0, path.length - 1) : []
},
// 处理树形
handleType (list) {
list.map(e => {
if (e.children && e.children.length) {
this.handleType(e.children)
} else {
delete e.children
}
})
},
// 删除分类
async delType (row) {
try {
await this.$confirm(`<p>确认要删除【${row.name}】吗?</p><p style="color: #f56c6c;">删除后,其子级分类与知识点都将被删除!</p>`, '提示', {
confirmButtonText: '确定',
cancelButtonText: '取消',
type: 'warning',
closeOnClickModal: false,
dangerouslyUseHTMLString: true,
})
await this.$post(this.api.knowledgeHierarchyDel, [row.id])
Util.successMsg('删除成功')
this.getType()
} catch (e) { }
},
// 删除分类提交
async delTypeSubmit () {
await this.$post(this.api.knowledgeHierarchyDel, this.curId)
Util.successMsg('删除成功')
this.delVisible = false
this.getType()
},
// 分类新增/编辑
typeSubmit () {
this.$refs.typeForm.validate(async (valid) => {
if (valid) {
const form = this.typeForm
const cas = this.cascaderValue
const len = cas.length
if (cas && len) {
this.typeForm.parentId = cas[len - 1]
}
form.createSource = this.createSource
form.questionBankId = this.questionBankId
form.type = 0
await this.$post(this.api.knowledgeHierarchySave, form)
Util.successMsg('保存成功')
this.closeType()
}
});
},
// 点击树节点查询列表数据
handleNodeClick (data) {
this.isNotJoin = ''
this.initData()
},
// 关闭组织新增编辑弹框
closeType () {
this.typeVisible = false
this.cascaderValue = []
this.getType()
},
// 列表
async getList () {
try {
this.listLoading = true
const res = await this.$post(this.api.knowledgeHierarchyList, {
...this.filter,
isNotJoin: this.isNotJoin || '',
pageNum: this.page,
pageSize: this.pageSize,
questionBankId: this.questionBankId,
knowledgePointCategoryId: this.$refs.typeTree.getCurrentKey() || '',
})
this.list = res.message.records
this.total = res.message.total
} finally {
this.listLoading = false
}
},
// 切换页码
currentChange (val) {
this.page = val
this.getList()
},
handleSelectionChange (val) { // 多选
this.multipleSelection = val
},
initData () {
this.$refs.table.clearSelection()
this.page = 1
this.getList()
},
// 排序回调
sortChange (column) {
if (column.prop === 'createTime') this.filter.timeOrderBy = column.order ? column.order === 'ascending' ? 'asc' : 'desc' : ''
this.getList()
},
// 删除
async del (row) {
try {
await this.$confirm(`<p>确认要删除【${row.name}】吗?</p><p style="color: #f56c6c;">删除后,关联此知识点的试题将自动移除此知识点!</p>`, '提示', {
confirmButtonText: '确定',
cancelButtonText: '取消',
type: 'warning',
closeOnClickModal: false,
dangerouslyUseHTMLString: true,
})
await this.$post(this.api.knowledgeHierarchyDel, [row.id])
Util.successMsg('删除成功')
this.getList()
} catch (e) { }
},
// 添加知识点
add () {
const type = this.$refs.typeTree.getCurrentNode()
this.form = {
name: '',
parentId: type ? type.path.split('/').map(e => +e) : [],
}
this.knowledgeVisible = true
},
// 编辑知识点
async edit (row) {
const { data } = await this.$post(`${this.api.knowledgeHierarchyFind}?id=${row.id}`)
this.knowledgeVisible = true
const path = data.path.split('/').map(e => +e)
this.form = {
id: row.id,
name: row.name,
parentId: path.slice(0, path.length - 1),
}
console.log(11, this.form)
},
// 知识点提交
async knowledgeSubmit () {
if (this.submiting) return false
const form = _.cloneDeep(this.form)
if (!form.name) return Util.warningMsg('请输入题库名称')
this.submiting = true
form.parentId = form.parentId.length ? form.parentId[form.parentId.length - 1] : 0
form.createSource = 1
form.questionBankId = this.questionBankId
form.type = 1
try {
await this.$post(this.api.knowledgeHierarchySave, form)
Util.successMsg('保存成功')
this.knowledgeVisible = false
this.submiting = false
this.getList()
} catch (e) {
this.submiting = false
}
},
async delAllSelection () {
const list = this.multipleSelection
if (list.length) {
try {
await this.$confirm(`<p style="margin-bottom: 10px;">确认要删除已选定的${list.length}个知识点吗?</p><p style="color: #f56c6c;">删除后关联这些知识点的试题将自动移除这些知识点</p>`, '提示', {
confirmButtonText: '确定',
cancelButtonText: '取消',
type: 'warning',
closeOnClickModal: false,
dangerouslyUseHTMLString: true,
})
await this.$post(this.api.knowledgeHierarchyDel, list.map(e => e.id))
Util.successMsg('删除成功')
this.multipleSelection = []
this.$refs.table.clearSelection()
this.getList()
} catch (e) { }
} else {
Util.warningMsg('请选择数据')
}
},
}
};
</script>
<style lang="scss" scoped>
.m-b-20 {
margin-bottom: 20px;
}
.org-name {
margin-right: 20px;
}
.w-100 {
width: 100%;
}
.inQues {
padding: 0 25px;
}
.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;
}
}
</style>