324 lines
8.2 KiB
Vue
324 lines
8.2 KiB
Vue
<script setup lang="ts">
|
|
import { Icon } from "@iconify/vue"
|
|
import { NFlex, NTag } from "naive-ui"
|
|
import { useRouteQuery } from "@vueuse/router"
|
|
import { getProblemList, getRandomProblemID } from "oj/api"
|
|
import { getTagColor } from "utils/functions"
|
|
import { ProblemFiltered } from "utils/types"
|
|
import { getProblemTagList } from "shared/api"
|
|
import Hitokoto from "shared/components/Hitokoto.vue"
|
|
import Pagination from "shared/components/Pagination.vue"
|
|
import { useBreakpoints } from "shared/composables/breakpoints"
|
|
import { usePagination } from "shared/composables/pagination"
|
|
import { useUserStore } from "shared/store/user"
|
|
import { renderTableTitle } from "utils/renders"
|
|
import ProblemStatus from "./components/ProblemStatus.vue"
|
|
import AuthorSelect from "shared/components/AuthorSelect.vue"
|
|
import ProblemListTitle from "./components/ProblemListTitle.vue"
|
|
|
|
interface Tag {
|
|
id: number
|
|
name: string
|
|
checked: boolean
|
|
}
|
|
|
|
interface ProblemQuery {
|
|
keyword: string
|
|
difficulty: string
|
|
tag: string
|
|
author: string
|
|
sort: string
|
|
}
|
|
|
|
const difficultyOptions = [
|
|
{ label: "全部", value: "" },
|
|
{ label: "简单", value: "Low" },
|
|
{ label: "中等", value: "Mid" },
|
|
{ label: "困难", value: "High" },
|
|
]
|
|
|
|
const sortOptions = [
|
|
{ label: "最新创建", value: "" },
|
|
{ label: "最早创建", value: "create_time" },
|
|
{ label: "最多提交", value: "-submission_number" },
|
|
{ label: "最少提交", value: "submission_number" },
|
|
{ label: "最多通过", value: "-accepted_number" },
|
|
{ label: "最少通过", value: "accepted_number" },
|
|
{ label: "画流程图", value: "flowchart" },
|
|
]
|
|
|
|
const router = useRouter()
|
|
|
|
const userStore = useUserStore()
|
|
|
|
const { isDesktop } = useBreakpoints()
|
|
|
|
const problems = ref<ProblemFiltered[]>([])
|
|
const total = ref(0)
|
|
const tags = ref<Tag[]>([])
|
|
const [showTag, toggleShowTag] = useToggle(isDesktop.value)
|
|
|
|
// 使用分页 composable
|
|
const { query, clearQuery } = usePagination<ProblemQuery>({
|
|
keyword: useRouteQuery("keyword", "").value,
|
|
difficulty: useRouteQuery("difficulty", "").value,
|
|
tag: useRouteQuery("tag", "").value,
|
|
author: useRouteQuery("author", "").value,
|
|
sort: useRouteQuery("sort", "").value,
|
|
})
|
|
|
|
async function listProblems() {
|
|
if (query.page < 1) query.page = 1
|
|
const offset = (query.page - 1) * query.limit
|
|
const res = await getProblemList(offset, query.limit, {
|
|
keyword: query.keyword,
|
|
tag: query.tag,
|
|
difficulty: query.difficulty,
|
|
author: query.author,
|
|
sort: query.sort,
|
|
})
|
|
total.value = res.total
|
|
problems.value = res.results
|
|
}
|
|
|
|
async function listTags() {
|
|
const res = await getProblemTagList()
|
|
tags.value = res.data.map((r: Omit<Tag, "checked">) => ({
|
|
...r,
|
|
checked: query.tag === r.name,
|
|
}))
|
|
}
|
|
|
|
function chooseTag(tag: Tag) {
|
|
query.tag = tag.checked ? "" : tag.name
|
|
tags.value = tags.value.map((t) => {
|
|
if (t.id === tag.id) {
|
|
t.checked = !t.checked
|
|
} else {
|
|
t.checked = false
|
|
}
|
|
return t
|
|
})
|
|
}
|
|
|
|
async function getRandom() {
|
|
const res = await getRandomProblemID()
|
|
router.push("/problem/" + res.data)
|
|
}
|
|
|
|
// 监听搜索关键词变化(防抖)
|
|
watchDebounced(() => query.keyword, listProblems, {
|
|
debounce: 500,
|
|
maxWait: 1000,
|
|
})
|
|
|
|
// 监听其他查询条件变化
|
|
watch(
|
|
() => [
|
|
query.tag,
|
|
query.difficulty,
|
|
query.limit,
|
|
query.page,
|
|
query.author,
|
|
query.sort,
|
|
],
|
|
listProblems,
|
|
)
|
|
|
|
// 监听标签变化,更新标签选中状态
|
|
watch(
|
|
() => query.tag,
|
|
() => {
|
|
tags.value = tags.value.map((r: Omit<Tag, "checked">) => ({
|
|
...r,
|
|
checked: query.tag === r.name,
|
|
}))
|
|
},
|
|
)
|
|
|
|
watch(
|
|
() => userStore.isFinished && userStore.isAuthed,
|
|
(isAuthenticatedAndFinished) => {
|
|
if (isAuthenticatedAndFinished) {
|
|
listProblems()
|
|
}
|
|
},
|
|
)
|
|
|
|
onMounted(() => {
|
|
listProblems()
|
|
listTags()
|
|
})
|
|
|
|
const baseColumns: DataTableColumn<ProblemFiltered>[] = [
|
|
{
|
|
title: renderTableTitle("状态", "streamline-emojis:high-voltage"),
|
|
key: "status",
|
|
width: 80,
|
|
align: "center",
|
|
render: (row) => h(ProblemStatus, { status: row.status }),
|
|
},
|
|
{
|
|
title: renderTableTitle("编号", "streamline-emojis:game-dice"),
|
|
key: "_id",
|
|
width: 100,
|
|
},
|
|
{
|
|
title: renderTableTitle("题目", "streamline-emojis:watermelon-2"),
|
|
key: "title",
|
|
minWidth: 200,
|
|
render: (row) => h(ProblemListTitle, { problem: row }),
|
|
},
|
|
{
|
|
title: renderTableTitle("难度", "streamline-emojis:lady-beetle"),
|
|
key: "difficulty",
|
|
width: 100,
|
|
render: (row) =>
|
|
h(NTag, { type: getTagColor(row.difficulty) }, () => row.difficulty),
|
|
},
|
|
{
|
|
title: renderTableTitle("标签", "streamline-emojis:paperclip"),
|
|
key: "tags",
|
|
width: 260,
|
|
render: (row) =>
|
|
h(NFlex, () => row.tags.map((t) => h(NTag, { key: t }, () => t))),
|
|
},
|
|
{
|
|
title: renderTableTitle("出题者", "streamline-emojis:man-raising-hand-2"),
|
|
key: "author",
|
|
width: 130,
|
|
},
|
|
{
|
|
title: renderTableTitle("提交数", "streamline-emojis:writing-hand-2"),
|
|
key: "submission",
|
|
align: "center",
|
|
width: 100,
|
|
},
|
|
{
|
|
title: renderTableTitle("通过率", "streamline-emojis:victory-hand-2"),
|
|
key: "rate",
|
|
width: 100,
|
|
align: "center",
|
|
},
|
|
]
|
|
|
|
const columns = computed(() =>
|
|
userStore.isAuthed
|
|
? baseColumns
|
|
: baseColumns.filter((c: any) => c.key !== "status"),
|
|
)
|
|
|
|
function rowProps(row: ProblemFiltered) {
|
|
return {
|
|
style: "cursor: pointer",
|
|
onClick() {
|
|
router.push("/problem/" + row._id)
|
|
},
|
|
}
|
|
}
|
|
</script>
|
|
|
|
<template>
|
|
<n-flex vertical size="large">
|
|
<div class="problem-list-toolbar">
|
|
<n-space>
|
|
<n-form :show-feedback="false" inline label-placement="left">
|
|
<n-form-item label="难度">
|
|
<n-select
|
|
style="width: 100px"
|
|
v-model:value="query.difficulty"
|
|
:options="difficultyOptions"
|
|
/>
|
|
</n-form-item>
|
|
<n-form-item label="出题者">
|
|
<AuthorSelect v-model:value="query.author" />
|
|
</n-form-item>
|
|
</n-form>
|
|
<n-form :show-feedback="false" inline label-placement="left">
|
|
<n-form-item label="排序">
|
|
<n-select
|
|
style="width: 100px"
|
|
v-model:value="query.sort"
|
|
:options="sortOptions"
|
|
/>
|
|
</n-form-item>
|
|
<n-form-item>
|
|
<n-input
|
|
clearable
|
|
style="width: 160px"
|
|
v-model:value="query.keyword"
|
|
placeholder="题号或标题"
|
|
/>
|
|
</n-form-item>
|
|
</n-form>
|
|
<n-form :show-feedback="false" inline label-placement="left">
|
|
<n-form-item>
|
|
<n-button @click="clearQuery" quaternary>重置</n-button>
|
|
</n-form-item>
|
|
<!-- <n-form-item>
|
|
<n-button @click="getRandom" quaternary>随机</n-button>
|
|
</n-form-item> -->
|
|
<n-form-item>
|
|
<n-button
|
|
@click="toggleShowTag()"
|
|
quaternary
|
|
icon-placement="right"
|
|
>
|
|
<template #icon>
|
|
<Icon v-if="showTag" icon="ph:caret-down"></Icon>
|
|
<Icon v-else icon="ph:caret-up"></Icon>
|
|
</template>
|
|
标签
|
|
</n-button>
|
|
</n-form-item>
|
|
</n-form>
|
|
</n-space>
|
|
<Hitokoto v-if="isDesktop" class="problem-list-hitokoto" />
|
|
</div>
|
|
<n-collapse-transition :show="showTag">
|
|
<n-flex>
|
|
<n-tag
|
|
v-for="tag in tags"
|
|
:closable="tag.checked"
|
|
@close="chooseTag(tag)"
|
|
@click="chooseTag(tag)"
|
|
:key="tag.id"
|
|
:type="tag.checked ? 'success' : 'default'"
|
|
>
|
|
{{ tag.name }}
|
|
</n-tag>
|
|
</n-flex>
|
|
</n-collapse-transition>
|
|
<n-data-table
|
|
:bordered="false"
|
|
:data="problems"
|
|
:columns="columns"
|
|
:row-props="rowProps"
|
|
/>
|
|
</n-flex>
|
|
<Pagination
|
|
:total="total"
|
|
v-model:limit="query.limit"
|
|
v-model:page="query.page"
|
|
/>
|
|
</template>
|
|
|
|
<style scoped>
|
|
.problem-list-toolbar {
|
|
display: grid;
|
|
grid-template-columns: minmax(0, 1fr) auto;
|
|
align-items: start;
|
|
gap: 12px 16px;
|
|
}
|
|
|
|
.problem-list-toolbar :deep(.n-space) {
|
|
min-width: 0;
|
|
}
|
|
|
|
.problem-list-hitokoto {
|
|
justify-self: end;
|
|
width: min(400px, 30vw);
|
|
min-width: 0;
|
|
}
|
|
</style>
|