member_controller.dart 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556
  1. import 'dart:async';
  2. import 'dart:io';
  3. import 'dart:math';
  4. import 'package:agile_pay/flutter_pay.dart';
  5. import 'package:flutter/material.dart';
  6. import 'package:flutter_screenutil/flutter_screenutil.dart';
  7. import 'package:get/get.dart';
  8. import 'package:injectable/injectable.dart';
  9. import 'package:location/base/base_controller.dart';
  10. import 'package:location/data/bean/goods_bean.dart';
  11. import 'package:location/data/bean/pay_item_bean.dart';
  12. import 'package:location/data/repositories/account_repository.dart';
  13. import 'package:location/data/repositories/member_repository.dart';
  14. import 'package:location/dialog/alipay_qr_code_dialog.dart';
  15. import 'package:location/handler/error_handler.dart';
  16. import 'package:location/module/login/login_page.dart';
  17. import 'package:location/resource/assets.gen.dart';
  18. import 'package:location/utils/async_util.dart';
  19. import 'package:location/utils/common_expand.dart';
  20. import 'package:location/utils/toast_util.dart';
  21. import '../../data/api/response/subscription_check_response.dart';
  22. import '../../data/bean/member_status_info.dart';
  23. import '../../data/bean/wechat_payment_sign_bean.dart';
  24. import '../../data/consts/error_code.dart';
  25. import '../../data/consts/payment_type.dart';
  26. import '../../data/consts/web_url.dart';
  27. import '../../dialog/common_confirm_dialog_impl.dart';
  28. import '../../dialog/loading_dialog.dart';
  29. import '../../dialog/member_retain_dialog.dart';
  30. import '../../dialog/wechat_qr_code_dialog.dart';
  31. import '../../resource/string.gen.dart';
  32. import '../../utils/http_handler.dart';
  33. import '../../utils/payment_status_manager.dart';
  34. import '../../widget/animated_switcher_widget.dart';
  35. import '../browser/browser_view.dart';
  36. import 'member_evaluate_bean.dart';
  37. import 'member_fun_bean.dart';
  38. import 'package:apple_pay/apple_pay.dart';
  39. @injectable
  40. class MemberController extends BaseController implements PaymentStatusCallback {
  41. final AccountRepository accountRepository;
  42. final MemberRepository memberRepository;
  43. final PaymentStatusManager paymentStatusManager;
  44. final switcherController = SwitcherController();
  45. final ScrollController scrollController = ScrollController();
  46. StreamController? _changeStreamController;
  47. final random = Random();
  48. final RxDouble _toolBarOpacity = 0.0.obs;
  49. double get toolBarOpacity => _toolBarOpacity.value;
  50. final List<String> _storeTypes = ['终身会员', '年度会员', '月度会员'];
  51. String? get phone => accountRepository.loginPhoneNum.value;
  52. bool get isLogin => accountRepository.isLogin.value;
  53. MemberStatusInfo? get memberStatusInfo =>
  54. accountRepository.memberStatusInfo.value;
  55. final RxList<GoodsBean> goodsList = <GoodsBean>[].obs;
  56. final Rxn<GoodsBean> _selectedGoods = Rxn<GoodsBean>();
  57. GoodsBean? get selectedGoods => _selectedGoods.value;
  58. final Rxn<PayItemBean> _selectedPayWay = Rxn<PayItemBean>();
  59. PayItemBean? get selectedPayWay => _selectedPayWay.value;
  60. final RxList<PayItemBean> payItemList = <PayItemBean>[].obs;
  61. CancelableFuture? _memberDataFuture;
  62. ///广告弹窗防抖
  63. bool _isPopBackInProgress = false; // 操作进行中标志
  64. ///检查续订的状态
  65. final Rx<SubscriptionCheckResponse> _checkResponse =
  66. Rx(SubscriptionCheckResponse(outTradeNo: ""));
  67. SubscriptionCheckResponse? get requestCheckResponse => _checkResponse.value;
  68. final List<MemberFunBean> funList = [
  69. MemberFunBean(1, Assets.images.iconMemberFun1.path,
  70. StringName.memberFunName1, StringName.memberFunName1Desc),
  71. MemberFunBean(2, Assets.images.iconMemberFun2.path,
  72. StringName.memberFunName2, StringName.memberFunName2Desc),
  73. // MemberFunBean(3, Assets.images.iconMemberFun3.path,
  74. // StringName.memberFunName3, StringName.memberFunName3Desc),
  75. // MemberFunBean(4, Assets.images.iconMemberFun4.path,
  76. // StringName.memberFunName4, StringName.memberFunName4Desc), //该功能还未开发
  77. MemberFunBean(5, Assets.images.iconMemberFun5.path,
  78. StringName.memberFunName5, StringName.memberFunName5Desc),
  79. MemberFunBean(6, Assets.images.iconMemberFun6.path,
  80. StringName.memberFunName6, StringName.memberFunName6Desc),
  81. ];
  82. final List<MemberEvaluateBean> evaluateList = [
  83. MemberEvaluateBean(1, Assets.images.iconEvaluate1.path, '用户189****7913',
  84. "上班没时间,远程遛娃,非常方便很好用。"),
  85. MemberEvaluateBean(2, Assets.images.iconEvaluate2.path, '用户177****2345',
  86. "这个功能太棒了!尤其是夜间出行时,一键报警让我感觉特别安心。"),
  87. MemberEvaluateBean(3, Assets.images.iconEvaluate3.path, '用户138****6789',
  88. "强烈推荐!我和家人经常用这个功能来共享位置,尤其是旅游时,走散了也不怕。"),
  89. MemberEvaluateBean(4, Assets.images.iconEvaluate4.path, '用户159****3456',
  90. "实时定位非常精准,用来监控孩子的行踪特别方便,再也不用担心他们乱跑了。"),
  91. MemberEvaluateBean(5, Assets.images.iconEvaluate5.path, '用户182****9012',
  92. "用来遛狗也很方便,再也不用担心狗狗跑丢了,真是个好工具!"),
  93. ];
  94. MemberController(
  95. this.accountRepository, this.memberRepository, this.paymentStatusManager);
  96. @override
  97. void onReady() async {
  98. super.onReady();
  99. _startAnimationSwitcher();
  100. scrollController.addListener(() {
  101. double offset = scrollController.offset;
  102. double opacity = offset / 100;
  103. if (opacity > 1) {
  104. opacity = 1;
  105. } else if (opacity < 0) {
  106. opacity = 0;
  107. }
  108. _toolBarOpacity.value = opacity;
  109. });
  110. refreshMemberData();
  111. }
  112. void _startAnimationSwitcher() {
  113. _changeStreamController = AsyncUtil.interval(
  114. (time) => changeSwitcherContent(), Duration(seconds: 3), -1);
  115. }
  116. Future<void> changeSwitcherContent() async {
  117. int userId = random.nextInt(10000);
  118. String userIdStr = userId.toString();
  119. int padLength = 4 - userIdStr.length;
  120. for (int i = 0; i < padLength; i++) {
  121. userIdStr = random.nextInt(10).toString() + userIdStr;
  122. }
  123. bool isHour = random.nextBool();
  124. String secondsOrHour;
  125. if (isHour) {
  126. secondsOrHour = "${1 + random.nextInt(8)}小时";
  127. } else {
  128. secondsOrHour = "${1 + random.nextInt(59)}分钟";
  129. }
  130. int index = random.nextInt(_storeTypes.length);
  131. switcherController.updateWidget(Row(
  132. mainAxisAlignment: MainAxisAlignment.center,
  133. children: [
  134. RichText(
  135. text: TextSpan(
  136. style: TextStyle(fontSize: 12.sp, color: Colors.white),
  137. children: [
  138. TextSpan(text: userIdStr),
  139. TextSpan(text: '用户 '),
  140. TextSpan(text: secondsOrHour),
  141. TextSpan(text: '前购买了'),
  142. TextSpan(
  143. text: _storeTypes[index],
  144. style: TextStyle(color: '#FFC95D'.color)),
  145. ]))
  146. ],
  147. ));
  148. }
  149. String getUserName(String phone) {
  150. if (phone.length > 4) {
  151. phone = phone.substring(phone.length - 4);
  152. }
  153. return '${StringName.mineAccountLoggedDesc}$phone';
  154. }
  155. void back() {
  156. Get.back();
  157. }
  158. void onLoginClick() {
  159. if (accountRepository.isLogin.value) {
  160. return;
  161. }
  162. LoginPage.start();
  163. }
  164. void refreshMemberData() {
  165. _memberDataFuture?.cancel();
  166. _memberDataFuture =
  167. AsyncUtil.retryWithExponentialBackoff(() => _requestMemberData(), 4);
  168. _memberDataFuture?.catchError((error) {
  169. ErrorHandler.toastError(error);
  170. });
  171. }
  172. Future<void> _requestMemberData() {
  173. return memberRepository
  174. .getMemberList(itemListType: Platform.isIOS ? 2 : 0)
  175. .then((response) {
  176. goodsList.clear();
  177. payItemList.clear();
  178. _selectedGoods.value = null;
  179. if (response.goodsList?.isNotEmpty == true) {
  180. goodsList.addAll(response.goodsList!);
  181. _selectedGoods.value = goodsList.first;
  182. }
  183. if (response.payInfoList?.isNotEmpty == true) {
  184. payItemList.addAll(response.payInfoList!);
  185. _selectedPayWay.value = payItemList.first;
  186. }
  187. });
  188. }
  189. void onGoodsItemClick(GoodsBean item) {
  190. _selectedGoods.value = item;
  191. }
  192. void onPrivacyPolicyClick() {
  193. BrowserPage.start(WebUrl.privacyPolicy);
  194. }
  195. void onTermOfServiceClick() {
  196. BrowserPage.start(WebUrl.userAgreement);
  197. }
  198. void onPayWayItemClick(PayItemBean item) {
  199. _selectedPayWay.value = item;
  200. }
  201. void onPopBack() {
  202. if (Platform.isIOS) {
  203. back();
  204. if (accountRepository.memberIsExpired()) {
  205. FocusScope.of(Get.context!).unfocus();
  206. showRetainDialog();
  207. }
  208. } else {
  209. if (accountRepository.memberIsExpired()) {
  210. showRetainDialog();
  211. } else {
  212. back();
  213. }
  214. }
  215. }
  216. void showRetainDialog() {
  217. if (_isPopBackInProgress) {
  218. // 如果正在处理返回逻辑,直接跳过
  219. return;
  220. }
  221. _isPopBackInProgress = true; // 锁定状态
  222. MemberRetainDialog.show(payClick: () {
  223. onBuyClick();
  224. _isPopBackInProgress = false;
  225. }, cancelClick: () {
  226. _isPopBackInProgress = false;
  227. if (!Platform.isIOS) {
  228. back();
  229. }
  230. });
  231. }
  232. void onBuyClick() {
  233. if (selectedGoods == null) {
  234. ToastUtil.show(StringName.memberPleaseChoiceGoods);
  235. return;
  236. }
  237. if (selectedPayWay == null && !Platform.isIOS) {
  238. ToastUtil.show(StringName.memberPleaseChoicePayment);
  239. return;
  240. }
  241. final buyGoods = selectedGoods!;
  242. final buyPayWay = selectedPayWay!;
  243. int goodsId = buyGoods.id;
  244. int payPlatform = buyPayWay.payPlatform;
  245. int payMethod = buyPayWay.payMethod;
  246. int payWayType =
  247. getPayWayType(payMethod: payMethod, payPlatform: payPlatform);
  248. LoadingDialog.show(StringName.payLoading);
  249. memberRepository
  250. .submitAndRequestPay(
  251. goodsId: goodsId, payPlatform: payPlatform, payMethod: payMethod)
  252. .then((response) {
  253. if (payWayType == PayWayType.paymentWayWechat) {
  254. _onWeChatPay(response.outTradeNo, response.wechatPayPrepayJson!,
  255. payMethod, buyGoods, buyPayWay);
  256. } else if (payWayType == PayWayType.paymentWayWechatScan) {
  257. _onWechatScanPay(response.outTradeNo, response.wechatPayQrcodeUrl!,
  258. buyPayWay, buyGoods);
  259. } else if (payWayType == PayWayType.paymentWayAlipay) {
  260. _onAliPay(response.outTradeNo, response.alipayOrderString!, payMethod,
  261. buyGoods, buyPayWay);
  262. } else if (payWayType == PayWayType.paymentWayAlipayScan) {
  263. _onAliScanPay(response.outTradeNo, response.alipayQrcodeHtml!,
  264. buyPayWay, buyGoods);
  265. } else if (payWayType == PayWayType.paymentWayApple) {
  266. _onApplePay(response.outTradeNo, response.appAccountToken ?? "",
  267. buyPayWay, buyGoods);
  268. } else {
  269. ToastUtil.show(StringName.payNotSupport);
  270. }
  271. }).catchError((error) {
  272. if (error is ServerErrorException) {
  273. if (error.code == ErrorCode.payOrderError) {
  274. refreshMemberData();
  275. ToastUtil.show(error.message);
  276. } else if (error.code == ErrorCode.noLoginError) {
  277. ToastUtil.show(StringName.accountNoLogin);
  278. LoginPage.start();
  279. } else {
  280. ToastUtil.show(error.message);
  281. }
  282. } else {
  283. ToastUtil.show(StringName.memberPaymentFailed);
  284. }
  285. }).whenComplete(() {
  286. LoadingDialog.hide();
  287. });
  288. }
  289. ///发起购买请求
  290. Future<void> _onApplePay(String outTradeNo,
  291. String appAccountToken,
  292. PayItemBean payWayInfo,
  293. GoodsBean goodsInfo,) async {
  294. final result = await ApplePay().purchase(
  295. productId: goodsInfo.appleGoodsId ?? "",
  296. appAccountToken: appAccountToken,
  297. );
  298. if (result["success"] == true) {
  299. var receipt = result['receipt'];
  300. print('购买成功: ${result['receipt']}');
  301. checkPaymentStatus(
  302. outTradeNo,
  303. payWayInfo,
  304. goodsInfo,
  305. receiptData: receipt,
  306. );
  307. } else {
  308. LoadingDialog.hide();
  309. ToastUtil.show("支付失败,请稍后重试");
  310. print('购买失败: ${result['error']}');
  311. }
  312. }
  313. void _onAliScanPay(String outTradeNo, String qrHtml, PayItemBean paymentWay,
  314. GoodsBean goodsBean) {
  315. AlipayQrCodeDialog.show(
  316. qrCodeHtml: qrHtml,
  317. loadSuccessCallback: () {
  318. checkPaymentStatus(outTradeNo, paymentWay, goodsBean);
  319. },
  320. onCloseCallback: () async {
  321. //关闭后再持续查询几秒
  322. CustomLoadingDialog.show();
  323. await Future.delayed(Duration(seconds: 4));
  324. paymentStatusManager.removePollingSubscription(outTradeNo);
  325. CustomLoadingDialog.hide();
  326. });
  327. }
  328. void checkPaymentStatus(
  329. String orderNo, PayItemBean paymentWay, GoodsBean goodsBean,
  330. {String? receiptData}) {
  331. paymentStatusManager.registerPaymentSuccessCallback(orderNo, this);
  332. paymentStatusManager.checkPaymentStatus(orderNo, paymentWay, goodsBean,
  333. receiptData: receiptData);
  334. }
  335. void _onWeChatPay(String outTradeNo, String payJson, int payMethod,
  336. GoodsBean buyGoods, PayItemBean buyPayWay) {
  337. final bean = WechatPaymentSignBean.stringToBean(payJson);
  338. final payInfo = WechatPayInfo(
  339. appId: bean.appId,
  340. partnerId: bean.partnerId,
  341. prepayId: bean.prepayId,
  342. package: bean.package,
  343. noncestr: bean.nonceStr,
  344. timestamp: bean.timeStamp,
  345. sign: bean.sign);
  346. requestSdkPay(payInfo, outTradeNo, payMethod, buyGoods, buyPayWay);
  347. }
  348. void _onAliPay(String outTradeNo, String payJson, int payMethod,
  349. GoodsBean buyGoods, PayItemBean buyPayWay) {
  350. final payInfo = AliPayInfo(payJson);
  351. requestSdkPay(payInfo, outTradeNo, payMethod, buyGoods, buyPayWay);
  352. }
  353. void _onWechatScanPay(String outTradeNo, String qrCodeUrl,
  354. PayItemBean paymentWay, GoodsBean goodsBean) {
  355. WechatQrCodeDialog.show(
  356. qrCodeUrl: qrCodeUrl,
  357. loadSuccessCallback: () {
  358. checkPaymentStatus(outTradeNo, paymentWay, goodsBean);
  359. },
  360. onCloseCallback: () async {
  361. //关闭后再持续查询几秒
  362. CustomLoadingDialog.show();
  363. await Future.delayed(Duration(seconds: 4));
  364. paymentStatusManager.removePollingSubscription(outTradeNo);
  365. CustomLoadingDialog.hide();
  366. });
  367. }
  368. ///查询订阅状态
  369. Future<void> _requestCheckRestoreStatus(String? receiptData) async {
  370. memberRepository.subscriptionCheck(3, receiptData ?? "").then((value) {
  371. _checkResponse.value = value;
  372. }).catchError((error) {});
  373. }
  374. /// 点击了恢复订阅
  375. Future<void> clickRecoverSubscribe() async {
  376. PayItemBean? paymentWay = _selectedPayWay.value;
  377. if (paymentWay == null) {
  378. return;
  379. }
  380. int payPlatform = paymentWay.payPlatform;
  381. int payMethod = paymentWay.payMethod;
  382. CustomLoadingDialog.show();
  383. Future.delayed(const Duration(seconds: 20), () {
  384. CustomLoadingDialog.hide();
  385. //ToastUtil.show("没有发现可恢复的记录");
  386. });
  387. final result = await ApplePay().restore();
  388. if (result["success"] == true) {
  389. // CustomLoadingDialog.hide();
  390. var receipt = result['receipt'];
  391. print('查找恢复记录成功: ${result['receipt']}');
  392. checkRestoreStatus(receipt);
  393. } else {
  394. CustomLoadingDialog.hide();
  395. ToastUtil.show("恢复失败");
  396. print('恢复失败: ${result['error']}');
  397. }
  398. // 显示恢复订阅弹窗
  399. // RecoverSubscribeDialog.show("周会员2025年3月6日到期。", () {
  400. // AtmobLog.d(tag, "恢复订阅弹窗 => 点击确认");
  401. // });
  402. }
  403. /// 检查恢复订阅结果
  404. Future<void> checkRestoreStatus(String? receiptData) async {
  405. PayItemBean? paymentWay = _selectedPayWay.value;
  406. if (paymentWay == null) {
  407. // ToastUtil.showToast(StringName.storeChoicePayment.tr);
  408. return;
  409. }
  410. if (receiptData == null) {
  411. return;
  412. }
  413. int payPlatform = paymentWay.payPlatform;
  414. int payMethod = paymentWay.payMethod;
  415. // var code = await storeRepository.resume(payPlatform, payMethod, receiptData);
  416. memberRepository.subscriptionResume(3, receiptData).then((data) async {
  417. CustomLoadingDialog.hide();
  418. ToastUtil.show("恢复成功");
  419. await AccountRepository.getInstance().getMemberStatus();
  420. //accountRepository.refreshMemberStatus();
  421. Get.back();
  422. }).catchError((error) {
  423. CustomLoadingDialog.hide();
  424. ToastUtil.show("恢复失败");
  425. });
  426. // if (code == 0) {
  427. // CustomLoadingDialog.hide();
  428. // ToastUtil.show("Restore success");
  429. // userRepository.getUserInfo();
  430. // Get.back();
  431. // } else {
  432. // CustomLoadingDialog.hide();
  433. // ToastUtil.show("Restore fail");
  434. // }
  435. }
  436. void requestSdkPay(dynamic payInfo, String outTradeNo, int payMethod,
  437. GoodsBean buyGoods, PayItemBean buyPayWay) {
  438. AgilePay.startPay(payInfo, success: (String? result) {
  439. LoadingDialog.show(StringName.payQuerypayState);
  440. checkPaymentStatus(outTradeNo, buyPayWay, buyGoods, receiptData: result);
  441. }, payError: (int error, String? errorMessage) {
  442. debugPrint('zk---payError: $error, $errorMessage');
  443. errorPayToast(error);
  444. errorEventReport(payMethod);
  445. }, error: (int errno, String? error) {
  446. debugPrint('zk---error: $errno, $error');
  447. errorPayToast(errno);
  448. errorEventReport(payMethod);
  449. });
  450. }
  451. void errorEventReport(int payMethod) {
  452. if (payMethod == PayMethod.wechat) {
  453. // EventHandler.report();
  454. } else if (payMethod == PayMethod.alipay) {
  455. // EventHandler.report();
  456. } else if (payMethod == PayMethod.apple) {
  457. // EventHandler.report();
  458. }
  459. }
  460. void errorPayToast(int errno) {
  461. if (errno == AgilePayCode.payCodeNotSupport) {
  462. ToastUtil.show(StringName.payNotSupport);
  463. } else if (errno == AgilePayCode.payCodeCancelError) {
  464. ToastUtil.show(StringName.payUserCancel);
  465. } else if (errno == AgilePayCode.payCodeWxEnvError) {
  466. ToastUtil.show(StringName.payWxEvnError);
  467. } else if (errno == AgilePayCode.payCodeNotConnectStore) {
  468. ToastUtil.show(StringName.payNotConnectStore);
  469. } else {
  470. ToastUtil.show(StringName.payError);
  471. }
  472. }
  473. @override
  474. void onClose() {
  475. super.onClose();
  476. _changeStreamController?.close();
  477. _memberDataFuture?.cancel();
  478. scrollController.dispose();
  479. paymentStatusManager.unregisterPaymentSuccessCallback(this);
  480. }
  481. @override
  482. void onPaymentSuccess(
  483. String orderNo, PayItemBean paymentWay, GoodsBean storeItemBean) {
  484. try {
  485. WechatQrCodeDialog.dismiss();
  486. AlipayQrCodeDialog.dismiss();
  487. CustomLoadingDialog.hide();
  488. LoadingDialog.hide();
  489. } catch (e) {
  490. debugPrint('zk---onPaymentSuccess error: $e');
  491. }
  492. showPaymentSuccessDialog(onConfirm: back, onCancel: back);
  493. }
  494. }