mirror of
https://github.com/atdunbg/Nekosonic-Music.git
synced 2026-06-22 00:58:51 +08:00
## 后端 - 替换 rodio 为 symphonia + ringbuf,重构 audio.rs 播放引擎 - 重构 api.rs,使用 api_call! 宏统一 API 调用模式 - 新增 media_controls.rs,使用 souvlaki 实现跨平台系统媒体控制 (Linux MPRIS / Windows SMTC / macOS Now Playing) - 版本号升至 v0.5.0 ## 前端 - 新增 - 新增 SongListItem 通用组件 - 新增 useOnlineStatus composable,检测网络状态 - 新增 usePageCache composable,页面数据缓存与失效 - 新增 getCoverUrl()、formatDate() 工具函数 - 新增 emitPlaybackState() 同步播放状态到系统媒体控制 - 新增 mpris-command 事件监听,响应系统媒体控制命令 - 新增 Toast 离线/恢复在线提示 - 各页面新增断网恢复后自动重试加载 - 新增路由守卫:已登录用户访问 /login 重定向至首页 - 新增音量持久化(settings store + localStorage) - 新增禁用右键菜单与用户选择限制(输入框除外) ## 前端 - 变更 - Song 接口从 player.ts 迁移至 song.ts 并导出 - AlbumDetail/ArtistDetail/PlaylistDetail/RecentPlays/LocalMusic 迁移至 SongListItem - PlayerBar 队列列表迁移至 SongListItem,封面使用 getCoverUrl() - downloadSong 参数类型从内联对象改为 Song,使用 getCoverUrl() - 默认主题从 green 改为 blue,ThemeName 及相关列表中 blue 移至首位 - 全局快捷键从 Alt+Control 改为 Control+Alt 顺序 - formatShortcut 新增 KeyP → P 显示 - keep-alive 从 max=3 固定 include 改为 max=5 动态列表,窗口隐藏时释放 - App.vue 封面使用 getCoverUrl() 替代手动 al/album 回退 - formatPlayCount 提取常量 - Login.vue text-warning 改为 text-yellow-400 ## 前端 - 删除 - 删除 Search.vue(与 Discover.vue 重复) - 删除 SongItemMenu.vue(被 SongListItem 替代) ## 修复 - 更新器跳过版本逻辑:仅静默检查时跳过已忽略版本,手动检查不再跳过 - 重复播放同一首歌时无法恢复播放 - settings.ts 重复的 ThemeName 定义 - PlayerBar.vue modeTexts 缺少类型注解 - Home.vue map 回调参数缺少类型 - Settings.vue v-for key 类型不匹配
121 lines
4.2 KiB
Vue
121 lines
4.2 KiB
Vue
<template>
|
|
<div class="p-8 text-content flex flex-col items-center justify-center min-h-full">
|
|
<div v-if="!currentSong" class="text-center">
|
|
<p class="text-content-2 mb-4">私人漫游未启动</p>
|
|
<button
|
|
@click="startFm"
|
|
class="px-6 py-2 bg-muted hover:bg-emphasis rounded-full transition"
|
|
>
|
|
开始漫游
|
|
</button>
|
|
</div>
|
|
|
|
<template v-else>
|
|
<img
|
|
v-if="coverUrl && !coverError"
|
|
:src="coverUrl"
|
|
class="w-80 h-80 rounded-3xl object-cover shadow-2xl mb-8"
|
|
@error="coverError = true"
|
|
/>
|
|
<div
|
|
v-else
|
|
class="w-80 h-80 rounded-3xl bg-muted flex items-center justify-center shadow-2xl mb-8"
|
|
>
|
|
<svg width="64" height="64" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="1.5" stroke-linecap="round" stroke-linejoin="round" class="text-content-3"><path d="M9 18V5l12-2v13"/><circle cx="6" cy="18" r="3"/><circle cx="18" cy="16" r="3"/></svg>
|
|
</div>
|
|
|
|
<h1 class="text-3xl font-bold mb-2">{{ currentSong.name }}</h1>
|
|
<p class="text-lg text-content-2 mb-8">
|
|
<template v-for="(a, i) in currentSong.ar || []" :key="a.id || i">
|
|
<span v-if="i > 0" class="text-content-3">/</span>
|
|
<span class="hover:text-accent-text cursor-pointer transition" @click="a.id && router.push({ name: 'artist', params: { id: a.id } })">{{ a.name }}</span>
|
|
</template>
|
|
<template v-if="currentSong.al?.name">
|
|
<span class="text-content-3 mx-1">·</span>
|
|
<span class="hover:text-accent-text cursor-pointer transition" @click="currentSong.al.id && router.push({ name: 'album', params: { id: currentSong.al.id } })">{{ currentSong.al.name }}</span>
|
|
</template>
|
|
</p>
|
|
|
|
<div class="flex items-center gap-8">
|
|
<button
|
|
@click="player.toggle()"
|
|
class="w-16 h-16 flex items-center justify-center rounded-full bg-muted hover:bg-emphasis transition border border-emphasis"
|
|
>
|
|
<svg v-if="player.playing" width="28" height="28" viewBox="0 0 16 16" fill="currentColor">
|
|
<rect x="3" y="2" width="3" height="12" rx="0.5" />
|
|
<rect x="10" y="2" width="3" height="12" rx="0.5" />
|
|
</svg>
|
|
<svg v-else width="28" height="28" viewBox="0 0 16 16" fill="currentColor">
|
|
<path d="M4 2.5v11l9-5.5z" />
|
|
</svg>
|
|
</button>
|
|
<button
|
|
@click="nextSong"
|
|
class="text-content-2 hover:text-content transition"
|
|
>
|
|
<svg width="28" height="28" viewBox="0 0 24 24" fill="none" stroke="currentColor" stroke-width="2" stroke-linecap="round" stroke-linejoin="round"><polygon points="5 4 15 12 5 20 5 4"/><line x1="19" y1="5" x2="19" y2="19"/></svg>
|
|
</button>
|
|
</div>
|
|
</template>
|
|
</div>
|
|
</template>
|
|
|
|
<script setup lang="ts">
|
|
import { ref, computed, watch, onMounted } from 'vue';
|
|
import { usePlayerStore } from '../stores/player';
|
|
import { invoke } from '@tauri-apps/api/core';
|
|
import { normalizeSong, getCoverUrl } from '../utils/song';
|
|
import { useRouter } from 'vue-router';
|
|
import { useOnlineStatus } from '../composables/useOnlineStatus';
|
|
|
|
const player = usePlayerStore();
|
|
const router = useRouter();
|
|
const { isOnline } = useOnlineStatus();
|
|
const coverError = ref(false);
|
|
|
|
const currentSong = computed(() => {
|
|
if (player.isFmMode && player.currentSong) {
|
|
return player.currentSong;
|
|
}
|
|
return null;
|
|
});
|
|
|
|
const coverUrl = computed(() => {
|
|
if (!currentSong.value) return '';
|
|
return getCoverUrl(currentSong.value) || '';
|
|
});
|
|
|
|
watch(coverUrl, () => { coverError.value = false; });
|
|
|
|
onMounted(async () => {
|
|
if (!player.isFmMode || !player.currentSong) {
|
|
await startFm();
|
|
}
|
|
});
|
|
|
|
async function startFm() {
|
|
try {
|
|
const jsonStr: string = await invoke('personal_fm');
|
|
const data = JSON.parse(jsonStr);
|
|
const songs = data.data || data;
|
|
if (songs && songs.length > 0) {
|
|
const song = normalizeSong(songs[0]);
|
|
player.enableFmMode(nextSong);
|
|
await player.playFmSong(song);
|
|
}
|
|
} catch (e) {
|
|
console.error('启动漫游失败', e);
|
|
}
|
|
}
|
|
|
|
async function nextSong() {
|
|
await startFm();
|
|
}
|
|
|
|
watch(isOnline, (val, old) => {
|
|
if (val && !old && !currentSong.value) {
|
|
startFm();
|
|
}
|
|
});
|
|
</script>
|