优化 TorrentItem 列表性能

This commit is contained in:
jxxghp
2025-04-08 15:35:11 +08:00
parent 717b460246
commit 7ae9bbc4f0
3 changed files with 105 additions and 87 deletions

View File

@@ -64,6 +64,7 @@ const filterOptions: Record<string, string[]> = reactive({
// 完整的数据列表
let dataList: SearchTorrent[]
// 显示用的数据列表
const displayDataList = ref<Array<SearchTorrent>>([])
@@ -328,16 +329,6 @@ function toggleFilterMenu(key: string) {
}
}
// 切换过滤器选项
function toggleFilter(key: string, value: string) {
const index = filterForm[key].indexOf(value)
if (index === -1) {
filterForm[key].push(value)
} else {
filterForm[key].splice(index, 1)
}
}
// 清除所有过滤条件
function clearAllFilters() {
for (const key in filterForm) {
@@ -388,7 +379,8 @@ function removeFilter(key: string, value: string) {
}
function loadMore({ done }: { done: any }) {
const itemsToMove = dataList.splice(0, 20) // 从 dataList 中获取最前面的 20 个元素
// 从 dataList 中获取最前面的 20 个元素
const itemsToMove = dataList.splice(0, 20)
displayDataList.value.push(...itemsToMove)
done('ok')
}

View File

@@ -1,11 +1,6 @@
<script lang="ts" setup>
import type { Context } from '@/api/types'
import TorrentItem from '@/components/cards/TorrentItem.vue'
import { useDisplay } from 'vuetify'
// 设备模式
const display = useDisplay()
const appMode = inject('pwaMode') && display.mdAndDown.value
// 定义输入参数
const props = defineProps({
@@ -184,6 +179,9 @@ const sortField = ref('default')
// 数据列表
const dataList = ref<Array<Context>>([])
// 显示用的数据列表
const displayDataList = ref<Array<Context>>([])
// 计算已选择的过滤条件数量
const getFilterCount = computed(() => {
let count = 0
@@ -253,6 +251,7 @@ watchEffect(() => {
watchEffect(() => {
// 清空列表
dataList.value = []
displayDataList.value = []
// 匹配过滤函数
const match = (filter: Array<string>, value: string | undefined) =>
filter.length === 0 || (value && filter.includes(value))
@@ -264,6 +263,9 @@ watchEffect(() => {
initOptions(data)
})
// 筛选数据
const filteredData: Context[] = []
// 然后根据过滤条件筛选数据
props.items.forEach(data => {
const { meta_info, torrent_info } = data
@@ -283,9 +285,13 @@ watchEffect(() => {
// 质量过滤
match(filterForm.edition, meta_info.edition)
) {
dataList.value.push(data)
filteredData.push(data)
}
})
// 显示前20个
displayDataList.value = filteredData.slice(0, 20)
// 保存剩余数据
dataList.value = filteredData.slice(20)
}
})
@@ -337,6 +343,13 @@ function toggleFilterMenu(key: string) {
filterMenuOpen.value = true
}
}
function loadMore({ done }: { done: any }) {
// 从 dataList 中获取最前面的 20 个元素
const itemsToMove = dataList.value.splice(0, 20)
displayDataList.value.push(...itemsToMove)
done('ok')
}
</script>
<template>
@@ -553,18 +566,26 @@ function toggleFilterMenu(key: string) {
<!-- 资源列表容器 -->
<VCard class="resource-list-container">
<!-- 无结果时显示 -->
<div v-if="dataList.length === 0" class="no-results">
<div v-if="displayDataList.length === 0" class="no-results">
<VIcon icon="mdi-file-search-outline" size="64" color="grey-lighten-1" />
<div class="text-h6 text-grey mt-4">暂无符合条件的资源</div>
</div>
<!-- 资源列表 -->
<div v-else class="resource-list">
<div v-for="(item, index) in dataList" :key="`${item.torrent_info?.enclosure || ''}-${index}`">
<VInfiniteScroll
v-else
mode="intersect"
side="end"
:items="displayDataList"
class="resource-list"
@load="loadMore"
>
<template #loading />
<template #empty />
<div v-for="(item, index) in displayDataList" :key="`${item.torrent_info?.enclosure || ''}-${index}`">
<TorrentItem :torrent="item" />
<VDivider v-if="index < dataList.length - 1" class="my-2" />
<VDivider v-if="index < displayDataList.length - 1" class="my-2" />
</div>
</div>
</VInfiniteScroll>
</VCard>
</div>
</template>