@pisell/pisellos 2.1.117 → 2.1.118

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (98) hide show
  1. package/dist/index.d.ts +1 -0
  2. package/dist/index.js +1 -0
  3. package/dist/model/index.d.ts +1 -0
  4. package/dist/model/index.js +1 -0
  5. package/dist/model/strategy/adapter/index.d.ts +3 -0
  6. package/dist/model/strategy/adapter/index.js +4 -0
  7. package/dist/model/strategy/adapter/type.d.ts +28 -0
  8. package/dist/model/strategy/adapter/type.js +1 -0
  9. package/dist/model/strategy/adapter/walletPass/evaluator.d.ts +84 -0
  10. package/dist/model/strategy/adapter/walletPass/evaluator.js +499 -0
  11. package/dist/model/strategy/adapter/walletPass/example.d.ts +4 -0
  12. package/dist/model/strategy/adapter/walletPass/example.js +258 -0
  13. package/dist/model/strategy/adapter/walletPass/index.d.ts +32 -0
  14. package/dist/model/strategy/adapter/walletPass/index.js +182 -0
  15. package/dist/model/strategy/adapter/walletPass/locales.d.ts +1 -0
  16. package/dist/model/strategy/adapter/walletPass/locales.js +26 -0
  17. package/dist/model/strategy/adapter/walletPass/type.d.ts +187 -0
  18. package/dist/model/strategy/adapter/walletPass/type.js +1 -0
  19. package/dist/model/strategy/adapter/walletPass/utils.d.ts +50 -0
  20. package/dist/model/strategy/adapter/walletPass/utils.js +1104 -0
  21. package/dist/model/strategy/index.d.ts +94 -0
  22. package/dist/model/strategy/index.js +551 -0
  23. package/dist/model/strategy/strategy-example.d.ts +5 -0
  24. package/dist/model/strategy/strategy-example.js +331 -0
  25. package/dist/model/strategy/type.d.ts +228 -0
  26. package/dist/model/strategy/type.js +94 -0
  27. package/dist/modules/Cart/types.d.ts +2 -0
  28. package/dist/modules/Cart/utils/cartProduct.js +56 -24
  29. package/dist/modules/Discount/types.d.ts +18 -0
  30. package/dist/modules/Payment/index.d.ts +2 -1
  31. package/dist/modules/Payment/index.js +10 -7
  32. package/dist/modules/Payment/utils.js +3 -0
  33. package/dist/modules/Payment/walletpass.d.ts +23 -0
  34. package/dist/modules/Payment/walletpass.js +191 -95
  35. package/dist/modules/Rules/index.d.ts +9 -5
  36. package/dist/modules/Rules/index.js +674 -162
  37. package/dist/modules/Rules/types.d.ts +2 -0
  38. package/dist/modules/Summary/types.d.ts +2 -0
  39. package/dist/modules/Summary/utils.d.ts +6 -0
  40. package/dist/modules/Summary/utils.js +21 -0
  41. package/dist/plugins/window.d.ts +2 -0
  42. package/dist/solution/BookingByStep/index.d.ts +1 -0
  43. package/dist/solution/BookingTicket/index.d.ts +1 -1
  44. package/dist/solution/Checkout/index.js +2 -0
  45. package/dist/solution/ShopDiscount/index.d.ts +1 -0
  46. package/dist/solution/ShopDiscount/index.js +23 -7
  47. package/dist/solution/ShopDiscount/types.d.ts +1 -0
  48. package/dist/solution/ShopDiscount/types.js +2 -1
  49. package/dist/solution/ShopDiscount/utils.js +26 -12
  50. package/lib/index.d.ts +1 -0
  51. package/lib/index.js +3 -1
  52. package/lib/model/index.d.ts +1 -0
  53. package/lib/model/index.js +23 -0
  54. package/lib/model/strategy/adapter/index.d.ts +3 -0
  55. package/lib/model/strategy/adapter/index.js +45 -0
  56. package/lib/model/strategy/adapter/type.d.ts +28 -0
  57. package/lib/model/strategy/adapter/type.js +17 -0
  58. package/lib/model/strategy/adapter/walletPass/evaluator.d.ts +84 -0
  59. package/lib/model/strategy/adapter/walletPass/evaluator.js +423 -0
  60. package/lib/model/strategy/adapter/walletPass/example.d.ts +4 -0
  61. package/lib/model/strategy/adapter/walletPass/example.js +207 -0
  62. package/lib/model/strategy/adapter/walletPass/index.d.ts +32 -0
  63. package/lib/model/strategy/adapter/walletPass/index.js +142 -0
  64. package/lib/model/strategy/adapter/walletPass/locales.d.ts +1 -0
  65. package/lib/model/strategy/adapter/walletPass/locales.js +54 -0
  66. package/lib/model/strategy/adapter/walletPass/type.d.ts +187 -0
  67. package/lib/model/strategy/adapter/walletPass/type.js +17 -0
  68. package/lib/model/strategy/adapter/walletPass/utils.d.ts +50 -0
  69. package/lib/model/strategy/adapter/walletPass/utils.js +791 -0
  70. package/lib/model/strategy/index.d.ts +94 -0
  71. package/lib/model/strategy/index.js +413 -0
  72. package/lib/model/strategy/strategy-example.d.ts +5 -0
  73. package/lib/model/strategy/strategy-example.js +318 -0
  74. package/lib/model/strategy/type.d.ts +228 -0
  75. package/lib/model/strategy/type.js +44 -0
  76. package/lib/modules/Cart/types.d.ts +2 -0
  77. package/lib/modules/Cart/utils/cartProduct.js +39 -11
  78. package/lib/modules/Discount/types.d.ts +18 -0
  79. package/lib/modules/Payment/index.d.ts +2 -1
  80. package/lib/modules/Payment/index.js +1 -0
  81. package/lib/modules/Payment/utils.js +3 -0
  82. package/lib/modules/Payment/walletpass.d.ts +23 -0
  83. package/lib/modules/Payment/walletpass.js +94 -17
  84. package/lib/modules/Rules/index.d.ts +9 -5
  85. package/lib/modules/Rules/index.js +779 -284
  86. package/lib/modules/Rules/types.d.ts +2 -0
  87. package/lib/modules/Summary/types.d.ts +2 -0
  88. package/lib/modules/Summary/utils.d.ts +6 -0
  89. package/lib/modules/Summary/utils.js +15 -0
  90. package/lib/plugins/window.d.ts +2 -0
  91. package/lib/solution/BookingByStep/index.d.ts +1 -0
  92. package/lib/solution/BookingTicket/index.d.ts +1 -1
  93. package/lib/solution/Checkout/index.js +2 -0
  94. package/lib/solution/ShopDiscount/index.d.ts +1 -0
  95. package/lib/solution/ShopDiscount/index.js +22 -8
  96. package/lib/solution/ShopDiscount/types.d.ts +1 -0
  97. package/lib/solution/ShopDiscount/utils.js +10 -6
  98. package/package.json +1 -1
