mirror of
https://github.com/algerkong/AlgerMusicPlayer.git
synced 2026-04-03 14:20:50 +08:00
- 关键布局组件(AppMenu/TitleBar/SearchBar)改为同步导入,消除加载闪烁 - 新增全局 skeleton-shimmer 流光动画替代 animate-pulse 闪烁效果 - 用户页 loading 骨架屏避免使用 .left scoped 样式导致的深色背景 - 全部 n-skeleton 组件替换为原生 div + shimmer,统一圆角风格 - 菜单容器添加背景色防止加载穿透
194 lines
5.7 KiB
Vue
194 lines
5.7 KiB
Vue
<template>
|
|
<section class="playlist-section">
|
|
<!-- Section Header -->
|
|
<div class="mb-6 flex items-center justify-between">
|
|
<div class="flex items-center gap-3">
|
|
<h2 class="text-xl font-bold tracking-tight text-neutral-900 md:text-2xl dark:text-white">
|
|
{{ title }}
|
|
</h2>
|
|
<div class="h-1.5 w-1.5 rounded-full bg-primary" />
|
|
</div>
|
|
<button
|
|
class="group flex items-center gap-1.5 text-sm font-semibold text-neutral-400 transition-colors hover:text-primary dark:text-neutral-500 dark:hover:text-white"
|
|
@click="$emit('more')"
|
|
>
|
|
<span>{{ t('comp.more') }}</span>
|
|
<i class="ri-arrow-right-s-line text-base transition-transform group-hover:translate-x-1" />
|
|
</button>
|
|
</div>
|
|
|
|
<!-- Loading Skeleton -->
|
|
<div v-if="loading" class="grid gap-6" :style="gridStyle">
|
|
<div v-for="i in displayCount" :key="i" class="space-y-3">
|
|
<div class="aspect-square skeleton-shimmer rounded-2xl" />
|
|
<div class="h-4 w-3/4 skeleton-shimmer rounded-lg" />
|
|
<div class="h-3 w-1/2 skeleton-shimmer rounded-lg" />
|
|
</div>
|
|
</div>
|
|
|
|
<!-- Playlist Grid -->
|
|
<div v-else-if="displayPlaylists.length > 0" class="grid gap-6" :style="gridStyle">
|
|
<home-list-item
|
|
v-for="(item, index) in displayPlaylists"
|
|
:key="item.id"
|
|
:cover="item.picUrl"
|
|
:title="item.name"
|
|
:subtitle="item.copywriter"
|
|
:tracks="playlistTracksMap[item.id] || []"
|
|
:play-count="item.playCount"
|
|
:animation-delay="calculateAnimationDelay(index, 0.04)"
|
|
@click="handlePlaylistClick(item)"
|
|
@play="playPlaylist(item)"
|
|
/>
|
|
</div>
|
|
|
|
<!-- Empty State -->
|
|
<div v-else class="flex flex-col items-center justify-center py-20 text-neutral-400">
|
|
<i class="ri-play-list-2-line mb-4 text-5xl opacity-20" />
|
|
<p class="text-sm font-medium">{{ t('comp.recommendSonglist.empty') }}</p>
|
|
</div>
|
|
</section>
|
|
</template>
|
|
|
|
<script setup lang="ts">
|
|
import { computed, onMounted, reactive, ref } from 'vue';
|
|
import { useI18n } from 'vue-i18n';
|
|
import { useRouter } from 'vue-router';
|
|
|
|
import { getPersonalizedPlaylist } from '@/api/home';
|
|
import { getListDetail } from '@/api/list';
|
|
import { navigateToMusicList } from '@/components/common/MusicListNavigator';
|
|
import { usePlayerCoreStore } from '@/store/modules/playerCore';
|
|
import { usePlaylistStore } from '@/store/modules/playlist';
|
|
import { calculateAnimationDelay, isElectron, isMobile } from '@/utils';
|
|
|
|
import HomeListItem from './HomeListItem.vue';
|
|
|
|
const props = withDefaults(
|
|
defineProps<{
|
|
title: string;
|
|
limit?: number;
|
|
columns?: number;
|
|
rows?: number;
|
|
}>(),
|
|
{
|
|
limit: 15,
|
|
columns: 5,
|
|
rows: 3
|
|
}
|
|
);
|
|
|
|
defineEmits<{
|
|
(e: 'more'): void;
|
|
}>();
|
|
|
|
const { t } = useI18n();
|
|
const router = useRouter();
|
|
const playlists = ref<any[]>([]);
|
|
const loading = ref(true);
|
|
const playlistTracksMap = reactive<Record<number, any[]>>({});
|
|
|
|
const effectiveColumns = computed(() =>
|
|
isMobile.value ? Math.min(2, props.columns) : props.columns
|
|
);
|
|
const effectiveRows = computed(() => (isMobile.value ? 2 : props.rows));
|
|
|
|
// Calculate display count to fill exactly N rows
|
|
const displayCount = computed(() => effectiveColumns.value * effectiveRows.value);
|
|
|
|
const displayPlaylists = computed(() => {
|
|
const count = displayCount.value;
|
|
return playlists.value.slice(0, count);
|
|
});
|
|
|
|
const gridStyle = computed(() => ({
|
|
gridTemplateColumns: `repeat(${effectiveColumns.value}, minmax(0, 1fr))`
|
|
}));
|
|
|
|
const fetchPlaylists = async () => {
|
|
try {
|
|
const { data } = await getPersonalizedPlaylist(props.limit || displayCount.value + 5);
|
|
if (data.code === 200) {
|
|
playlists.value = data.result || [];
|
|
// Preload tracks for displayed playlists (Electron only)
|
|
if (isElectron) {
|
|
preloadAllTracks();
|
|
}
|
|
}
|
|
} catch (error) {
|
|
console.error('Failed to fetch playlists:', error);
|
|
} finally {
|
|
loading.value = false;
|
|
}
|
|
};
|
|
|
|
const preloadAllTracks = async () => {
|
|
const playlistsToLoad = displayPlaylists.value;
|
|
|
|
// Load tracks in parallel with concurrency limit
|
|
const batchSize = 4;
|
|
for (let i = 0; i < playlistsToLoad.length; i += batchSize) {
|
|
const batch = playlistsToLoad.slice(i, i + batchSize);
|
|
await Promise.all(
|
|
batch.map(async (item) => {
|
|
if (playlistTracksMap[item.id]) return;
|
|
try {
|
|
const { data } = await getListDetail(item.id);
|
|
if (data.playlist?.tracks) {
|
|
playlistTracksMap[item.id] = data.playlist.tracks.slice(0, 3).map((s: any) => ({
|
|
id: s.id,
|
|
name: s.name
|
|
}));
|
|
}
|
|
} catch (error) {
|
|
console.debug('Failed to load tracks for playlist:', item.id, error);
|
|
}
|
|
})
|
|
);
|
|
}
|
|
};
|
|
|
|
const handlePlaylistClick = async (item: any) => {
|
|
try {
|
|
navigateToMusicList(router, {
|
|
id: item.id,
|
|
type: 'playlist',
|
|
name: item.name,
|
|
listInfo: item,
|
|
canRemove: false
|
|
});
|
|
} catch (error) {
|
|
console.error('Failed to navigate to playlist:', error);
|
|
}
|
|
};
|
|
|
|
const playPlaylist = async (item: any) => {
|
|
try {
|
|
const { data } = await getListDetail(item.id);
|
|
if (data.playlist?.tracks?.length > 0) {
|
|
const playerCore = usePlayerCoreStore();
|
|
const playlistStore = usePlaylistStore();
|
|
|
|
const playlist = data.playlist.tracks.map((s: any) => ({
|
|
id: s.id,
|
|
name: s.name,
|
|
picUrl: s.al?.picUrl || item.picUrl,
|
|
source: 'netease',
|
|
song: s,
|
|
...s,
|
|
playLoading: false
|
|
}));
|
|
|
|
playlistStore.setPlayList(playlist, false, false);
|
|
await playerCore.handlePlayMusic(playlist[0], true);
|
|
}
|
|
} catch (error) {
|
|
console.error('Failed to play playlist:', error);
|
|
}
|
|
};
|
|
|
|
onMounted(() => {
|
|
fetchPlaylists();
|
|
});
|
|
</script>
|