diff --git a/src/api/index.js b/src/api/index.js index 3313d89..403493a 100644 --- a/src/api/index.js +++ b/src/api/index.js @@ -143,6 +143,8 @@ export default { stageTeamScoreDetails: `competition/competition/rank/stageTeamScoreDetails`, getRedisCacheCompetition: `competition/competition/management/getRedisCache`, getCompetitionStageRankingTime: `competition/competitionReleaseTime/getCompetitionStageRankingTime`, + cCompetitionStageFileSave: `competition/cCompetitionStageFile/save`, + cCompetitionStageFileDel: `competition/cCompetitionStageFile/batchDeletion`, // 创业活动 activityList: `occupationlab/occupationlab/activity/activityList`, diff --git a/src/pages/match/details/index.vue b/src/pages/match/details/index.vue index 6b500d0..38d37dd 100644 --- a/src/pages/match/details/index.vue +++ b/src/pages/match/details/index.vue @@ -616,6 +616,58 @@ @click="memberVisible = false">确定 + + + + + 提交 + 取消 + + @@ -629,6 +681,9 @@ export default { name: 'matchdetail', data () { return { + headers: { + token: util.local.get(Setting.tokenKey) + }, token: util.local.get(Setting.tokenKey), id: +this.$route.query.id, end: '', @@ -733,6 +788,10 @@ export default { peopleSignupForm: { registrationInvitationCode: '' }, + submiting: false, + + stageVisible: false, + filesResult: {} }; }, computed: { @@ -815,99 +874,105 @@ export default { let signUpEndTime = new Date(this.core.dateCompatible(form.signUpEndTime)) // 报名结束时间 let playStartTime = new Date(this.core.dateCompatible(form.playStartTime)) // 比赛开始时间 let playEndTime = new Date(this.core.dateCompatible(form.playEndTime)) // 比赛结束时间 - this.timer = setInterval(() => { - const now = new Date() - if (now < signUpStartTime) { // 报名没开始 - status = 0 - total = signUpStartTime - now - } else if (now > signUpStartTime && now < signUpEndTime) { // 报名进行中 - // 1已报名,2立即报名(没登录的情况下,直接显示立即报名,登录了则取报名信息,有则已报名,无则立即报名) - status = this.token ? - (form.competitionRegistration ? - 1 : - 2) : - 2 - total = signUpEndTime - now - } else if (now > signUpEndTime && now < playStartTime) { // 报名结束了,但比赛没开始 - status = 3 - total = playStartTime - now - } else if (now > playStartTime && now < playEndTime) { // 比赛进行中 - // 如果是完整比赛 - if (form.releaseType) { - // 进行中的赛事,则遍历每个阶段的开始结束时间,看阶段比赛是否开始 - let curStage = null - const stages = form.competitionStage - if (stages) { - this.playingStages = [] - form.competitionRegistration && stages.forEach(e => { - if (now >= new Date(e.startTime) && now <= new Date(e.endTime) && (e.method !== 2 || this.isLink(e))) this.playingStages.push(e) - }) - let endText = '' - for (const i in stages) { - const e = stages[i] - const startTime = new Date(e.startTime) - const endTime = new Date(e.endTime) - if (now < startTime) { // 阶段比赛未开始,不显示进入比赛按钮 - endText = '阶段开始' - total = startTime - now - break - } else if (now >= startTime && now <= endTime && (e.method !== 2 || this.isLink(e))) { // 阶段比赛进行中,显示进入比赛按钮 - if (e.method !== 2) { - if (form.competitionRegistration) { // 报名了才能进入比赛 - this.statusList[4] = e.count ? '已提交' : '进入' + e.stageName - curStage = e - } else if (!this.token) { - this.statusList[4] = '进入' + e.stageName - curStage = e - } - } else { + // this.timer = setInterval(() => { + const now = new Date() + if (now < signUpStartTime) { // 报名没开始 + status = 0 + total = signUpStartTime - now + } else if (now > signUpStartTime && now < signUpEndTime) { // 报名进行中 + // 1已报名,2立即报名(没登录的情况下,直接显示立即报名,登录了则取报名信息,有则已报名,无则立即报名) + status = this.token ? + (form.competitionRegistration ? + 1 : + 2) : + 2 + total = signUpEndTime - now + } else if (now > signUpEndTime && now < playStartTime) { // 报名结束了,但比赛没开始 + status = 3 + total = playStartTime - now + } else if (now > playStartTime && now < playEndTime) { // 比赛进行中 + // 如果是完整比赛 + if (form.releaseType) { + // 进行中的赛事,则遍历每个阶段的开始结束时间,看阶段比赛是否开始 + let curStage = null + const stages = form.competitionStage + if (stages) { + this.playingStages = [] + form.competitionRegistration && stages.forEach(e => { + if (now >= new Date(e.startTime) && now <= new Date(e.endTime) && (e.method !== 2 || this.offlineCanEntry(e))) this.playingStages.push(e) + }) + let endText = '' + for (const i in stages) { + const e = stages[i] + const startTime = new Date(e.startTime) + const endTime = new Date(e.endTime) + console.log("🚀 ~ file: index.vue:901 ~ //this.timer=setInterval ~ form.competitionRegistration:", stages, startTime, endTime, now >= startTime && now <= endTime) + if (now < startTime) { // 阶段比赛未开始,不显示进入比赛按钮 + endText = '阶段开始' + total = startTime - now + break + } else if (now >= startTime && now <= endTime) { // 阶段比赛进行中,显示进入比赛按钮 + // 非线下赛事 + if (e.method !== 2) { + if (form.competitionRegistration) { // 报名了才能进入比赛 + this.statusList[4] = e.count ? '已提交' : '进入' + e.stageName + curStage = e + } else if (!this.token) { this.statusList[4] = '进入' + e.stageName curStage = e } - endText = '阶段结束' - total = endTime - now - break - } else if (stages[i + 1] && now > endTime && now < new Date(stages[i + 1].startTime)) { // 过了该阶段的结束时间,但是没到下个阶段的开始时间,不显示进入比赛按钮 - endText = '阶段开始' - total = new Date(stages[i + 1].startTime) - now - break - } else if (i === stages.length - 1) { // 当前时间在比赛开始结束时间之间,并且是最后一个阶段结束时间之后 - this.$set(form, 'stageName', '') - endText = '竞赛结束' - total = playEndTime - now - break + } else if (this.offlineCanEntry(e)) { // 线下(输入了系统链接或者上传文件选择了是,才需要显示进入按钮) + // 当系统链接为空,且上传文件为否时,无需展示入口 + // 当系统链接不为空,且上传文件为否时,点击入口,直接跳转到链接页面,无需弹窗 + // 当上传文件为是时,点击入口需弹窗,共两种样式 + this.statusList[4] = '进入' + e.stageName + console.log("🚀 ~ file: index.vue:924 ~ //this.timer=setInterval ~ e:", e, this.playingStages) + curStage = e } + endText = '阶段结束' + total = endTime - now + break + } else if (stages[i + 1] && now > endTime && now < new Date(stages[i + 1].startTime)) { // 过了该阶段的结束时间,但是没到下个阶段的开始时间,不显示进入比赛按钮 + endText = '阶段开始' + total = new Date(stages[i + 1].startTime) - now + break + } else if (i === stages.length - 1) { // 当前时间在比赛开始结束时间之间,并且是最后一个阶段结束时间之后 + this.$set(form, 'stageName', '') + endText = '竞赛结束' + total = playEndTime - now + break } - this.endList[4] = endText } - this.curStage = curStage - } else { // 仅发布信息 - total = playEndTime - now + this.endList[4] = endText } - status = 4 - } else if (now > playEndTime) { // 比赛结束 - status = 5 - } - this.status = status - total = total / 1000 - --total - if (total > 86400) { // 超过一天则显示天数 - // clearInterval(this.timer) - this.end = Math.floor(total / 86400) + '天' - } else if (total > 0) { // 一天之内,显示时分秒 - let hours = Math.floor(total / (60 * 60)) - let minutes = Math.floor(total % (60 * 60) / 60) - let seconds = Math.floor(total % (60 * 60) % 60) - time = `${this.core.formateTime(hours)}:${this.core.formateTime(minutes)}:${this.core.formateTime(seconds)}` - if (total > 0) this.end = time - } else if (this.status === 5) { // 竞赛结束,清除定时器 - clearInterval(this.timer) + this.curStage = curStage + } else { // 仅发布信息 + total = playEndTime - now } - }, 1000) + status = 4 + } else if (now > playEndTime) { // 比赛结束 + status = 5 + } + this.status = status + total = total / 1000 + --total + if (total > 86400) { // 超过一天则显示天数 + // clearInterval(this.timer) + this.end = Math.floor(total / 86400) + '天' + } else if (total > 0) { // 一天之内,显示时分秒 + let hours = Math.floor(total / (60 * 60)) + let minutes = Math.floor(total % (60 * 60) / 60) + let seconds = Math.floor(total % (60 * 60) % 60) + time = `${this.core.formateTime(hours)}:${this.core.formateTime(minutes)}:${this.core.formateTime(seconds)}` + if (total > 0) this.end = time + } else if (this.status === 5) { // 竞赛结束,清除定时器 + clearInterval(this.timer) + } + // }, 1000) }, - isLink (stage) { - console.log("🚀 ~ file: index.vue:904 ~ isLink ~ stage:", stage) - return stage.method === 2 && /[a-zA-Z\d]+\.[a-zA-Z]+/.test(stage.offlineAddress) + // 该阶段是否符合线下能进入比赛的条件 + offlineCanEntry (stage) { + return stage.method === 2 && stage.competitionStageContentSetting && !!(stage.competitionStageContentSetting.systemLink || stage.competitionStageContentSetting.whetherToUploadFiles) }, // 获取竞赛信息 getInfo () { @@ -1066,7 +1131,7 @@ export default { }, // 定时调获取排名接口 intervalRank () { - this.rankTimer = setInterval(this.getRank, 1000) + // this.rankTimer = setInterval(this.getRank, 1000) }, // 删除指导老师 @@ -1348,6 +1413,56 @@ export default { } }, + + // 附件上传前 + beforeUpload (file) { + const oversize = file.size / 1024 / 1024 < 1000 + if (!oversize) util.warningMsg('请上传小于1GB的附件!') + if (oversize) { + return true + } else { + return false + } + }, + uploadError (err, file, fileList) { + this.$message({ + message: "上传出错,请重试!", + type: "error", + center: true + }) + }, + beforeRemove (file, fileList) { + return this.$confirm(`确定移除 ${file.name}?`); + }, + handleExceed (files, fileList) { + util.warningMsg(`当前限制选择 1 个文件,如需更换,请删除上一个文件再重新选择!`); + }, + handleRemove (file, fileList) { }, + uploadSuccess (res) { + this.filesResult = res.filesResult + }, + // 提交阶段内容 + stageSubmit () { + if (this.submiting) return false + this.submiting = true + // 有上传记录,则删除上次的文件 + this.curStage.competitionStageFile && this.$post(this.api.cCompetitionStageFileDel, [this.curStage.competitionStageFile.id]).then(res => { }).catch(res => { }) + this.$post(this.api.cCompetitionStageFileSave, { + competitionId: this.id, + fileFormat: this.filesResult.fileType, + fileName: this.filesResult.originalFileName, + filePath: this.filesResult.fileUrl, + fileSize: this.filesResult.fileSize, + ossFileName: this.filesResult.ossFileName, + stageId: this.curStage.stageId, + teamId: this.form.competitionRegistration.teamId + }).then(res => { + util.successMsg('提交成功!') + this.stageVisible = false + this.submiting = false + }).catch(res => { }) + }, + // 选择要进入的阶段 chooseStage (e) { this.curStage = e @@ -1376,9 +1491,15 @@ export default { if (util.local.get(Setting.tokenKey)) { const { competitionType } = form.completeCompetitionSetup if (status == 4) { // 进入比赛 - // 线下 + // 线下(规则见handleStatus方法) if (this.curStage.method == 2) { - window.open(this.curStage.offlineAddress) + // 直接打开系统链接 + if (!this.curStage.competitionStageContentSetting.whetherToUploadFiles) { + window.open(this.curStage.competitionStageContentSetting.systemLink) + } else { // 显示上传文件弹框 + this.stageVisible = true + console.log("🚀 ~ file: index.vue:1499 ~ signup ~ this.curStage:", this.curStage) + } } else { // 参加过比赛不让参加 if (this.curStage && this.curStage.count) return util.errorMsg('您已经参加过该阶段竞赛!') @@ -1435,7 +1556,7 @@ export default { }, // 线下比赛方式点击进入跳转<比赛地点> toOffline () { - window.open(this.curStage.offlineAddress) + window.open(this.curStage.competitionStageContentSetting.systemLink) }, // 进入python系统 toPython () { diff --git a/src/pages/match/list/index.vue b/src/pages/match/list/index.vue index f7a6dbe..81cfcd9 100644 --- a/src/pages/match/list/index.vue +++ b/src/pages/match/list/index.vue @@ -263,6 +263,58 @@ @click="teamVisible = false">取消 + + + + + 提交 + 取消 + + @@ -276,6 +328,9 @@ export default { name: "match", data () { return { + headers: { + token: util.local.get(Setting.tokenKey) + }, timer: null, token: util.local.get(Setting.tokenKey), statusList: ["待报名", "取消报名", "马上报名", "报名截止", "进入初赛", "已结束"], @@ -384,7 +439,12 @@ export default { peopleSignupForm: { registrationInvitationCode: '' }, - curRow: {} + curRow: {}, + submiting: false, + + stageVisible: false, + filesResult: {}, + curStageItem: {} }; }, computed: { @@ -494,7 +554,7 @@ export default { item.playingStages = [] // 报了名才算作可以进入的阶段 item.whetherToSignUp === 0 && stages.forEach(e => { - if (now >= new Date(e.startTime) && now <= new Date(e.endTime) && (e.method !== 2 || this.isLink(e))) item.playingStages.push(e) + if (now >= new Date(e.startTime) && now <= new Date(e.endTime) && (e.method !== 2 || this.offlineCanEntry(e))) item.playingStages.push(e) }) for (const i in stages) { const e = stages[i] @@ -506,7 +566,7 @@ export default { total = startTime - now break } else if (now >= startTime && now <= endTime) { // 阶段比赛进行中,显示进入比赛按钮 - if (item.whetherToSignUp === 0 && (e.method !== 2 || this.isLink(e))) this.$set(item, 'stageName', (!this.isLink(e) && e.count) ? '已提交' : '进入' + e.stageName) // 报名了并且没参加比赛才能进入比赛 + if (item.whetherToSignUp === 0 && (e.method !== 2 || this.offlineCanEntry(e))) this.$set(item, 'stageName', (!this.offlineCanEntry(e) && e.count) ? '已提交' : '进入' + e.stageName) // 报名了并且没参加比赛才能进入比赛 this.$set(item, 'endText', '阶段结束') curStage = e total = endTime - now @@ -553,9 +613,6 @@ export default { } }) }, - isLink (stage) { - return stage.method === 2 && /[a-zA-Z\d]+\.[a-zA-Z]+/.test(stage.offlineAddress) - }, // 清除定时器 clearTimer () { this.timerList.forEach(n => { @@ -679,6 +736,71 @@ export default { util.successMsg('报名成功!') }).catch(res => { }) }, + + + + + // 该阶段是否符合线下能进入比赛的条件 + offlineCanEntry (stage) { + return stage.method === 2 && stage.competitionStageContentSetting && !!(stage.competitionStageContentSetting.systemLink || stage.competitionStageContentSetting.whetherToUploadFiles) + }, + // 附件上传前 + beforeUpload (file) { + const oversize = file.size / 1024 / 1024 < 1000 + if (!oversize) util.warningMsg('请上传小于1GB的附件!') + if (oversize) { + return true + } else { + return false + } + }, + uploadError (err, file, fileList) { + this.$message({ + message: "上传出错,请重试!", + type: "error", + center: true + }) + }, + beforeRemove (file, fileList) { + return this.$confirm(`确定移除 ${file.name}?`); + }, + handleExceed (files, fileList) { + util.warningMsg(`当前限制选择 1 个文件,如需更换,请删除上一个文件再重新选择!`); + }, + handleRemove (file, fileList) { }, + uploadSuccess (res) { + this.filesResult = res.filesResult + }, + // 提交阶段内容 + stageSubmit () { + if (this.submiting) return false + this.submiting = true + // 有上传记录,则删除上次的文件 + this.curStageItem.competitionStageFile && this.$post(this.api.cCompetitionStageFileDel, [this.curStageItem.competitionStageFile.id]).then(res => { }).catch(res => { }) + this.$post(this.api.cCompetitionStageFileSave, { + competitionId: this.curItem.id, + fileFormat: this.filesResult.fileType, + fileName: this.filesResult.originalFileName, + filePath: this.filesResult.fileUrl, + fileSize: this.filesResult.fileSize, + ossFileName: this.filesResult.ossFileName, + stageId: this.curStageItem.stageId, + teamId: this.curItem.teamId + }).then(res => { + util.successMsg('提交成功!') + this.stageVisible = false + this.submiting = false + }).catch(res => { }) + }, + // 下载附件 + download (item) { + util.downloadFile(item.fileName, item.filePath) + }, + // 线下比赛方式点击进入跳转<比赛地点> + toOffline () { + window.open(this.curStageItem.competitionStageContentSetting.systemLink) + }, + stageClick (e) { }, // 选择要进入的阶段 chooseStage (e, item) { @@ -709,9 +831,16 @@ export default { if (util.local.get(Setting.tokenKey)) { this.curItem = item if (status == 4) { // 进入比赛 - // 线下 + // 线下(规则见handleStatus方法) if (item.curStage.method == 2) { - window.open(item.curStage.offlineAddress) + // 直接打开系统链接 + if (!item.curStage.competitionStageContentSetting.whetherToUploadFiles) { + window.open(item.curStage.competitionStageContentSetting.systemLink) + } else { // 显示上传文件弹框 + this.curStageItem = item.curStage + this.stageVisible = true + console.log("🚀 ~ file: index.vue:830 ~ signup ~ this.curItem:", this.curItem.curStage.competitionStageContentSetting.systemLink) + } } else { // 参加过比赛不让参加 if (item.curStage.count) return util.errorMsg('您已经参加过该阶段竞赛!') diff --git a/src/pages/station/list/index.vue b/src/pages/station/list/index.vue index 9ac9a30..a9f275a 100644 --- a/src/pages/station/list/index.vue +++ b/src/pages/station/list/index.vue @@ -24,10 +24,10 @@