add countdown.

This commit is contained in:
2023-03-03 12:46:58 +08:00
parent b07489723c
commit 18a20c04c5
10 changed files with 414 additions and 368 deletions

View File

@@ -1,5 +1,7 @@
import { formatISO, getTime, intervalToDuration, parseISO } from "date-fns"
import { useUserStore } from "~/shared/store/user"
import { ContestType } from "~/utils/constants"
import { ContestStatus, ContestType } from "~/utils/constants"
import { duration } from "~/utils/functions"
import { Contest, Problem } from "~/utils/types"
import {
getContest,
@@ -11,27 +13,58 @@ import {
export const useContestStore = defineStore("contest", () => {
const userStore = useUserStore()
const message = useMessage()
const [access, toggleAccess] = useToggle()
const contest = ref<Contest>()
const [access, toggleAccess] = useToggle(false)
const contest = ref<Contest | null>(null)
const problems = ref<Problem[]>([])
const now = ref(0)
const contestStatus = computed(() => {
return false
let timer = 0
const contestStatus = computed<ContestStatus>(() => {
const start = getTime(parseISO(contest.value!.start_time.toString()))
const end = getTime(parseISO(contest.value!.end_time.toString()))
if (start > now.value) {
return ContestStatus.not_started
} else if (end < now.value) {
return ContestStatus.finished
} else {
return ContestStatus.underway
}
})
const countdown = computed(() => {
if (contestStatus.value === ContestStatus.finished) {
return "已结束"
} else if (contestStatus.value === ContestStatus.not_started) {
const d = duration(formatISO(now.value), contest.value!.start_time, true)
return "距离比赛开始 " + d
} else {
const d = duration(formatISO(now.value), contest.value!.end_time, true)
return "距离比赛结束 " + d
}
})
const isContestAdmin = computed(
() =>
userStore.isSuperAdmin ||
(userStore.isAuthed && contest.value?.created_by.id === userStore.user.id)
(userStore.isAuthed &&
contest.value?.created_by.id === userStore.user!.id)
)
const isPrivate = computed(
() => contest.value?.contest_type === ContestType.private
() => contest.value!.contest_type === ContestType.private
)
async function init(contestID: string) {
problems.value = []
const res = await getContest(contestID)
contest.value = res.data
now.value = getTime(parseISO(res.data.now))
if (contestStatus.value !== ContestStatus.finished) {
timer = setInterval(() => {
now.value = now.value + 1000
}, 1000)
}
if (contest.value?.contest_type === ContestType.private) {
const res = await getContestAccess(contestID)
toggleAccess(res.data.access)
@@ -39,6 +72,14 @@ export const useContestStore = defineStore("contest", () => {
_getProblems(contestID)
}
function clear() {
contest.value = null
problems.value = []
toggleAccess(false)
now.value = 0
if (timer) clearInterval(timer)
}
async function checkPassword(contestID: string, password: string) {
try {
const res = await checkContestPassword(contestID, password)
@@ -53,7 +94,6 @@ export const useContestStore = defineStore("contest", () => {
}
async function _getProblems(contestID: string) {
problems.value = []
try {
problems.value = await getContestProblems(contestID)
} catch (err) {
@@ -69,7 +109,9 @@ export const useContestStore = defineStore("contest", () => {
access,
problems,
isPrivate,
countdown,
init,
clear,
checkPassword,
}
})