member_controller.dart 18 KB

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