mirror of
https://github.com/fawney19/Aether.git
synced 2026-01-09 03:02:26 +08:00
Initial commit
This commit is contained in:
100
frontend/src/App.vue
Normal file
100
frontend/src/App.vue
Normal file
@@ -0,0 +1,100 @@
|
||||
<template>
|
||||
<RouterView />
|
||||
<ToastContainer />
|
||||
<ConfirmContainer />
|
||||
</template>
|
||||
|
||||
<script setup lang="ts">
|
||||
import { onMounted, onErrorCaptured } from 'vue'
|
||||
import { useAuthStore } from '@/stores/auth'
|
||||
import ToastContainer from '@/components/ToastContainer.vue'
|
||||
import ConfirmContainer from '@/components/ConfirmContainer.vue'
|
||||
import apiClient from '@/api/client'
|
||||
import { NETWORK_CONFIG, AUTH_CONFIG } from '@/config/constants'
|
||||
import { log } from '@/utils/logger'
|
||||
|
||||
const authStore = useAuthStore()
|
||||
|
||||
// 立即检查token,如果存在就设置到store中
|
||||
const storedToken = apiClient.getToken()
|
||||
if (storedToken) {
|
||||
authStore.token = storedToken
|
||||
}
|
||||
|
||||
// 全局错误处理器 - 只处理特定错误,避免完全吞掉所有错误
|
||||
onErrorCaptured((error: Error) => {
|
||||
log.error('Error captured in component', error)
|
||||
// 对于非关键错误,不阻止传播
|
||||
return true
|
||||
})
|
||||
|
||||
// 统一的模块加载错误处理
|
||||
let moduleLoadFailureCount = 0
|
||||
|
||||
if (typeof window !== 'undefined') {
|
||||
// 处理未捕获的 Promise 拒绝
|
||||
window.addEventListener('unhandledrejection', (event) => {
|
||||
const error = event.reason
|
||||
|
||||
// 只处理模块加载失败的情况
|
||||
if (error?.message?.includes('Failed to fetch dynamically imported module')) {
|
||||
event.preventDefault() // 阻止控制台显示这个特定错误
|
||||
|
||||
if (moduleLoadFailureCount < NETWORK_CONFIG.MODULE_LOAD_RETRY_LIMIT) {
|
||||
moduleLoadFailureCount++
|
||||
log.info(`模块加载失败,尝试刷新页面 (${moduleLoadFailureCount}/${NETWORK_CONFIG.MODULE_LOAD_RETRY_LIMIT})`)
|
||||
window.location.reload()
|
||||
} else {
|
||||
// 超过最大重试次数,显示友好提示
|
||||
alert('页面加载失败,请手动刷新浏览器。如问题持续,请清除浏览器缓存后重试。')
|
||||
}
|
||||
return
|
||||
}
|
||||
|
||||
// 其他 Promise 错误记录日志
|
||||
log.error('Unhandled promise rejection', event.reason)
|
||||
})
|
||||
|
||||
// 处理全局错误
|
||||
window.addEventListener('error', (event) => {
|
||||
// 过滤掉常见的无害警告
|
||||
const harmlessWarnings = [
|
||||
'ResizeObserver loop completed with undelivered notifications',
|
||||
'ResizeObserver loop limit exceeded'
|
||||
]
|
||||
|
||||
const isHarmless = harmlessWarnings.some(warning =>
|
||||
event.message?.includes(warning)
|
||||
)
|
||||
|
||||
if (isHarmless) {
|
||||
event.preventDefault()
|
||||
return
|
||||
}
|
||||
|
||||
// 记录其他错误
|
||||
if (event.error) {
|
||||
log.error('Global error', event.error)
|
||||
} else {
|
||||
log.warn('Global error event', {
|
||||
message: event.message,
|
||||
filename: event.filename,
|
||||
lineno: event.lineno,
|
||||
colno: event.colno
|
||||
})
|
||||
}
|
||||
})
|
||||
}
|
||||
|
||||
onMounted(async () => {
|
||||
// 延迟检查认证状态,让页面先加载
|
||||
setTimeout(async () => {
|
||||
try {
|
||||
await authStore.checkAuth()
|
||||
} catch (error) {
|
||||
// 即使checkAuth失败,也不要做任何会导致退出的操作
|
||||
log.warn('Auth check failed, but keeping session', error)
|
||||
}
|
||||
}, AUTH_CONFIG.TOKEN_REFRESH_INTERVAL)
|
||||
})
|
||||
</script>
|
||||
Reference in New Issue
Block a user