payment_status_manager.dart 3.8 KB

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