主要功能: 1. 添加推送依赖 - jpush_flutter: ^2.5.0 (极光推送) - flutter_local_notifications: ^16.0.0 (本地通知) - permission_handler: ^11.0.0 (权限管理) 2. 创建PushService服务类 - 初始化极光推送SDK - 处理推送消息接收 - 处理推送点击事件 - 显示本地通知 - 设置/删除别名(用户ID绑定) 3. 在main.dart中初始化推送服务 - 应用启动时自动初始化 - 登录后可设置用户别名 - 退出登录时删除别名 4. 推送场景支持 - 充值审批通知 - 提现审批通知 - 资产变动通知 - 自定义消息推送 5. 文档 - PUSH_NOTIFICATION_GUIDE.md: 完整的集成指南 - 包含Android/iOS配置说明 - 后端接口示例 - 测试方法 技术栈: - 极光推送 (JPush) - 国内推送到达率高 - 本地通知 - 支持前台和后台推送 - 别名机制 - 按用户ID精准推送 待完成: - [ ] 配置极光推送APPKEY - [ ] Android权限配置 - [ ] iOS证书配置 - [ ] 后端推送接口开发
171 lines
4.9 KiB
Dart
171 lines
4.9 KiB
Dart
import 'package:flutter/material.dart';
|
|
import 'package:shadcn_ui/shadcn_ui.dart';
|
|
import 'package:flutter_localizations/flutter_localizations.dart';
|
|
import 'package:provider/provider.dart';
|
|
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_color_scheme.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';
|
|
import 'services/push_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';
|
|
|
|
void main() async {
|
|
WidgetsFlutterBinding.ensureInitialized();
|
|
Provider.debugCheckInvalidValueType = null;
|
|
|
|
await SharedPreferences.getInstance();
|
|
await LocalStorage.init();
|
|
|
|
// 初始化推送服务
|
|
await PushService().init();
|
|
debugPrint('✅ 推送服务已初始化');
|
|
|
|
runApp(const MyApp());
|
|
}
|
|
|
|
class MyApp extends StatelessWidget {
|
|
const MyApp({super.key});
|
|
|
|
@override
|
|
Widget build(BuildContext context) {
|
|
return MultiProvider(
|
|
providers: _buildProviders(),
|
|
child: Consumer<ThemeProvider>(
|
|
builder: (context, themeProvider, _) {
|
|
return AuthNavigator(
|
|
child: ShadApp.custom(
|
|
themeMode: themeProvider.themeMode,
|
|
theme: createLightShadTheme(),
|
|
darkTheme: createDarkShadTheme(),
|
|
appBuilder: _buildMaterialApp,
|
|
),
|
|
);
|
|
},
|
|
),
|
|
);
|
|
}
|
|
|
|
List<SingleChildWidget> _buildProviders() {
|
|
final dioClient = DioClient();
|
|
|
|
return [
|
|
// Theme Provider (必须放在最前面)
|
|
ChangeNotifierProvider<ThemeProvider>(
|
|
create: (_) => ThemeProvider()..init(),
|
|
),
|
|
// 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)),
|
|
// 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>(),
|
|
),
|
|
),
|
|
];
|
|
}
|
|
|
|
Widget _buildMaterialApp(BuildContext context) {
|
|
return MaterialApp(
|
|
debugShowCheckedModeBanner: false,
|
|
theme: Theme.of(context),
|
|
localizationsDelegates: const [
|
|
GlobalShadLocalizations.delegate,
|
|
GlobalMaterialLocalizations.delegate,
|
|
GlobalCupertinoLocalizations.delegate,
|
|
GlobalWidgetsLocalizations.delegate,
|
|
],
|
|
builder: (context, child) => ShadAppBuilder(child: child!),
|
|
home: _buildHome(),
|
|
);
|
|
}
|
|
|
|
Widget _buildHome() {
|
|
return Consumer<AuthProvider>(
|
|
builder: (context, auth, _) {
|
|
if (auth.isLoading) {
|
|
return const Scaffold(
|
|
body: Center(child: CircularProgressIndicator()),
|
|
);
|
|
}
|
|
return auth.isLoggedIn ? const MainPage() : const LoginPage();
|
|
},
|
|
);
|
|
}
|
|
}
|
|
|
|
/// 认证路由守卫 - 监听认证状态并自动导航
|
|
class AuthNavigator extends StatefulWidget {
|
|
final Widget child;
|
|
|
|
const AuthNavigator({super.key, required this.child});
|
|
|
|
@override
|
|
State<AuthNavigator> createState() => _AuthNavigatorState();
|
|
}
|
|
|
|
class _AuthNavigatorState extends State<AuthNavigator> {
|
|
bool? _wasLoggedIn;
|
|
|
|
@override
|
|
void didChangeDependencies() {
|
|
super.didChangeDependencies();
|
|
final isLoggedIn = context.watch<AuthProvider>().isLoggedIn;
|
|
|
|
if (_wasLoggedIn == null) {
|
|
_wasLoggedIn = isLoggedIn;
|
|
return;
|
|
}
|
|
|
|
if (_wasLoggedIn != isLoggedIn) {
|
|
_wasLoggedIn = isLoggedIn;
|
|
_navigateToAuthPage(isLoggedIn);
|
|
}
|
|
}
|
|
|
|
void _navigateToAuthPage(bool isLoggedIn) {
|
|
WidgetsBinding.instance.addPostFrameCallback((_) {
|
|
if (!mounted) return;
|
|
|
|
// 退出登录时重置其他 Provider 的状态
|
|
if (!isLoggedIn) {
|
|
context.read<AssetProvider>().resetLoadState();
|
|
context.read<MarketProvider>().resetLoadState();
|
|
}
|
|
|
|
Navigator.of(context).pushAndRemoveUntil(
|
|
MaterialPageRoute(
|
|
builder: (_) => isLoggedIn ? const MainPage() : const LoginPage(),
|
|
),
|
|
(route) => false,
|
|
);
|
|
});
|
|
}
|
|
|
|
@override
|
|
Widget build(BuildContext context) => widget.child;
|
|
}
|