Initial project

This commit is contained in:
Timi
2025-07-16 16:35:28 +08:00
parent 1e8213575b
commit 1c1f2f6594
36 changed files with 11610 additions and 129 deletions

238
src/views/ArticleIndex.vue Normal file
View File

@ -0,0 +1,238 @@
<template>
<div class="article-index">
<div class="list">
<article-list
v-if="pageResult"
:list="pageResult.list"
/>
<t-pagination
class="pages"
v-if="pageResult"
:total="pageResult.total"
:pageSize="16"
:showPageSize="false"
:totalContent="isShowPageTotal"
:current="currentPage"
:onCurrentChange="onPageChangeEvent"
>
<template #totalContent v-if="isShowPageTotal">
<div style="flex: 1" v-text="`共 ${pageResult.total} 个项目`"></div>
</template>
</t-pagination>
</div>
<aside class="aside">
<section class="ranking">
<h4 class="title">
<icon class="icon" name="LIST" />
<span>排行</span>
</h4>
<ul class="items">
<li
class="item"
v-for="item in ranking"
:key="item.id"
>
<a
class="title black clip-text"
:href="`/aid${item.id}.html`"
v-text="item.title"
:title="item.title"
@click.prevent="Toolkit.leftClickCallback($event, () => router.push(`/aid${item.id}.html`))"
></a>
</li>
</ul>
</section>
<section class="friend">
<h4 class="title">
<icon class="icon" name="LINK_0" />
<span>友链</span>
</h4>
<div class="items">
<div
class="item"
v-for="friend in friends"
:key="friend.link"
>
<img class="icon ir-pixelated" :src="friend.icon" :alt="friend.name" />
<a
class="name black"
:href="friend.link"
target="_blank"
v-text="friend.name"
></a>
</div>
</div>
</section>
</aside>
</div>
</template>
<script lang="ts" setup>
import { deviceStore, Icon, Page, PageResult, Toolkit } from "timi-web";
import ArticleAPI from "@/api/ArticleAPI";
import { Article, ArticleRanking } from "@/types/Article";
import ArticleList from "@/components/article/ArticleList.vue";
import { Friend } from "@/types/Friend.ts";
import BlogAPI from "@/api/BlogAPI.ts";
const router = useRouter();
const route = useRoute();
// 页码
const currentPage = computed(() => page.index + 1);
// 分页参数
const page = reactive<Page>({
index: 0,
size: 16
});
// 数据列表
const pageResult = ref<PageResult<Article<any>>>();
// 获取列表
const fetchList = Toolkit.debounce(async () => pageResult.value = await ArticleAPI.page(page));
// 监听路由分页
watch(
() => route.query.p,
async (newP) => {
const pNum = Number(newP);
if (Number.isInteger(pNum) && 0 <= pNum) {
page.index = pNum;
await fetchList();
} else {
// 参数无效时重置为第一页
await router.push({
path: route.path,
query: {
...route.query,
p: 0
}
});
}
},
{ immediate: true }
);
function onPageChangeEvent(current: number) {
const newIndex = current - 1;
router.push({
path: route.path,
query: {
...route.query,
p: newIndex
}
});
}
// 分页总数
const isShowPageTotal = computed(() => !deviceStore.isPhone.value);
// 排行
const ranking = reactive<ArticleRanking[]>([]);
onMounted(async () => {
ranking.push(...await ArticleAPI.listRanking());
});
// 友链
const friends = reactive<Friend[]>([]);
onMounted(async () => {
friends.push(...await BlogAPI.friends());
});
</script>
<style lang="less" scoped>
.article-index {
display: flex;
border-bottom: var(--tui-border);
.list {
border-right: var(--tui-border);
.pages {
bottom: -1px;
padding: var(--tui-page-padding);
position: sticky;
background: rgba(255, 255, 255, .8);
border-top: var(--tui-border);
backdrop-filter: blur(10px);
-webkit-backdrop-filter: blur(10px);
}
}
.aside {
top: 49px;
width: 240px;
height: fit-content;
position: sticky;
min-width: 240px;
h4.title {
margin: 0;
display: flex;
padding: .5rem;
align-items: center;
border-bottom: var(--tui-border);
.icon {
margin-right: .5rem;
}
}
.ranking {
.items {
margin: 0;
padding: .5rem .5rem .5rem 2rem;
.item {
.title {
width: 100%;
display: block;
font-size: 14px;
}
}
}
}
.friend {
margin-top: 2rem;
.items {
display: flex;
padding: .5rem;
flex-direction: column;
.item {
display: flex;
margin-bottom: .5rem;
.icon {
width: 16px;
height: 16px;
margin-right: .25rem;
}
.name {
margin: 0;
font-size: 13px;
}
}
}
}
}
}
@media screen and (max-width: 680px) {
.article-index {
.list {
border-right: none;
}
.aside {
display: none;
}
}
}
</style>