member_controller.dart 18 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555
  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. showRetainDialog();
  206. }
  207. } else {
  208. if (accountRepository.memberIsExpired()) {
  209. showRetainDialog();
  210. } else {
  211. back();
  212. }
  213. }
  214. }
  215. void showRetainDialog() {
  216. if (_isPopBackInProgress) {
  217. // 如果正在处理返回逻辑,直接跳过
  218. return;
  219. }
  220. _isPopBackInProgress = true; // 锁定状态
  221. MemberRetainDialog.show(payClick: () {
  222. onBuyClick();
  223. _isPopBackInProgress = false;
  224. }, cancelClick: () {
  225. _isPopBackInProgress = false;
  226. if (!Platform.isIOS) {
  227. back();
  228. }
  229. });
  230. }
  231. void onBuyClick() {
  232. if (selectedGoods == null) {
  233. ToastUtil.show(StringName.memberPleaseChoiceGoods);
  234. return;
  235. }
  236. if (selectedPayWay == null && !Platform.isIOS) {
  237. ToastUtil.show(StringName.memberPleaseChoicePayment);
  238. return;
  239. }
  240. final buyGoods = selectedGoods!;
  241. final buyPayWay = selectedPayWay!;
  242. int goodsId = buyGoods.id;
  243. int payPlatform = buyPayWay.payPlatform;
  244. int payMethod = buyPayWay.payMethod;
  245. int payWayType =
  246. getPayWayType(payMethod: payMethod, payPlatform: payPlatform);
  247. LoadingDialog.show(StringName.payLoading);
  248. memberRepository
  249. .submitAndRequestPay(
  250. goodsId: goodsId, payPlatform: payPlatform, payMethod: payMethod)
  251. .then((response) {
  252. if (payWayType == PayWayType.paymentWayWechat) {
  253. _onWeChatPay(response.outTradeNo, response.wechatPayPrepayJson!,
  254. payMethod, buyGoods, buyPayWay);
  255. } else if (payWayType == PayWayType.paymentWayWechatScan) {
  256. _onWechatScanPay(response.outTradeNo, response.wechatPayQrcodeUrl!,
  257. buyPayWay, buyGoods);
  258. } else if (payWayType == PayWayType.paymentWayAlipay) {
  259. _onAliPay(response.outTradeNo, response.alipayOrderString!, payMethod,
  260. buyGoods, buyPayWay);
  261. } else if (payWayType == PayWayType.paymentWayAlipayScan) {
  262. _onAliScanPay(response.outTradeNo, response.alipayQrcodeHtml!,
  263. buyPayWay, buyGoods);
  264. } else if (payWayType == PayWayType.paymentWayApple) {
  265. _onApplePay(response.outTradeNo, response.appAccountToken ?? "",
  266. buyPayWay, buyGoods);
  267. } else {
  268. ToastUtil.show(StringName.payNotSupport);
  269. }
  270. }).catchError((error) {
  271. if (error is ServerErrorException) {
  272. if (error.code == ErrorCode.payOrderError) {
  273. refreshMemberData();
  274. ToastUtil.show(error.message);
  275. } else if (error.code == ErrorCode.noLoginError) {
  276. ToastUtil.show(StringName.accountNoLogin);
  277. LoginPage.start();
  278. } else {
  279. ToastUtil.show(error.message);
  280. }
  281. } else {
  282. ToastUtil.show(StringName.memberPaymentFailed);
  283. }
  284. }).whenComplete(() {
  285. LoadingDialog.hide();
  286. });
  287. }
  288. ///发起购买请求
  289. Future<void> _onApplePay(String outTradeNo,
  290. String appAccountToken,
  291. PayItemBean payWayInfo,
  292. GoodsBean goodsInfo,) async {
  293. final result = await ApplePay().purchase(
  294. productId: goodsInfo.appleGoodsId ?? "",
  295. appAccountToken: appAccountToken,
  296. );
  297. if (result["success"] == true) {
  298. var receipt = result['receipt'];
  299. print('购买成功: ${result['receipt']}');
  300. checkPaymentStatus(
  301. outTradeNo,
  302. payWayInfo,
  303. goodsInfo,
  304. receiptData: receipt,
  305. );
  306. } else {
  307. LoadingDialog.hide();
  308. ToastUtil.show("支付失败,请稍后重试");
  309. print('购买失败: ${result['error']}');
  310. }
  311. }
  312. void _onAliScanPay(String outTradeNo, String qrHtml, PayItemBean paymentWay,
  313. GoodsBean goodsBean) {
  314. AlipayQrCodeDialog.show(
  315. qrCodeHtml: qrHtml,
  316. loadSuccessCallback: () {
  317. checkPaymentStatus(outTradeNo, paymentWay, goodsBean);
  318. },
  319. onCloseCallback: () async {
  320. //关闭后再持续查询几秒
  321. CustomLoadingDialog.show();
  322. await Future.delayed(Duration(seconds: 4));
  323. paymentStatusManager.removePollingSubscription(outTradeNo);
  324. CustomLoadingDialog.hide();
  325. });
  326. }
  327. void checkPaymentStatus(
  328. String orderNo, PayItemBean paymentWay, GoodsBean goodsBean,
  329. {String? receiptData}) {
  330. paymentStatusManager.registerPaymentSuccessCallback(orderNo, this);
  331. paymentStatusManager.checkPaymentStatus(orderNo, paymentWay, goodsBean,
  332. receiptData: receiptData);
  333. }
  334. void _onWeChatPay(String outTradeNo, String payJson, int payMethod,
  335. GoodsBean buyGoods, PayItemBean buyPayWay) {
  336. final bean = WechatPaymentSignBean.stringToBean(payJson);
  337. final payInfo = WechatPayInfo(
  338. appId: bean.appId,
  339. partnerId: bean.partnerId,
  340. prepayId: bean.prepayId,
  341. package: bean.package,
  342. noncestr: bean.nonceStr,
  343. timestamp: bean.timeStamp,
  344. sign: bean.sign);
  345. requestSdkPay(payInfo, outTradeNo, payMethod, buyGoods, buyPayWay);
  346. }
  347. void _onAliPay(String outTradeNo, String payJson, int payMethod,
  348. GoodsBean buyGoods, PayItemBean buyPayWay) {
  349. final payInfo = AliPayInfo(payJson);
  350. requestSdkPay(payInfo, outTradeNo, payMethod, buyGoods, buyPayWay);
  351. }
  352. void _onWechatScanPay(String outTradeNo, String qrCodeUrl,
  353. PayItemBean paymentWay, GoodsBean goodsBean) {
  354. WechatQrCodeDialog.show(
  355. qrCodeUrl: qrCodeUrl,
  356. loadSuccessCallback: () {
  357. checkPaymentStatus(outTradeNo, paymentWay, goodsBean);
  358. },
  359. onCloseCallback: () async {
  360. //关闭后再持续查询几秒
  361. CustomLoadingDialog.show();
  362. await Future.delayed(Duration(seconds: 4));
  363. paymentStatusManager.removePollingSubscription(outTradeNo);
  364. CustomLoadingDialog.hide();
  365. });
  366. }
  367. ///查询订阅状态
  368. Future<void> _requestCheckRestoreStatus(String? receiptData) async {
  369. memberRepository.subscriptionCheck(3, receiptData ?? "").then((value) {
  370. _checkResponse.value = value;
  371. }).catchError((error) {});
  372. }
  373. /// 点击了恢复订阅
  374. Future<void> clickRecoverSubscribe() async {
  375. PayItemBean? paymentWay = _selectedPayWay.value;
  376. if (paymentWay == null) {
  377. return;
  378. }
  379. int payPlatform = paymentWay.payPlatform;
  380. int payMethod = paymentWay.payMethod;
  381. CustomLoadingDialog.show();
  382. Future.delayed(const Duration(seconds: 20), () {
  383. CustomLoadingDialog.hide();
  384. //ToastUtil.show("没有发现可恢复的记录");
  385. });
  386. final result = await ApplePay().restore();
  387. if (result["success"] == true) {
  388. // CustomLoadingDialog.hide();
  389. var receipt = result['receipt'];
  390. print('查找恢复记录成功: ${result['receipt']}');
  391. checkRestoreStatus(receipt);
  392. } else {
  393. CustomLoadingDialog.hide();
  394. ToastUtil.show("恢复失败");
  395. print('恢复失败: ${result['error']}');
  396. }
  397. // 显示恢复订阅弹窗
  398. // RecoverSubscribeDialog.show("周会员2025年3月6日到期。", () {
  399. // AtmobLog.d(tag, "恢复订阅弹窗 => 点击确认");
  400. // });
  401. }
  402. /// 检查恢复订阅结果
  403. Future<void> checkRestoreStatus(String? receiptData) async {
  404. PayItemBean? paymentWay = _selectedPayWay.value;
  405. if (paymentWay == null) {
  406. // ToastUtil.showToast(StringName.storeChoicePayment.tr);
  407. return;
  408. }
  409. if (receiptData == null) {
  410. return;
  411. }
  412. int payPlatform = paymentWay.payPlatform;
  413. int payMethod = paymentWay.payMethod;
  414. // var code = await storeRepository.resume(payPlatform, payMethod, receiptData);
  415. memberRepository.subscriptionResume(3, receiptData).then((data) async {
  416. CustomLoadingDialog.hide();
  417. ToastUtil.show("恢复成功");
  418. await AccountRepository.getInstance().getMemberStatus();
  419. //accountRepository.refreshMemberStatus();
  420. Get.back();
  421. }).catchError((error) {
  422. CustomLoadingDialog.hide();
  423. ToastUtil.show("恢复失败");
  424. });
  425. // if (code == 0) {
  426. // CustomLoadingDialog.hide();
  427. // ToastUtil.show("Restore success");
  428. // userRepository.getUserInfo();
  429. // Get.back();
  430. // } else {
  431. // CustomLoadingDialog.hide();
  432. // ToastUtil.show("Restore fail");
  433. // }
  434. }
  435. void requestSdkPay(dynamic payInfo, String outTradeNo, int payMethod,
  436. GoodsBean buyGoods, PayItemBean buyPayWay) {
  437. AgilePay.startPay(payInfo, success: (String? result) {
  438. LoadingDialog.show(StringName.payQuerypayState);
  439. checkPaymentStatus(outTradeNo, buyPayWay, buyGoods, receiptData: result);
  440. }, payError: (int error, String? errorMessage) {
  441. debugPrint('zk---payError: $error, $errorMessage');
  442. errorPayToast(error);
  443. errorEventReport(payMethod);
  444. }, error: (int errno, String? error) {
  445. debugPrint('zk---error: $errno, $error');
  446. errorPayToast(errno);
  447. errorEventReport(payMethod);
  448. });
  449. }
  450. void errorEventReport(int payMethod) {
  451. if (payMethod == PayMethod.wechat) {
  452. // EventHandler.report();
  453. } else if (payMethod == PayMethod.alipay) {
  454. // EventHandler.report();
  455. } else if (payMethod == PayMethod.apple) {
  456. // EventHandler.report();
  457. }
  458. }
  459. void errorPayToast(int errno) {
  460. if (errno == AgilePayCode.payCodeNotSupport) {
  461. ToastUtil.show(StringName.payNotSupport);
  462. } else if (errno == AgilePayCode.payCodeCancelError) {
  463. ToastUtil.show(StringName.payUserCancel);
  464. } else if (errno == AgilePayCode.payCodeWxEnvError) {
  465. ToastUtil.show(StringName.payWxEvnError);
  466. } else if (errno == AgilePayCode.payCodeNotConnectStore) {
  467. ToastUtil.show(StringName.payNotConnectStore);
  468. } else {
  469. ToastUtil.show(StringName.payError);
  470. }
  471. }
  472. @override
  473. void onClose() {
  474. super.onClose();
  475. _changeStreamController?.close();
  476. _memberDataFuture?.cancel();
  477. scrollController.dispose();
  478. paymentStatusManager.unregisterPaymentSuccessCallback(this);
  479. }
  480. @override
  481. void onPaymentSuccess(
  482. String orderNo, PayItemBean paymentWay, GoodsBean storeItemBean) {
  483. try {
  484. WechatQrCodeDialog.dismiss();
  485. AlipayQrCodeDialog.dismiss();
  486. CustomLoadingDialog.hide();
  487. LoadingDialog.hide();
  488. } catch (e) {
  489. debugPrint('zk---onPaymentSuccess error: $e');
  490. }
  491. showPaymentSuccessDialog(onConfirm: back, onCancel: back);
  492. }
  493. }