Files
AlgerMusicPlayer/src/renderer/views/home/components/HomePlaylistSection.vue
T
alger 0cfec3dd82 refactor(player): 重构播放控制系统,移除 Howler.js 改用原生 HTMLAudioElement
- 新建 playbackController.ts,使用 generation-based 取消替代 playbackRequestManager 状态机
- audioService 重写:单一持久 HTMLAudioElement + Web Audio API,createMediaElementSource 只调一次
- playerCore 瘦身为纯状态管理,移除 handlePlayMusic/playAudio/checkPlaybackState
- playlist next/prev 简化,区分用户手动切歌和歌曲自然播完
- MusicHook 适配 HTMLAudioElement API(.currentTime/.duration/.paused)
- preloadService 从 Howl 实例缓存改为 URL 可用性验证
- 所有 view/component 调用者迁移到 playbackController.playTrack()

修复:快速切歌竞态、seek 到未缓冲位置失败、重启后自动播放循环提示、EQ 重建崩溃
2026-03-29 13:18:05 +08:00

181 lines
5.3 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="isElectron ? playlistTracksMap[item.id] || [] : []"
:show-hover-tracks="isElectron"
:play-count="item.playCount"
:animation-delay="calculateAnimationDelay(index, 0.04)"
@click="handlePlaylistClick(item)"
@play="playPlaylist(item)"
@mouseenter="isElectron && loadTracksOnHover(item.id)"
/>
</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 { playTrack } from '@/services/playbackController';
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 || [];
}
} catch (error) {
console.error('Failed to fetch playlists:', error);
} finally {
loading.value = false;
}
};
/** Lazy load tracks for a single playlist on hover */
const loadTracksOnHover = async (id: number) => {
if (playlistTracksMap[id]) return;
try {
const { data } = await getListDetail(id);
if (data.playlist?.tracks) {
playlistTracksMap[id] = data.playlist.tracks.slice(0, 3).map((s: any) => ({
id: s.id,
name: s.name
}));
}
} catch {
// silent — user can retry by hovering again
}
};
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 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 playTrack(playlist[0], true);
}
} catch (error) {
console.error('Failed to play playlist:', error);
}
};
onMounted(() => {
fetchPlaylists();
});
</script>