< template >
< div class = "wrap index" >
< div class = "banner" : style = "{backgroundImage: 'url(' + (form.carouselUrl || 'https://huoran.oss-cn-shenzhen.aliyuncs.com/20220613/png/1536269450851409920.png') + ')'}" > < / div >
< div class = "center" >
< div class = "center-wrap" >
< breadcrumb ref = "breadcrumb" : data = "'全部赛事/' + form.name" > < / breadcrumb >
< div class = "content" >
< div class = "flex-between" >
< el -tabs v-model ="curType" @tab-click="typeChange" >
< el -tab -pane v-for ="(item, index) in typeList" :key="index" :label="item.name" :name="item.id" > < / el -tab -pane >
< / e l - t a b s >
< div class = "action" >
< p class = "end-text" v-if ="status != 5" >
距离 { { endList [ status ] } } 还有 < em > { { end } } < / em >
< / p >
< a class = "status" : class = "{wait: status == 0 || status == 4,signing: status == 2,signed: status == 1,finish: status == 3 || status == 5}" @click.stop ="signup" > { { statusList [ status ] } } < / a >
< / div >
< / div >
< div class = "info" >
< h6 class = "title" > { { form . name } } < / h6 >
< div class = "meta" > 最近编辑时间 : { { form . updateTime } } < / div >
< / div >
< div class = "l-title" id = "part1" > < img src = "@/assets/img/label.png" alt = "" > 竞赛信息 < / div >
< div v-if ="form.description" class="texts ql-editor" v-html="form.description" > < / div >
< template v-if ="form.contestAnnexList" >
< h6 class = "p-title" > 附件下载 < / h6 >
< ul class = "files" >
< li v-for ="(item, i) in form.contestAnnexList" :key ="i" >
< el -link v-if ="item.canPreview" class="m-r-10" type="primary" @click="preview(item)" > {{ item.fileName }} < / el -link >
< span v -else class = "fileName" > { { item . fileName } } < / span >
< el -link type = "primary" :underline ="false" @click ="download(item)" > 下载 < / el -link >
< / li >
< / ul >
< / template >
<!-- 进展 -- >
< div class = "l-title" id = "part2" > < img src = "@/assets/img/label.png" alt = "" > 竞赛进展 < / div >
< ul class = "progress" v-if ="progress.length" >
< li v-for ="(item,index) in progress" :key="index" :class="item.status == 0 ? 'not' : (item.status == 1 ? 'ing' : 'done')" >
< i class = "dot" > < / i >
< p class = "name" > { { item . title } } < / p >
< p class = "desc" > { { item . description } } < / p >
< / li >
< img class = "rocket" src = "@/assets/img/rocket.png" alt = "" >
< / ul >
< template v-else >
< div class = "empty" >
< div >
< img src = "@/assets/img/none.png" alt = "" >
< p > 暂无数据 < / p >
< / div >
< / div >
< / template >
<!-- 公告 -- >
< div class = "l-title" id = "part3" > < img src = "@/assets/img/label.png" alt = "" > 通知公告 < / div >
< ul class = "notice-list" v-if ="notices.length" >
< li v-for ="(item, i) in notices" :key="i" @click="toNotice(item)" >
< h6 > { { item . announcementTitle } } < / h6 >
< p class = "meta" > { { item . updateTime } } < / p >
< div class = "des" v-html ="item.announcementText" > < / div >
< / li >
< / ul >
< template v-else >
< div class = "empty" >
< div >
< img src = "@/assets/img/none.png" alt = "" >
< p > 暂无通知公告 < / p >
< / div >
< / div >
< / template >
< / div >
< / div >
< / div >
< / div >
< / template >
< script >
import { mapMutations } from "vuex" ;
import breadcrumb from '@/components/breadcrumb'
import util from '@/libs/util'
import Setting from "@/setting"
export default {
name : 'matchdetail' ,
data ( ) {
return {
id : this . $route . query . id ,
end : this . $route . query . end ,
status : this . $route . query . status ,
statusList : [ "等待报名" , "已报名" , "立即报名" , "报名截止" , "比赛中" , "已结束" ] ,
endList : [ "报名开始" , "报名截止" , "报名截止" , "竞赛开始" , "竞赛结束" , "" ] ,
form : {
name : '' ,
coverUrl : '' ,
description : '' ,
signUpStartTime : '' ,
signUpEndTime : '' ,
playStartTime : '' ,
playEndTime : '' ,
} ,
curType : '1' ,
typeList : [
{
id : '1' ,
name : '竞赛信息'
} ,
{
id : '2' ,
name : '竞赛进展'
} ,
{
id : '3' ,
name : '通知公告'
}
] ,
progress : [ ] ,
statusList : [ "等待报名" , "已报名" , "立即报名" , "报名截止" , "比赛中" , "已结束" ] ,
timer : null ,
notices : [ ] ,
noticeDetail : { }
} ;
} ,
components : {
breadcrumb
} ,
directives : {
countdown : {
bind : function ( el , binding , vnode ) {
let that = vnode . context ;
let time = ''
let second = 1000 ;
let minute = second * 60 ;
let hour = minute * 60 ;
let now = new Date ( ) . getTime ( ) ;
let signUpStartTime = new Date ( that . core . dateCompatible ( that . signUpStartTime ) ) . getTime ( ) ; // 报名开始时间
let signUpEndTime = new Date ( that . core . dateCompatible ( that . signUpEndTime ) ) . getTime ( ) ; // 报名结束时间
let playStartTime = new Date ( that . core . dateCompatible ( that . playStartTime ) ) . getTime ( ) ; // 比赛开始时间
let playEndTime = new Date ( that . core . dateCompatible ( that . playEndTime ) ) . getTime ( ) ; // 比赛结束时间
switch ( that . status ) {
// status每个值的解释请看getData方法
case 0 :
if ( now > signUpStartTime ) {
that . status = 1 ;
} else {
time = signUpStartTime - now ;
}
break ;
case 1 :
if ( now > signUpEndTime ) {
that . status = 3 ;
} else {
time = signUpEndTime - now ;
}
break ;
case 2 :
if ( now > signUpEndTime ) {
that . status = 3 ;
} else {
time = signUpEndTime - now ;
}
break ;
case 3 :
if ( now > playStartTime ) {
that . status = 4 ;
} else {
time = playStartTime - now ;
}
break ;
case 4 :
if ( now > playEndTime ) {
that . status = 5 ;
} else {
time = playEndTime - now ;
}
break ;
}
time = ` ${ Math . floor ( time / hour ) } : ${ Math . floor ( time % hour / minute ) } : ${ Math . floor ( time % hour % minute / second ) } ` ;
that . timer = setInterval ( ( ) => {
let timeList = time . split ( ":" ) ;
let total = Number . parseInt ( timeList [ 0 ] * 60 * 60 ) + Number . parseInt ( timeList [ 1 ] * 60 ) + Number . parseInt ( timeList [ 2 ] ) ;
if ( total > 0 ) {
-- total ;
let hours = Math . floor ( total / ( 60 * 60 ) ) ;
let minutes = Math . floor ( total % ( 60 * 60 ) / 60 ) ;
let seconds = Math . floor ( total % ( 60 * 60 ) % 60 ) ;
time = ` ${ that . core . formateTime ( hours ) } : ${ that . core . formateTime ( minutes ) } : ${ that . core . formateTime ( seconds ) } ` ;
} else {
clearInterval ( that . timer ) ;
}
el . innerHTML = time ;
} , 1000 )
}
}
} ,
mounted ( ) {
this . getData ( )
this . getProgress ( )
this . getNotice ( )
} ,
methods : {
... mapMutations ( 'match' , [
'SET_SOURCE'
] ) ,
getData ( ) { // 获取竞赛信息
this . $post ( ` ${ this . api . getContest } ?contestId= ${ this . id } ` ) . then ( ( { contest } ) => {
const list = contest . contestAnnexList
if ( list ) {
list . map ( e => {
const { filePath } = e
e . canPreview = util . canPreview ( filePath . substr ( filePath . lastIndexOf ( '.' ) + 1 ) )
} )
}
this . form = contest
this . $refs . breadcrumb . update ( '全部赛事/' + contest . name )
} ) . catch ( err => { } )
} ,
getProgress ( ) { // 获取竞赛进展
this . $get ( this . api . getContestProgress , {
contestId : this . id
} ) . then ( res => {
this . progress = res . contestProgressList . reverse ( )
} ) . catch ( err => { } ) ;
} ,
// 公告列表
getNotice ( ) {
this . $post ( ` ${ this . api . queryAnnouncementByContestId } ?pageNum=1&pageSize=1000&contestId= ${ this . id } ` ) . then ( ( { data } ) => {
const records = data . records . filter ( e => e . status ) // 只显示已发布的(status 0草稿 1为已发布)
records . map ( e => {
e . announcementText = e . announcementText . replace ( /<img.*?(?:>|\/>)/gi , '' )
} )
this . notices = records
} ) . catch ( res => { } )
} ,
// 预览附件
preview ( item ) {
const { filePath } = item
const suffix = filePath . substr ( filePath . lastIndexOf ( '.' ) + 1 )
window . open ( ( util . isDoc ( suffix ) ? 'https://view.officeapps.live.com/op/view.aspx?src=' : '' ) + item . filePath )
} ,
// 下载附件
download ( item ) {
util . downloadFile ( item . fileName , item . filePath )
} ,
// tab切换
typeChange ( ) {
document . querySelector ( ` #part ${ this . curType } ` ) . scrollIntoView ( )
} ,
// 跳转公告详情
toNotice ( item ) {
this . $router . push ( ` noticeDetail?id= ${ item . id } &matchId= ${ this . id } &name= ${ this . form . name } &end= ${ this . end } &status= ${ this . status } ` )
} ,
// 立即报名
signup ( ) {
const { status } = this
// 如果没登录,提示去登录
if ( util . local . get ( Setting . tokenKey ) ) {
if ( this . status == 2 ) {
let data = {
contestId : this . id
}
this . $post ( this . api . addApplicant , data ) . then ( res => {
this . $message . success ( '报名成功' )
this . status = 1 ;
} ) . catch ( res => { } )
} else if ( status == 1 ) {
// 已报名,点击取消报名
this . $confirm ( '是否要取消报名?' , '提示' , {
type : 'success'
} ) . then ( ( ) => {
this . $post ( ` ${ this . api . cancelRegistration } ?contestId= ${ this . id } ` ) . then ( res => {
this . $message . success ( '取消报名成功' )
this . status = 2
} ) . catch ( res => { } )
} ) . catch ( ( ) => { } )
}
} else {
this . $confirm ( '请先登录,是否直接前往登录?' , "提示" , {
type : 'success'
} ) . then ( ( ) => {
this . SET _SOURCE ( this . id )
this . $router . push ( '/login' )
} ) . catch ( ( ) => { } )
}
}
}
} ;
< / script >
< style lang = "scss" scoped >
. banner {
width : 100 % ;
height : 350 px ;
padding : 120 px 0 0 20 % ;
color : # fff ;
background - size : 100 % 350 px ;
background - repeat : no - repeat ;
}
. l - title {
font - size : 18 px ;
}
. main . center {
background : url ( . . / . . / . . / assets / img / match - bg1 . png ) ( 0 px 95 px ) / auto auto no - repeat ,
url ( . . / . . / . . / assets / img / match - bg2 . png ) ( 98 % 300 px ) / auto auto no - repeat ;
}
. main . center - wrap {
margin - top : 30 px ;
}
. content {
position : relative ;
padding : 20 px 40 px ;
margin - top : 30 px ;
background - color : # fff ;
. title {
width : 67 % ;
margin : 0 auto ;
font - size : 28 px ;
text - align : center ;
color : # 0B1 D30 ;
}
. info . meta {
padding : 16 px 0 ;
font - size : 12 px ;
color : # 999 ;
text - align : center ;
}
. action {
display : inline - flex ;
align - items : center ;
}
. status {
padding : 0 16 px ;
margin - left : 20 px ;
line - height : 34 px ;
font - size : 14 px ;
color : # fff ;
background - color : # 52 C41A ;
border - radius : 4 px ;
cursor : pointer ;
& . wait {
background - color : # FAAD14 ;
}
& . signing {
background - color : $main - color ;
}
& . signed {
background - color : # 52 C41A ;
}
& . finish {
background - color : # ccc ;
}
}
. end - text {
font - size : 12 px ;
color : # 666 ;
em {
font - style : normal ;
color : # f00 ;
}
}
. texts {
margin : 20 px 0 50 px ;
font - size : 14 px ;
line - height : 1.6 ;
text - indent : 2 em ;
overflow : hidden ;
/deep/img {
max - width : 100 % ;
}
}
. progress {
position : relative ;
width : 95 % ;
padding : 50 px 0 ;
margin : 40 px auto 80 px ;
text - align : left ;
& : before {
content : '' ;
position : absolute ;
top : 0 ;
left : 50 % ;
width : 2 px ;
height : 100 % ;
background - color : # E1E6F2 ;
}
& : after {
content : '' ;
position : absolute ;
top : - 10 px ;
left : 430 px ;
border : 8 px solid transparent ;
border - bottom - color : # E1E6F2 ;
}
. rocket {
position : absolute ;
bottom : - 50 px ;
left : 425 px ;
}
li {
position : relative ;
width : 400 px ;
margin - bottom : 42 px ;
. dot {
position : absolute ;
top : 12 px ;
left : 431 px ;
width : 15 px ;
height : 15 px ;
background - color : # DCDCDC ;
border - radius : 50 % ;
}
. name {
display : inline - block ;
padding : 0 19 px ;
margin - bottom : 16 px ;
line - height : 40 px ;
text - align : center ;
font - size : 16 px ;
color : # fff ;
border - radius : 20 px ;
background - color : # C4C4C4 ;
}
. desc {
position : relative ;
color : # 333 ;
font - size : 14 px ;
}
& . ing , & . done {
. dot {
top : 8 px ;
background - color : # 007 EFF ;
}
. name {
background - color : # 007 EFF ;
}
}
& . ing {
. dot {
width : 27 px ;
height : 27 px ;
border : 6 px solid # E2F1FB ;
}
}
& : nth - child ( odd ) {
text - align : right ;
& . ing {
. dot {
left : auto ;
right : - 51 px ;
}
}
. name {
& : before {
content : '' ;
z - index : 2 ;
position : absolute ;
top : 14 px ;
right : - 35 px ;
border : 18 px solid transparent ;
border - top - width : 6 px ;
border - bottom - width : 6 px ;
border - left - color : # C4C4C4 ;
}
}
. desc {
text - align : right ;
}
& . ing , & . done {
. name {
& : before {
border - left - color : # 007 EFF ;
}
}
}
}
& : nth - child ( even ) {
margin - left : 482 px ;
. dot {
left : - 51 px ;
}
& . ing {
. dot {
left : - 57 px ;
}
}
. name {
text - align : left ;
& : after {
content : '' ;
z - index : 2 ;
position : absolute ;
top : 14 px ;
left : - 35 px ;
border : 18 px solid transparent ;
border - top - width : 6 px ;
border - bottom - width : 6 px ;
border - right - color : # C4C4C4 ;
}
}
. desc {
& : before {
left : auto ;
right : - 16 px ;
border : 8 px solid transparent ;
border - left - color : # fff ;
}
& : after {
left : auto ;
right : - 18 px ;
border : 9 px solid transparent ;
border - left - color : # E6E6E6 ;
}
}
& . ing , & . done {
. name {
& : after {
border - right - color : # 007 EFF ;
}
}
}
}
& : last - child {
margin - bottom : 0 ;
}
}
}
}
. files {
margin - bottom : 30 px ;
li {
display : flex ;
align - items : center ;
margin : 10 px 0 ;
}
. fileName {
margin - right : 10 px ;
font - size : 12 px ;
}
}
. notice - list {
text - align : left ;
li {
padding : 16 px ;
margin - bottom : 12 px ;
transition : all 0.3 s ;
cursor : pointer ;
border - radius : 6 px ;
background - color : # fff ;
border - bottom : 1 px dashed # ebebeb ;
& : last - child {
border - bottom : 0 ;
}
}
h6 {
font - size : 20 px ;
font - weight : 500 ;
color : # 0B1 D30 ;
& : hover {
color : # 007 EFF ;
}
}
. meta {
margin : 10 px 0 ;
font - size : 14 px ;
color : # 666 ;
}
. des {
font - size : 14 px ;
color : # 333 ;
line - height : 24 px ;
display : - webkit - box ;
display : - moz - box ;
- webkit - box - orient : vertical ;
- moz - box - orient : vertical ;
- webkit - line - clamp : 2 ;
- moz - line - clamp : 2 ;
overflow : hidden ;
text - overflow : ellipsis ;
}
}
< / style >