626 lines
26 KiB
Dart
Executable File
626 lines
26 KiB
Dart
Executable File
import 'package:get/get.dart';
|
||
import 'dart:convert';
|
||
import 'package:openinstall_flutter_plugin/openinstall_flutter_plugin.dart';
|
||
|
||
import 'dart:io' show Platform, SocketException;
|
||
import 'dart:math';
|
||
import 'dart:async';
|
||
import 'package:dio/dio.dart';
|
||
import 'package:flutter/foundation.dart' show kDebugMode;
|
||
import 'package:kaer_with_panels/app/utils/kr_network_check.dart';
|
||
import 'package:kaer_with_panels/app/utils/kr_log_util.dart';
|
||
import 'package:kaer_with_panels/app/utils/kr_secure_storage.dart';
|
||
import 'package:kaer_with_panels/app/routes/app_pages.dart';
|
||
import 'package:kaer_with_panels/app/common/app_config.dart';
|
||
import 'package:kaer_with_panels/app/common/app_run_data.dart';
|
||
import 'package:kaer_with_panels/app/services/singbox_imp/kr_sing_box_imp.dart';
|
||
import 'package:kaer_with_panels/app/services/kr_site_config_service.dart';
|
||
import 'package:kaer_with_panels/app/services/kr_device_info_service.dart';
|
||
import 'package:kaer_with_panels/app/services/api_service/kr_auth_api.dart';
|
||
import 'package:kaer_with_panels/app/model/enum/kr_request_type.dart';
|
||
import 'package:kaer_with_panels/app/utils/kr_secure_storage.dart';
|
||
import 'package:kaer_with_panels/app/services/kr_subscribe_service.dart';
|
||
import 'package:kaer_with_panels/app/widgets/dialogs/hi_dialog.dart';
|
||
import 'package:kaer_with_panels/app/widgets/kr_app_text_style.dart';
|
||
import 'package:flutter/material.dart';
|
||
|
||
import 'package:kaer_with_panels/app/localization/app_translations.dart';
|
||
import 'package:kaer_with_panels/app/modules/kr_home/controllers/kr_home_controller.dart';
|
||
import 'package:kaer_with_panels/app/modules/kr_home/models/kr_home_views_status.dart';
|
||
import 'dart:async';
|
||
import 'package:flutter/foundation.dart';
|
||
import 'package:kaer_with_panels/app/utils/kr_init_log_collector.dart';
|
||
|
||
class KRSplashController extends GetxController {
|
||
// 🔧 新增:初始化日志收集器
|
||
final _initLog = KRInitLogCollector();
|
||
// 加载状态
|
||
final RxBool kr_isLoading = true.obs;
|
||
|
||
// 错误状态
|
||
final RxBool kr_hasError = false.obs;
|
||
|
||
// 错误信息
|
||
final RxString kr_errorMessage = ''.obs;
|
||
|
||
/// 订阅服务
|
||
final KRSubscribeService kr_subscribeService = KRSubscribeService();
|
||
|
||
// 倒计时
|
||
// final count = 0.obs;
|
||
// 是否正在加载
|
||
final isLoading = true.obs;
|
||
// // 是否初始化成功
|
||
// final isInitialized = false.obs;
|
||
|
||
// 启动开始时间(用于计算总耗时)
|
||
late final DateTime _startTime;
|
||
late DateTime _stepStartTime;
|
||
|
||
@override
|
||
void onInit() {
|
||
super.onInit();
|
||
|
||
// 记录启动开始时间
|
||
_startTime = DateTime.now();
|
||
_stepStartTime = _startTime;
|
||
|
||
// 使用 print 确保日志一定会输出
|
||
print('[SPLASH_TIMING] 🎬 KRSplashController onInit 被调用了!');
|
||
print('[SPLASH_TIMING] ⏰ 启动开始时间: ${_startTime.toIso8601String()}');
|
||
|
||
// 🔧 修复1.0:新增 - DEBUG模式下清理旧数据
|
||
// ⚠️ 仅在DEBUG模式下执行,防止误删生产环境用户数据
|
||
if (kDebugMode) {
|
||
KRLogUtil.kr_i('🧹 DEBUG模式:准备清理旧本地存储数据', tag: 'SplashController');
|
||
_kr_clearOldLocalData();
|
||
}
|
||
|
||
// 🔧 修复1:应用启动时清理域名检测静态状态
|
||
print('🧹 清理域名检测状态...');
|
||
KRDomain.kr_resetDomainState();
|
||
|
||
_initializeAndStart();
|
||
}
|
||
|
||
/// 🔧 新增:初始化日志收集器并启动主流程
|
||
Future<void> _initializeAndStart() async {
|
||
try {
|
||
// 等待日志收集器初始化完成
|
||
await _initLog.initialize();
|
||
_initLog.logPhaseStart('应用启动');
|
||
_initLog.log('KRSplashController.onInit 被调用', tag: 'Splash');
|
||
|
||
if (kDebugMode) {
|
||
print('✅ 日志收集器初始化完成');
|
||
print('🔧 立即开始主初始化流程...');
|
||
}
|
||
|
||
// 开始主初始化流程
|
||
_kr_initialize();
|
||
} catch (e) {
|
||
if (kDebugMode) {
|
||
print('❌ 日志收集器初始化失败: $e,继续执行主流程');
|
||
}
|
||
// 即使日志收集器失败,也要继续主流程
|
||
_kr_initialize();
|
||
}
|
||
}
|
||
|
||
// 记录步骤耗时的辅助方法
|
||
void _logStepTiming(String stepName) {
|
||
final now = DateTime.now();
|
||
final stepDuration = now.difference(_stepStartTime);
|
||
final totalDuration = now.difference(_startTime);
|
||
final stepMs = stepDuration.inMilliseconds;
|
||
final totalMs = totalDuration.inMilliseconds;
|
||
|
||
print('[SPLASH_TIMING] ⏱️ $stepName 耗时: ${stepMs}ms | 累计: ${totalMs}ms');
|
||
KRLogUtil.kr_i(
|
||
'[SPLASH_TIMING] ⏱️ $stepName 耗时: ${stepMs}ms | 累计: ${totalMs}ms',
|
||
tag: 'SplashController');
|
||
|
||
_stepStartTime = now; // 更新步骤开始时间
|
||
}
|
||
|
||
/// 初始化网站配置(后台执行,不阻塞主流程)
|
||
Future<void> _kr_initSiteConfig() async {
|
||
try {
|
||
// 🔧 Android 15 优化:延长超时保护到25秒,匹配网络请求超时(20秒 + 5秒缓冲)
|
||
await Future.any([
|
||
_executeSiteConfigInit(),
|
||
Future.delayed(const Duration(seconds: 25), () {
|
||
throw TimeoutException('网站配置加载超时(25秒)');
|
||
}),
|
||
]);
|
||
} on TimeoutException catch (e) {
|
||
print('⏱️ 网站配置加载超时,应用将使用默认配置: $e');
|
||
KRLogUtil.kr_w('⏱️ 网站配置加载超时: $e', tag: 'SplashController');
|
||
} catch (e) {
|
||
print('❌ 网站配置初始化异常: $e');
|
||
KRLogUtil.kr_e('❌ 网站配置初始化异常: $e', tag: 'SplashController');
|
||
}
|
||
|
||
print('━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━');
|
||
KRLogUtil.kr_i('━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━━',
|
||
tag: 'SplashController');
|
||
}
|
||
|
||
/// 执行网站配置初始化
|
||
Future<void> _executeSiteConfigInit() async {
|
||
print('📞 准备调用 KRDomain.kr_loadBaseDomain()...');
|
||
await KRDomain.kr_loadBaseDomain();
|
||
print('📞 KRDomain.kr_loadBaseDomain() 完成');
|
||
}
|
||
|
||
|
||
Future<void> _kr_initialize() async {
|
||
try {
|
||
KRLogUtil.kr_i('🔧 开始执行 _kr_initialize', tag: 'SplashController');
|
||
|
||
// 🔧 Android 15 优化:添加总体超时保护(15秒),更快响应
|
||
await Future.any([
|
||
_executeInitialization(),
|
||
Future.delayed(const Duration(seconds: 15), () {
|
||
throw TimeoutException('初始化超时:15秒内未完成');
|
||
}),
|
||
]);
|
||
|
||
// 静默邀请初始化(在主流程完成后进行,不影响启动速度)
|
||
_kr_initOpenInstall();
|
||
|
||
_initLog.logPhaseEnd('主初始化流程', success: true);
|
||
} on TimeoutException catch (e) {
|
||
// 🔧 P2优化:超时错误提供更友好的提示
|
||
KRLogUtil.kr_e('⏱️ 初始化超时: $e', tag: 'SplashController');
|
||
print('⏱️ 初始化超时,直接跳转到主页');
|
||
// 超时后直接跳转到主页,让用户可以手动重试
|
||
// Get.offAllNamed(Routes.KR_HOME);
|
||
HIDialog.show(
|
||
message: '初始化超时,请检查网络或重试',
|
||
confirmText: '重试',
|
||
preventBackDismiss: true,
|
||
onConfirm: () async {
|
||
await _kr_initialize(); // 递归重试
|
||
},
|
||
);
|
||
return;
|
||
} on SocketException catch (e) {
|
||
// 🔧 P2优化:网络连接错误
|
||
KRLogUtil.kr_e('❌ 网络连接错误: $e', tag: 'SplashController');
|
||
_handleError('网络连接失败', '请检查您的网络连接是否正常,然后重试。');
|
||
} on DioException catch (e) {
|
||
// 🔧 P2优化:HTTP请求错误
|
||
KRLogUtil.kr_e('❌ HTTP请求错误: ${e.type} - ${e.message}',
|
||
tag: 'SplashController');
|
||
final errorMsg = _getDioErrorMessage(e);
|
||
_handleError('服务请求失败', errorMsg);
|
||
} catch (e) {
|
||
// 🔧 P2优化:其他未知错误
|
||
KRLogUtil.kr_e('❌ 未知初始化异常: $e', tag: 'SplashController');
|
||
_handleError('初始化失败', '应用启动时发生错误,请尝试重启应用或清除缓存。');
|
||
}
|
||
}
|
||
|
||
/// 🔧 Android 15 新增:最小化初始化(降级策略)
|
||
/// 确保即使网络失败,应用也能启动到可用状态
|
||
Future<void> _executeMinimalInitialization() async {
|
||
try {
|
||
_initLog.logPhaseStart('降级初始化(Minimal Initialization)');
|
||
_initLog.logWarning('网络初始化失败,执行降级策略', tag: 'Splash');
|
||
KRLogUtil.kr_i('🛡️ 执行最小化初始化(降级模式)', tag: 'SplashController');
|
||
if (kDebugMode) {
|
||
print('🛡️ 执行最小化初始化(降级模式)');
|
||
}
|
||
|
||
// 1. 设置默认域名(确保应用有可用域名)
|
||
try {
|
||
_initLog.log('步骤1: 设置默认域名', tag: 'Minimal');
|
||
if (KRDomain.kr_currentDomain.isEmpty &&
|
||
KRDomain.kr_baseDomains.isNotEmpty) {
|
||
KRDomain.kr_currentDomain = KRDomain.kr_baseDomains[0];
|
||
_initLog.logSuccess('设置默认域名: ${KRDomain.kr_currentDomain}',
|
||
tag: 'Minimal');
|
||
KRLogUtil.kr_i('✅ 设置默认域名: ${KRDomain.kr_currentDomain}',
|
||
tag: 'SplashController');
|
||
if (kDebugMode) {
|
||
print('✅ 设置默认域名: ${KRDomain.kr_currentDomain}');
|
||
}
|
||
} else {
|
||
_initLog.log('当前域名: ${KRDomain.kr_currentDomain}', tag: 'Minimal');
|
||
}
|
||
} catch (e) {
|
||
_initLog.logError('设置默认域名失败', tag: 'Minimal', error: e);
|
||
KRLogUtil.kr_w('⚠️ 设置默认域名失败: $e', tag: 'SplashController');
|
||
}
|
||
|
||
// 2. 初始化 SingBox(核心功能,不依赖网络)
|
||
try {
|
||
_initLog.log('步骤2: 初始化 SingBox', tag: 'Minimal');
|
||
await KRSingBoxImp.instance.init().timeout(
|
||
const Duration(seconds: 3),
|
||
onTimeout: () {
|
||
_initLog.logWarning('SingBox 初始化超时(3秒)', tag: 'Minimal');
|
||
KRLogUtil.kr_w('SingBox 初始化超时', tag: 'SplashController');
|
||
if (kDebugMode) {
|
||
print('⏱️ SingBox 初始化超时');
|
||
}
|
||
},
|
||
);
|
||
_initLog.logSuccess('SingBox 初始化完成', tag: 'Minimal');
|
||
KRLogUtil.kr_i('✅ SingBox 初始化完成', tag: 'SplashController');
|
||
if (kDebugMode) {
|
||
print('✅ SingBox 初始化完成');
|
||
}
|
||
} catch (e) {
|
||
_initLog.logError('SingBox 初始化失败', tag: 'Minimal', error: e);
|
||
KRLogUtil.kr_w('⚠️ SingBox 初始化失败: $e', tag: 'SplashController');
|
||
if (kDebugMode) {
|
||
print('⚠️ SingBox 初始化失败: $e');
|
||
}
|
||
}
|
||
|
||
// 3. 尝试加载本地用户信息(不依赖网络)
|
||
try {
|
||
_initLog.log('步骤3: 加载本地用户信息', tag: 'Minimal');
|
||
await KRAppRunData.getInstance().kr_initializeUserInfo().timeout(
|
||
const Duration(seconds: 2),
|
||
onTimeout: () {
|
||
_initLog.logWarning('用户信息加载超时(2秒)', tag: 'Minimal');
|
||
KRLogUtil.kr_w('用户信息加载超时', tag: 'SplashController');
|
||
if (kDebugMode) {
|
||
print('⏱️ 用户信息加载超时');
|
||
}
|
||
},
|
||
);
|
||
_initLog.logSuccess('本地用户信息加载完成', tag: 'Minimal');
|
||
KRLogUtil.kr_i('✅ 本地用户信息加载完成', tag: 'SplashController');
|
||
if (kDebugMode) {
|
||
print('✅ 本地用户信息加载完成');
|
||
}
|
||
} catch (e) {
|
||
_initLog.logError('用户信息加载失败', tag: 'Minimal', error: e);
|
||
KRLogUtil.kr_w('⚠️ 用户信息加载失败: $e', tag: 'SplashController');
|
||
if (kDebugMode) {
|
||
print('⚠️ 用户信息加载失败: $e');
|
||
}
|
||
}
|
||
|
||
_initLog.logPhaseEnd('降级初始化(Minimal Initialization)', success: true);
|
||
_initLog.log('应用将以降级模式启动,用户将看到基本UI', tag: 'Minimal');
|
||
_initLog.log('预期显示:订阅卡片 + 连接选项', tag: 'Minimal');
|
||
|
||
KRLogUtil.kr_i('✅ 最小化初始化完成,应用将以降级模式启动', tag: 'SplashController');
|
||
if (kDebugMode) {
|
||
print('✅ 最小化初始化完成,应用将以降级模式启动');
|
||
}
|
||
if (kDebugMode) {
|
||
print('📱 用户将看到基本UI(订阅卡片 + 连接选项)');
|
||
}
|
||
|
||
// 🔧 最小化初始化也不关闭日志
|
||
// await _initLog.finalize(); // ❌ 注释掉
|
||
if (kDebugMode && _initLog.getLogFilePath() != null) {
|
||
print('📁 初始化日志文件(保持打开): ${_initLog.getLogFilePath()}');
|
||
}
|
||
} catch (e) {
|
||
_initLog.logError('最小化初始化也失败', tag: 'Minimal', error: e);
|
||
KRLogUtil.kr_e('❌ 最小化初始化也失败: $e', tag: 'SplashController');
|
||
if (kDebugMode) {
|
||
print('❌ 最小化初始化也失败: $e');
|
||
}
|
||
// 即使降级初始化失败也不关闭日志
|
||
// await _initLog.finalize(); // ❌ 注释掉
|
||
}
|
||
}
|
||
|
||
/// 处理错误并显示
|
||
void _handleError(String title, String message) {
|
||
kr_hasError.value = true;
|
||
kr_errorMessage.value = '$title\n\n$message';
|
||
}
|
||
|
||
/// 获取Dio错误的友好提示
|
||
String _getDioErrorMessage(DioException e) {
|
||
switch (e.type) {
|
||
case DioExceptionType.connectionTimeout:
|
||
case DioExceptionType.sendTimeout:
|
||
case DioExceptionType.receiveTimeout:
|
||
return '请求超时,请检查网络连接后重试。';
|
||
case DioExceptionType.badResponse:
|
||
final statusCode = e.response?.statusCode;
|
||
if (statusCode == 404) {
|
||
return '服务地址未找到(404),请联系管理员。';
|
||
} else if (statusCode == 500) {
|
||
return '服务器内部错误(500),请稍后重试。';
|
||
} else if (statusCode == 401 || statusCode == 403) {
|
||
return '访问被拒绝($statusCode),请检查配置。';
|
||
}
|
||
return '服务器返回错误($statusCode),请稍后重试。';
|
||
case DioExceptionType.cancel:
|
||
return '请求已取消';
|
||
case DioExceptionType.connectionError:
|
||
return '无法连接到服务器,请检查网络设置。';
|
||
default:
|
||
return '网络请求失败,请检查网络后重试。';
|
||
}
|
||
}
|
||
|
||
// 🔧 P0修复:将原来的初始化逻辑提取到单独方法
|
||
Future<void> _executeInitialization() async {
|
||
_initLog.log('🔍 开始执行核心初始化流程', tag: 'Init');
|
||
|
||
// 只在手机端检查网络权限
|
||
if (Platform.isIOS || Platform.isAndroid) {
|
||
_initLog.log('📱 检测到移动平台,检查网络权限', tag: 'Init');
|
||
KRLogUtil.kr_i('📱 移动平台,检查网络权限...', tag: 'SplashController');
|
||
|
||
try {
|
||
_initLog.log('开始网络权限检查(超时8秒)', tag: 'Init');
|
||
// 🔧 Android 15 优化:网络权限检查添加 8 秒超时
|
||
final bool hasNetworkPermission = await KRNetworkCheck.kr_initialize(
|
||
Get.context!,
|
||
onPermissionGranted: () async {
|
||
_initLog.logSuccess('网络权限已授予', tag: 'Init');
|
||
await _kr_continueInitialization();
|
||
},
|
||
).timeout(
|
||
const Duration(seconds: 8),
|
||
onTimeout: () {
|
||
_initLog.logWarning('网络权限检查超时(8秒)', tag: 'Init');
|
||
KRLogUtil.kr_w('⏱️ 网络权限检查超时,执行降级初始化', tag: 'SplashController');
|
||
return false;
|
||
},
|
||
);
|
||
|
||
if (!hasNetworkPermission) {
|
||
// 🔧 Android 15 优化:权限检查失败时执行降级策略而非显示错误
|
||
_initLog.logWarning('网络权限检查失败或超时,执行降级初始化', tag: 'Init');
|
||
KRLogUtil.kr_w('⚠️ 网络权限检查失败或超时,执行降级初始化', tag: 'SplashController');
|
||
await _executeMinimalInitialization();
|
||
Get.offAllNamed(Routes.KR_HOME);
|
||
return;
|
||
}
|
||
|
||
_initLog.logSuccess('网络权限检查通过', tag: 'Init');
|
||
} catch (e) {
|
||
_initLog.logError('网络权限检查异常,执行降级初始化', tag: 'Init', error: e);
|
||
KRLogUtil.kr_w('⚠️ 网络权限检查异常: $e,执行降级初始化', tag: 'SplashController');
|
||
await _executeMinimalInitialization();
|
||
Get.offAllNamed(Routes.KR_HOME);
|
||
return;
|
||
}
|
||
} else {
|
||
// 非手机端直接继续初始化
|
||
_initLog.log('💻 检测到桌面平台,直接执行初始化', tag: 'Init');
|
||
KRLogUtil.kr_i('💻 桌面平台,直接执行初始化', tag: 'SplashController');
|
||
await _kr_continueInitialization();
|
||
}
|
||
}
|
||
|
||
Future<void> _kr_continueInitialization() async {
|
||
try {
|
||
KRLogUtil.kr_i('🚀 启动页主流程开始...', tag: 'SplashController');
|
||
// 🔧 Android 15 优化:网络连接检查改为非阻塞
|
||
if (Platform.isIOS || Platform.isAndroid) {
|
||
_initLog.log('📱 检查网络连接状态(超时5秒)', tag: 'Continue');
|
||
KRLogUtil.kr_i('📱 检查网络连接...', tag: 'SplashController');
|
||
try {
|
||
// 添加5秒超时保护
|
||
final bool isConnected =
|
||
await KRNetworkCheck.kr_checkNetworkConnection().timeout(
|
||
const Duration(seconds: 5),
|
||
onTimeout: () {
|
||
_initLog.logWarning('网络连接检查超时(5秒),继续初始化', tag: 'Continue');
|
||
KRLogUtil.kr_w('⏱️ 网络连接检查超时,继续初始化', tag: 'SplashController');
|
||
return true; // 超时时允许继续
|
||
},
|
||
);
|
||
|
||
if (isConnected) {
|
||
_initLog.logSuccess('网络连接正常', tag: 'Continue');
|
||
KRLogUtil.kr_i('✅ 网络连接正常', tag: 'SplashController');
|
||
} else {
|
||
// 🔧 Android 15 优化:网络未连接也允许继续,改为警告而非错误
|
||
_initLog.logWarning('网络未连接,应用将以离线模式启动', tag: 'Continue');
|
||
KRLogUtil.kr_w('⚠️ 网络未连接,应用将以离线模式启动', tag: 'SplashController');
|
||
}
|
||
} catch (e) {
|
||
// 🔧 Android 15 优化:网络检查异常不阻塞启动
|
||
_initLog.logError('网络检查异常,继续初始化', tag: 'Continue', error: e);
|
||
KRLogUtil.kr_w('⚠️ 网络检查异常: $e,继续初始化', tag: 'SplashController');
|
||
}
|
||
} else {
|
||
_initLog.log('💻 桌面平台,跳过网络连接检查', tag: 'Continue');
|
||
KRLogUtil.kr_i('💻 桌面平台,跳过网络连接检查', tag: 'SplashController');
|
||
}
|
||
|
||
// 🔧 关键修复:在设备登录前先初始化站点配置,确保域名可用
|
||
// 这样如果主域名不可用,initSiteConfig 会切换到备用域名
|
||
// 后续的设备登录就能使用正确的域名
|
||
_logStepTiming('开始站点配置初始化');
|
||
await _kr_initSiteConfig();
|
||
_logStepTiming('站点配置初始化完成');
|
||
|
||
_logStepTiming('开始设备登录检查');
|
||
// 5️⃣ 执行设备登录
|
||
final success =
|
||
await KRAppRunData.getInstance().kr_checkAndPerformDeviceLogin();
|
||
|
||
if (!success) {
|
||
// 设备登录失败 → 提示用户重试
|
||
HIDialog.show(
|
||
message: '设备登录失败,请检查网络或重试',
|
||
confirmText: '重试',
|
||
preventBackDismiss: true,
|
||
onConfirm: () async {
|
||
await _kr_initialize(); // 递归重试
|
||
},
|
||
);
|
||
return;
|
||
}
|
||
_logStepTiming('设备登录检查完成');
|
||
|
||
// 初始化配置 (注意:这里再次调用init会导致重复,但AppConfig内部有防重,
|
||
// 不过为了逻辑清晰,既然上面已经调用了_kr_initSiteConfig(获取了KRSiteConfig),
|
||
// 下面AppConfig().initConfig主要是做AppConfig层面的配置加载,可能包含重复的请求。
|
||
// 鉴于 AppConfig().initConfig 也会调用 API,为了保险起见,
|
||
// 我们可以让 AppConfig().initConfig 使用已经切换好的域名。)
|
||
|
||
_initLog.log('⚙️ 开始初始化应用配置(域名加载等)', tag: 'Continue');
|
||
KRLogUtil.kr_i('⚙️ 开始初始化应用配置...', tag: 'SplashController');
|
||
await AppConfig().initConfig(
|
||
onSuccess: () async {
|
||
// 配置初始化成功,继续后续步骤
|
||
_initLog.logSuccess('应用配置初始化成功', tag: 'Continue');
|
||
KRLogUtil.kr_i('✅ 应用配置初始化成功,继续后续步骤', tag: 'SplashController');
|
||
await _kr_continueAfterConfig();
|
||
},
|
||
);
|
||
} catch (e) {
|
||
// 🔧 Android 15 优化:配置初始化失败时执行降级策略
|
||
_initLog.logError('启动页初始化异常,执行降级策略', tag: 'Continue', error: e);
|
||
KRLogUtil.kr_w('⚠️ 启动页初始化异常,执行降级策略: $e', tag: 'SplashController');
|
||
await _executeMinimalInitialization();
|
||
Get.offAllNamed(Routes.KR_HOME);
|
||
}
|
||
}
|
||
|
||
// 配置初始化成功后的后续步骤
|
||
Future<void> _kr_continueAfterConfig() async {
|
||
try {
|
||
// 初始化SingBox(必须先于订阅服务,避免目录未初始化)
|
||
_logStepTiming('开始SingBox初始化');
|
||
await KRSingBoxImp.instance.init();
|
||
_logStepTiming('SingBox初始化完成');
|
||
|
||
// 验证登录状态是否已正确设置
|
||
final loginStatus = KRAppRunData.getInstance().kr_isLogin.value;
|
||
final token = KRAppRunData.getInstance().kr_token;
|
||
final hasToken = token != null && token.isNotEmpty;
|
||
|
||
if (hasToken) {
|
||
print('🎫 Token前缀: ${token.substring(0, min(20, token.length))}...');
|
||
}
|
||
KRLogUtil.kr_i('🎯 准备进入主页', tag: 'SplashController');
|
||
KRLogUtil.kr_i('📊 最终登录状态: $loginStatus', tag: 'SplashController');
|
||
KRLogUtil.kr_i('🎫 Token存在: $hasToken', tag: 'SplashController');
|
||
|
||
// 🔧 记录最终状态到日志
|
||
_initLog.logSeparator();
|
||
_initLog.log('准备进入主页', tag: 'Splash');
|
||
_initLog.log('最终登录状态: $loginStatus', tag: 'Splash');
|
||
_initLog.log('Token存在: $hasToken', tag: 'Splash');
|
||
_initLog.logSuccess('正常初始化流程完成', tag: 'Splash');
|
||
|
||
// 🔧 关键修复:不要在这里关闭日志文件!
|
||
// 日志文件需要保持打开状态,让 HomeController 也能写入日志
|
||
// 日志文件将在 HomeController 初始化完成后关闭
|
||
// await _initLog.finalize(); // ❌ 注释掉
|
||
if (kDebugMode && _initLog.getLogFilePath() != null) {
|
||
print('📁 初始化日志文件(保持打开): ${_initLog.getLogFilePath()}');
|
||
}
|
||
|
||
// 直接导航到主页(无论是否登录,主页会根据登录状态显示不同内容)
|
||
_logStepTiming('开始页面导航');
|
||
if (loginStatus) {
|
||
// 直接导航到主页
|
||
Get.offAllNamed(Routes.KR_HOME);
|
||
} else {
|
||
kr_hasError.value = true;
|
||
kr_errorMessage.value =
|
||
'登录:${AppTranslations.kr_splash.kr_initializationFailed}';
|
||
}
|
||
_logStepTiming('页面导航完成');
|
||
} catch (e) {
|
||
// 计算错误时的总耗时
|
||
final endTime = DateTime.now();
|
||
final totalDuration = endTime.difference(_startTime);
|
||
final totalMs = totalDuration.inMilliseconds;
|
||
|
||
print('❌ 启动失败时间: ${endTime.toIso8601String()}');
|
||
print(
|
||
'🕐 启动失败总耗时: ${totalMs}ms (${totalDuration.inSeconds}.${(totalMs % 1000).toString().padLeft(3, '0')}s)');
|
||
|
||
// 后续步骤失败,显示错误信息
|
||
_initLog.logError('启动初始化失败', tag: 'Splash', error: e);
|
||
KRLogUtil.kr_e('启动初始化失败: $e', tag: 'SplashController');
|
||
KRLogUtil.kr_e('⏰ 启动失败总耗时: ${totalMs}ms', tag: 'SplashController');
|
||
kr_hasError.value = true;
|
||
kr_errorMessage.value =
|
||
'${AppTranslations.kr_splash.kr_initializationFailed}$e';
|
||
|
||
// 🔧 错误情况下也不关闭日志,让后续的错误处理也能写入
|
||
// await _initLog.finalize(); // ❌ 注释掉
|
||
}
|
||
}
|
||
|
||
// 重试按钮点击事件
|
||
void kr_retry() {
|
||
kr_hasError.value = false;
|
||
kr_errorMessage.value = '';
|
||
_kr_initialize();
|
||
}
|
||
|
||
/// 初始化 OpenInstall
|
||
void _kr_initOpenInstall() {
|
||
if (Platform.isAndroid || Platform.isIOS) {
|
||
try {
|
||
KRLogUtil.kr_i('🚀 初始化 OpenInstall...', tag: 'SplashController');
|
||
OpeninstallFlutterPlugin().init((data) async {
|
||
KRLogUtil.kr_i('收到 OpenInstall 唤醒数据: $data', tag: 'SplashController');
|
||
// 处理唤醒时的邀请绑定
|
||
KRAppRunData().kr_handleOpenInstallData(data);
|
||
});
|
||
} catch (e) {
|
||
KRLogUtil.kr_e('OpenInstall 初始化失败: $e', tag: 'SplashController');
|
||
}
|
||
}
|
||
}
|
||
|
||
// 🔧 P3优化:跳过初始化,直接进入主页
|
||
void kr_skipInitialization() {
|
||
KRLogUtil.kr_i('⏭️ 用户选择跳过初始化', tag: 'SplashController');
|
||
print('⏭️ 用户跳过初始化,直接进入主页');
|
||
|
||
// 重置状态
|
||
kr_hasError.value = false;
|
||
kr_errorMessage.value = '';
|
||
kr_isLoading.value = false;
|
||
|
||
// 直接跳转到主页
|
||
Get.offAllNamed(Routes.KR_HOME);
|
||
}
|
||
|
||
/// 🔧 修复1.1:清理旧的本地存储数据(DEBUG模式专用)
|
||
/// 防止旧的测试账号在新安装时被恢复
|
||
Future<void> _kr_clearOldLocalData() async {
|
||
try {
|
||
// KRLogUtil.kr_i('🧹 开始清理旧本地存储数据...', tag: 'SplashController');
|
||
|
||
// 清理用户信息存储
|
||
await KRSecureStorage().kr_deleteData(key: 'USER_INFO');
|
||
// KRLogUtil.kr_i('✅ 已清理USER_INFO', tag: 'SplashController');
|
||
|
||
// 清理设备登录状态
|
||
await KRSecureStorage().kr_deleteData(key: 'DEVICE_INFO');
|
||
// KRLogUtil.kr_i('✅ 已清理DEVICE_INFO', tag: 'SplashController');
|
||
|
||
KRLogUtil.kr_i('✅ 旧本地存储数据已全部清理', tag: 'SplashController');
|
||
} catch (e) {
|
||
KRLogUtil.kr_w('⚠️ 清理旧数据过程中出错: $e', tag: 'SplashController');
|
||
// 继续执行,不阻塞初始化流程
|
||
}
|
||
}
|
||
|
||
@override
|
||
void onReady() {
|
||
super.onReady();
|
||
}
|
||
|
||
@override
|
||
void onClose() {
|
||
super.onClose();
|
||
}
|
||
}
|