截流自动化的商城平台
You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

WithdrawalLogic.php 15KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409
  1. <?php
  2. namespace app\shop\logic\finance;
  3. use app\common\basics\Logic;
  4. use app\common\enum\NoticeEnum;
  5. use app\common\enum\OrderEnum;
  6. use app\common\enum\ShopEnum;
  7. use app\common\enum\PayEnum;
  8. use app\common\enum\OrderRefundEnum;
  9. use app\common\enum\ShopWithdrawEnum;
  10. use app\common\enum\WithdrawalEnum;
  11. use app\common\enum\AfterSaleEnum;
  12. use app\common\model\order\Order;
  13. use app\common\model\shop\Shop;
  14. use app\common\model\shop\ShopAccountLog;
  15. use app\common\model\shop\ShopAlipay;
  16. use app\common\model\shop\ShopBank;
  17. use app\common\model\shop\ShopWithdrawal;
  18. use app\common\model\shop\ShopSettlement;
  19. use app\common\server\ConfigServer;
  20. use app\common\server\ExportExcelServer;
  21. use think\Exception;
  22. use think\facade\Db;
  23. class WithdrawalLogic extends Logic
  24. {
  25. /**
  26. * @Notes: 商家提现列表
  27. * @Author: 张无忌
  28. * @param $get
  29. * @param $shop_id
  30. * @return array
  31. */
  32. public static function lists($get, $shop_id, $is_export = false)
  33. {
  34. try {
  35. $where[] = ['shop_id', '=', $shop_id];
  36. $where[] = ['status', '=', $get['type'] ?? 0];
  37. if (!empty($get['start_time']) and $get['start_time']) {
  38. $where[] = ['create_time', '>=', strtotime($get['start_time'])];
  39. }
  40. if (!empty($get['end_time']) and $get['end_time']) {
  41. $where[] = ['create_time', '<=', strtotime($get['end_time'])];
  42. }
  43. // 导出
  44. if (true === $is_export) {
  45. return self::export($where);
  46. }
  47. $model = new ShopWithdrawal();
  48. $lists = $model->field(true)
  49. ->where($where)
  50. ->order('id desc')
  51. ->paginate([
  52. 'page' => $get['page'],
  53. 'list_rows' => $get['limit'],
  54. 'var_page' => 'page'
  55. ])->toArray();
  56. foreach ($lists['data'] as &$item) {
  57. $item['status'] = WithdrawalEnum::getStatusDesc($item['status']);
  58. }
  59. return ['count' => $lists['total'], 'lists' => $lists['data']];
  60. } catch (\Exception $e) {
  61. return ['error' => $e->getMessage()];
  62. }
  63. }
  64. /**
  65. * @Notes: 财务汇总
  66. * @Author: 张无忌
  67. * @param $shop_id
  68. */
  69. public static function summary($shop_id)
  70. {
  71. $orderModel = new Order();
  72. // 成交订单笔数
  73. $detail['dealOrderCount'] = $orderModel->where(['shop_id' => $shop_id, 'pay_status' => OrderEnum::PAY_STATUS_PAID])->count();
  74. // 商家营业金额
  75. $detail['businessMoney'] = $orderModel->where(['shop_id' => $shop_id, 'pay_status' => OrderEnum::PAY_STATUS_PAID])->sum('order_amount');
  76. // 退款订单金额
  77. $detail['refundMoney'] = $orderModel->where(['shop_id' => $shop_id])->where('refund_status', '>', 0)->sum('order_amount');
  78. }
  79. /**
  80. * @Notes: 提现详细
  81. * @Author: 张无忌
  82. * @param $id
  83. * @param $shop_id
  84. * @return array
  85. */
  86. public static function detail($id, $shop_id)
  87. {
  88. $withdrawal = (new ShopWithdrawal())->findOrEmpty($id)->toArray();
  89. $shop = (new Shop())->with(['category'])->findOrEmpty($shop_id)->toArray();
  90. $bank = (new ShopBank())->findOrEmpty($withdrawal['bank_id'])->toArray();
  91. $alipay = (new ShopAlipay())->findOrEmpty($withdrawal['alipay_id'])->toArray();
  92. $shop['type'] = ShopEnum::getShopTypeDesc($shop['type']);
  93. $withdrawal['status_text'] = WithdrawalEnum::getStatusDesc($withdrawal['status']);
  94. $withdrawal['type_text'] = ShopWithdrawEnum::getTypeText($withdrawal['type']);
  95. return [ 'withdrawal' => $withdrawal, 'shop' => $shop, 'bank' => $bank, 'alipay' => $alipay ];
  96. }
  97. /**
  98. * @Notes: 选项卡与汇总统计详细
  99. * @param $shop_id
  100. * @return array
  101. */
  102. public static function statistics($shop_id)
  103. {
  104. $model = new ShopWithdrawal();
  105. $apply = $model->where(['status' => WithdrawalEnum::APPLY_STATUS, 'shop_id' => $shop_id])->count();
  106. $handle = $model->where(['status' => WithdrawalEnum::HANDLE_STATUS, 'shop_id' => $shop_id])->count();
  107. $success = $model->where(['status' => WithdrawalEnum::SUCCESS_STATUS, 'shop_id' => $shop_id])->count();
  108. $error = $model->where(['status' => WithdrawalEnum::ERROR_STATUS, 'shop_id' => $shop_id])->count();
  109. //成交订单笔数
  110. $modelOrder = new Order();
  111. $modelShopWithdrawal = new ShopWithdrawal();
  112. $orderNum = $modelOrder
  113. ->where([
  114. ['shop_id', '=', $shop_id],
  115. ['pay_status', '>', PayEnum::UNPAID]
  116. ])
  117. ->count('id');
  118. //营业额
  119. $orderAmount = $modelOrder
  120. ->where([
  121. ['shop_id', '=', $shop_id],
  122. ['pay_status', '>', PayEnum::UNPAID]
  123. ])
  124. ->sum('order_amount');
  125. //退款订单金额
  126. $refundAmount = $modelOrder
  127. ->where([
  128. ['shop_id', '=', $shop_id],
  129. ['shipping_status', '=', OrderEnum::SHIPPING_NO],
  130. ['pay_status', '=', PayEnum::REFUNDED],
  131. ['refund_status', 'in', [OrderEnum::REFUND_STATUS_PART_REFUND, OrderEnum::REFUND_STATUS_ALL_REFUND]],
  132. ])
  133. ->sum('refund_amount');
  134. //待退款订单金额
  135. $refundAmountIng = $modelOrder->alias('o')
  136. ->join('order_refund or', 'or.order_id = o.id')
  137. ->where([
  138. ['o.shop_id', '=', $shop_id],
  139. ['o.shipping_status', '=', OrderEnum::SHIPPING_NO],
  140. ['or.refund_status', '<>', OrderRefundEnum::REFUND_STATUS_COMPLETE]
  141. ])
  142. ->sum('or.refund_amount');
  143. //售后退款金额
  144. $salesRefundAmount = $modelOrder->alias('o')
  145. ->join('after_sale as', 'as.order_id = o.id')
  146. ->where([
  147. ['o.shop_id', '=', $shop_id],
  148. ['o.shipping_status', '=', OrderEnum::SHIPPING_FINISH],
  149. ['as.status', '=', AfterSaleEnum::STATUS_COMPLETE]
  150. ])
  151. ->sum('as.refund_price');
  152. //待售后退款金额
  153. $salesRefundAmountIng = $modelOrder->alias('o')
  154. ->join('after_sale as', 'as.order_id = o.id')
  155. ->where([
  156. ['o.shop_id', '=', $shop_id],
  157. ['o.shipping_status', '=', OrderEnum::SHIPPING_FINISH],
  158. ['as.status', '=', AfterSaleEnum::STATUS_WAITING]
  159. ])
  160. ->sum('as.refund_price');
  161. //已结算成交订单数
  162. $modelShopSettlement = new ShopSettlement();
  163. $settleOrederNum = $modelShopSettlement
  164. ->where([
  165. ['shop_id', '=', $shop_id],
  166. ])
  167. ->sum('deal_order_count');
  168. //已结算营业额
  169. $settleOrederAmount = $modelShopSettlement
  170. ->where([
  171. ['shop_id', '=', $shop_id],
  172. ])
  173. ->sum('business_money');
  174. //待结算营业额
  175. $settleOrederAmountWait = $modelOrder
  176. ->where([
  177. ['shop_id', '=', $shop_id],
  178. ['refund_status', '=', 0 ],
  179. ['settle_id', '=', OrderEnum::SETTLE_WAIT]
  180. ])
  181. ->sum('order_amount');
  182. //已结算分销佣金金额
  183. $settleDistributionAmount = $modelShopSettlement
  184. ->where([
  185. ['shop_id', '=', $shop_id],
  186. ])
  187. ->sum('distribution_money');
  188. //已结算入账金额
  189. $settleWithdrawalAmount = $modelShopSettlement
  190. ->where([
  191. ['shop_id', '=', $shop_id],
  192. ])
  193. ->sum('entry_account_money');
  194. //已结算交易服务费
  195. $settlePoundageAmount = $modelShopSettlement
  196. ->where([
  197. ['shop_id', '=', $shop_id],
  198. ])
  199. ->sum('trade_service_fee');
  200. //已提现金额
  201. $withdrawaLeftamount = $modelShopWithdrawal
  202. ->where([
  203. ['shop_id', '=', $shop_id],
  204. ['status', '=', WithdrawalEnum::SUCCESS_STATUS]
  205. ])
  206. ->sum('apply_amount');
  207. //提现中金额
  208. $withdrawaLeftamountIng = $modelShopWithdrawal
  209. ->where([
  210. ['shop_id', '=', $shop_id],
  211. ['status', 'in', [WithdrawalEnum::APPLY_STATUS, WithdrawalEnum::HANDLE_STATUS]]
  212. ])
  213. ->sum('apply_amount');
  214. //可提现金额
  215. $modelShop = new Shop();
  216. $shopWallet = $modelShop
  217. ->where([
  218. ['id', '=', $shop_id]
  219. ])
  220. ->sum('wallet');
  221. //提现手续费
  222. $procedMoney = $modelShopWithdrawal
  223. ->where([
  224. ['shop_id', '=', $shop_id],
  225. ['status', '=', WithdrawalEnum::SUCCESS_STATUS]
  226. ])
  227. ->sum('poundage_amount');
  228. return [
  229. 'apply' => $apply,
  230. 'handle' => $handle,
  231. 'success' => $success,
  232. 'error' => $error,
  233. 'orderNum' => $orderNum, //成交订单笔数
  234. 'orderAmount' => $orderAmount, //营业额
  235. 'refundAmount' => $refundAmount, //退款订单金额
  236. 'refundAmountIng' => $refundAmountIng, //待退款订单金额
  237. 'salesRefundAmount' => $salesRefundAmount, //售后退款金额
  238. 'salesRefundAmountIng' => $salesRefundAmountIng, //待售后退款金额
  239. 'settleOrederNum' => $settleOrederNum, //已结算成交订单数
  240. 'settleOrederAmount' => $settleOrederAmount, //已结算营业额
  241. 'settleOrederAmountWait' => $settleOrederAmountWait, //待结算营业额
  242. 'settleDistributionAmount' => $settleDistributionAmount, //已结算分销佣金金额
  243. 'settleWithdrawalAmount' => $settleWithdrawalAmount, //已结算入账金额
  244. 'settlePoundageAmount' => $settlePoundageAmount, //已结算交易服务费
  245. 'withdrawaLeftamount' => $withdrawaLeftamount, //已提现金额
  246. 'procedMoney' => $procedMoney, //提现手续费
  247. 'withdrawaLeftamountIng' => $withdrawaLeftamountIng, //提现中金额
  248. 'shopWallet' => $shopWallet, //可提现金额
  249. ];
  250. }
  251. /**
  252. * @Notes: 申请提现
  253. * @Author: 张无忌
  254. * @param $post
  255. * @param $shop_id
  256. * @return bool
  257. */
  258. public static function add($post, $shop_id)
  259. {
  260. Db::startTrans();
  261. try {
  262. // 1、获取提现条件
  263. $min_withdrawal_money = ConfigServer::get('shop_withdrawal', 'min_withdrawal_money', 0);
  264. $max_withdrawal_money = ConfigServer::get('shop_withdrawal', 'max_withdrawal_money', 0);
  265. $withdrawal_service_charge = ConfigServer::get('shop_withdrawal', 'withdrawal_service_charge', 0);
  266. // 2、获取商家信息
  267. $shop = (new Shop())->findOrEmpty($shop_id)->toArray();
  268. $wallet = $shop['wallet'];
  269. // 3、验证条件是否满足
  270. if (floatval($post['apply_amount']) > $wallet)
  271. throw new Exception('账户余额不足');
  272. if (floatval($post['apply_amount']) < $min_withdrawal_money)
  273. throw new Exception('最低提现金额不能少于' . $min_withdrawal_money . '元');
  274. if (floatval($post['apply_amount']) > $max_withdrawal_money)
  275. throw new Exception('最高提现金额不能大于' . $max_withdrawal_money . '元');
  276. // 4、获取商家提现手续费
  277. $poundage_amount = 0;
  278. if ($withdrawal_service_charge > 0) {
  279. $proportion = $withdrawal_service_charge / 100;
  280. $poundage_amount = $post['apply_amount'] * $proportion;
  281. $poundage_amount = $poundage_amount <= 0 ? 0 : $poundage_amount;
  282. }
  283. // 5、创建申请记录
  284. $withdrawal = ShopWithdrawal::create([
  285. 'sn' => createSn('shop_withdrawal', 'sn'),
  286. 'bank_id' => $post['bank_id'] ?? 0,
  287. 'alipay_id' => $post['alipay_id'] ?? 0,
  288. 'type' => $post['type'],
  289. 'shop_id' => $shop_id,
  290. 'apply_amount' => floatval($post['apply_amount']),
  291. 'left_amount' => $post['apply_amount'] - $poundage_amount,
  292. 'poundage_amount' => $poundage_amount,
  293. 'poundage_ratio' => $withdrawal_service_charge,
  294. 'status' => WithdrawalEnum::APPLY_STATUS
  295. ]);
  296. // 6、扣除商家可提现金额
  297. Shop::update([
  298. 'wallet' => ['dec', floatval($post['apply_amount'])],
  299. 'update_time' => time()
  300. ], ['id' => $shop_id]);
  301. $left_amount = Shop::where(['id' => $shop_id])->value('wallet');
  302. // 7、增加提现流水记录(待提现)
  303. $logType = ShopAccountLog::withdrawal_stay_money;
  304. ShopAccountLog::decData($shop_id, $logType, $post['apply_amount'], $left_amount, [
  305. 'source_id' => $withdrawal['id'],
  306. 'source_sn' => $withdrawal['sn'],
  307. 'remark' => '商家提现'
  308. ]);
  309. $platform_contacts = ConfigServer::get('website_platform', 'platform_mobile');
  310. if (!empty($platform_contacts)) {
  311. event('Notice', [
  312. 'scene' => NoticeEnum::SHOP_WITHDRAWAL_NOTICE_PLATFORM,
  313. 'mobile' => $platform_contacts,
  314. 'params' => [
  315. 'shop_withdrawal_sn' => $withdrawal['sn'],
  316. 'shop_name' => $shop['name'],
  317. ]
  318. ]);
  319. }
  320. Db::commit();
  321. return true;
  322. } catch (\Exception $e) {
  323. Db::rollback();
  324. static::$error = $e->getMessage();
  325. return false;
  326. }
  327. }
  328. /**
  329. * @notes 导出
  330. * @param $where
  331. * @return array|false
  332. * @author 段誉
  333. * @date 2022/4/24 11:59
  334. */
  335. public static function export($where)
  336. {
  337. try {
  338. $lists = (new ShopWithdrawal())
  339. ->field(true)
  340. ->where($where)
  341. ->select()->toArray();
  342. foreach ($lists as &$item) {
  343. $item['status'] = WithdrawalEnum::getStatusDesc($item['status']);
  344. }
  345. $excelFields = [
  346. 'sn' => '提现单号',
  347. 'apply_amount' => '提现金额',
  348. 'poundage_amount' => '提现手续费',
  349. 'left_amount' => '到账金额',
  350. 'status' => '提现状态',
  351. 'create_time' => '提现时间',
  352. ];
  353. $export = new ExportExcelServer();
  354. $export->setFileName('财务中心');
  355. $result = $export->createExcel($excelFields, $lists);
  356. return ['url' => $result];
  357. } catch (\Exception $e) {
  358. self::$error = $e->getMessage();
  359. return false;
  360. }
  361. }
  362. }