@@ -0,0 +1,1104 @@
1
+ function _typeof(o) { "@babel/helpers - typeof"; return _typeof = "function" == typeof Symbol && "symbol" == typeof Symbol.iterator ? function (o) { return typeof o; } : function (o) { return o && "function" == typeof Symbol && o.constructor === Symbol && o !== Symbol.prototype ? "symbol" : typeof o; }, _typeof(o); }
2
+ function ownKeys(e, r) { var t = Object.keys(e); if (Object.getOwnPropertySymbols) { var o = Object.getOwnPropertySymbols(e); r && (o = o.filter(function (r) { return Object.getOwnPropertyDescriptor(e, r).enumerable; })), t.push.apply(t, o); } return t; }
3
+ function _objectSpread(e) { for (var r = 1; r < arguments.length; r++) { var t = null != arguments[r] ? arguments[r] : {}; r % 2 ? ownKeys(Object(t), !0).forEach(function (r) { _defineProperty(e, r, t[r]); }) : Object.getOwnPropertyDescriptors ? Object.defineProperties(e, Object.getOwnPropertyDescriptors(t)) : ownKeys(Object(t)).forEach(function (r) { Object.defineProperty(e, r, Object.getOwnPropertyDescriptor(t, r)); }); } return e; }
4
+ function _defineProperty(obj, key, value) { key = _toPropertyKey(key); if (key in obj) { Object.defineProperty(obj, key, { value: value, enumerable: true, configurable: true, writable: true }); } else { obj[key] = value; } return obj; }
5
+ function _toPropertyKey(t) { var i = _toPrimitive(t, "string"); return "symbol" == _typeof(i) ? i : String(i); }
6
+ function _toPrimitive(t, r) { if ("object" != _typeof(t) || !t) return t; var e = t[Symbol.toPrimitive]; if (void 0 !== e) { var i = e.call(t, r || "default"); if ("object" != _typeof(i)) return i; throw new TypeError("@@toPrimitive must return a primitive value."); } return ("string" === r ? String : Number)(t); }
7
+ function _createForOfIteratorHelper(o, allowArrayLike) { var it = typeof Symbol !== "undefined" && o[Symbol.iterator] || o["@@iterator"]; if (!it) { if (Array.isArray(o) || (it = _unsupportedIterableToArray(o)) || allowArrayLike && o && typeof o.length === "number") { if (it) o = it; var i = 0; var F = function F() {}; return { s: F, n: function n() { if (i >= o.length) return { done: true }; return { done: false, value: o[i++] }; }, e: function e(_e) { throw _e; }, f: F }; } throw new TypeError("Invalid attempt to iterate non-iterable instance.\nIn order to be iterable, non-array objects must have a [Symbol.iterator]() method."); } var normalCompletion = true, didErr = false, err; return { s: function s() { it = it.call(o); }, n: function n() { var step = it.next(); normalCompletion = step.done; return step; }, e: function e(_e2) { didErr = true; err = _e2; }, f: function f() { try { if (!normalCompletion && it.return != null) it.return(); } finally { if (didErr) throw err; } } }; }
8
+ function _toConsumableArray(arr) { return _arrayWithoutHoles(arr) || _iterableToArray(arr) || _unsupportedIterableToArray(arr) || _nonIterableSpread(); }
9
+ function _nonIterableSpread() { throw new TypeError("Invalid attempt to spread non-iterable instance.\nIn order to be iterable, non-array objects must have a [Symbol.iterator]() method."); }
10
+ function _unsupportedIterableToArray(o, minLen) { if (!o) return; if (typeof o === "string") return _arrayLikeToArray(o, minLen); var n = Object.prototype.toString.call(o).slice(8, -1); if (n === "Object" && o.constructor) n = o.constructor.name; if (n === "Map" || n === "Set") return Array.from(o); if (n === "Arguments" || /^(?:Ui|I)nt(?:8|16|32)(?:Clamped)?Array$/.test(n)) return _arrayLikeToArray(o, minLen); }
11
+ function _iterableToArray(iter) { if (typeof Symbol !== "undefined" && iter[Symbol.iterator] != null || iter["@@iterator"] != null) return Array.from(iter); }
12
+ function _arrayWithoutHoles(arr) { if (Array.isArray(arr)) return _arrayLikeToArray(arr); }
13
+ function _arrayLikeToArray(arr, len) { if (len == null || len > arr.length) len = arr.length; for (var i = 0, arr2 = new Array(len); i < len; i++) arr2[i] = arr[i]; return arr2; }
14
+ import Decimal from 'decimal.js';
15
+ /** 订单商品数量 */
16
+ export var getProductQuantity = function getProductQuantity(product) {
17
+ return product.quantity || product.product_quantity || 1;
18
+ };
19
+
20
+ /**
21
+ * 订单商品行唯一键,用于 maxPassesPerItem 按行、按件配额(同 SPU 不同规格视为不同行)。
22
+ */
23
+ export function resolveWalletPassLineKey(product, indexInOrder) {
24
+ var m = (product === null || product === void 0 ? void 0 : product.metadata) || {};
25
+ var u1 = m.product_unique;
26
+ if (u1 != null && String(u1).length > 0) return String(u1);
27
+ var u2 = m.unique_identification_number;
28
+ if (u2 != null && String(u2).length > 0) return String(u2);
29
+ if ((product === null || product === void 0 ? void 0 : product.id) != null && String(product.id).length > 0) return "order_item:".concat(product.id);
30
+ if ((product === null || product === void 0 ? void 0 : product.product_unique_string) != null && String(product.product_unique_string).length > 0) {
31
+ return String(product.product_unique_string);
32
+ }
33
+ return "order_line_".concat(indexInOrder);
34
+ }
35
+ function getExpandedOrderLineQuantity(p) {
36
+ if ((p === null || p === void 0 ? void 0 : p._orderLineQuantity) != null && p._orderLineQuantity > 0) return p._orderLineQuantity;
37
+ return getProductQuantity(p);
38
+ }
39
+
40
+ /** 该行允许消耗的「券次」上限:maxPassesPerItem × 订单行件数 */
41
+ function getMaxPassSlotsForExpandedLine(maxPassesPerItem, p) {
42
+ if (maxPassesPerItem <= 0) return Infinity;
43
+ return maxPassesPerItem * getExpandedOrderLineQuantity(p);
44
+ }
45
+
46
+ /**
47
+ * 本次抵扣占用多少「券次」配额:有抵扣时至少占 1,并与按金额折算的件数挂钩,不超过剩余额度。
48
+ */
49
+ function computePassSlotsIncrement(deductAmount, deductQty, maxPassesPerItem, orderLineQuantity, currentUsage) {
50
+ if (maxPassesPerItem <= 0 || !deductAmount.greaterThan(0)) return 0;
51
+ var cap = maxPassesPerItem * orderLineQuantity;
52
+ var room = Math.max(0, cap - currentUsage);
53
+ if (room <= 0) return 0;
54
+ var raw = Math.max(1, Math.ceil(Number(deductQty)) || 1);
55
+ return Math.min(raw, room);
56
+ }
57
+ function applyMaxPassUsageIncrements(usageMap, walletPassProductId, maxPassesPerItem, deductionDetails) {
58
+ deductionDetails.forEach(function (detail) {
59
+ var _usageMap$get;
60
+ var lineKey = detail.lineKey;
61
+ if (!lineKey || maxPassesPerItem <= 0) return;
62
+ var orderLineQty = detail.orderLineQuantity != null && detail.orderLineQuantity > 0 ? detail.orderLineQuantity : 1;
63
+ var cur = ((_usageMap$get = usageMap.get(walletPassProductId)) === null || _usageMap$get === void 0 ? void 0 : _usageMap$get.get(lineKey)) || 0;
64
+ var inc = computePassSlotsIncrement(new Decimal(detail.deductAmount), detail.deductQuantity, maxPassesPerItem, orderLineQty, cur);
65
+ if (inc <= 0) return;
66
+ if (!usageMap.has(walletPassProductId)) usageMap.set(walletPassProductId, new Map());
67
+ var inner = usageMap.get(walletPassProductId);
68
+ inner.set(lineKey, cur + inc);
69
+ });
70
+ }
71
+
72
+ // 辅助函数:根据 deductTaxAndFee 配置获取推荐使用金额
73
+ var getRecommendedAmount = function getRecommendedAmount(voucher) {
74
+ var _config$deductTaxAndF;
75
+ console.log('voucher312', voucher);
76
+ var config = voucher.config,
77
+ recommended_usage_amount = voucher.recommended_usage_amount,
78
+ recommended_pure_product_usage_amount = voucher.recommended_pure_product_usage_amount;
79
+ var deductTaxAndFee = (_config$deductTaxAndF = config === null || config === void 0 ? void 0 : config.deductTaxAndFee) !== null && _config$deductTaxAndF !== void 0 ? _config$deductTaxAndF : true; // 默认开启抵扣税费
80
+
81
+ // 如果开启抵扣税费,使用 recommended_usage_amount(包含税费)
82
+ // 如果关闭抵扣税费,使用 recommended_pure_product_usage_amount(仅商品金额)
83
+ return deductTaxAndFee ? recommended_usage_amount : recommended_pure_product_usage_amount !== null && recommended_pure_product_usage_amount !== void 0 ? recommended_pure_product_usage_amount : recommended_usage_amount;
84
+ };
85
+
86
+ // 辅助函数:获取适用商品ID列表
87
+ export var getApplicableProductIds = function getApplicableProductIds(voucher) {
88
+ var _ref = voucher,
89
+ available_product_type = _ref.available_product_type,
90
+ available_product_ids = _ref.available_product_ids;
91
+
92
+ // 如果没有 available_product_type,默认为 product_all
93
+ var productType = available_product_type || 'product_all';
94
+
95
+ // product_all 时,返回 null 表示适用所有商品
96
+ if (productType === 'product_all') {
97
+ return null;
98
+ }
99
+
100
+ // product_collection 或 products 时,返回指定的商品ID列表
101
+ if (productType === 'product_collection' || productType === 'products') {
102
+ return available_product_ids || [];
103
+ }
104
+
105
+ // 其他情况默认返回空数组(不适用任何商品)
106
+ return [];
107
+ };
108
+
109
+ // 辅助函数:计算适用商品的总金额(基于剩余金额)
110
+ var getApplicableProductsAmount = function getApplicableProductsAmount(voucher, productsData) {
111
+ var _config$deductTaxAndF2;
112
+ var applicableProductIds = getApplicableProductIds(voucher);
113
+ var config = voucher.config;
114
+ var deductTaxAndFee = (_config$deductTaxAndF2 = config === null || config === void 0 ? void 0 : config.deductTaxAndFee) !== null && _config$deductTaxAndF2 !== void 0 ? _config$deductTaxAndF2 : true;
115
+
116
+ // 根据券的配置选择使用哪个金额字段
117
+ var amountField = deductTaxAndFee ? 'remainingAmountWithTax' : 'remainingAmountPure';
118
+
119
+ // 如果为 null,适用于所有商品
120
+ if (applicableProductIds === null) {
121
+ return productsData.filter(function (p) {
122
+ return p[amountField].greaterThan(0);
123
+ }).reduce(function (sum, p) {
124
+ return sum.plus(p[amountField]);
125
+ }, new Decimal(0));
126
+ }
127
+
128
+ // 如果为空数组,不适用任何商品
129
+ if (applicableProductIds.length === 0) {
130
+ return new Decimal(0);
131
+ }
132
+
133
+ // 计算指定商品的总金额
134
+ return productsData.filter(function (p) {
135
+ return applicableProductIds.includes(p.product_id) && p[amountField].greaterThan(0);
136
+ }).reduce(function (sum, p) {
137
+ return sum.plus(p[amountField]);
138
+ }, new Decimal(0));
139
+ };
140
+
141
+ // 辅助函数:获取适用的商品列表
142
+ var getApplicableProducts = function getApplicableProducts(voucher, productsData) {
143
+ var applicableProductIds = getApplicableProductIds(voucher);
144
+
145
+ // 如果为 null,适用于所有商品
146
+ if (applicableProductIds === null) {
147
+ return productsData;
148
+ }
149
+
150
+ // 如果为空数组,不适用任何商品
151
+ if (applicableProductIds.length === 0) {
152
+ return [];
153
+ }
154
+
155
+ // 返回指定的商品列表
156
+ return productsData.filter(function (p) {
157
+ return applicableProductIds.includes(p.product_id);
158
+ });
159
+ };
160
+
161
+ /**
162
+ * 优惠券处理函数
163
+ * @param applicableVouchers 可用的券列表
164
+ * @param orderTotalAmount 订单总金额
165
+ * @param products 订单商品列表
166
+ * @returns 返回推荐券列表和全部列表,每个券包含 _available_max_amount 和 _unified_available_status
167
+ */
168
+ export function processVouchers(applicableVouchers, orderTotalAmount, products) {
169
+ console.log(products, 'products123');
170
+
171
+ // 拆分商品数据,同时维护含税和不含税两个金额池
172
+ // remainingAmountWithTax: 含税费的剩余可抵扣金额
173
+ // remainingAmountPure: 纯商品金额(不含税费)的剩余可抵扣金额
174
+ // 在抵扣时,会根据券的 deductTaxAndFee 配置选择从哪个金额池扣除
175
+ var productsCopy = expandProductsWithBundleItems(products, true);
176
+ var remainingOrderAmount = new Decimal(orderTotalAmount); // 订单剩余应付金额
177
+
178
+ // ========== 辅助工具:maxPassesPerItem(按订单行 + 件数)追踪 ==========
179
+ var getItemPassUsage = function getItemPassUsage(usageMap, walletPassProductId, lineKey) {
180
+ var _usageMap$get2;
181
+ return ((_usageMap$get2 = usageMap.get(walletPassProductId)) === null || _usageMap$get2 === void 0 ? void 0 : _usageMap$get2.get(lineKey)) || 0;
182
+ };
183
+
184
+ // 按 maxPassesPerItem 过滤:排除已达到该行「券次」上限的展开行(按 _walletPassLineKey)
185
+ var filterByMaxPassesPerItem = function filterByMaxPassesPerItem(products, usageMap, walletPassProductId, maxPassesPerItem) {
186
+ if (maxPassesPerItem <= 0) return products;
187
+ return products.filter(function (p) {
188
+ var lineKey = p._walletPassLineKey;
189
+ if (!lineKey) return true;
190
+ return getItemPassUsage(usageMap, walletPassProductId, lineKey) < getMaxPassSlotsForExpandedLine(maxPassesPerItem, p);
191
+ });
192
+ };
193
+ // ================================================================
194
+
195
+ // 辅助函数:计算单张券的 _available_max_amount
196
+ var calculateAvailableMaxAmount = function calculateAvailableMaxAmount(voucher, productsData, itemPassUsage) {
197
+ var config = voucher.config;
198
+ var _ref2 = config !== null && config !== void 0 ? config : {},
199
+ _ref2$maxDeductionAmo = _ref2.maxDeductionAmount,
200
+ maxDeductionAmount = _ref2$maxDeductionAmo === void 0 ? 0 : _ref2$maxDeductionAmo,
201
+ _ref2$allowCrossProdu = _ref2.allowCrossProduct,
202
+ allowCrossProduct = _ref2$allowCrossProdu === void 0 ? true : _ref2$allowCrossProdu,
203
+ _ref2$applicableProdu = _ref2.applicableProductLimit,
204
+ applicableProductLimit = _ref2$applicableProdu === void 0 ? 0 : _ref2$applicableProdu,
205
+ _ref2$deductTaxAndFee = _ref2.deductTaxAndFee,
206
+ deductTaxAndFee = _ref2$deductTaxAndFee === void 0 ? true : _ref2$deductTaxAndFee,
207
+ _ref2$maxPassesPerIte = _ref2.maxPassesPerItem,
208
+ maxPassesPerItem = _ref2$maxPassesPerIte === void 0 ? 0 : _ref2$maxPassesPerIte;
209
+
210
+ // 根据 deductTaxAndFee 配置获取推荐金额
211
+ var recommendedAmount = getRecommendedAmount(voucher);
212
+
213
+ // 基础值 = min(recommendedAmount, maxDeductionAmount)
214
+ var baseAmount = Decimal.min(new Decimal(recommendedAmount), new Decimal(maxDeductionAmount));
215
+
216
+ // 根据券的配置选择使用哪个单价字段和金额字段
217
+ var unitPriceField = deductTaxAndFee ? 'unitPriceWithTax' : 'unitPricePure';
218
+ var amountField = deductTaxAndFee ? 'remainingAmountWithTax' : 'remainingAmountPure';
219
+
220
+ // 获取适用商品列表(只筛选有剩余金额的商品)
221
+ var applicableProducts = getApplicableProducts(voucher, productsData).filter(function (p) {
222
+ return p[amountField].greaterThan(0);
223
+ });
224
+
225
+ // 按 maxPassesPerItem 过滤:排除已达到单商品可用卡券上限的商品
226
+ if (itemPassUsage) {
227
+ applicableProducts = filterByMaxPassesPerItem(applicableProducts, itemPassUsage, voucher.product_id, maxPassesPerItem);
228
+ }
229
+ if (applicableProducts.length === 0) {
230
+ return new Decimal(0);
231
+ }
232
+
233
+ // 根据跨商品配置计算适用商品金额
234
+ var finalApplicableAmount = new Decimal(0);
235
+ if (allowCrossProduct) {
236
+ // 跨商品券:可以抵扣多个商品
237
+ if (applicableProductLimit > 0) {
238
+ // 有数量限制:按剩余金额从高到低排序,按动态计算的可抵扣数量累计直到达到 limit
239
+ var sortedProducts = _toConsumableArray(applicableProducts).sort(function (a, b) {
240
+ return a[amountField].comparedTo(b[amountField]) > 0 ? -1 : 1;
241
+ });
242
+ var remainingLimit = applicableProductLimit;
243
+ var _iterator = _createForOfIteratorHelper(sortedProducts),
244
+ _step;
245
+ try {
246
+ for (_iterator.s(); !(_step = _iterator.n()).done;) {
247
+ var product = _step.value;
248
+ if (remainingLimit <= 0) break;
249
+ // 动态计算当前可抵扣数量 = ceil(剩余金额 / 单价)
250
+ var currentAvailableQty = Math.ceil(product[amountField].dividedBy(product[unitPriceField]).toNumber());
251
+ var deductQty = Math.min(currentAvailableQty, remainingLimit);
252
+ // 实际可抵扣金额 = min(数量 * 单价, 剩余金额)
253
+ var deductAmount = Decimal.min(product[unitPriceField].times(deductQty), product[amountField]);
254
+ finalApplicableAmount = finalApplicableAmount.plus(deductAmount);
255
+ remainingLimit -= deductQty;
256
+ }
257
+ } catch (err) {
258
+ _iterator.e(err);
259
+ } finally {
260
+ _iterator.f();
261
+ }
262
+ } else {
263
+ // 无数量限制:所有适用商品的剩余金额总和
264
+ finalApplicableAmount = applicableProducts.reduce(function (sum, p) {
265
+ return sum.plus(p[amountField]);
266
+ }, new Decimal(0));
267
+ }
268
+ } else {
269
+ // 非跨商品券:选择单价最高的商品(优先抵扣高价商品)
270
+ var maxProduct = applicableProducts.reduce(function (max, p) {
271
+ return p[unitPriceField].greaterThan(max[unitPriceField]) ? p : max;
272
+ });
273
+ // maxPassesPerItem 限制每张券最多抵扣的单位数
274
+ if (maxPassesPerItem > 0) {
275
+ finalApplicableAmount = Decimal.min(maxProduct[unitPriceField].times(maxPassesPerItem), maxProduct[amountField]);
276
+ } else {
277
+ finalApplicableAmount = maxProduct[amountField];
278
+ }
279
+ }
280
+
281
+ // 返回最小值
282
+ return Decimal.min(baseAmount, finalApplicableAmount, remainingOrderAmount);
283
+ };
284
+
285
+ // 辅助函数:判断券是否可用,返回可用性和原因代码
286
+ var isVoucherAvailable = function isVoucherAvailable(voucher, productsData, usedVoucherCounts, itemPassUsage) {
287
+ var config = voucher.config,
288
+ id = voucher.id,
289
+ product_id = voucher.product_id;
290
+
291
+ // 根据 deductTaxAndFee 配置获取推荐金额
292
+ var recommendedAmount = getRecommendedAmount(voucher);
293
+
294
+ // 余额为 0
295
+ if (recommendedAmount <= 0) {
296
+ return {
297
+ isAvailable: false,
298
+ reasonCode: 'not_meet_the_required_conditions'
299
+ };
300
+ }
301
+
302
+ // 订单金额已抵扣完
303
+ if (remainingOrderAmount.lessThanOrEqualTo(0)) {
304
+ return {
305
+ isAvailable: false,
306
+ reasonCode: 'exceeds_the_maximum_deduction_limit'
307
+ };
308
+ }
309
+
310
+ // 适用商品金额已被抵扣完
311
+ var applicableAmount = getApplicableProductsAmount(voucher, productsData);
312
+ if (applicableAmount.lessThanOrEqualTo(0)) {
313
+ return {
314
+ isAvailable: false,
315
+ reasonCode: 'not_meet_the_required_conditions'
316
+ };
317
+ }
318
+
319
+ // 检查 maxUsagePerOrder(同 product_id 的券),0 表示不限制
320
+ if (((config === null || config === void 0 ? void 0 : config.maxUsagePerOrder) || 0) > 0) {
321
+ var usedCount = usedVoucherCounts.get(product_id) || 0;
322
+ if (usedCount >= ((config === null || config === void 0 ? void 0 : config.maxUsagePerOrder) || 0)) {
323
+ return {
324
+ isAvailable: false,
325
+ reasonCode: 'usage_limit_reached'
326
+ };
327
+ }
328
+ }
329
+
330
+ // 检查 maxPassesPerItem:如果所有适用商品都已达到单商品可用卡券上限,则不可用
331
+ var maxPassesPerItem = (config === null || config === void 0 ? void 0 : config.maxPassesPerItem) || 0;
332
+ if (maxPassesPerItem > 0 && itemPassUsage) {
333
+ var _config$deductTaxAndF3;
334
+ var deductTaxAndFee = (_config$deductTaxAndF3 = config === null || config === void 0 ? void 0 : config.deductTaxAndFee) !== null && _config$deductTaxAndF3 !== void 0 ? _config$deductTaxAndF3 : true;
335
+ var amountField = deductTaxAndFee ? 'remainingAmountWithTax' : 'remainingAmountPure';
336
+ var availableAfterPassLimit = getApplicableProducts(voucher, productsData).filter(function (p) {
337
+ return p[amountField].greaterThan(0);
338
+ }).filter(function (p) {
339
+ var lineKey = p._walletPassLineKey;
340
+ if (!lineKey) return true;
341
+ return getItemPassUsage(itemPassUsage, product_id, lineKey) < getMaxPassSlotsForExpandedLine(maxPassesPerItem, p);
342
+ });
343
+ if (availableAfterPassLimit.length === 0) {
344
+ return {
345
+ isAvailable: false,
346
+ reasonCode: 'max_passes_per_item_reached'
347
+ };
348
+ }
349
+ }
350
+ return {
351
+ isAvailable: true
352
+ };
353
+ };
354
+
355
+ // ========== 第一步:为所有券计算初始的 _available_max_amount 和 _unified_available_status ==========
356
+ var usedVoucherCountsForAll = new Map();
357
+ var allVouchersWithStatus = applicableVouchers.map(function (voucher) {
358
+ var _available_max_amount = calculateAvailableMaxAmount(voucher, productsCopy);
359
+ var availabilityResult = isVoucherAvailable(voucher, productsCopy, usedVoucherCountsForAll);
360
+ var _unified_available_status = availabilityResult.isAvailable ? 1 : 0;
361
+ return _objectSpread(_objectSpread({}, voucher), {}, {
362
+ _available_max_amount: _available_max_amount.toNumber(),
363
+ // 转换为数字
364
+ _unified_available_status: _unified_available_status
365
+ }, availabilityResult.reasonCode && {
366
+ reasonCode: availabilityResult.reasonCode
367
+ });
368
+ });
369
+
370
+ // ========== 第二步:计算推荐券列表 ==========
371
+ var recommendedVouchers = [];
372
+ var usedVoucherCounts = new Map(); // 跟踪每个券 ID 的使用次数
373
+
374
+ // 重置商品余额追踪(同时维护含税和不含税两个金额池)
375
+ var productsForRecommendation = expandProductsWithBundleItems(products, true);
376
+ remainingOrderAmount = new Decimal(orderTotalAmount);
377
+ // 追踪推荐阶段每个 Wallet Pass 商品对每个订单商品行的已使用卡券次数(按行唯一键)
378
+ var itemPassUsageMap = new Map();
379
+
380
+ /**
381
+ // 分类券:未跨商品券 vs 跨商品券
382
+ const nonCrossProductVouchers = applicableVouchers.filter(
383
+ (v) => !v.config?.allowCrossProduct,
384
+ );
385
+ const crossProductVouchers = applicableVouchers.filter(
386
+ (v) => v.config?.allowCrossProduct,
387
+ );
388
+ // 按 tag 分组
389
+ const groupByType = (vouchers: any[]) => {
390
+ const groups = new Map<string, any[]>();
391
+ vouchers.forEach((v) => {
392
+ const tag = v.tag;
393
+ if (!groups.has(tag)) {
394
+ groups.set(tag, []);
395
+ }
396
+ groups.get(tag)!.push(v);
397
+ });
398
+ return groups;
399
+ };
400
+ // 对每组券按金额(balance)从大到小排序
401
+ const sortVouchersByAmount = (vouchers: Voucher[]) => {
402
+ return [...vouchers].sort((a, b) => getRecommendedAmount(b) - getRecommendedAmount(a));
403
+ };
404
+ */
405
+
406
+ // 处理单张券的抵扣
407
+ var applyVoucher = function applyVoucher(voucher) {
408
+ // 检查是否可用
409
+ var availabilityCheck = isVoucherAvailable(voucher, productsForRecommendation, usedVoucherCounts, itemPassUsageMap);
410
+ if (!availabilityCheck.isAvailable) {
411
+ return false;
412
+ }
413
+ var config = voucher.config,
414
+ id = voucher.id,
415
+ product_id = voucher.product_id;
416
+ var _ref3 = config !== null && config !== void 0 ? config : {},
417
+ _ref3$maxDeductionAmo = _ref3.maxDeductionAmount,
418
+ maxDeductionAmount = _ref3$maxDeductionAmo === void 0 ? 0 : _ref3$maxDeductionAmo,
419
+ _ref3$allowCrossProdu = _ref3.allowCrossProduct,
420
+ allowCrossProduct = _ref3$allowCrossProdu === void 0 ? true : _ref3$allowCrossProdu,
421
+ _ref3$applicableProdu = _ref3.applicableProductLimit,
422
+ applicableProductLimit = _ref3$applicableProdu === void 0 ? 0 : _ref3$applicableProdu,
423
+ _ref3$deductTaxAndFee = _ref3.deductTaxAndFee,
424
+ deductTaxAndFee = _ref3$deductTaxAndFee === void 0 ? true : _ref3$deductTaxAndFee,
425
+ _ref3$maxPassesPerIte = _ref3.maxPassesPerItem,
426
+ maxPassesPerItem = _ref3$maxPassesPerIte === void 0 ? 0 : _ref3$maxPassesPerIte;
427
+
428
+ // 根据券的配置选择使用哪个单价字段和金额字段
429
+ var unitPriceField = deductTaxAndFee ? 'unitPriceWithTax' : 'unitPricePure';
430
+ var amountField = deductTaxAndFee ? 'remainingAmountWithTax' : 'remainingAmountPure';
431
+
432
+ // 获取适用商品(只筛选有剩余金额的商品)
433
+ var applicableProducts = getApplicableProducts(voucher, productsForRecommendation).filter(function (p) {
434
+ return p[amountField].greaterThan(0);
435
+ });
436
+
437
+ // 按 maxPassesPerItem 过滤:排除已达到单商品可用卡券上限的商品
438
+ applicableProducts = filterByMaxPassesPerItem(applicableProducts, itemPassUsageMap, product_id, maxPassesPerItem);
439
+ if (applicableProducts.length === 0) return false;
440
+
441
+ // ========== 关键修改:在应用券之前,基于当前剩余金额计算 _available_max_amount ==========
442
+ // 优先使用用户手动编辑的金额,否则根据 deductTaxAndFee 配置获取推荐金额
443
+ var usageAmount = typeof voucher.edit_current_amount === 'number' ? voucher.edit_current_amount : getRecommendedAmount(voucher);
444
+ var baseAmount = Decimal.min(new Decimal(usageAmount), new Decimal(maxDeductionAmount));
445
+ var calculatedAvailableMaxAmount = new Decimal(0);
446
+ if (allowCrossProduct) {
447
+ // 跨商品券:按 quantity 限制计算可抵扣金额
448
+ if (applicableProductLimit > 0) {
449
+ // 按剩余金额从高到低排序
450
+ var sortedProducts = _toConsumableArray(applicableProducts).sort(function (a, b) {
451
+ return a[amountField].comparedTo(b[amountField]) > 0 ? -1 : 1;
452
+ });
453
+ var remainingLimit = applicableProductLimit;
454
+ var _iterator2 = _createForOfIteratorHelper(sortedProducts),
455
+ _step2;
456
+ try {
457
+ for (_iterator2.s(); !(_step2 = _iterator2.n()).done;) {
458
+ var product = _step2.value;
459
+ if (remainingLimit <= 0) break;
460
+ // 动态计算当前可抵扣数量 = ceil(剩余金额 / 单价)
461
+ var currentAvailableQty = Math.ceil(product[amountField].dividedBy(product[unitPriceField]).toNumber());
462
+ var deductQty = Math.min(currentAvailableQty, remainingLimit);
463
+ // 实际可抵扣金额 = min(数量 * 单价, 剩余金额)
464
+ var deductAmount = Decimal.min(product[unitPriceField].times(deductQty), product[amountField]);
465
+ calculatedAvailableMaxAmount = calculatedAvailableMaxAmount.plus(deductAmount);
466
+ remainingLimit -= deductQty;
467
+ }
468
+ } catch (err) {
469
+ _iterator2.e(err);
470
+ } finally {
471
+ _iterator2.f();
472
+ }
473
+ } else {
474
+ // 无数量限制:所有适用商品的剩余金额总和
475
+ calculatedAvailableMaxAmount = applicableProducts.reduce(function (sum, p) {
476
+ return sum.plus(p[amountField]);
477
+ }, new Decimal(0));
478
+ }
479
+ } else {
480
+ // 非跨商品券:选择单价最高的商品(优先抵扣高价商品)
481
+ var maxProduct = applicableProducts.reduce(function (max, p) {
482
+ return p[unitPriceField].greaterThan(max[unitPriceField]) ? p : max;
483
+ });
484
+ // maxPassesPerItem 限制每张券最多抵扣的单位数
485
+ if (maxPassesPerItem > 0) {
486
+ calculatedAvailableMaxAmount = Decimal.min(maxProduct[unitPriceField].times(maxPassesPerItem), maxProduct[amountField]);
487
+ } else {
488
+ calculatedAvailableMaxAmount = maxProduct[amountField];
489
+ }
490
+ }
491
+
492
+ // 取最小值:min(recommended_usage_amount, maxDeductionAmount, 适用商品金额, 订单剩余金额)
493
+ var availableMaxAmount = Decimal.min(baseAmount, calculatedAvailableMaxAmount, remainingOrderAmount);
494
+ // ======================================================================================
495
+
496
+ // 计算本次抵扣金额(使用同样的 usageAmount)
497
+ var maxDeduction = Decimal.min(new Decimal(usageAmount), new Decimal(maxDeductionAmount));
498
+ var deductionLeft = maxDeduction;
499
+ var deductionDetails = [];
500
+ if (allowCrossProduct) {
501
+ // 跨商品券:按剩余金额从高到低抵扣,受 applicableProductLimit 限制
502
+ var _sortedProducts = _toConsumableArray(applicableProducts).sort(function (a, b) {
503
+ return a[amountField].comparedTo(b[amountField]) > 0 ? -1 : 1;
504
+ });
505
+ var _remainingLimit = applicableProductLimit > 0 ? applicableProductLimit : Infinity;
506
+ var _iterator3 = _createForOfIteratorHelper(_sortedProducts),
507
+ _step3;
508
+ try {
509
+ for (_iterator3.s(); !(_step3 = _iterator3.n()).done;) {
510
+ var _product = _step3.value;
511
+ if (deductionLeft.lessThanOrEqualTo(0) || _remainingLimit <= 0) break;
512
+
513
+ // 动态计算当前可抵扣数量 = ceil(剩余金额 / 单价)
514
+ var _currentAvailableQty = Math.ceil(_product[amountField].dividedBy(_product[unitPriceField]).toNumber());
515
+ var availableQty = Math.min(_currentAvailableQty, _remainingLimit);
516
+
517
+ // 计算本商品最大可抵扣金额 = min(数量 * 单价, 剩余金额)
518
+ var maxDeductForProduct = Decimal.min(_product[unitPriceField].times(availableQty), _product[amountField]);
519
+ var actualDeductAmount = Decimal.min(deductionLeft, maxDeductForProduct);
520
+
521
+ // 计算实际抵扣的数量(用于记录和配额计算)
522
+ var actualDeductQty = Math.ceil(actualDeductAmount.dividedBy(_product[unitPriceField]).toNumber());
523
+
524
+ // 只更新商品的剩余金额,不更新 remainingQuantity
525
+ _product[amountField] = _product[amountField].minus(actualDeductAmount);
526
+ deductionLeft = deductionLeft.minus(actualDeductAmount);
527
+ _remainingLimit -= actualDeductQty;
528
+ deductionDetails.push({
529
+ product_id: _product.product_id,
530
+ parent_product_id: _product.parent_product_id || null,
531
+ is_bundle_item: _product.is_bundle_item || false,
532
+ lineKey: _product._walletPassLineKey,
533
+ orderLineQuantity: getExpandedOrderLineQuantity(_product),
534
+ deductAmount: actualDeductAmount.toNumber(),
535
+ // 转换为数字
536
+ deductQuantity: actualDeductQty // 抵扣涉及的数量(用于记录)
537
+ });
538
+ }
539
+ } catch (err) {
540
+ _iterator3.e(err);
541
+ } finally {
542
+ _iterator3.f();
543
+ }
544
+ } else {
545
+ // 非跨商品券:选择单价最高的商品(优先抵扣高价商品)
546
+ var targetProduct = applicableProducts.reduce(function (max, p) {
547
+ return p[unitPriceField].greaterThan(max[unitPriceField]) ? p : max;
548
+ });
549
+
550
+ // maxPassesPerItem 限制每张券最多抵扣的单位数
551
+ var _maxDeductForProduct = targetProduct[amountField];
552
+ if (maxPassesPerItem > 0) {
553
+ _maxDeductForProduct = Decimal.min(targetProduct[unitPriceField].times(maxPassesPerItem), targetProduct[amountField]);
554
+ }
555
+ var _actualDeductAmount = Decimal.min(deductionLeft, _maxDeductForProduct);
556
+
557
+ // 计算实际抵扣的数量
558
+ var _actualDeductQty = Math.ceil(_actualDeductAmount.dividedBy(targetProduct[unitPriceField]).toNumber());
559
+
560
+ // 只更新商品的剩余金额,不更新 remainingQuantity
561
+ targetProduct[amountField] = targetProduct[amountField].minus(_actualDeductAmount);
562
+ deductionLeft = deductionLeft.minus(_actualDeductAmount);
563
+ deductionDetails.push({
564
+ product_id: targetProduct.product_id,
565
+ parent_product_id: targetProduct.parent_product_id || null,
566
+ is_bundle_item: targetProduct.is_bundle_item || false,
567
+ lineKey: targetProduct._walletPassLineKey,
568
+ orderLineQuantity: getExpandedOrderLineQuantity(targetProduct),
569
+ deductAmount: _actualDeductAmount.toNumber(),
570
+ deductQuantity: _actualDeductQty
571
+ });
572
+ }
573
+ var totalDeducted = maxDeduction.minus(deductionLeft);
574
+ if (totalDeducted.greaterThan(0)) {
575
+ // 更新订单剩余金额
576
+ remainingOrderAmount = remainingOrderAmount.minus(totalDeducted);
577
+
578
+ // 更新券使用次数(按 product_id 统计)
579
+ usedVoucherCounts.set(product_id, (usedVoucherCounts.get(product_id) || 0) + 1);
580
+ applyMaxPassUsageIncrements(itemPassUsageMap, product_id, maxPassesPerItem, deductionDetails);
581
+
582
+ // 添加到推荐列表(包含基于当前剩余金额计算的 available_max_amount)
583
+ recommendedVouchers.push(_objectSpread(_objectSpread({}, voucher), {}, {
584
+ actualDeduction: totalDeducted.toNumber(),
585
+ // 转换为数字
586
+ deductionDetails: deductionDetails,
587
+ _available_max_amount: availableMaxAmount.toNumber(),
588
+ // 转换为数字
589
+ _unified_available_status: 1
590
+ }));
591
+ return true;
592
+ }
593
+ return false;
594
+ };
595
+
596
+ /**
597
+ // 第一轮:处理未跨商品券
598
+ const nonCrossGroups = groupByType(nonCrossProductVouchers);
599
+ nonCrossGroups.forEach((vouchersInGroup) => {
600
+ const sortedVouchers = sortVouchersByAmount(vouchersInGroup);
601
+ sortedVouchers.forEach((voucher) => {
602
+ applyVoucher(voucher);
603
+ });
604
+ });
605
+ // 第二轮:处理跨商品券
606
+ const crossGroups = groupByType(crossProductVouchers);
607
+ crossGroups.forEach((vouchersInGroup) => {
608
+ const sortedVouchers = sortVouchersByAmount(vouchersInGroup);
609
+ sortedVouchers.forEach((voucher) => {
610
+ applyVoucher(voucher);
611
+ });
612
+ });
613
+ */
614
+
615
+ // 直接按 applicableVouchers 的顺序处理所有券
616
+ applicableVouchers.forEach(function (voucher) {
617
+ applyVoucher(voucher);
618
+ });
619
+
620
+ // recommendedVouchers 已经包含了基于应用时计算的 _available_max_amount 和 _unified_available_status
621
+
622
+ // ========== 第三步:构建 all 数组,以 recommended 为基础 ==========
623
+ // 创建 recommended 券的 Map,以 id 为 key
624
+ var recommendedMap = new Map();
625
+ recommendedVouchers.forEach(function (v) {
626
+ recommendedMap.set(v.id, v);
627
+ });
628
+
629
+ // 遍历所有券,如果在 recommended 中则使用增强数据,否则使用基础数据
630
+ var allWithEnhancedData = allVouchersWithStatus.map(function (voucher) {
631
+ if (recommendedMap.has(voucher.id)) {
632
+ // 使用 recommended 中的增强数据
633
+ return recommendedMap.get(voucher.id);
634
+ } else {
635
+ // 不在 recommended 中的券,补充基础字段
636
+ return _objectSpread(_objectSpread({}, voucher), {}, {
637
+ actualDeduction: 0,
638
+ deductionDetails: []
639
+ // 保留 reasonCode(如果不可用的话)
640
+ });
641
+ }
642
+ });
643
+ return {
644
+ recommended: recommendedVouchers,
645
+ transformList: allWithEnhancedData
646
+ };
647
+ }
648
+
649
+ /**
650
+ * 重新计算优惠券状态(基于已选券的增量计算)
651
+ * @param allVouchers 所有原始券列表
652
+ * @param selectedVouchers 已选中的券列表(按选中顺序)
653
+ * @param orderTotalAmount 订单总金额
654
+ * @param products 订单商品列表
655
+ * @returns 返回更新后的所有券列表和已选券的详细抵扣信息
656
+ */
657
+ export function recalculateVouchers(allVouchers, selectedVouchers, orderTotalAmount, products) {
658
+ // 深拷贝并拆分商品列表(包含主商品和原价子商品,同时维护含税和不含税两个金额池),用于计算
659
+ var productsForCalc = expandProductsWithBundleItems(products, true);
660
+ var remainingOrderAmount = new Decimal(orderTotalAmount);
661
+ var selectedWithDetails = [];
662
+
663
+ // 追踪每个 Wallet Pass 商品对每个订单商品行的已使用卡券次数(按行唯一键)
664
+ var itemPassUsageMap = new Map();
665
+ var getItemPassUsageRecalc = function getItemPassUsageRecalc(walletPassProductId, lineKey) {
666
+ var _itemPassUsageMap$get;
667
+ return ((_itemPassUsageMap$get = itemPassUsageMap.get(walletPassProductId)) === null || _itemPassUsageMap$get === void 0 ? void 0 : _itemPassUsageMap$get.get(lineKey)) || 0;
668
+ };
669
+ var filterByMaxPassesPerItemRecalc = function filterByMaxPassesPerItemRecalc(products, walletPassProductId, maxPassesPerItem) {
670
+ if (maxPassesPerItem <= 0) return products;
671
+ return products.filter(function (p) {
672
+ var lineKey = p._walletPassLineKey;
673
+ if (!lineKey) return true;
674
+ return getItemPassUsageRecalc(walletPassProductId, lineKey) < getMaxPassSlotsForExpandedLine(maxPassesPerItem, p);
675
+ });
676
+ };
677
+
678
+ // 第一步:按顺序应用已选中的券,计算实际抵扣
679
+ selectedVouchers.forEach(function (selectedVoucher) {
680
+ var config = selectedVoucher.config,
681
+ id = selectedVoucher.id;
682
+ var maxDeductionAmount = config.maxDeductionAmount,
683
+ allowCrossProduct = config.allowCrossProduct,
684
+ applicableProductLimit = config.applicableProductLimit,
685
+ _config$deductTaxAndF4 = config.deductTaxAndFee,
686
+ deductTaxAndFee = _config$deductTaxAndF4 === void 0 ? true : _config$deductTaxAndF4,
687
+ _config$maxPassesPerI = config.maxPassesPerItem,
688
+ maxPassesPerItem = _config$maxPassesPerI === void 0 ? 0 : _config$maxPassesPerI;
689
+
690
+ // 根据券的配置选择使用哪个单价字段和金额字段
691
+ var unitPriceField = deductTaxAndFee ? 'unitPriceWithTax' : 'unitPricePure';
692
+ var amountField = deductTaxAndFee ? 'remainingAmountWithTax' : 'remainingAmountPure';
693
+
694
+ // 获取适用商品(只筛选有剩余金额的商品)
695
+ var applicableProducts = getApplicableProducts(selectedVoucher, productsForCalc).filter(function (p) {
696
+ return p[amountField].greaterThan(0);
697
+ });
698
+
699
+ // 按 maxPassesPerItem 过滤:排除已达到该行券次上限的展开行
700
+ applicableProducts = filterByMaxPassesPerItemRecalc(applicableProducts, selectedVoucher.product_id, maxPassesPerItem);
701
+ if (applicableProducts.length === 0) {
702
+ // 无适用商品,跳过
703
+ selectedWithDetails.push(_objectSpread(_objectSpread({}, selectedVoucher), {}, {
704
+ actualDeduction: 0,
705
+ deductionDetails: [],
706
+ _available_max_amount: 0,
707
+ _unified_available_status: 0,
708
+ reasonCode: 'not_meet_the_required_conditions'
709
+ }));
710
+ return;
711
+ }
712
+
713
+ // 计算本次抵扣金额
714
+ // 优先使用用户手动编辑的金额,否则根据 deductTaxAndFee 配置获取推荐金额
715
+ var usageAmount = typeof selectedVoucher.edit_current_amount === 'number' ? selectedVoucher.edit_current_amount : getRecommendedAmount(selectedVoucher);
716
+ var maxDeduction = Decimal.min(new Decimal(usageAmount), new Decimal(maxDeductionAmount), remainingOrderAmount);
717
+ var deductionLeft = maxDeduction;
718
+ var deductionDetails = [];
719
+ if (allowCrossProduct) {
720
+ // 跨商品券:按剩余金额从高到低抵扣,受 applicableProductLimit 限制
721
+ var sortedProducts = _toConsumableArray(applicableProducts).sort(function (a, b) {
722
+ return a[amountField].comparedTo(b[amountField]) > 0 ? -1 : 1;
723
+ });
724
+ var remainingLimit = applicableProductLimit > 0 ? applicableProductLimit : Infinity;
725
+ var _iterator4 = _createForOfIteratorHelper(sortedProducts),
726
+ _step4;
727
+ try {
728
+ for (_iterator4.s(); !(_step4 = _iterator4.n()).done;) {
729
+ var product = _step4.value;
730
+ if (deductionLeft.lessThanOrEqualTo(0) || remainingLimit <= 0) break;
731
+
732
+ // 动态计算当前可抵扣数量 = ceil(剩余金额 / 单价)
733
+ var currentAvailableQty = Math.ceil(product[amountField].dividedBy(product[unitPriceField]).toNumber());
734
+ var availableQty = Math.min(currentAvailableQty, remainingLimit);
735
+
736
+ // 计算本商品最大可抵扣金额 = min(数量 * 单价, 剩余金额)
737
+ var maxDeductForProduct = Decimal.min(product[unitPriceField].times(availableQty), product[amountField]);
738
+ var actualDeductAmount = Decimal.min(deductionLeft, maxDeductForProduct);
739
+
740
+ // 计算实际抵扣的数量(用于记录和配额计算)
741
+ var actualDeductQty = Math.ceil(actualDeductAmount.dividedBy(product[unitPriceField]).toNumber());
742
+
743
+ // 只更新商品的剩余金额,不更新 remainingQuantity
744
+ product[amountField] = product[amountField].minus(actualDeductAmount);
745
+ deductionLeft = deductionLeft.minus(actualDeductAmount);
746
+ remainingLimit -= actualDeductQty;
747
+ deductionDetails.push({
748
+ product_id: product.product_id,
749
+ parent_product_id: product.parent_product_id || null,
750
+ is_bundle_item: product.is_bundle_item || false,
751
+ lineKey: product._walletPassLineKey,
752
+ orderLineQuantity: getExpandedOrderLineQuantity(product),
753
+ deductAmount: actualDeductAmount.toNumber(),
754
+ // 转换为数字
755
+ deductQuantity: actualDeductQty // 抵扣涉及的数量(用于记录)
756
+ });
757
+ }
758
+ } catch (err) {
759
+ _iterator4.e(err);
760
+ } finally {
761
+ _iterator4.f();
762
+ }
763
+ } else {
764
+ // 非跨商品券:选择单价最高的商品(优先抵扣高价商品)
765
+ var targetProduct = applicableProducts.reduce(function (max, p) {
766
+ return p[unitPriceField].greaterThan(max[unitPriceField]) ? p : max;
767
+ });
768
+
769
+ // maxPassesPerItem 限制每张券最多抵扣的单位数
770
+ var _maxDeductForProduct2 = targetProduct[amountField];
771
+ if (maxPassesPerItem > 0) {
772
+ _maxDeductForProduct2 = Decimal.min(targetProduct[unitPriceField].times(maxPassesPerItem), targetProduct[amountField]);
773
+ }
774
+ var _actualDeductAmount2 = Decimal.min(deductionLeft, _maxDeductForProduct2);
775
+
776
+ // 计算实际抵扣的数量
777
+ var _actualDeductQty2 = Math.ceil(_actualDeductAmount2.dividedBy(targetProduct[unitPriceField]).toNumber());
778
+
779
+ // 只更新商品的剩余金额,不更新 remainingQuantity
780
+ targetProduct[amountField] = targetProduct[amountField].minus(_actualDeductAmount2);
781
+ deductionLeft = deductionLeft.minus(_actualDeductAmount2);
782
+ deductionDetails.push({
783
+ product_id: targetProduct.product_id,
784
+ parent_product_id: targetProduct.parent_product_id || null,
785
+ is_bundle_item: targetProduct.is_bundle_item || false,
786
+ lineKey: targetProduct._walletPassLineKey,
787
+ orderLineQuantity: getExpandedOrderLineQuantity(targetProduct),
788
+ deductAmount: _actualDeductAmount2.toNumber(),
789
+ deductQuantity: _actualDeductQty2
790
+ });
791
+ }
792
+ var totalDeducted = maxDeduction.minus(deductionLeft);
793
+
794
+ // 更新订单剩余金额
795
+ remainingOrderAmount = remainingOrderAmount.minus(totalDeducted);
796
+ applyMaxPassUsageIncrements(itemPassUsageMap, selectedVoucher.product_id, maxPassesPerItem, deductionDetails);
797
+ selectedWithDetails.push(_objectSpread(_objectSpread({}, selectedVoucher), {}, {
798
+ actualDeduction: totalDeducted.toNumber(),
799
+ // 转换为数字
800
+ deductionDetails: deductionDetails,
801
+ _available_max_amount: totalDeducted.toNumber(),
802
+ // 转换为数字
803
+ _unified_available_status: totalDeducted.greaterThan(0) ? 1 : 0
804
+ }));
805
+ });
806
+
807
+ // 第二步:为所有券重新计算 _available_max_amount 和 _unified_available_status
808
+ var selectedIds = new Set(selectedVouchers.map(function (v) {
809
+ return v.id;
810
+ }));
811
+ var usedVoucherCounts = new Map();
812
+ selectedVouchers.forEach(function (v) {
813
+ usedVoucherCounts.set(v.product_id, (usedVoucherCounts.get(v.product_id) || 0) + 1);
814
+ });
815
+ var allWithUpdatedStatus = allVouchers.map(function (voucher) {
816
+ // 如果是已选中的券,使用详细计算结果
817
+ if (selectedIds.has(voucher.id)) {
818
+ var selectedDetail = selectedWithDetails.find(function (v) {
819
+ return v.id === voucher.id;
820
+ });
821
+ return selectedDetail || voucher;
822
+ }
823
+
824
+ // 未选中的券,重新计算其可用状态
825
+ var config = voucher.config,
826
+ id = voucher.id,
827
+ product_id = voucher.product_id;
828
+ var maxDeductionAmount = config.maxDeductionAmount,
829
+ allowCrossProduct = config.allowCrossProduct,
830
+ applicableProductLimit = config.applicableProductLimit,
831
+ _config$deductTaxAndF5 = config.deductTaxAndFee,
832
+ deductTaxAndFee = _config$deductTaxAndF5 === void 0 ? true : _config$deductTaxAndF5,
833
+ _config$maxPassesPerI2 = config.maxPassesPerItem,
834
+ maxPassesPerItem = _config$maxPassesPerI2 === void 0 ? 0 : _config$maxPassesPerI2;
835
+
836
+ // 根据券的配置选择使用哪个单价字段和金额字段
837
+ var unitPriceField = deductTaxAndFee ? 'unitPriceWithTax' : 'unitPricePure';
838
+ var amountField = deductTaxAndFee ? 'remainingAmountWithTax' : 'remainingAmountPure';
839
+
840
+ // 根据 deductTaxAndFee 配置获取推荐金额
841
+ var recommendedAmount = getRecommendedAmount(voucher);
842
+
843
+ // 检查基本可用性
844
+ var isAvailable = true;
845
+ var calculatedMaxAmount = new Decimal(0);
846
+ var reasonCode;
847
+ if (recommendedAmount <= 0) {
848
+ isAvailable = false;
849
+ reasonCode = 'not_meet_the_required_conditions';
850
+ } else if (remainingOrderAmount.lessThanOrEqualTo(0)) {
851
+ isAvailable = false;
852
+ reasonCode = 'exceeds_the_maximum_deduction_limit';
853
+ } else {
854
+ // 检查 maxUsagePerOrder(按 product_id 统计),0 表示不限制
855
+ if (config.maxUsagePerOrder > 0) {
856
+ var usedCount = usedVoucherCounts.get(product_id) || 0;
857
+ if (usedCount >= config.maxUsagePerOrder) {
858
+ isAvailable = false;
859
+ reasonCode = 'usage_limit_reached';
860
+ }
861
+ }
862
+ if (isAvailable) {
863
+ // 获取适用商品(只筛选有剩余金额的商品)
864
+ var applicableProducts = getApplicableProducts(voucher, productsForCalc).filter(function (p) {
865
+ return p[amountField].greaterThan(0);
866
+ });
867
+
868
+ // 按 maxPassesPerItem 过滤:排除已达到单商品可用卡券上限的商品
869
+ applicableProducts = filterByMaxPassesPerItemRecalc(applicableProducts, product_id, maxPassesPerItem);
870
+ if (applicableProducts.length === 0) {
871
+ isAvailable = false;
872
+ reasonCode = 'not_meet_the_required_conditions';
873
+ } else {
874
+ // 计算 _available_max_amount
875
+ var baseAmount = Decimal.min(new Decimal(recommendedAmount), new Decimal(maxDeductionAmount));
876
+ if (allowCrossProduct) {
877
+ if (applicableProductLimit > 0) {
878
+ // 按剩余金额从高到低排序,按动态计算的可抵扣数量累计直到达到 limit
879
+ var sortedProducts = _toConsumableArray(applicableProducts).sort(function (a, b) {
880
+ return a[amountField].comparedTo(b[amountField]) > 0 ? -1 : 1;
881
+ });
882
+ var remainingLimit = applicableProductLimit;
883
+ var _iterator5 = _createForOfIteratorHelper(sortedProducts),
884
+ _step5;
885
+ try {
886
+ for (_iterator5.s(); !(_step5 = _iterator5.n()).done;) {
887
+ var product = _step5.value;
888
+ if (remainingLimit <= 0) break;
889
+ // 动态计算当前可抵扣数量 = ceil(剩余金额 / 单价)
890
+ var currentAvailableQty = Math.ceil(product[amountField].dividedBy(product[unitPriceField]).toNumber());
891
+ var deductQty = Math.min(currentAvailableQty, remainingLimit);
892
+ // 实际可抵扣金额 = min(数量 * 单价, 剩余金额)
893
+ var deductAmount = Decimal.min(product[unitPriceField].times(deductQty), product[amountField]);
894
+ calculatedMaxAmount = calculatedMaxAmount.plus(deductAmount);
895
+ remainingLimit -= deductQty;
896
+ }
897
+ } catch (err) {
898
+ _iterator5.e(err);
899
+ } finally {
900
+ _iterator5.f();
901
+ }
902
+ } else {
903
+ // 无数量限制:所有适用商品的剩余金额总和
904
+ calculatedMaxAmount = applicableProducts.reduce(function (sum, p) {
905
+ return sum.plus(p[amountField]);
906
+ }, new Decimal(0));
907
+ }
908
+ } else {
909
+ // 非跨商品券:选择单价最高的商品(优先抵扣高价商品)
910
+ var maxProduct = applicableProducts.reduce(function (max, p) {
911
+ return p[unitPriceField].greaterThan(max[unitPriceField]) ? p : max;
912
+ });
913
+ // maxPassesPerItem 限制每张券最多抵扣的单位数
914
+ if (maxPassesPerItem > 0) {
915
+ calculatedMaxAmount = Decimal.min(maxProduct[unitPriceField].times(maxPassesPerItem), maxProduct[amountField]);
916
+ } else {
917
+ calculatedMaxAmount = maxProduct[amountField];
918
+ }
919
+ }
920
+ calculatedMaxAmount = Decimal.min(baseAmount, calculatedMaxAmount, remainingOrderAmount);
921
+ if (calculatedMaxAmount.lessThanOrEqualTo(0)) {
922
+ isAvailable = false;
923
+ reasonCode = 'exceeds_the_maximum_deduction_limit';
924
+ }
925
+ }
926
+ }
927
+ }
928
+ return _objectSpread(_objectSpread({}, voucher), {}, {
929
+ _available_max_amount: calculatedMaxAmount.toNumber(),
930
+ // 转换为数字
931
+ _unified_available_status: isAvailable ? 1 : 0
932
+ }, reasonCode && {
933
+ reasonCode: reasonCode
934
+ });
935
+ });
936
+ return {
937
+ allWithUpdatedStatus: allWithUpdatedStatus,
938
+ selectedWithDetails: selectedWithDetails
939
+ };
940
+ }
941
+
942
+ /**
943
+ * 获取主商品价格(单价,不含舍入余数)
944
+ * @param product 商品
945
+ * @param isDeductTaxAndFee 是否抵扣税费与附加费
946
+ * @returns 商品单价
947
+ */
948
+ export var getMainProductPrice = function getMainProductPrice(product, isDeductTaxAndFee) {
949
+ var _product$metadata, _product$metadata2, _product$metadata3, _product$metadata4, _product$metadata5;
950
+ var mainProductPrice = new Decimal((product === null || product === void 0 ? void 0 : product.main_product_selling_price) || ((_product$metadata = product.metadata) === null || _product$metadata === void 0 ? void 0 : _product$metadata.main_product_selling_price) || 0);
951
+ var _iterator6 = _createForOfIteratorHelper((product === null || product === void 0 ? void 0 : product.product_bundle) || []),
952
+ _step6;
953
+ try {
954
+ for (_iterator6.s(); !(_step6 = _iterator6.n()).done;) {
955
+ var bundleItem = _step6.value;
956
+ // 子商品是加价或减价才需要计算
957
+ if (getBundleItemIsMarkupOrDiscountPrice(bundleItem)) {
958
+ var _bundleItem$bundle_se;
959
+ // 子商品价格
960
+ var bundleItemPrice = new Decimal((_bundleItem$bundle_se = bundleItem.bundle_selling_price) !== null && _bundleItem$bundle_se !== void 0 ? _bundleItem$bundle_se : 0);
961
+ mainProductPrice = mainProductPrice.add(bundleItemPrice.times(bundleItem.num));
962
+ }
963
+ }
964
+ } catch (err) {
965
+ _iterator6.e(err);
966
+ } finally {
967
+ _iterator6.f();
968
+ }
969
+ var taxFee = new Decimal((product === null || product === void 0 ? void 0 : product.tax_fee) || (product === null || product === void 0 || (_product$metadata2 = product.metadata) === null || _product$metadata2 === void 0 ? void 0 : _product$metadata2.main_product_attached_bundle_tax_fee) || 0).add((product === null || product === void 0 || (_product$metadata3 = product.metadata) === null || _product$metadata3 === void 0 ? void 0 : _product$metadata3.tax_fee_rounding_remainder) || 0);
970
+ if (product.is_price_include_tax === 1) {
971
+ taxFee = new Decimal(0);
972
+ }
973
+
974
+ // 税费
975
+ // 附加费
976
+ var surchargeFee = new Decimal((product === null || product === void 0 || (_product$metadata4 = product.metadata) === null || _product$metadata4 === void 0 ? void 0 : _product$metadata4.main_product_attached_bundle_surcharge_fee) || 0).add((product === null || product === void 0 || (_product$metadata5 = product.metadata) === null || _product$metadata5 === void 0 ? void 0 : _product$metadata5.surcharge_rounding_remainder) || 0);
977
+
978
+ // 税费附加费总额
979
+ var taxAndFeeTotal = taxFee.add(surchargeFee);
980
+
981
+ // 如果需要抵扣税费与附加费,则加上税费附加费总额
982
+ if (isDeductTaxAndFee) {
983
+ mainProductPrice = mainProductPrice.add(taxAndFeeTotal);
984
+ }
985
+ return mainProductPrice.toNumber();
986
+ };
987
+
988
+ /**
989
+ * 获取套餐子商品价格(不含舍入余数)
990
+ * @param bundleItem 套餐子商品
991
+ * @param parentQuantity 父商品数量
992
+ * @param isDeductTaxAndFee 是否抵扣税费与附加费
993
+ * @returns 子商品总价格(不含舍入余数)
994
+ */
995
+ export var getBundleItemPrice = function getBundleItemPrice(bundleItem, parentQuantity, isDeductTaxAndFee) {
996
+ var _bundleItem$bundle_se2;
997
+ // 子商品基础价格 = bundle_selling_price * num * 主商品数量
998
+ var totalQuantity = bundleItem.num * parentQuantity;
999
+ var bundleItemPrice = new Decimal((_bundleItem$bundle_se2 = bundleItem.bundle_selling_price) !== null && _bundleItem$bundle_se2 !== void 0 ? _bundleItem$bundle_se2 : 0).times(totalQuantity);
1000
+
1001
+ // 根据 deductTaxAndFee 配置决定是否加上税费和附加费
1002
+ if (isDeductTaxAndFee) {
1003
+ var _bundleItem$tax_fee, _bundleItem$metadata$, _bundleItem$metadata;
1004
+ // 税费
1005
+ var taxFee = new Decimal((_bundleItem$tax_fee = bundleItem.tax_fee) !== null && _bundleItem$tax_fee !== void 0 ? _bundleItem$tax_fee : 0).times(totalQuantity);
1006
+ if (bundleItem.is_price_include_tax === 1) {
1007
+ taxFee = new Decimal(0);
1008
+ }
1009
+ // 附加费
1010
+ var surchargeFee = new Decimal((_bundleItem$metadata$ = (_bundleItem$metadata = bundleItem.metadata) === null || _bundleItem$metadata === void 0 ? void 0 : _bundleItem$metadata.surcharge_fee) !== null && _bundleItem$metadata$ !== void 0 ? _bundleItem$metadata$ : 0).times(totalQuantity);
1011
+ bundleItemPrice = bundleItemPrice.add(taxFee).add(surchargeFee);
1012
+ }
1013
+ return bundleItemPrice.toNumber();
1014
+ };
1015
+
1016
+ /**
1017
+ * 将商品数据拆分,包含主商品和原价子商品
1018
+ * @param products 原始商品列表
1019
+ * @param deductTaxAndFee 是否抵扣税费与附加费(已废弃,保留参数以兼容旧代码)
1020
+ * @returns 拆分后的商品数据(包含主商品和子商品,同时维护含税和不含税两个金额池,以及单价和剩余数量)
1021
+ */
1022
+ var expandProductsWithBundleItems = function expandProductsWithBundleItems(products, deductTaxAndFee) {
1023
+ var expandedProducts = [];
1024
+ products.forEach(function (product, indexInOrder) {
1025
+ var productQuantity = getProductQuantity(product);
1026
+ var parentLineKey = resolveWalletPassLineKey(product, indexInOrder);
1027
+
1028
+ // 计算主商品单价(含税和不含税)
1029
+ var unitPriceWithTax = getMainProductPrice(product, true);
1030
+ var unitPricePure = getMainProductPrice(product, false);
1031
+
1032
+ // 1. 添加主商品(同时计算含税和不含税两个金额,以及单价和剩余数量)
1033
+ expandedProducts.push(_objectSpread(_objectSpread({}, product), {}, {
1034
+ is_bundle_item: false,
1035
+ parent_product_id: null,
1036
+ _walletPassLineKey: parentLineKey,
1037
+ _orderLineQuantity: productQuantity,
1038
+ // 单价(用于按 quantity 计算抵扣)
1039
+ unitPriceWithTax: new Decimal(unitPriceWithTax),
1040
+ unitPricePure: new Decimal(unitPricePure),
1041
+ // 剩余可抵扣数量
1042
+ remainingQuantity: productQuantity,
1043
+ // 含税费的剩余金额
1044
+ remainingAmountWithTax: new Decimal(unitPriceWithTax).times(productQuantity),
1045
+ // 纯商品金额(不含税费)
1046
+ remainingAmountPure: new Decimal(unitPricePure).times(productQuantity)
1047
+ }));
1048
+
1049
+ // 2. 添加原价子商品(作为独立商品项)
1050
+ if (product.product_bundle && product.product_bundle.length > 0) {
1051
+ product.product_bundle.forEach(function (bundleItem) {
1052
+ if (getBundleItemIsOriginalPrice(bundleItem)) {
1053
+ var bundleQuantity = bundleItem.num * productQuantity;
1054
+ var bundleLineKey = "".concat(parentLineKey, "#bundle:").concat(bundleItem.bundle_id, ":").concat(bundleItem.bundle_product_id);
1055
+ // 计算子商品单价(注意:getBundleItemPrice 返回的是总价,需要除以数量得到单价)
1056
+ var bundleUnitPriceWithTax = new Decimal(getBundleItemPrice(bundleItem, 1, true)).dividedBy(bundleItem.num);
1057
+ var bundleUnitPricePure = new Decimal(getBundleItemPrice(bundleItem, 1, false)).dividedBy(bundleItem.num);
1058
+
1059
+ // 原价子商品作为独立商品
1060
+ expandedProducts.push(_objectSpread(_objectSpread({}, bundleItem), {}, {
1061
+ product_id: bundleItem.bundle_product_id,
1062
+ // 使用 bundle_product_id 作为 product_id
1063
+ is_bundle_item: true,
1064
+ parent_product_id: product.product_id,
1065
+ quantity: bundleQuantity,
1066
+ // 子商品数量 * 主商品数量
1067
+ _walletPassLineKey: bundleLineKey,
1068
+ _orderLineQuantity: bundleQuantity,
1069
+ // 单价(用于按 quantity 计算抵扣)
1070
+ unitPriceWithTax: bundleUnitPriceWithTax,
1071
+ unitPricePure: bundleUnitPricePure,
1072
+ // 剩余可抵扣数量
1073
+ remainingQuantity: bundleQuantity,
1074
+ // 含税费的剩余金额
1075
+ remainingAmountWithTax: new Decimal(getBundleItemPrice(bundleItem, productQuantity, true)),
1076
+ // 纯商品金额(不含税费)
1077
+ remainingAmountPure: new Decimal(getBundleItemPrice(bundleItem, productQuantity, false))
1078
+ }));
1079
+ }
1080
+ });
1081
+ }
1082
+ });
1083
+ return expandedProducts;
1084
+ };
1085
+
1086
+ // bundle商品是否是原价
1087
+ export var getBundleItemIsOriginalPrice = function getBundleItemIsOriginalPrice(item) {
1088
+ return (item === null || item === void 0 ? void 0 : item.price_type) === 'markup' && (item === null || item === void 0 ? void 0 : item.price_type_ext) === 'product_price';
1089
+ };
1090
+
1091
+ // bundle商品是否是加价
1092
+ export var getBundleItemIsMarkupPrice = function getBundleItemIsMarkupPrice(item) {
1093
+ return (item === null || item === void 0 ? void 0 : item.price_type) === 'markup' && ((item === null || item === void 0 ? void 0 : item.price_type_ext) === '' || !(item !== null && item !== void 0 && item.price_type_ext));
1094
+ };
1095
+
1096
+ // bundle商品是否是减价
1097
+ export var getBundleItemIsDiscountPrice = function getBundleItemIsDiscountPrice(item) {
1098
+ return (item === null || item === void 0 ? void 0 : item.price_type) === 'markdown' && ((item === null || item === void 0 ? void 0 : item.price_type_ext) === '' || !(item !== null && item !== void 0 && item.price_type_ext));
1099
+ };
1100
+
1101
+ // bundle商品是否是加价或减价
1102
+ export var getBundleItemIsMarkupOrDiscountPrice = function getBundleItemIsMarkupOrDiscountPrice(item) {
1103
+ return getBundleItemIsMarkupPrice(item) || getBundleItemIsDiscountPrice(item);
1104
+ };