gallery_screen.dart 8.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242
  1. import 'dart:async';
  2. import 'dart:io';
  3. import 'package:app_tracking_transparency/app_tracking_transparency.dart';
  4. import 'package:firebase_crashlytics/firebase_crashlytics.dart';
  5. import 'package:firebase_messaging/firebase_messaging.dart';
  6. import 'package:flutter/material.dart';
  7. import 'package:logging/logging.dart';
  8. import 'package:lottie/lottie.dart';
  9. import 'package:provider/provider.dart';
  10. import 'package:puzzleweave/ads/applovin_ads_controller.dart';
  11. import 'package:puzzleweave/audio/jc_audio_controller.dart';
  12. import 'package:puzzleweave/config/device.dart';
  13. import 'package:puzzleweave/firebase/adjust_helper.dart';
  14. import 'package:puzzleweave/gallery/grid_item.dart';
  15. import 'package:puzzleweave/models/cached_request.dart';
  16. import 'package:puzzleweave/models/data.dart';
  17. import 'package:puzzleweave/models/items.dart';
  18. import 'package:puzzleweave/persistence/persistence.dart';
  19. final Logger _log = Logger('gallery_screen');
  20. class GalleryScreen extends StatefulWidget {
  21. const GalleryScreen({super.key});
  22. @override
  23. State<StatefulWidget> createState() => _GalleryScreen();
  24. }
  25. const int minimumRemoteLoadCount = 30; // 假设加载到 30 张图才算网络畅通
  26. class _GalleryScreen extends State<GalleryScreen> {
  27. late Device device;
  28. late JcAudioController audio;
  29. late Data data;
  30. List<ListItem>? latest;
  31. late CachedRequest latestCachedRequest;
  32. late StreamSubscription? latestSubscription;
  33. @override
  34. void initState() {
  35. super.initState();
  36. device = context.read<Device>();
  37. audio = context.read<JcAudioController>();
  38. data = context.read<Data>();
  39. latestCachedRequest = data.latest;
  40. // 主动获取缓存数据(关键)
  41. final cachedData = latestCachedRequest.cachedData;
  42. if (cachedData != null) {
  43. _onLatestDataUpdate(cachedData);
  44. }
  45. latestSubscription = latestCachedRequest.stream.listen(_onLatestDataUpdate, onError: _onLatestDataError);
  46. audio.startMusic();
  47. }
  48. _onLatestDataUpdate(datalist) {
  49. _log.info('_onLatestDataUpdate.... ');
  50. if (datalist != null) {
  51. latest = datalist as List<ListItem>;
  52. setState(() {});
  53. final bool hasSufficientData = datalist.length >= minimumRemoteLoadCount;
  54. if (hasSufficientData) {
  55. // 如果数据完整,无论是否是缓存数据,都尝试初始化第三方服务(因为主页已经可以显示了)
  56. if (!hasInit) {
  57. initThird();
  58. }
  59. }
  60. }
  61. }
  62. _onLatestDataError(error) {
  63. _log.info('_onLatestDataError.... $error');
  64. if (latest == null || latest!.isEmpty || latest!.length < minimumRemoteLoadCount) {
  65. _log.warning("_onLatestDataError, retry again");
  66. // refresh();
  67. Future.delayed(Duration(seconds: 3), () => refresh());
  68. }
  69. }
  70. Future<void> refresh() async {
  71. _log.info('refresh...');
  72. await latestCachedRequest.refresh();
  73. }
  74. @override
  75. Widget build(BuildContext context) {
  76. final device = context.read<Device>();
  77. final isTablet = device.isTablet;
  78. return Scaffold(
  79. body: latest == null
  80. ? scrollableDummy
  81. : RefreshIndicator(
  82. onRefresh: refresh,
  83. child: CustomScrollView(
  84. slivers: <Widget>[
  85. SliverPadding(
  86. sliver: SliverGrid(
  87. gridDelegate: SliverGridDelegateWithMaxCrossAxisExtent(maxCrossAxisExtent: isTablet ? 300 : 210, childAspectRatio: 2 / 3),
  88. delegate: SliverChildBuilderDelegate((BuildContext context, int index) {
  89. return _buildItem(context, index);
  90. }, childCount: latest!.length),
  91. ),
  92. padding: const EdgeInsets.only(left: 10.0, right: 10.0),
  93. ),
  94. ],
  95. ),
  96. ),
  97. );
  98. }
  99. Widget get scrollableDummy => LayoutBuilder(
  100. builder: (p0, p1) {
  101. return SingleChildScrollView(
  102. physics: const AlwaysScrollableScrollPhysics(),
  103. child: SizedBox(
  104. height: p1.maxHeight,
  105. child: Center(
  106. child: ListView(
  107. shrinkWrap: true,
  108. children: [
  109. Lottie.asset('assets/lottie/loading.json', height: 100),
  110. const Center(child: Text("loading...")),
  111. ],
  112. ),
  113. ),
  114. ),
  115. );
  116. },
  117. );
  118. Widget _buildItem(context, index) {
  119. ListItem item = latest![index];
  120. return Padding(
  121. padding: const EdgeInsets.all(10.0),
  122. child: GridItem(item: item, lock: false, index: index),
  123. );
  124. }
  125. ///////////////////////// 初始化相关 /////////////////////////
  126. static bool hasInit = false;
  127. // 在列表刷出来后才正式初始化admod等组件
  128. void initThird() async {
  129. if (hasInit) return;
  130. hasInit = true;
  131. // 有了UMP后, 这里的ATT就不需要了
  132. // bool auth = await initATT();
  133. // if (auth) {
  134. // await platform.setHasUserConsent(true);
  135. // await platform.setAdvertiserTrackingEnabled(true);
  136. // }
  137. // await initUMP(); // 征询欧洲用户同意 // applovin max 已经可以自动处理,这里不需要了
  138. TrackingStatus attStatus = await AppTrackingTransparency.trackingAuthorizationStatus;
  139. if (attStatus == TrackingStatus.authorized && Platform.isIOS) {
  140. // ATT 通过之后,ios需要调用相关的原生sdk接口做进一步的初始化
  141. // await platform.setHasUserConsent(true);
  142. // await platform.setAdvertiserTrackingEnabled(true);
  143. }
  144. initFCM(); // 消息推送许可弹窗
  145. initAd(); // admod 的广告加载安排在iOS ATT 之后,以便能够加载到个性化广告
  146. AdjustHelper.init(Persistence().uuid); // 初始化Adjust
  147. final idfa = await AppTrackingTransparency.getAdvertisingIdentifier();
  148. _log.info("idfa: $idfa");
  149. }
  150. /////////////////////////// ATT ///////////////////////////
  151. // Platform messages are asynchronous, so we initialize in an async method.
  152. Future<bool> initATT() async {
  153. TrackingStatus status = await AppTrackingTransparency.trackingAuthorizationStatus;
  154. _log.info('initATT111 $status');
  155. // If the system can show an authorization request dialog
  156. if (status == TrackingStatus.notDetermined) {
  157. // Show a custom explainer dialog before the system dialog
  158. // await showCustomTrackingDialog(context);
  159. // Wait for dialog popping animation
  160. // await Future.delayed(const Duration(milliseconds: 200));
  161. // Request system's tracking authorization dialog
  162. status = await AppTrackingTransparency.requestTrackingAuthorization();
  163. _log.info('initATT222 $status');
  164. }
  165. if (status == TrackingStatus.authorized) {
  166. return true;
  167. }
  168. return false;
  169. }
  170. // no need
  171. Future<void> showCustomTrackingDialog(BuildContext context) async => await showDialog<void>(
  172. context: context,
  173. builder: (context) => AlertDialog(
  174. title: const Text('Dear User'),
  175. content: const Text(
  176. 'We care about your privacy and data security. We keep this app free by showing ads. '
  177. 'Can we continue to use your data to tailor ads for you?\n\nYou can change your choice anytime in the app settings. '
  178. 'Our partners will collect data and use a unique identifier on your device to show you ads.',
  179. ),
  180. actions: [TextButton(onPressed: () => Navigator.pop(context), child: const Text('Continue'))],
  181. ),
  182. );
  183. /////////////////////////////////////////////////////////
  184. ///
  185. /// 初始化广告模块
  186. initAd() {
  187. _log.info('initAd');
  188. ApplovinAdsController applovinAdsController = context.read<ApplovinAdsController>();
  189. applovinAdsController.initialize();
  190. }
  191. /////////////////////////// FCM ///////////////////////////
  192. // 消息推送许可弹框
  193. initFCM() async {
  194. try {
  195. final fcmToken = await FirebaseMessaging.instance.getToken();
  196. _log.info("FCM Token: $fcmToken");
  197. FirebaseMessaging messaging = FirebaseMessaging.instance;
  198. NotificationSettings settings = await messaging.requestPermission(
  199. alert: true,
  200. announcement: false,
  201. badge: true,
  202. carPlay: false,
  203. criticalAlert: false,
  204. provisional: false,
  205. sound: true,
  206. );
  207. _log.warning('User granted permission: ${settings.authorizationStatus}');
  208. } catch (e) {
  209. FirebaseCrashlytics.instance.log("FCM FirebaseMessaging.instance.getToken error: $e");
  210. _log.warning(e);
  211. }
  212. }
  213. }