mirror of
https://github.com/algerkong/AlgerMusicPlayer.git
synced 2026-04-14 23:11:00 +08:00
281 lines
7.7 KiB
Vue
281 lines
7.7 KiB
Vue
<template>
|
||
<Teleport to="body">
|
||
<Transition name="update-modal">
|
||
<div
|
||
v-if="showModal"
|
||
class="fixed inset-0 z-[999999] flex items-end justify-center bg-black/50 backdrop-blur-sm"
|
||
>
|
||
<!-- 弹窗内容 -->
|
||
<div
|
||
class="w-full max-w-lg bg-white dark:bg-gray-900 rounded-t-3xl overflow-hidden animate-slide-up"
|
||
>
|
||
<!-- 顶部装饰条 -->
|
||
<div class="h-1 bg-gradient-to-r from-green-400 via-green-500 to-emerald-600"></div>
|
||
|
||
<!-- 关闭条 -->
|
||
<div class="flex justify-center pt-3 pb-2">
|
||
<div class="w-10 h-1 rounded-full bg-gray-300 dark:bg-gray-700"></div>
|
||
</div>
|
||
|
||
<!-- 头部信息 -->
|
||
<div class="px-6 pb-5">
|
||
<div class="flex items-center gap-4">
|
||
<!-- 应用图标 -->
|
||
<div
|
||
class="w-20 h-20 rounded-2xl overflow-hidden shadow-lg flex-shrink-0 ring-2 ring-green-500/20"
|
||
>
|
||
<img src="@/assets/logo.png" alt="App Icon" class="w-full h-full object-cover" />
|
||
</div>
|
||
|
||
<!-- 版本信息 -->
|
||
<div class="flex-1 min-w-0">
|
||
<div class="flex items-center gap-2 mb-2">
|
||
<span
|
||
class="px-3 py-1 text-xs font-medium text-white bg-gradient-to-r from-green-500 to-emerald-600 rounded-full"
|
||
>
|
||
{{ t('comp.update.newVersion') }}
|
||
</span>
|
||
</div>
|
||
<h2 class="text-2xl font-bold text-gray-900 dark:text-white truncate">
|
||
v{{ updateInfo.latestVersion }}
|
||
</h2>
|
||
<p class="text-sm text-gray-500 dark:text-gray-400 mt-1">
|
||
{{ t('comp.update.currentVersion') }}: v{{ updateInfo.currentVersion }}
|
||
</p>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
|
||
<!-- 更新内容 -->
|
||
<div
|
||
class="mx-6 mb-6 max-h-80 overflow-y-auto rounded-2xl bg-gray-50 dark:bg-gray-800/50"
|
||
>
|
||
<div
|
||
class="p-5 text-sm text-gray-600 dark:text-gray-300 leading-relaxed"
|
||
v-html="parsedReleaseNotes"
|
||
></div>
|
||
</div>
|
||
|
||
<!-- 操作按钮 -->
|
||
<div
|
||
class="px-6 pb-8 flex gap-3"
|
||
:style="{ paddingBottom: `calc(32px + var(--safe-area-inset-bottom, 0px))` }"
|
||
>
|
||
<button
|
||
@click="handleLater"
|
||
class="flex-1 py-4 px-4 rounded-2xl text-base font-medium text-gray-700 dark:text-gray-300 bg-gray-100 dark:bg-gray-800 hover:bg-gray-200 dark:hover:bg-gray-700 active:scale-[0.98] transition-all duration-200"
|
||
>
|
||
{{ t('comp.update.later') }}
|
||
</button>
|
||
<button
|
||
@click="handleUpdate"
|
||
class="flex-1 py-4 px-4 rounded-2xl text-base font-medium text-white bg-gradient-to-r from-green-500 to-emerald-600 hover:from-green-600 hover:to-emerald-700 active:scale-[0.98] transition-all duration-200 shadow-lg shadow-green-500/25"
|
||
>
|
||
<span class="flex items-center justify-center gap-2">
|
||
<i class="ri-download-2-line text-lg"></i>
|
||
{{ t('comp.update.updateNow') }}
|
||
</span>
|
||
</button>
|
||
</div>
|
||
</div>
|
||
</div>
|
||
</Transition>
|
||
</Teleport>
|
||
</template>
|
||
|
||
<script setup lang="ts">
|
||
import { marked } from 'marked';
|
||
import { computed, onMounted, ref } from 'vue';
|
||
import { useI18n } from 'vue-i18n';
|
||
|
||
import { checkUpdate, getProxyNodes, UpdateResult } from '@/utils/update';
|
||
|
||
import config from '../../../../package.json';
|
||
|
||
const { t } = useI18n();
|
||
|
||
// 缓存键:记录用户点击"稍后提醒"的时间
|
||
const REMIND_LATER_KEY = 'update_remind_later_timestamp';
|
||
|
||
// 配置 marked
|
||
marked.setOptions({
|
||
breaks: true,
|
||
gfm: true
|
||
});
|
||
|
||
const showModal = ref(false);
|
||
|
||
const updateInfo = ref<UpdateResult>({
|
||
hasUpdate: false,
|
||
latestVersion: '',
|
||
currentVersion: config.version,
|
||
releaseInfo: null
|
||
});
|
||
|
||
// 解析 Markdown
|
||
const parsedReleaseNotes = computed(() => {
|
||
if (!updateInfo.value.releaseInfo?.body) return '';
|
||
try {
|
||
return marked.parse(updateInfo.value.releaseInfo.body);
|
||
} catch (error) {
|
||
console.error('Error parsing markdown:', error);
|
||
return updateInfo.value.releaseInfo.body;
|
||
}
|
||
});
|
||
|
||
// 检查是否应该显示更新提醒
|
||
const shouldShowReminder = (): boolean => {
|
||
const remindLaterTime = localStorage.getItem(REMIND_LATER_KEY);
|
||
if (!remindLaterTime) return true;
|
||
|
||
const savedTime = parseInt(remindLaterTime, 10);
|
||
const now = Date.now();
|
||
const oneDayInMs = 24 * 60 * 60 * 1000; // 24小时
|
||
|
||
// 如果距离上次点击"稍后提醒"超过24小时,则显示
|
||
return now - savedTime >= oneDayInMs;
|
||
};
|
||
|
||
// 处理"稍后提醒"
|
||
const handleLater = () => {
|
||
// 记录当前时间
|
||
localStorage.setItem(REMIND_LATER_KEY, Date.now().toString());
|
||
showModal.value = false;
|
||
};
|
||
|
||
const closeModal = () => {
|
||
showModal.value = false;
|
||
};
|
||
|
||
const checkForUpdates = async () => {
|
||
// 检查是否应该显示提醒
|
||
if (!shouldShowReminder()) {
|
||
console.log('更新提醒被延迟,等待24小时后再提醒');
|
||
return;
|
||
}
|
||
|
||
try {
|
||
const result = await checkUpdate(config.version);
|
||
if (result && result.hasUpdate) {
|
||
updateInfo.value = result;
|
||
showModal.value = true;
|
||
}
|
||
} catch (error) {
|
||
console.error('检查更新失败:', error);
|
||
}
|
||
};
|
||
|
||
const handleUpdate = async () => {
|
||
const version = updateInfo.value.latestVersion;
|
||
|
||
// Android APK 下载地址
|
||
const downloadUrl = `https://github.com/algerkong/AlgerMusicPlayer/releases/download/v${version}/AlgerMusicPlayer-${version}.apk`;
|
||
|
||
try {
|
||
// 获取代理节点
|
||
const proxyHosts = await getProxyNodes();
|
||
const proxyDownloadUrl = `${proxyHosts[0]}/${downloadUrl}`;
|
||
|
||
// 清除"稍后提醒"记录(用户选择更新后,下次应该正常提醒)
|
||
localStorage.removeItem(REMIND_LATER_KEY);
|
||
|
||
// 使用系统浏览器打开下载链接
|
||
window.open(proxyDownloadUrl, '_blank');
|
||
|
||
// 关闭弹窗
|
||
closeModal();
|
||
} catch (error) {
|
||
console.error('打开下载链接失败:', error);
|
||
// 回退到直接打开 GitHub Releases
|
||
const releaseUrl =
|
||
updateInfo.value.releaseInfo?.html_url ||
|
||
'https://github.com/algerkong/AlgerMusicPlayer/releases/latest';
|
||
window.open(releaseUrl, '_blank');
|
||
closeModal();
|
||
}
|
||
};
|
||
|
||
onMounted(() => {
|
||
// 延迟检查更新,确保应用完全加载
|
||
setTimeout(() => {
|
||
checkForUpdates();
|
||
}, 2000);
|
||
});
|
||
</script>
|
||
|
||
<style scoped>
|
||
/* 动画 */
|
||
.update-modal-enter-active,
|
||
.update-modal-leave-active {
|
||
transition: opacity 0.3s ease;
|
||
}
|
||
|
||
.update-modal-enter-from,
|
||
.update-modal-leave-to {
|
||
opacity: 0;
|
||
}
|
||
|
||
.update-modal-enter-active .animate-slide-up,
|
||
.update-modal-leave-active .animate-slide-up {
|
||
transition: transform 0.3s cubic-bezier(0.32, 0.72, 0, 1);
|
||
}
|
||
|
||
.update-modal-enter-from .animate-slide-up {
|
||
transform: translateY(100%);
|
||
}
|
||
|
||
.update-modal-leave-to .animate-slide-up {
|
||
transform: translateY(100%);
|
||
}
|
||
|
||
/* 更新内容样式 */
|
||
:deep(h1) {
|
||
font-size: 1.25rem;
|
||
font-weight: 700;
|
||
margin-bottom: 0.75rem;
|
||
}
|
||
|
||
:deep(h2) {
|
||
font-size: 1.125rem;
|
||
font-weight: 600;
|
||
margin-bottom: 0.5rem;
|
||
}
|
||
|
||
:deep(h3) {
|
||
font-size: 1rem;
|
||
font-weight: 600;
|
||
margin-bottom: 0.5rem;
|
||
}
|
||
|
||
:deep(ul) {
|
||
list-style-type: disc;
|
||
padding-left: 1.25rem;
|
||
margin-bottom: 0.75rem;
|
||
}
|
||
|
||
:deep(ol) {
|
||
list-style-type: decimal;
|
||
padding-left: 1.25rem;
|
||
margin-bottom: 0.75rem;
|
||
}
|
||
|
||
:deep(li) {
|
||
margin-bottom: 0.25rem;
|
||
}
|
||
|
||
:deep(p) {
|
||
margin-bottom: 0.5rem;
|
||
}
|
||
|
||
:deep(code) {
|
||
padding: 0.125rem 0.375rem;
|
||
border-radius: 0.25rem;
|
||
background-color: rgba(0, 0, 0, 0.05);
|
||
font-size: 0.875rem;
|
||
}
|
||
|
||
:deep(a) {
|
||
color: #22c55e;
|
||
}
|
||
</style>
|