Files
monisuo/flutter_monisuo/lib/main.dart

188 lines
6.0 KiB
Dart
Raw Normal View History

2026-03-24 22:51:10 +08:00
import 'dart:ui';
import 'package:flutter/material.dart';
import 'package:bot_toast/bot_toast.dart';
2026-03-22 02:14:55 +08:00
import 'package:flutter_localizations/flutter_localizations.dart';
import 'package:provider/provider.dart';
2026-03-23 00:43:19 +08:00
import 'package:provider/single_child_widget.dart';
import 'package:shared_preferences/shared_preferences.dart';
import 'core/network/dio_client.dart';
2026-04-18 10:02:47 +08:00
import 'core/network/domain_navigator.dart';
import 'core/storage/local_storage.dart';
2026-04-18 10:02:47 +08:00
import 'core/constants/api_endpoints.dart';
import 'core/theme/app_theme.dart';
2026-04-04 21:19:29 +08:00
import 'core/event/app_event_bus.dart';
import 'data/services/user_service.dart';
import 'data/services/market_service.dart';
import 'data/services/trade_service.dart';
import 'data/services/asset_service.dart';
import 'data/services/fund_service.dart';
2026-03-29 16:11:01 +08:00
import 'data/services/bonus_service.dart';
2026-04-21 08:09:45 +08:00
import 'data/services/config_service.dart';
import 'providers/auth_provider.dart';
import 'providers/market_provider.dart';
import 'providers/asset_provider.dart';
2026-04-21 08:09:45 +08:00
import 'providers/trade_provider.dart';
import 'providers/theme_provider.dart';
import 'ui/pages/auth/login_page.dart';
import 'ui/pages/main/main_page.dart';
2026-03-25 23:56:23 +08:00
import 'ui/pages/onboarding/onboarding_page.dart';
2026-04-21 08:09:45 +08:00
/// 全局導航 Key用於從任意位置跳轉到登錄頁
final GlobalKey<NavigatorState> navigatorKey = GlobalKey<NavigatorState>();
void main() async {
2026-04-07 01:05:05 +08:00
// 確保 Flutter 綁定初始化
WidgetsFlutterBinding.ensureInitialized();
2026-03-22 02:14:55 +08:00
2026-04-21 08:09:45 +08:00
// 防截图Android 端通过 FLAG_SECURE在 MainActivity 中已设置)
// Web 端通过 index.html 中的 JS/CSS 防护
// iOS 端暂无原生防截图能力Web 模式下同样由 JS 保护)
2026-04-07 01:05:05 +08:00
// 全局錯誤處理 - Flutter 框架錯誤
FlutterError.onError = (FlutterErrorDetails details) {
FlutterError.presentError(details);
};
2026-04-07 01:05:05 +08:00
// 全局錯誤處理 - 異步未捕獲錯誤
2026-03-24 22:51:10 +08:00
PlatformDispatcher.instance.onError = (error, stack) {
return true;
};
Provider.debugCheckInvalidValueType = null;
2026-03-24 22:51:10 +08:00
try {
await SharedPreferences.getInstance();
await LocalStorage.init();
2026-04-18 10:02:47 +08:00
// 域名導航Debug 用 localhostRelease 競速 CF Workers
final resolvedUrl = await DomainNavigator.init();
ApiEndpoints.init(resolvedUrl);
2026-03-24 22:51:10 +08:00
} catch (e, stack) {
2026-04-18 10:02:47 +08:00
// 域名解析失敗時使用兜底地址
ApiEndpoints.init(DomainNavigator.activeUrl);
2026-03-24 22:51:10 +08:00
}
2026-03-24 22:51:10 +08:00
runApp(const MyApp());
}
class MyApp extends StatelessWidget {
const MyApp({super.key});
@override
Widget build(BuildContext context) {
return MultiProvider(
2026-03-23 00:08:19 +08:00
providers: _buildProviders(),
child: Consumer<ThemeProvider>(
builder: (context, themeProvider, _) {
return _buildMaterialApp(context, themeProvider.themeMode);
},
2026-03-23 00:08:19 +08:00
),
);
}
List<SingleChildWidget> _buildProviders() {
2026-04-18 10:02:47 +08:00
final dioClient = DioClient();
2026-03-23 00:08:19 +08:00
return [
2026-04-07 01:05:05 +08:00
// Theme Provider (必須放在最前面)
ChangeNotifierProvider<ThemeProvider>(
create: (_) => ThemeProvider()..init(),
),
2026-03-23 00:08:19 +08:00
// Services
Provider<DioClient>.value(value: dioClient),
2026-04-04 21:19:29 +08:00
Provider<AppEventBus>(create: (_) => AppEventBus()),
2026-03-23 00:08:19 +08:00
Provider<UserService>(create: (_) => UserService(dioClient)),
Provider<MarketService>(create: (_) => MarketService(dioClient)),
Provider<TradeService>(create: (_) => TradeService(dioClient)),
Provider<AssetService>(create: (_) => AssetService(dioClient)),
Provider<FundService>(create: (_) => FundService(dioClient)),
2026-03-29 16:11:01 +08:00
Provider<BonusService>(create: (_) => BonusService(dioClient)),
2026-04-21 08:09:45 +08:00
Provider<ConfigService>(create: (_) => ConfigService(dioClient)),
2026-03-23 00:08:19 +08:00
// State Management
ChangeNotifierProvider<AuthProvider>(
create: (ctx) {
final authProvider = AuthProvider(ctx.read<UserService>());
2026-04-07 01:05:05 +08:00
// token 過期時DioClient 回調 AuthProvider 強制登出
2026-04-18 10:02:47 +08:00
dioClient.onUnauthorized = authProvider.forceLogout;
return authProvider;
},
2026-03-23 00:08:19 +08:00
),
ChangeNotifierProvider<MarketProvider>(
create: (ctx) => MarketProvider(ctx.read<MarketService>()),
),
ChangeNotifierProvider<AssetProvider>(
create: (ctx) => AssetProvider(
ctx.read<AssetService>(),
ctx.read<FundService>(),
2026-04-21 08:09:45 +08:00
ctx.read<TradeService>(),
2026-04-04 21:19:29 +08:00
ctx.read<AppEventBus>(),
),
2026-03-23 00:08:19 +08:00
),
2026-04-21 08:09:45 +08:00
ChangeNotifierProvider<TradeProvider>(
create: (ctx) => TradeProvider(
ctx.read<MarketService>(),
ctx.read<TradeService>(),
),
),
2026-03-23 00:08:19 +08:00
];
}
Widget _buildMaterialApp(BuildContext context, ThemeMode themeMode) {
2026-03-23 00:08:19 +08:00
return MaterialApp(
2026-04-21 08:09:45 +08:00
navigatorKey: navigatorKey,
2026-03-23 00:08:19 +08:00
debugShowCheckedModeBanner: false,
theme: AppTheme.lightTheme,
darkTheme: AppTheme.darkTheme,
themeMode: themeMode,
2026-03-23 00:08:19 +08:00
localizationsDelegates: const [
GlobalMaterialLocalizations.delegate,
GlobalCupertinoLocalizations.delegate,
GlobalWidgetsLocalizations.delegate,
],
builder: (context, child) {
2026-04-07 01:05:05 +08:00
// 配置 BotToast 確保顯示在所有內容之上
final botToastBuilder = BotToastInit();
child = botToastBuilder(context, child);
return child;
},
navigatorObservers: [BotToastNavigatorObserver()],
initialRoute: '/',
routes: {
2026-03-25 23:56:23 +08:00
'/': (context) => const RootPage(),
'/login': (context) => const LoginPage(),
'/main': (context) => const MainPage(),
},
2026-03-23 00:08:19 +08:00
);
}
2026-03-25 23:56:23 +08:00
}
2026-04-07 01:05:05 +08:00
/// 根頁面 - 決定顯示引導頁還是主頁面
2026-03-25 23:56:23 +08:00
class RootPage extends StatelessWidget {
const RootPage({super.key});
@override
Widget build(BuildContext context) {
2026-04-07 01:05:05 +08:00
// 檢查是否需要顯示引導頁
2026-03-25 23:56:23 +08:00
if (!LocalStorage.isOnboardingCompleted) {
return OnboardingPage(
onComplete: () {
Navigator.of(context).pushReplacementNamed('/login');
},
);
}
2026-03-23 00:08:19 +08:00
return Consumer<AuthProvider>(
builder: (context, auth, _) {
if (auth.isLoading) {
return const Scaffold(
body: Center(child: CircularProgressIndicator()),
2026-03-22 02:14:55 +08:00
);
2026-03-23 00:08:19 +08:00
}
return auth.isLoggedIn ? const MainPage() : const LoginPage();
},
);
}
}