2026-03-24 22:51:10 +08:00
|
|
|
|
import 'dart:ui';
|
2026-03-22 00:21:21 +08:00
|
|
|
|
import 'package:flutter/material.dart';
|
2026-03-30 11:26:47 +08:00
|
|
|
|
import 'package:bot_toast/bot_toast.dart';
|
2026-03-22 02:14:55 +08:00
|
|
|
|
import 'package:flutter_localizations/flutter_localizations.dart';
|
2026-03-22 00:21:21 +08:00
|
|
|
|
import 'package:provider/provider.dart';
|
2026-03-23 00:43:19 +08:00
|
|
|
|
import 'package:provider/single_child_widget.dart';
|
2026-03-22 00:21:21 +08:00
|
|
|
|
import 'package:shared_preferences/shared_preferences.dart';
|
|
|
|
|
|
|
|
|
|
|
|
import 'core/network/dio_client.dart';
|
|
|
|
|
|
import 'core/storage/local_storage.dart';
|
2026-04-05 23:47:56 +08:00
|
|
|
|
import 'core/theme/app_theme.dart';
|
2026-04-04 21:19:29 +08:00
|
|
|
|
import 'core/event/app_event_bus.dart';
|
2026-03-22 00:21:21 +08:00
|
|
|
|
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-03-22 00:21:21 +08:00
|
|
|
|
import 'providers/auth_provider.dart';
|
|
|
|
|
|
import 'providers/market_provider.dart';
|
|
|
|
|
|
import 'providers/asset_provider.dart';
|
2026-03-23 14:12:00 +08:00
|
|
|
|
import 'providers/theme_provider.dart';
|
2026-03-22 00:21:21 +08:00
|
|
|
|
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-03-22 00:21:21 +08:00
|
|
|
|
|
|
|
|
|
|
void main() async {
|
2026-04-07 01:05:05 +08:00
|
|
|
|
// 確保 Flutter 綁定初始化
|
2026-03-22 00:21:21 +08:00
|
|
|
|
WidgetsFlutterBinding.ensureInitialized();
|
2026-03-22 02:14:55 +08:00
|
|
|
|
|
2026-04-07 01:05:05 +08:00
|
|
|
|
// 全局錯誤處理 - Flutter 框架錯誤
|
2026-03-24 18:54:01 +08:00
|
|
|
|
FlutterError.onError = (FlutterErrorDetails details) {
|
|
|
|
|
|
FlutterError.presentError(details);
|
|
|
|
|
|
debugPrint('Flutter Error: ${details.exception}');
|
|
|
|
|
|
debugPrint('Stack trace: ${details.stack}');
|
|
|
|
|
|
};
|
2026-03-22 00:21:21 +08:00
|
|
|
|
|
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 18:54:01 +08:00
|
|
|
|
|
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 18:54:01 +08:00
|
|
|
|
|
2026-03-24 22:51:10 +08:00
|
|
|
|
runApp(const MyApp());
|
2026-03-22 00:21:21 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
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(),
|
2026-03-23 14:12:00 +08:00
|
|
|
|
child: Consumer<ThemeProvider>(
|
|
|
|
|
|
builder: (context, themeProvider, _) {
|
2026-04-08 12:24:24 +08:00
|
|
|
|
return _buildMaterialApp(context, themeProvider.themeMode);
|
2026-03-23 14:12:00 +08:00
|
|
|
|
},
|
2026-03-23 00:08:19 +08:00
|
|
|
|
),
|
|
|
|
|
|
);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
List<SingleChildWidget> _buildProviders() {
|
|
|
|
|
|
final dioClient = DioClient();
|
|
|
|
|
|
|
|
|
|
|
|
return [
|
2026-04-07 01:05:05 +08:00
|
|
|
|
// Theme Provider (必須放在最前面)
|
2026-03-23 14:12:00 +08:00
|
|
|
|
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>(
|
2026-04-01 11:26:07 +08:00
|
|
|
|
create: (ctx) {
|
|
|
|
|
|
final authProvider = AuthProvider(ctx.read<UserService>());
|
2026-04-07 01:05:05 +08:00
|
|
|
|
// token 過期時,DioClient 回調 AuthProvider 強制登出
|
2026-04-01 11:26:07 +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-04 21:19:29 +08:00
|
|
|
|
ctx.read<AppEventBus>(),
|
2026-03-22 00:21:21 +08:00
|
|
|
|
),
|
2026-03-23 00:08:19 +08:00
|
|
|
|
),
|
|
|
|
|
|
];
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2026-04-06 01:58:08 +08:00
|
|
|
|
Widget _buildMaterialApp(BuildContext context, ThemeMode themeMode) {
|
2026-03-23 00:08:19 +08:00
|
|
|
|
return MaterialApp(
|
|
|
|
|
|
debugShowCheckedModeBanner: false,
|
2026-04-06 01:58:08 +08:00
|
|
|
|
theme: AppTheme.lightTheme,
|
|
|
|
|
|
darkTheme: AppTheme.darkTheme,
|
|
|
|
|
|
themeMode: themeMode,
|
2026-03-23 00:08:19 +08:00
|
|
|
|
localizationsDelegates: const [
|
|
|
|
|
|
GlobalMaterialLocalizations.delegate,
|
|
|
|
|
|
GlobalCupertinoLocalizations.delegate,
|
|
|
|
|
|
GlobalWidgetsLocalizations.delegate,
|
2026-03-22 00:21:21 +08:00
|
|
|
|
],
|
2026-03-30 11:26:47 +08:00
|
|
|
|
builder: (context, child) {
|
2026-04-07 01:05:05 +08:00
|
|
|
|
// 配置 BotToast 確保顯示在所有內容之上
|
2026-03-30 11:26:47 +08:00
|
|
|
|
final botToastBuilder = BotToastInit();
|
|
|
|
|
|
child = botToastBuilder(context, child);
|
|
|
|
|
|
return child;
|
|
|
|
|
|
},
|
|
|
|
|
|
navigatorObservers: [BotToastNavigatorObserver()],
|
2026-03-25 09:14:10 +08:00
|
|
|
|
initialRoute: '/',
|
|
|
|
|
|
routes: {
|
2026-03-25 23:56:23 +08:00
|
|
|
|
'/': (context) => const RootPage(),
|
2026-03-25 09:14:10 +08:00
|
|
|
|
'/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();
|
|
|
|
|
|
},
|
2026-03-22 00:21:21 +08:00
|
|
|
|
);
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|