member_controller.dart 18 KB

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