Files
monisuo/flutter_monisuo/lib/main.dart

172 lines
5.5 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';
import 'core/storage/local_storage.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';
import 'providers/auth_provider.dart';
import 'providers/market_provider.dart';
import 'providers/asset_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';
/// 全局導航 Key用於 token 過期時全局跳轉登錄頁
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-07 01:05:05 +08:00
// 全局錯誤處理 - Flutter 框架錯誤
FlutterError.onError = (FlutterErrorDetails details) {
FlutterError.presentError(details);
debugPrint('Flutter Error: ${details.exception}');
debugPrint('Stack trace: ${details.stack}');
};
2026-04-07 01:05:05 +08:00
// 全局錯誤處理 - 異步未捕獲錯誤
2026-03-24 22:51:10 +08:00
PlatformDispatcher.instance.onError = (error, stack) {
debugPrint('Uncaught error: $error');
debugPrint('Stack: $stack');
return true;
};
Provider.debugCheckInvalidValueType = null;
2026-03-24 22:51:10 +08:00
try {
await SharedPreferences.getInstance();
await LocalStorage.init();
debugPrint('App initialized successfully');
} catch (e, stack) {
debugPrint('Initialization error: $e');
debugPrint('Stack: $stack');
}
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() {
final dioClient = DioClient(navigatorKey: navigatorKey);
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-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 強制登出
dioClient.onForceLogout = 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-04 21:19:29 +08:00
ctx.read<AppEventBus>(),
),
2026-03-23 00:08:19 +08:00
),
];
}
Widget _buildMaterialApp(BuildContext context, ThemeMode themeMode) {
2026-03-23 00:08:19 +08:00
return MaterialApp(
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();
},
);
}
}