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-22 02:14:55 +08:00
|
|
|
import 'package:shadcn_ui/shadcn_ui.dart';
|
|
|
|
|
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-03-23 02:43:35 +08:00
|
|
|
import 'core/theme/app_color_scheme.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-03-24 18:54:01 +08:00
|
|
|
// 确保 Flutter 绑定初始化
|
2026-03-22 00:21:21 +08:00
|
|
|
WidgetsFlutterBinding.ensureInitialized();
|
2026-03-22 02:14:55 +08:00
|
|
|
|
2026-03-24 22:51:10 +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-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-03-25 00:47:37 +08:00
|
|
|
return ShadApp.custom(
|
|
|
|
|
themeMode: themeProvider.themeMode,
|
|
|
|
|
theme: createLightShadTheme(),
|
|
|
|
|
darkTheme: createDarkShadTheme(),
|
|
|
|
|
appBuilder: _buildMaterialApp,
|
2026-03-23 14:12:00 +08:00
|
|
|
);
|
|
|
|
|
},
|
2026-03-23 00:08:19 +08:00
|
|
|
),
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
List<SingleChildWidget> _buildProviders() {
|
|
|
|
|
final dioClient = DioClient();
|
|
|
|
|
|
|
|
|
|
return [
|
2026-03-23 14:12:00 +08:00
|
|
|
// Theme Provider (必须放在最前面)
|
|
|
|
|
ChangeNotifierProvider<ThemeProvider>(
|
|
|
|
|
create: (_) => ThemeProvider()..init(),
|
|
|
|
|
),
|
2026-03-23 00:08:19 +08:00
|
|
|
// Services
|
|
|
|
|
Provider<DioClient>.value(value: dioClient),
|
|
|
|
|
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) => AuthProvider(ctx.read<UserService>()),
|
|
|
|
|
),
|
|
|
|
|
ChangeNotifierProvider<MarketProvider>(
|
|
|
|
|
create: (ctx) => MarketProvider(ctx.read<MarketService>()),
|
|
|
|
|
),
|
|
|
|
|
ChangeNotifierProvider<AssetProvider>(
|
|
|
|
|
create: (ctx) => AssetProvider(
|
|
|
|
|
ctx.read<AssetService>(),
|
|
|
|
|
ctx.read<FundService>(),
|
2026-03-22 00:21:21 +08:00
|
|
|
),
|
2026-03-23 00:08:19 +08:00
|
|
|
),
|
|
|
|
|
];
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
Widget _buildMaterialApp(BuildContext context) {
|
|
|
|
|
return MaterialApp(
|
|
|
|
|
debugShowCheckedModeBanner: false,
|
|
|
|
|
theme: Theme.of(context),
|
|
|
|
|
localizationsDelegates: const [
|
|
|
|
|
GlobalShadLocalizations.delegate,
|
|
|
|
|
GlobalMaterialLocalizations.delegate,
|
|
|
|
|
GlobalCupertinoLocalizations.delegate,
|
|
|
|
|
GlobalWidgetsLocalizations.delegate,
|
2026-03-22 00:21:21 +08:00
|
|
|
],
|
2026-03-23 00:08:19 +08:00
|
|
|
builder: (context, child) => ShadAppBuilder(child: child!),
|
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
|
|
|
}
|
|
|
|
|
|
|
|
|
|
/// 根页面 - 决定显示引导页还是主页面
|
|
|
|
|
class RootPage extends StatelessWidget {
|
|
|
|
|
const RootPage({super.key});
|
|
|
|
|
|
|
|
|
|
@override
|
|
|
|
|
Widget build(BuildContext context) {
|
|
|
|
|
// 检查是否需要显示引导页
|
|
|
|
|
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
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
}
|