order = $order; $this->default_refund_type = $order->ext['refund_type'] ?? 'back'; } /** * 全额退款(无条件退款, 优惠券积分全退) * * @param \think\Model $user * @param string $remark * @return void */ public function fullRefund($user = null, $data = []) { $items = OrderItem::where('order_id', $this->order->id)->lock(true)->select(); foreach ($items as $key => $item) { if (in_array($item['refund_status'], [ OrderItem::REFUND_STATUS_AGREE, OrderItem::REFUND_STATUS_COMPLETED, ])) { error_stop('订单有退款,不能全额退款'); } } // 返还库存,减少销量 $stockSale = new StockSale(); $stockSale->backStockSale($this->order, $items); if ($this->order->apply_refund_status == Order::APPLY_REFUND_STATUS_APPLY) { // 如果订单申请了全额退款,这里将全额退款状态改为 已完成 $this->order->apply_refund_status = Order::APPLY_REFUND_STATUS_FINISH; $this->order->save(); } if ($this->order->coupon_id) { // 订单使用了优惠券,退回用户优惠券 $this->backUserCoupon($this->order->coupon_id); } if ($this->order->activity_id || $this->order->promo_types) { // 有活动,执行活动失败 ActivityFacade::buyFail($this->order, 'refund'); } $total_refund_fee = '0'; // 已退金额 foreach ($items as $key => $item) { $is_last = (count($items) - 1) == $key ? true : false; // 是否最后一个商品 // 计算 refund_fee $refund_fee = $this->calcRefundFee($item, $total_refund_fee, $is_last); $item->refund_status = OrderItem::REFUND_STATUS_AGREE; // 同意退款 $item->refund_fee = $refund_fee; // 实时计算,总 退款金额 等于 商品实际支付金额 $item->refund_type = $data['refund_type'] ?? $this->default_refund_type; $item->ext = array_merge($item->ext, ['refund_time' => time()]); // 退款时间 $item->save(); // 累加已退金额 if ($refund_fee > 0) { $total_refund_fee = bcadd($total_refund_fee, $refund_fee, 2); } Action::add($this->order, $item, $user, ($user ? (($user instanceof User || $user instanceof CommonUser) ? 'user' : 'admin') : 'system'), (isset($data['remark']) && $data['remark'] ? $data['remark'] . ',' : '') . '退款金额:¥' . $item->refund_fee); // 订单商品退款后 $eventData = ['order' => $this->order, 'item' => $item]; \think\Hook::listen('order_item_refund_after', $eventData);//orderItemRefundAfter ,就只是更新一下订单的退款时间 } // 退回已支付的所有金额(积分,余额,在线支付原路返回等) $this->refundAllPays($data); // 订单商品退款后 //1 将开票申请取消, //2 扣除退款金额的累计消费金额, //3 通知用户 $eventData = [ 'order' => $this->order, 'items' => $items, 'refund_type' => $data['refund_type'] ?? $this->default_refund_type, 'refund_method' => 'full_refund' ]; \think\Hook::listen('order_refund_after', $eventData);//orderRefundAfter } /** * 部分退款 (通过订单商品退款) * * @param \think\Model $item * @param string $refund_money * @param \think\Model $user * @param string $remark * @return void */ public function refund($item, $refund_money, $user = null, $data = []) { $item->refund_status = OrderItem::REFUND_STATUS_AGREE; // 同意退款 $item->refund_fee = $refund_money; $item->refund_type = $data['refund_type'] ?? $this->default_refund_type; $item->ext = array_merge($item->ext, ['refund_time' => time()]); // 退款时间 $item->save(); Action::add($this->order, $item, $user, ($user ? (($user instanceof User || $user instanceof CommonUser) ? 'user' : 'admin') : 'system'), (isset($data['remark']) && $data['remark'] ? $data['remark'] . ',' : '') . '退款金额:¥' . $refund_money); // 订单商品退款后 $eventData = ['order' => $this->order, 'item' => $item]; \think\Hook::listen('order_item_refund_after', $eventData); // 查找符合条件的 pays 并从中退指定金额 $this->refundPaysByMoney((string)$refund_money, $data); // 订单商品退款后 $eventData = [ 'order' => $this->order, 'items' => [$item], 'refund_type' => $data['refund_type'] ?? $this->default_refund_type, 'refund_method' => 'item_refund' ]; \think\Hook::listen('order_refund_after', $eventData); } /** * 退回已支付的所有 pays 记录 * * @param string $remark * @return void */ public function refundAllPays($data = []) { // 商城订单,已支付的 pay 记录 $pays = PayModel::typeOrder()->paid()->where('order_id', $this->order->id)->lock(true)->select(); $refund = new PayRefund($this->order->user_id); foreach ($pays as $key => $pay) { $refund->fullRefund($pay, [ 'refund_type' => $data['refund_type'] ?? $this->default_refund_type, 'platform' => $this->order->platform, 'remark' => $data['remark'] ?? '' ]); } } /** * 查找符合条件的 pays 并从中退指定金额 (不退积分,包括积分抵扣的积分) * * @param string $refund_money * @param string $remark * @return void */ protected function refundPaysByMoney(string $refund_money, $data = []) { $payOper = new PayOper($this->order->user_id); $pays = $payOper->getCanRefundPays($this->order->id); $remain_max_refund_money = $payOper->getRemainRefundMoney($pays); if (bccomp($refund_money, $remain_max_refund_money, 2) === 1) { // 退款金额超出最大支付金额 error_stop('退款金额超出最大可退款金额'); } $current_refunded_money = '0'; // 本次退款,已退金额累计 $refund = new PayRefund($this->order->user_id); foreach ($pays as $key => $pay) { $current_remain_money = bcsub($refund_money, $current_refunded_money, 2); // 剩余应退款金额 if ($current_remain_money <= 0) { // 退款完成 break; } $current_pay_remain_money = bcsub($pay->pay_fee, $pay->refund_fee, 2); // 当前 pay 记录剩余可退金额 if ($current_pay_remain_money <= 0) { // 当前 pay 支付的金额已经退完了,循环下一个 continue; } $current_refund_money = min($current_remain_money, $current_pay_remain_money); // 取最小值 $refund->refund($pay, $current_refund_money, [ 'refund_type' => $data['refund_type'] ?? $this->default_refund_type, 'platform' => $this->order->platform, 'remark' => $data['remark'] ?? '' ]); $current_refunded_money = bcadd($current_refunded_money, $current_refund_money, 2); } if ($refund_money > $current_refunded_money) { // 退款金额超出最大支付金额 error_stop('退款金额超出最大可退款金额'); } } /** * 计算 item 应退金额 * * @param \think\Model $item * @param string $total_refund_fee * @param boolean $is_last * @return string */ private function calcRefundFee($item, $total_refund_fee, $is_last = false) { $pay_fee = $this->order->pay_fee; // 支付总金额 $current_goods_amount = bcmul($item->goods_price, (string)$item->goods_num, 2); $total_amount = bcadd($current_goods_amount, $item->dispatch_fee, 2); $refund_fee = bcsub($total_amount, $item->discount_fee, 2); // (商品金额 + 运费金额) - 总优惠(活动,优惠券,包邮优惠) if ($total_refund_fee >= $pay_fee) { $refund_fee = 0; } else { $remain_fee = bcsub($pay_fee, $total_refund_fee, 2); $refund_fee = $remain_fee > $refund_fee ? ($is_last ? $remain_fee : $refund_fee) : $remain_fee; } return $refund_fee; } }