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.

361 lines
11 KiB

4 years ago
<template>
<div v-show="loaded">
2 years ago
<div class="header"
:class="{hh: $config.isHh}">
<img v-if="$config.isHh"
src="@/assets/images/logo-hh.png"
alt=""
class="logo">
<p v-else>{{curriculumName}}</p>
2 years ago
<el-button class="back btn"
type="primary"
@click="back">退出实验</el-button>
4 years ago
</div>
3 years ago
<div class="top">
<div class="language">
2 years ago
<p v-if="$config.isHh"
style="font-size: 18px">{{$config.title}}</p>
3 years ago
<p>编程语言</p>
2 years ago
<el-select v-model="language"
@change="languageChange">
<el-option v-for="(item, i) in languages"
:key="i"
:label="item.name"
:value="item.id"></el-option>
</el-select>
1 year ago
<el-button v-if="language"
class="open-vs"
type="primary"
size="small"
@click="openVscode">打开Vscode新窗口</el-button>
3 years ago
</div>
<div class="inline-center">
2 years ago
<el-tooltip class="item"
effect="dark"
content="支持导入其它数据用于实验,总上传的文件大小不可以超过10M"
placement="bottom">
<i class="info el-icon-warning"
style="margin-right: 10px"></i>
</el-tooltip>
2 years ago
<el-button type="primary"
size="small"
icon="el-icon-document"
@click="toData">我的数据</el-button>
</div>
4 years ago
</div>
<div class="tab">
2 years ago
<el-tabs v-model="curTab"
type="card"
@tab-click="judChange">
<el-tab-pane v-for="(item, i) in workbench"
:key="item.judgmentId"
:label="item.name"
:value="item.judgmentId">
2 years ago
<codemirror v-if="!language"
:ref="'code' + i"
2 years ago
:key="codeKey"
:projectId.sync="projectId"
:systemId.sync="systemId"
:code.sync="item.code"
:judgmentId="item.judgmentId"
:codeId.sync="item.codeId"
:answer.sync="item.answer"
:retResult.sync="item.retResult"
:modelIsShow.sync="modelIsShow"
@cache="leavePage"></codemirror>
2 years ago
<iframe v-else
class="vscode"
:src="Config.vscodeUrl"
2 years ago
frameborder="0"
width="100%"></iframe>
4 years ago
</el-tab-pane>
</el-tabs>
</div>
<div class="menu">
2 years ago
<testPanel @tell="setPoints"
@recoveryCode="recoveryCode"
ref="mainindex"
:workbench.sync="workbench"></testPanel>
4 years ago
</div>
</div>
</template>
<script>
import testPanel from "../components/TestPanel";
import codemirror from "../components/codemirror";
import Cookie from 'js-cookie'
import { Loading } from 'element-ui'
2 years ago
import Config from '@/config'
4 years ago
export default {
2 years ago
data () {
4 years ago
return {
2 years ago
Config,
loaded: false, // 页面是否加载完的标识,页面默认隐藏,一进来先显示加载条,接口加载完后再显示页面,不然一开始会一闪而过没有样式的页面
loadIns: null, // loading实例
competitionId: Cookie.get('admin-competitionId'),
fromManager: Cookie.get('admin-fromManager'), // 是否是从教师端进入
courseId: Cookie.get('admin-courseId'), // 课程id
curriculumName: Cookie.get('admin-curriculumName') ? decodeURIComponent(Cookie.get('admin-curriculumName')) : 'python', // 课程名称
assessmentId: Cookie.get('admin-assessmentId'), // 考核id
2 years ago
mallId: Cookie.get('admin-mallId'),
2 years ago
language: +Cookie.get('admin-language') || 0, // 编程语言
projectId: '',
systemId: '',
modelIsShow: false, // 导入模型按钮是否显示
3 years ago
projectPermissions: 0, // 项目权限(0、练习 1、考核 2、竞赛)
2 years ago
languages: [
{
id: 0,
name: 'Python'
},
{
id: 1,
name: 'Vscode'
},
],
curTab: '', // 选中后绑定的对象
3 years ago
workbench: [], // 判分点切换列表
codeKey: 1 // 编辑器索引
4 years ago
};
},
components: {
codemirror,
testPanel
},
2 years ago
mounted () {
1 year ago
document.onkeydown = function (event) {
1 year ago
var e = event || window.event || arguments.callee.caller.arguments[0];
if (e && e.keyCode == 123) {
e.returnValue = false;
return (false);
}
}
this.loadIns = Loading.service({
background: 'rgba(255, 255, 255, .1)'
})
4 years ago
this.autoLogout()
// 绑定页面离开监听
3 years ago
window.onbeforeunload = () => {
this.leavePage()
4 years ago
}
},
2 years ago
beforeRouteLeave (to, from, next) {
next(this.leavePage())
},
4 years ago
methods: {
3 years ago
// 页面离开的时候缓存未提交的代码(只有未提交状态下才会调用该接口)
2 years ago
leavePage () {
3 years ago
const list = this.workbench
// 未提交,且有判分点,才需要缓存
if (!this.$refs.mainindex.isSubmit && list.length) {
const cache = {
projectId: Number(this.projectId),
judgmentIdList: this.workbench.map(e => e.judgmentId)
4 years ago
}
// 如果有输入代码,则要入库,否则,只保存到本地
if (list.some(e => e.code)) {
list.map(e => {
if (e.code) {
let data = {
1 year ago
competitionId: this.competitionId,
assessmentId: this.assessmentId,
code: e.code,
bcId: e.judgmentId,
cid: this.courseId,
projectId: this.projectId
}
2 years ago
this.$post(this.api.saveCache, data).then(res => { }).catch(e => { })
}
})
} else {
cache.empty = true // 没有输入代码的,加个标识,回到页面的时候不需要调接口了,只需要恢复项目和判分点即可
}
localStorage.setItem('codeCache', JSON.stringify(cache)) // 在本地缓存里保存项目id和判分点id,下次进来通过这两个id去查找缓存代码。之所以不在前端缓存代码是预防更换了浏览器或电脑后依然可以找回缓存代码,只不过更换了客户端后本地缓存就消失了,这种情况就做一个向下处理,就切换项目的时候去恢复代码
4 years ago
}
},
// 获取导入模型按钮展示状态
2 years ago
getModelStatus (systemId) {
2 years ago
this.$post(`${this.api.displayListOrNotByStudent}?systemId=${systemId}`).then(res => {
this.modelIsShow = res.studentSideShowsTheStatus == 'true' ? true : false
2 years ago
}).catch(res => { })
2 years ago
},
// 编程语言选择回调
languageChange (id) {
2 years ago
Cookie.set('admin-language', id)
this.$refs.mainindex.reportVisible = false
},
1 year ago
// 打开Vscode新窗口
openVscode () {
window.open(Config.vscodeUrl)
},
3 years ago
// 跳转我的数据
2 years ago
toData () {
3 years ago
this.$router.push('/data')
},
// 长时间未点击页面,就自动退出页面
2 years ago
autoLogout () {
4 years ago
let lastTime = new Date().getTime()
let logout = false
3 years ago
// 页面点击后赋值当前时间
4 years ago
document.onmousedown = () => {
lastTime = new Date().getTime()
}
// 每秒钟判断一次,如果当前时间距离上次点击页面的试卷超过了设置的时间,就退出登录
4 years ago
setInterval(() => {
2 years ago
if ((new Date().getTime() - lastTime) > this.$config.autoLogoutTime) {
logout || this.$message.error('用户登录过期,请重新登录')
logout = true
setTimeout(this.back, 1500)
4 years ago
}
2 years ago
}, 1000)
4 years ago
},
// 判分规则切换
2 years ago
judChange () {
this.$nextTick(_ => {
const code = this.$refs['code' + this.curTab][0].$refs.codemirror.codemirror // 获取codemirror实例
code.focus() // 编辑器聚焦
code.setCursor(code.lineCount(), 0) // 焦点移动至最后面
})
},
// 重置编辑器
2 years ago
recoveryCode (curTab = '0') {
this.curTab = curTab
this.codeKey++
4 years ago
},
3 years ago
// 退出实验
2 years ago
back () {
let href = this.$config.isDev ?
3 years ago
`http://${location.hostname}:8082/#/` :
`${location.origin}${this.fromManager ?
'/admin' :
2 years ago
(!this.$config.isTest) ?
'' :
'/student'}/#/`
3 years ago
// 考核
if (this.assessmentId) {
href += `ass/list`
} else if (this.competitionId) {
// 竞赛
href += `match/list`
3 years ago
} else {
// 练习
2 years ago
href += `station/preview?courseId=${this.courseId}&curriculumName=${this.curriculumName}&mallId=${this.mallId}`
3 years ago
}
location.href = href
4 years ago
},
3 years ago
// 赋值项目id、项目列表
2 years ago
setPoints (projectId, systemId, workbench) {
this.projectId = projectId
this.systemId = systemId
this.workbench = workbench
4 years ago
},
}
};
</script>
<style lang="scss" scoped>
[v-cloak] {
display: none;
}
.header {
display: flex;
justify-content: space-between;
height: 58px;
line-height: 58px;
background-color: #f8f8f8;
2 years ago
&.hh {
padding: 10px 0;
line-height: normal;
}
p {
margin-left: 18px;
font-size: 20px;
color: rgba(51, 51, 51, 1);
}
2 years ago
.logo {
width: 200px;
margin-left: 10px;
}
.back {
padding: 23px 50px;
border: none;
border-radius: 0;
}
}
3 years ago
/deep/.top {
display: flex;
3 years ago
justify-content: space-between;
align-items: center;
height: 60px;
3 years ago
padding: 0 15px;
line-height: 60px;
3 years ago
.language {
display: inline-flex;
1 year ago
align-items: center;
3 years ago
}
p {
font-size: 16px;
margin-right: 15px;
}
.el-input {
3 years ago
width: 200px;
}
2 years ago
.el-input .el-input__inner {
border-radius: 30px;
}
1 year ago
.open-vs {
margin-left: 20px;
}
}
2 years ago
.vscode {
height: calc(100vh - 186px);
}
4 years ago
.menu {
position: relative;
z-index: 1000;
}
::v-deep .el-dialog--center {
width: 400px;
}
::v-deep .el-dialog__headerbtn .el-icon-close:before {
padding: 3px;
border-radius: 50%;
}
/deep/.tab {
4 years ago
height: 50px;
line-height: 50px;
.el-tabs__header {
margin-bottom: 0;
}
.el-tabs__item.is-active {
color: #333;
}
.el-icon-circle-plus-outline:before {
font-size: 16px;
}
.el-tabs--card > .el-tabs__header {
border-bottom: none;
4 years ago
}
.el-tabs--card > .el-tabs__header .el-tabs__nav {
border: none;
}
.el-tabs--card > .el-tabs__header .el-tabs__item {
border-left: none;
border-bottom: none;
4 years ago
}
4 years ago
}
.info {
color: #bfbfbf;
cursor: pointer;
&:hover {
2 years ago
opacity: 0.9;
}
}
4 years ago
</style>