MerchantReconciliationRepository.php 17 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394
  1. <?php
  2. // +----------------------------------------------------------------------
  3. // | CRMEB [ CRMEB赋能开发者,助力企业发展 ]
  4. // +----------------------------------------------------------------------
  5. // | Copyright (c) 2016~2024 https://www.crmeb.com All rights reserved.
  6. // +----------------------------------------------------------------------
  7. // | Licensed CRMEB并不是自由软件,未经许可不能去掉CRMEB相关版权
  8. // +----------------------------------------------------------------------
  9. // | Author: CRMEB Team <admin@crmeb.com>
  10. // +----------------------------------------------------------------------
  11. namespace app\common\repositories\store\order;
  12. use app\common\model\store\order\MerchantReconciliationOrder;
  13. use app\common\model\system\merchant\Merchant;
  14. use app\common\repositories\BaseRepository;
  15. use app\common\dao\store\order\MerchantReconciliationDao as dao;
  16. use app\common\repositories\store\order\StoreOrderRepository;
  17. use app\common\repositories\store\order\StoreRefundOrderRepository;
  18. use app\common\repositories\system\admin\AdminRepository;
  19. use app\common\repositories\system\merchant\FinancialRecordRepository;
  20. use app\common\repositories\system\merchant\MerchantRepository;
  21. use crmeb\services\SwooleTaskService;
  22. use think\exception\ValidateException;
  23. use think\facade\Db;
  24. use FormBuilder\Factory\Elm;
  25. use think\facade\Route;
  26. /**
  27. * 商户对账记录
  28. */
  29. class MerchantReconciliationRepository extends BaseRepository
  30. {
  31. public function __construct(dao $dao)
  32. {
  33. $this->dao = $dao;
  34. }
  35. /**
  36. * 根据ID获取核对记录的数量,以判断是否存在特定状态的核对记录。
  37. *
  38. * 此方法用于检查是否存在指定ID的核对记录,并且其状态为2(表示某种特定的状态,具体含义依赖于业务逻辑)。
  39. * 如果存在至少一条满足条件的记录,则返回true,否则返回false。
  40. *
  41. * @param int $id 核对记录的ID,用于查询特定的核对记录。
  42. * @return bool 如果存在满足条件的核对记录则返回true,否则返回false。
  43. */
  44. public function getWhereCountById($id)
  45. {
  46. // 定义查询条件,包括核对记录的ID和状态。
  47. $where = ['reconciliation_id' => $id, 'status' => 2];
  48. // 调用DAO层的方法查询满足条件的记录数量,并检查数量是否大于0。
  49. // 如果大于0,说明存在满足条件的记录,返回true;否则,返回false。
  50. return $this->dao->getWhereCount($where) > 0;
  51. }
  52. /**
  53. * 检查指定ID和商家ID对应的未结算记录是否存在
  54. *
  55. * 本函数用于查询特定reconciliation_id和mer_id对应的未结算记录是否存在于数据库中。
  56. * 通过判断返回的记录数是否大于0来确定是否存在符合条件的未结算记录。
  57. *
  58. * @param int $id 对应的reconciliation_id,用于查询记录。
  59. * @param int $merId 商家ID,用于查询特定商家的记录。
  60. * @return bool 如果存在符合条件的未结算记录则返回true,否则返回false。
  61. */
  62. public function merWhereCountById($id, $merId)
  63. {
  64. // 定义查询条件,包括reconciliation_id、mer_id、is_accounts和status
  65. $where = ['reconciliation_id' => $id, 'mer_id' => $merId, 'is_accounts' => 0, 'status' => 0];
  66. // 调用dao层的getWhereCount方法查询符合条件的记录数,并检查是否大于0
  67. return ($this->dao->getWhereCount($where) > 0);
  68. }
  69. /**
  70. * 列表
  71. * @param $where
  72. * @param $page
  73. * @param $limit
  74. * @return array
  75. * @author Qinii
  76. * @day 2020-06-15
  77. */
  78. public function getList($where, $page, $limit)
  79. {
  80. $query = $this->dao->search($where)->with([
  81. 'merchant' => function ($query) {
  82. $query->field('mer_id,mer_name');
  83. },
  84. 'admin' => function ($query) {
  85. $query->field('admin_id,real_name');
  86. }]);
  87. $count = $query->count();
  88. $list = $query->page($page, $limit)->select()->append(['charge'])->each(function ($item) {
  89. if ($item->type == 1) return $item->price = '-' . $item->price;
  90. });
  91. return compact('count', 'list');
  92. }
  93. /**
  94. * 创建对账单
  95. * @param $id
  96. * @param $data
  97. * @author Qinii
  98. * @day 2020-06-15
  99. */
  100. public function create(int $id, array $data)
  101. {
  102. $orderMake = app()->make(StoreOrderRepository::class);
  103. $refundMake = app()->make(StoreRefundOrderRepository::class);
  104. $bank = merchantConfig($id, 'bank');
  105. $bank_name = merchantConfig($id, 'bank_name');
  106. $bank_number = merchantConfig($id, 'bank_number');
  107. $bank_address = merchantConfig($id, 'bank_address');
  108. if (!$bank || !$bank_name || !$bank_number || !$bank_address)
  109. throw new ValidateException('商户未填写银行卡信息');
  110. $order_ids = $data['order_ids'];
  111. $refund_order_ids = $data['refund_order_ids'];
  112. if ($data['order_type']) { //全选
  113. $order_ids = $orderMake->search([
  114. 'date' => $data['date'],
  115. 'mer_id' => $id,
  116. 'paid' => 1,
  117. 'order_id'
  118. ], null
  119. )->whereNotIn('order_id', $data['order_out_ids'])->column('order_id');
  120. }
  121. if ($data['refund_type']) { //全选
  122. $refund_order_ids = $refundMake->search([
  123. 'date' => $data['date'],
  124. 'mer_id' => $id,
  125. 'reconciliation_type' => 0,
  126. 'status' => 3
  127. ])->whereNotIn('refund_order_id', $data['refund_out_ids'])->column('refund_order_id');
  128. }
  129. if (is_array($order_ids) && (count($order_ids) < 1) && is_array($refund_order_ids) && (count($refund_order_ids) < 1)) {
  130. throw new ValidateException('没有数据可对账');
  131. }
  132. $compute = $this->compute($id, $order_ids, $refund_order_ids);
  133. $createData = [
  134. 'status' => 0,
  135. 'mer_id' => $id,
  136. 'is_accounts' => 0,
  137. 'mer_admin_id' => 0,
  138. 'bank' => $bank,
  139. 'bank_name' => $bank_name,
  140. 'bank_number' => $bank_number,
  141. 'bank_address' => $bank_address,
  142. 'admin_id' => $data['adminId'],
  143. 'price' => round($compute['price'], 2),
  144. 'order_price' => round($compute['order_price'], 2),
  145. 'refund_price' => round($compute['refund_price'], 2),
  146. //'refund_rate' => round($compute['refund_rate'],2),
  147. 'order_rate' => round($compute['order_rate'], 2),
  148. 'order_extension' => round($compute['order_extension'], 2),
  149. 'refund_extension' => round($compute['refund_extension'], 2),
  150. ];
  151. Db::transaction(function () use ($order_ids, $refund_order_ids, $orderMake, $refundMake, $createData) {
  152. $res = $this->dao->create($createData);
  153. $orderMake->updates($order_ids, ['reconciliation_id' => $res['reconciliation_id']]);
  154. $refundMake->updates($refund_order_ids, ['reconciliation_id' => $res['reconciliation_id']]);
  155. $this->reconciliationOrder($order_ids, $refund_order_ids, $res['reconciliation_id']);
  156. SwooleTaskService::merchant('notice', [
  157. 'type' => 'accoubts',
  158. 'data' => [
  159. 'title' => '新对账',
  160. 'message' => '您有一条新的对账单',
  161. 'id' => $res['reconciliation_id']
  162. ]
  163. ], $createData['mer_id']);
  164. });
  165. }
  166. /**
  167. * 计算对账单金额
  168. * @param $merId
  169. * @param $order_ids
  170. * @param $refund_order_ids
  171. * @return array
  172. * @author Qinii
  173. * @day 2020-06-23
  174. */
  175. public function compute($merId, $order_ids, $refund_order_ids)
  176. {
  177. $order_price = $refund_price = $order_extension = $refund_extension = $order_rate = $refund_rate = 0;
  178. $orderMake = app()->make(StoreOrderRepository::class);
  179. $refundMake = app()->make(StoreRefundOrderRepository::class);
  180. foreach ($order_ids as $item) {
  181. if (!$order = $orderMake->getWhere(['order_id' => $item, 'mer_id' => $merId, 'paid' => 1]))
  182. throw new ValidateException('订单信息不存在或状态错误');
  183. if ($order['reconciliation_id']) throw new ValidateException('订单重复提交');
  184. //(实付金额 - 一级佣金 - 二级佣金) * 抽成
  185. $commission_rate = ($order['commission_rate'] / 100);
  186. //佣金
  187. $_order_extension = bcadd($order['extension_one'], $order['extension_two'], 3);
  188. $order_extension = bcadd($order_extension, $_order_extension, 3);
  189. //手续费 = (实付金额 - 一级佣金 - 二级佣金) * 比例
  190. $_order_rate = bcmul(bcsub($order['pay_price'], $_order_extension, 3), $commission_rate, 3);
  191. $order_rate = bcadd($order_rate, $_order_rate, 3);
  192. //金额
  193. $_order_price = bcsub(bcsub($order['pay_price'], $_order_extension, 3), $_order_rate, 3);
  194. $order_price = bcadd($order_price, $_order_price, 3);
  195. }
  196. foreach ($refund_order_ids as $item) {
  197. if (!$refundOrder = $refundMake->getWhere(['refund_order_id' => $item, 'mer_id' => $merId, 'status' => 3], '*', ['order']))
  198. throw new ValidateException('退款订单信息不存在或状态错误');
  199. if ($refundOrder['reconciliation_id']) throw new ValidateException('退款订单重复提交');
  200. //退款金额 + 一级佣金 + 二级佣金
  201. $refund_commission_rate = ($refundOrder['order']['commission_rate'] / 100);
  202. //佣金
  203. $_refund_extension = bcadd($refundOrder['extension_one'], $refundOrder['extension_two'], 3);
  204. $refund_extension = bcadd($refund_extension, $_refund_extension, 3);
  205. //手续费
  206. // $_refund_rate = bcmul(bcsub($refundOrder['refund_price'],$_refund_extension,3),$refund_commission_rate,3);
  207. // $refund_rate = bcadd($refund_rate,$_refund_rate,3);
  208. //金额
  209. $_refund_price = bcadd($refundOrder['refund_price'], $_refund_extension, 3);
  210. $refund_price = bcadd($refund_price, $_refund_price, 3);
  211. }
  212. $price = bcsub($order_price, $refund_price, 3);
  213. return compact('price', 'refund_price', 'order_extension', 'refund_extension', 'order_price', 'order_rate');
  214. }
  215. /**
  216. * 同步调账订单和退款订单到调账表
  217. *
  218. * 本函数用于将订单和退款订单的信息同步到调账表中,以确保调账表中的数据与实际发生的订单和退款一致。
  219. * 这对于财务对账来说是非常重要的,可以确保所有的交易都能在调账表中找到相应的记录。
  220. *
  221. * @param array $order_ids 订单ID列表,包含需要同步到调账表的订单ID。
  222. * @param array $refund_ids 退款ID列表,包含需要同步到调账表的退款ID。
  223. * @param int $reconciliation_id 调账ID,用于标识这次调账操作。
  224. *
  225. * @return bool 返回操作结果,true表示成功插入所有数据,false表示插入数据失败。
  226. */
  227. public function reconciliationOrder($order_ids, $refund_ids, $reconciliation_id)
  228. {
  229. // 初始化数据数组,用于存储待插入调账表的订单和退款信息。
  230. $data = [];
  231. // 遍历订单ID列表,构建订单类型的调账数据,并加入到数据数组中。
  232. foreach ($order_ids as $item) {
  233. $data[] = [
  234. 'order_id' => $item,
  235. 'reconciliation_id' => $reconciliation_id,
  236. 'type' => 0, // 订单类型标识。
  237. ];
  238. }
  239. // 遍历退款ID列表,构建退款类型的调账数据,并加入到数据数组中。
  240. foreach ($refund_ids as $item) {
  241. $data[] = [
  242. 'order_id' => $item,
  243. 'reconciliation_id' => $reconciliation_id,
  244. 'type' => 1, // 退款类型标识。
  245. ];
  246. }
  247. // 使用依赖注入的方式,创建调账订单仓库实例,并调用其insertAll方法,批量插入调账数据。
  248. // 返回操作结果,true表示成功插入所有数据,false表示插入数据失败。
  249. return app()->make(MerchantReconciliationOrderRepository::class)->insertAll($data);
  250. }
  251. /**
  252. * 修改状态
  253. * @param $id
  254. * @param $data
  255. * @param $type
  256. * @author Qinii
  257. * @day 2020-06-15
  258. */
  259. public function switchStatus($id, $data)
  260. {
  261. Db::transaction(function () use ($id, $data) {
  262. if (isset($data['status']) && $data['status'] == 1) {
  263. app()->make(StoreRefundOrderRepository::class)->reconciliationUpdate($id);
  264. app()->make(StoreOrderRepository::class)->reconciliationUpdate($id);
  265. }
  266. $this->dao->update($id, $data);
  267. });
  268. $res = $this->dao->get($id);
  269. $mer = app()->make(MerchantRepository::class)->get($res['mer_id']);
  270. if (isset($data['is_accounts']) && $data['is_accounts']) {
  271. // $make = app()->make(FinancialRecordRepository::class);
  272. //
  273. // $make->dec([
  274. // 'order_id' => $id,
  275. // 'order_sn' => $id,
  276. // 'user_info' => $mer['mer_name'],
  277. // 'user_id' => $res['mer_id'],
  278. // 'financial_type' => 'sys_accoubts',
  279. // 'number' => $res->price,
  280. // ],0);
  281. //
  282. // $make->inc([
  283. // 'order_id' => $id,
  284. // 'order_sn' => $id,
  285. // 'user_info' => '总平台',
  286. // 'user_id' => 0,
  287. // 'financial_type' => 'mer_accoubts',
  288. // 'number' => $res->price,
  289. // ],$res->mer_id);
  290. SwooleTaskService::merchant('notice', [
  291. 'type' => 'accoubts',
  292. 'data' => [
  293. 'title' => '新对账打款',
  294. 'message' => '您有一条新对账打款通知',
  295. 'id' => $id
  296. ]
  297. ], $res['mer_id']);
  298. }
  299. }
  300. /**
  301. * 标记商户结算记录的表单生成方法
  302. *
  303. * 本方法用于生成一个用于修改商户结算记录备注的表单。通过给定的结算记录ID,获取当前备注信息,
  304. * 并构建一个表单以允许用户输入新的备注。表单提交的URL是根据当前商户结算记录ID动态生成的。
  305. *
  306. * @param int $id 商户结算记录的唯一标识ID
  307. * @return \Phper6\Elm\Form 表单对象,已经配置好表单的URL、表单规则和默认值
  308. */
  309. public function markForm($id)
  310. {
  311. // 根据$id获取商户结算记录的详细信息
  312. $data = $this->dao->get($id);
  313. // 构建表单的提交URL,URL指向处理商户结算记录备注修改的接口
  314. $form = Elm::createForm(Route::buildUrl('merchantReconciliationMark', ['id' => $id])->build());
  315. // 配置表单的规则,这里只包含一个文本输入框用于输入备注信息
  316. $form->setRule([
  317. // 文本输入框用于输入备注,初始值从获取的商户结算记录中取得
  318. Elm::text('mark', '备注:', $data['mark'])->placeholder('请输入备注')->required(),
  319. ]);
  320. // 设置表单的标题为“修改备注”
  321. return $form->setTitle('修改备注');
  322. }
  323. /**
  324. * 创建管理员标记表单
  325. *
  326. * 该方法用于生成一个用于修改管理员标记的表单。通过给定的商家ID,获取当前的管理员标记,
  327. * 并构建一个表单以允许管理员输入新的标记。此表单通常在后台管理系统中使用,以提供
  328. * 对商家进行备注和标记的功能,帮助管理员更好地管理商家信息。
  329. *
  330. * @param int $id 商家ID,用于获取当前商家的管理员标记信息。
  331. * @return string 返回构建好的表单HTML代码。
  332. */
  333. public function adminMarkForm($id)
  334. {
  335. // 根据商家ID获取当前商家的管理员标记信息
  336. $data = $this->dao->get($id);
  337. // 构建表单URL,指向系统中的商家对账标记页面
  338. $formUrl = Route::buildUrl('systemMerchantReconciliationMark', ['id' => $id])->build();
  339. // 创建表单对象,并设置表单提交的URL
  340. $form = Elm::createForm($formUrl);
  341. // 设置表单的验证规则,包括一个文本输入框用于输入管理员标记
  342. $form->setRule([
  343. Elm::text('admin_mark', '备注:', $data['admin_mark'])
  344. ->placeholder('请输入备注')
  345. ->required(), // 输入框必须填写
  346. ]);
  347. // 设置表单的标题为“修改备注”
  348. return $form->setTitle('修改备注');
  349. }
  350. }