截流自动化的商城平台
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.

ShopWithdrawalLogic.php 16KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446
  1. <?php
  2. namespace app\admin\logic\finance;
  3. use app\common\basics\Logic;
  4. use app\common\enum\ShopEnum;
  5. use app\common\enum\ShopWithdrawEnum;
  6. use app\common\enum\WithdrawalEnum;
  7. use app\common\model\shop\Shop;
  8. use app\common\model\shop\ShopAccountLog;
  9. use app\common\model\shop\ShopAlipay;
  10. use app\common\model\shop\ShopBank;
  11. use app\common\model\shop\ShopWithdrawal;
  12. use app\common\server\ExportExcelServer;
  13. use app\common\server\UrlServer;
  14. use app\common\server\YansongdaAliPayTransferServer;
  15. class ShopWithdrawalLogic extends Logic
  16. {
  17. /**
  18. * @Notes: 申请提现记录列表
  19. * @Author: 张无忌
  20. * @param $get
  21. * @return array
  22. */
  23. public static function lists($get, $is_export = false)
  24. {
  25. try {
  26. $where[] = ['status', '=', $get['type'] ?? 0];
  27. if (!empty($get['start_time']) and $get['start_time']) {
  28. $where[] = ['create_time', '>=', strtotime($get['start_time'])];
  29. }
  30. if (!empty($get['end_time']) and $get['end_time']) {
  31. $where[] = ['create_time', '<=', strtotime($get['start_time'])];
  32. }
  33. // 导出
  34. if (true === $is_export) {
  35. return self::withdrawalExport($where);
  36. }
  37. $model = new ShopWithdrawal();
  38. $lists = $model->field(true)
  39. ->where($where)
  40. ->with(['shop'])
  41. ->order('id desc')
  42. ->paginate([
  43. 'page' => $get['page'],
  44. 'list_rows' => $get['limit'],
  45. 'var_page' => 'page'
  46. ])->toArray();
  47. foreach ($lists['data'] as &$item) {
  48. $item['status_text'] = WithdrawalEnum::getStatusDesc($item['status']);
  49. }
  50. return ['count'=>$lists['total'], 'lists'=>$lists['data']];
  51. } catch (\Exception $e) {
  52. static::$error = $e->getMessage();
  53. return ['error'=>$e->getMessage()];
  54. }
  55. }
  56. /**
  57. * @Notes: 统计
  58. * @Author: 张无忌
  59. * @return array
  60. */
  61. public static function statistics()
  62. {
  63. $model = new ShopWithdrawal();
  64. $apply = $model->where(['status'=>WithdrawalEnum::APPLY_STATUS])->count();
  65. $handle = $model->where(['status'=>WithdrawalEnum::HANDLE_STATUS])->count();
  66. $success = $model->where(['status'=>WithdrawalEnum::SUCCESS_STATUS])->count();
  67. $error = $model->where(['status'=>WithdrawalEnum::ERROR_STATUS])->count();
  68. return ['apply'=>$apply, 'handle'=>$handle, 'success'=>$success, 'error'=>$error];
  69. }
  70. /**
  71. * @Notes: 数据汇总
  72. * @Author: 张无忌
  73. */
  74. public static function summary()
  75. {
  76. $model = new ShopWithdrawal();
  77. $successWithdrawn = $model->where(['status'=>WithdrawalEnum::SUCCESS_STATUS])->sum('apply_amount');
  78. $handleWithdrawn = $model->where(['status'=>WithdrawalEnum::HANDLE_STATUS])->sum('apply_amount');
  79. $totalWallet = (new Shop())->where(['del'=>0])->sum('wallet');
  80. return ['successWithdrawn'=>$successWithdrawn, 'handleWithdrawn'=>$handleWithdrawn, 'totalWallet'=>$totalWallet];
  81. }
  82. /**
  83. * @Notes: 提现详细
  84. * @Author: 张无忌
  85. * @param $id
  86. * @return array
  87. */
  88. public static function detail($id)
  89. {
  90. $withdrawal = (new ShopWithdrawal())->findOrEmpty($id)->toArray();
  91. $shop = (new Shop())->with(['category'])->findOrEmpty($withdrawal['shop_id'])->toArray();
  92. $bank = (new ShopBank())->findOrEmpty($withdrawal['bank_id'])->toArray();
  93. $alipay = (new ShopAlipay())->findOrEmpty($withdrawal['alipay_id'])->toArray();
  94. $shop['type'] = ShopEnum::getShopTypeDesc($shop['type']);
  95. $withdrawal['status_text'] = WithdrawalEnum::getStatusDesc($withdrawal['status']);
  96. $withdrawal['type_text'] = ShopWithdrawEnum::getTypeText($withdrawal['type']);
  97. return [ 'withdrawal' => $withdrawal, 'shop' => $shop, 'bank' => $bank, 'alipay' => $alipay ];
  98. }
  99. /**
  100. * @Notes: 审核提现
  101. * @Author: 张无忌
  102. * @param $post
  103. * @return bool
  104. */
  105. public static function examine($post)
  106. {
  107. try {
  108. if ($post['is_examine']) {
  109. // 同意提现
  110. ShopWithdrawal::update([
  111. 'explain' => $post['explain'] ?? '',
  112. 'status' => WithdrawalEnum::HANDLE_STATUS,
  113. 'update_time' => time()
  114. ], ['id'=>$post['id']]);
  115. } else {
  116. // 拒绝提现
  117. $withdrawal = (new ShopWithdrawal())->findOrEmpty($post['id'])->toArray();
  118. ShopWithdrawal::update([
  119. 'explain' => $post['explain'] ?? '',
  120. 'status' => WithdrawalEnum::ERROR_STATUS,
  121. 'update_time' => time()
  122. ], ['id'=>$post['id']]);
  123. Shop::update([
  124. 'wallet' => ['inc', $withdrawal['apply_amount']],
  125. 'update_time' => time()
  126. ], ['id'=>$withdrawal['shop_id']]);
  127. (new ShopAccountLog())->where([
  128. 'source_id' => $withdrawal['id'],
  129. 'source_sn' => $withdrawal['sn']
  130. ])->update([
  131. 'change_type' => 1,
  132. 'left_amount' => ['inc', $withdrawal['apply_amount']],
  133. 'source_type' => ShopAccountLog::withdrawal_fail_money
  134. ]);
  135. }
  136. return true;
  137. } catch (\Exception $e) {
  138. static::$error = $e->getMessage();
  139. return false;
  140. }
  141. }
  142. /**
  143. * @Notes: 审核提现转账
  144. * @Author: 张无忌
  145. * @param $post
  146. * @return bool
  147. */
  148. public static function transfer($post)
  149. {
  150. try {
  151. if ($post['is_examine']) {
  152. // 转账成功
  153. ShopWithdrawal::update([
  154. 'transfer_content' => $post['transfer_content'] ?? '',
  155. 'status' => WithdrawalEnum::SUCCESS_STATUS,
  156. 'transfer_voucher' => $post['image'] ?? '',
  157. 'transfer_time' => time(),
  158. 'update_time' => time()
  159. ], ['id'=>(int)$post['id']]);
  160. $withdrawal = (new ShopWithdrawal())->findOrEmpty($post['id'])->toArray();
  161. (new ShopAccountLog())->where([
  162. 'source_id' => $withdrawal['id'],
  163. 'source_sn' => $withdrawal['sn']
  164. ])->update([
  165. 'change_type' => 2,
  166. 'source_type' => ShopAccountLog::withdrawal_dec_money
  167. ]);
  168. } else {
  169. // 转账失败
  170. $withdrawal = (new ShopWithdrawal())->findOrEmpty($post['id'])->toArray();
  171. ShopWithdrawal::update([
  172. 'transfer_content' => $post['transfer_content'] ?? '',
  173. 'status' => WithdrawalEnum::ERROR_STATUS,
  174. 'transfer_voucher' => $post['image'] ?? '',
  175. 'transfer_time' => time(),
  176. 'update_time' => time()
  177. ], ['id'=>$post['id']]);
  178. Shop::update([
  179. 'wallet' => ['inc', $withdrawal['apply_amount']],
  180. 'update_time' => time()
  181. ], ['id'=>$withdrawal['shop_id']]);
  182. (new ShopAccountLog())->where([
  183. 'source_id' => $withdrawal['id'],
  184. 'source_sn' => $withdrawal['sn']
  185. ])->update([
  186. 'change_type' => 1,
  187. 'left_amount' => ['inc', $withdrawal['apply_amount']],
  188. 'source_type' => ShopAccountLog::withdrawal_fail_money
  189. ]);
  190. }
  191. return true;
  192. } catch (\Exception $e) {
  193. static::$error = $e->getMessage();
  194. return false;
  195. }
  196. }
  197. static function transfer_online($post) : bool
  198. {
  199. try {
  200. $detail = ShopWithdrawal::with([ 'alipay' ])->findOrEmpty($post['id']);
  201. $result = (new YansongdaAliPayTransferServer())->shopWithdrawTransfer($detail);
  202. if (true === $result) {
  203. // 转账成功
  204. ShopWithdrawal::update([
  205. 'explain' => '',
  206. 'status' => WithdrawalEnum::SUCCESS_STATUS,
  207. 'transfer_voucher' => '',
  208. 'transfer_time' => time(),
  209. 'update_time' => time()
  210. ], [ 'id' => (int) $post['id'] ]);
  211. $withdrawal = (new ShopWithdrawal())->findOrEmpty($post['id'])->toArray();
  212. (new ShopAccountLog())->where([
  213. 'source_id' => $withdrawal['id'],
  214. 'source_sn' => $withdrawal['sn']
  215. ])->update([
  216. 'change_type' => 2,
  217. 'source_type' => ShopAccountLog::withdrawal_dec_money
  218. ]);
  219. } else {
  220. static::$error = (string) $result;
  221. // 转账失败
  222. $withdrawal = (new ShopWithdrawal())->findOrEmpty($post['id'])->toArray();
  223. ShopWithdrawal::update([
  224. 'explain' => '支付宝转账失败',
  225. 'status' => WithdrawalEnum::ERROR_STATUS,
  226. 'transfer_voucher' => '',
  227. 'transfer_time' => time(),
  228. 'update_time' => time()
  229. ], [ 'id' => $post['id'] ]);
  230. Shop::update([
  231. 'wallet' => ['inc', $withdrawal['apply_amount']],
  232. 'update_time' => time()
  233. ], [ 'id' => $withdrawal['shop_id'] ]);
  234. (new ShopAccountLog())->where([
  235. 'source_id' => $withdrawal['id'],
  236. 'source_sn' => $withdrawal['sn']
  237. ])->update([
  238. 'change_type' => 1,
  239. 'left_amount' => ['inc', $withdrawal['apply_amount']],
  240. 'source_type' => ShopAccountLog::withdrawal_fail_money
  241. ]);
  242. }
  243. return true;
  244. } catch (\Throwable $e) {
  245. static::$error = $e->getMessage();
  246. return false;
  247. }
  248. }
  249. /**
  250. * @Notes: 账户明细
  251. * @Author: 张无忌
  252. * @param $get
  253. * @return array
  254. */
  255. public static function account($get, $is_export = false)
  256. {
  257. $where = [];
  258. if (isset($get['shop_name']) && $get['shop_name']) {
  259. $where[] = ['S.name', 'like', '%' . $get['shop_name'] . '%'];
  260. }
  261. if (isset($get['search_key']) && $get['search_key']) {
  262. switch($get['search_key']){
  263. case 'settle':
  264. $where[] = ['SAL.source_type', '=', ShopAccountLog::settlement_add_money];
  265. break;
  266. case 'withdrawal':
  267. $where[] = ['SAL.source_type', '=', ShopAccountLog::withdrawal_dec_money];
  268. break;
  269. case 'withdrawal_stay':
  270. $where[] = ['SAL.source_type', '=', ShopAccountLog::withdrawal_stay_money];
  271. break;
  272. case 'withdrawal_error':
  273. $where[] = ['SAL.source_type', '=', ShopAccountLog::withdrawal_fail_money];
  274. break;
  275. }
  276. }
  277. if (!empty($get['start_time']) and $get['start_time']) {
  278. $where[] = ['SAL.create_time', '>=', strtotime($get['start_time'])];
  279. }
  280. if (!empty($get['end_time']) and $get['end_time']) {
  281. $where[] = ['SAL.create_time', '<=', strtotime($get['end_time'])];
  282. }
  283. // 导出
  284. if (true === $is_export) {
  285. return self::accountExport($where);
  286. }
  287. $model = new ShopAccountLog();
  288. $lists = $model->alias('SAL')
  289. ->field(['SAL.*', 'S.name,S.logo,S.type'])
  290. ->join('shop S', 'S.id = SAL.shop_id')
  291. ->order('SAL.id desc')
  292. ->where($where)
  293. ->paginate([
  294. 'page' => $get['page'],
  295. 'list_rows' => $get['limit'],
  296. 'var_page' => 'page'
  297. ])->toArray();
  298. foreach ($lists['data'] as &$item) {
  299. $item['logo'] = empty($item['logo']) ? '' : UrlServer::getFileUrl($item['logo']);
  300. $item['type'] = ShopEnum::getShopTypeDesc($item['type']);
  301. $item['source_type'] = ShopAccountLog::getSourceType($item['source_type']);
  302. $item['change_amount'] = $item['change_type'] == 1 ? '+'.$item['change_amount'] : '-'.$item['change_amount'];
  303. $item['logo'] = !empty($item['logo']) ? UrlServer::getFileUrl($item['logo']) : "";
  304. }
  305. return ['count'=>$lists['total'], 'lists'=>$lists['data']];
  306. }
  307. /**
  308. * @notes 导出商家明细Excel
  309. * @param array $where
  310. * @return array|false
  311. * @author 段誉
  312. * @date 2022/4/24 10:10
  313. */
  314. public static function withdrawalExport($where)
  315. {
  316. try {
  317. $model = new ShopWithdrawal();
  318. $lists = $model->field(true)
  319. ->where($where)
  320. ->with(['shop'])
  321. ->select()->toArray();
  322. foreach ($lists as &$item) {
  323. $item['status_text'] = WithdrawalEnum::getStatusDesc($item['status']);
  324. $item['shop_name'] = $item['shop']['name'];
  325. $item['shop_type'] = ShopEnum::getShopTypeDesc($item['shop']['type']);
  326. }
  327. $excelFields = [
  328. 'shop_name' => '商家名称',
  329. 'shop_type' => '商家类型',
  330. 'sn' => '提现单号',
  331. 'apply_amount' => '提现金额',
  332. 'poundage_amount' => '提现手续费',
  333. 'left_amount' => '到账金额',
  334. 'status_text' => '提现状态',
  335. 'create_time' => '提现时间',
  336. ];
  337. $export = new ExportExcelServer();
  338. $export->setFileName('商家提现');
  339. $result = $export->createExcel($excelFields, $lists);
  340. return ['url' => $result];
  341. } catch (\Exception $e) {
  342. self::$error = $e->getMessage();
  343. return false;
  344. }
  345. }
  346. /**
  347. * @notes 导出商家账户明细Excel
  348. * @param array $where
  349. * @return array|false
  350. * @author 段誉
  351. * @date 2022/4/24 10:10
  352. */
  353. public static function accountExport($where)
  354. {
  355. try {
  356. $model = new ShopAccountLog();
  357. $lists = $model->alias('SAL')
  358. ->field(['SAL.*', 'S.name,S.type'])
  359. ->join('shop S', 'S.id = SAL.shop_id')
  360. ->order('SAL.id desc')
  361. ->where($where)
  362. ->select()->toArray();
  363. foreach ($lists as &$item) {
  364. $item['type'] = ShopEnum::getShopTypeDesc($item['type']);
  365. $item['source_type'] = ShopAccountLog::getSourceType($item['source_type']);
  366. $item['change_amount'] = $item['change_type'] == 1 ? '+'.$item['change_amount'] : '-'.$item['change_amount'];
  367. }
  368. $excelFields = [
  369. 'name' => '商家名称',
  370. 'type' => '商家类型',
  371. 'log_sn' => '明细流水号',
  372. 'source_sn' => '来源单号',
  373. 'source_type' => '明细类型',
  374. 'change_amount' => '变动金额',
  375. 'left_amount' => '剩余金额',
  376. 'create_time' => '记录时间',
  377. ];
  378. $export = new ExportExcelServer();
  379. $export->setFileName('商家账户明细');
  380. $export->setExportNumber(['log_sn', 'source_sn']);
  381. $result = $export->createExcel($excelFields, $lists);
  382. return ['url' => $result];
  383. } catch (\Exception $e) {
  384. self::$error = $e->getMessage();
  385. return false;
  386. }
  387. }
  388. }