payment_status_manager.dart 3.8 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121
  1. import 'package:flutter/cupertino.dart';
  2. import 'package:synchronized/synchronized.dart';
  3. import '../../data/bean/payment_way.dart';
  4. import '../../data/bean/store_item.dart';
  5. import '../../data/repositories/store_repository.dart';
  6. import '../../utils/async_util.dart';
  7. class PaymentStatusManager {
  8. PaymentStatusManager._();
  9. //订单状态
  10. //0-查询失败,继续轮询
  11. //1-未支付,继续轮询
  12. //2-支付成功
  13. //3-支付关闭
  14. //4-已退款
  15. static const int payStatusFail = 0;
  16. static const int payStatusUnpaid = 1;
  17. static const int payStatusSuccess = 2;
  18. static const int payStatusClose = 3;
  19. static const int payStatusRefund = 4;
  20. final Map<String, PaymentStatusCallback> callbackMap = {};
  21. final Map<String, CancelableFuture> pollingSubscriptionMap = {};
  22. final _lock = Lock();
  23. void _startCheckPolling(
  24. String orderNo, PaymentWay paymentWay, StoreItem storeItemBean,
  25. {String? receiptData}) async {
  26. await _lock.synchronized(() async {
  27. pollingSubscriptionMap[orderNo]?.cancel();
  28. debugPrint('开始轮询支付状态: orderNo = $orderNo');
  29. CancelableFuture orderFuture = AsyncUtil.retryWithExponentialBackoff(
  30. () {
  31. return storeRepository
  32. .orderStatus(orderNo, receiptData: receiptData)
  33. .then((status) {
  34. if (status == payStatusSuccess) {
  35. return true;
  36. } else {
  37. throw PaymentStatusException(status);
  38. }
  39. });
  40. },
  41. 10,
  42. predicate: (error) {
  43. if (error is PaymentStatusException) {
  44. return error.status == payStatusFail ||
  45. error.status == payStatusUnpaid;
  46. }
  47. return true;
  48. });
  49. orderFuture.then((data) async {
  50. debugPrint('支付成功: orderNo = $orderNo');
  51. // accountRepository.refreshUserInfo();
  52. await _lock.synchronized(() {
  53. callbackMap[orderNo]
  54. ?.onPaymentSuccess(orderNo, paymentWay, storeItemBean);
  55. callbackMap.remove(orderNo);
  56. });
  57. reportPaySuccess(storeItemBean.amount, orderNo, storeItemBean.name,
  58. paymentWay.payMethod);
  59. }).catchError((error) async {
  60. debugPrint('支付失败: orderNo = $orderNo, error = $error');
  61. await _lock.synchronized(() {
  62. callbackMap[orderNo]
  63. ?.onPaymentError(error);
  64. callbackMap.remove(orderNo);
  65. });
  66. });
  67. pollingSubscriptionMap[orderNo] = orderFuture;
  68. });
  69. }
  70. void reportPaySuccess(
  71. int price, String orderId, String itemName, int paymentWay) {
  72. // EventHandler.reportPay(price, orderId, itemName, paymentWay);
  73. }
  74. void checkPaymentStatus(
  75. String orderNo, PaymentWay paymentWay, StoreItem storeItemBean,
  76. {String? receiptData}) {
  77. // recordKeyInfoToDisk(orderNo, paymentWay, storeItemBean);
  78. _startCheckPolling(orderNo, paymentWay, storeItemBean,
  79. receiptData: receiptData);
  80. }
  81. void registerPaymentSuccessCallback(
  82. String orderNo, PaymentStatusCallback callback) async {
  83. await _lock.synchronized(() {
  84. callbackMap[orderNo] = callback;
  85. });
  86. }
  87. void unregisterPaymentSuccessCallback(PaymentStatusCallback callback) async {
  88. await _lock.synchronized(() {
  89. callbackMap.removeWhere((key, value) => value == callback);
  90. });
  91. }
  92. }
  93. class PaymentStatusException implements Exception {
  94. final int status;
  95. PaymentStatusException(this.status);
  96. @override
  97. String toString() {
  98. return '支付状态异常: status = $status';
  99. }
  100. }
  101. abstract class PaymentStatusCallback {
  102. void onPaymentSuccess(
  103. String orderNo, PaymentWay paymentWay, StoreItem storeItemBean);
  104. void onPaymentError(Error error);
  105. }
  106. final paymentStatusManager = PaymentStatusManager._();