TransferOrderHelper.php 9.5 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283
  1. <?php
  2. namespace App\Module\Transfer\AdminControllers\Helper;
  3. use App\Module\Transfer\Models\TransferOrder;
  4. use App\Module\Transfer\Enums\TransferStatus;
  5. use App\Module\Transfer\Enums\TransferType;
  6. use Dcat\Admin\Grid;
  7. use Dcat\Admin\Show;
  8. use Dcat\Admin\Form;
  9. /**
  10. * 划转订单管理辅助类
  11. */
  12. class TransferOrderHelper
  13. {
  14. /**
  15. * 配置Grid表格
  16. */
  17. public static function grid(Grid $grid): void
  18. {
  19. $grid->column('id', 'ID')->sortable();
  20. $grid->column('transfer_app.title', '应用名称')->limit(15);
  21. $grid->column('out_order_id', '外部订单ID')->copyable()->limit(20);
  22. $grid->column('user_id', '用户ID')->link(function ($value) {
  23. return admin_url("users/{$value}");
  24. });
  25. $grid->column('out_user_id', '外部用户ID')->limit(15);
  26. $grid->column('type', '类型')->using([
  27. 1 => '转入',
  28. 2 => '转出',
  29. ])->label([
  30. 1 => 'success',
  31. 2 => 'warning',
  32. ]);
  33. $grid->column('status', '状态')->using([
  34. 1 => '已创建',
  35. 20 => '处理中',
  36. 30 => '已回调',
  37. 100 => '已完成',
  38. -1 => '失败',
  39. ])->label([
  40. 1 => 'info',
  41. 20 => 'warning',
  42. 30 => 'primary',
  43. 100 => 'success',
  44. -1 => 'danger',
  45. ]);
  46. $grid->column('out_amount', '外部金额')->display(function ($value) {
  47. return number_format($value, 4);
  48. });
  49. $grid->column('amount', '内部金额')->display(function ($value) {
  50. return number_format($value, 4);
  51. });
  52. $grid->column('exchange_rate', '汇率')->display(function ($value) {
  53. return number_format($value, 4);
  54. });
  55. $grid->column('created_at', '创建时间')->sortable();
  56. $grid->column('completed_at', '完成时间');
  57. // 处理时长
  58. $grid->column('duration', '处理时长')->display(function () {
  59. if ($this->completed_at && $this->created_at) {
  60. $seconds = $this->completed_at->diffInSeconds($this->created_at);
  61. if ($seconds < 60) {
  62. return $seconds . '秒';
  63. } elseif ($seconds < 3600) {
  64. return round($seconds / 60, 1) . '分钟';
  65. } else {
  66. return round($seconds / 3600, 1) . '小时';
  67. }
  68. }
  69. return '-';
  70. });
  71. // 筛选器
  72. $grid->filter(function (Grid\Filter $filter) {
  73. $filter->equal('transfer_app_id', '应用')->select(
  74. \App\Module\Transfer\Models\TransferApp::pluck('title', 'id')
  75. );
  76. $filter->equal('type', '类型')->select([
  77. 1 => '转入',
  78. 2 => '转出',
  79. ]);
  80. $filter->equal('status', '状态')->select([
  81. 1 => '已创建',
  82. 20 => '处理中',
  83. 30 => '已回调',
  84. 100 => '已完成',
  85. -1 => '失败',
  86. ]);
  87. $filter->equal('user_id', '用户ID');
  88. $filter->like('out_order_id', '外部订单ID');
  89. $filter->like('out_user_id', '外部用户ID');
  90. $filter->between('amount', '内部金额');
  91. $filter->between('created_at', '创建时间')->datetime();
  92. });
  93. // 操作按钮
  94. $grid->actions(function (Grid\Displayers\Actions $actions) {
  95. $actions->disableDelete(); // 禁用删除
  96. $actions->disableEdit(); // 禁用编辑
  97. // 重试按钮
  98. if ($this->canRetry()) {
  99. $actions->append('<a href="javascript:void(0)" class="btn btn-xs btn-outline-warning retry-order" data-id="'.$this->id.'">重试</a>');
  100. }
  101. // 手动完成按钮(仅转出订单)
  102. if ($this->isTransferOut() && !$this->isFinalStatus()) {
  103. $actions->append('<a href="javascript:void(0)" class="btn btn-xs btn-outline-success manual-complete" data-id="'.$this->id.'">手动完成</a>');
  104. }
  105. });
  106. // 批量操作
  107. $grid->batchActions(function (Grid\Tools\BatchActions $batch) {
  108. $batch->add(new \App\Module\Transfer\AdminControllers\Tools\RetryOrderTool());
  109. });
  110. // 工具栏
  111. $grid->tools(function (Grid\Tools $tools) {
  112. $tools->append('<a href="javascript:void(0)" class="btn btn-sm btn-primary" onclick="showStats()">统计信息</a>');
  113. $tools->append(new \App\Module\Transfer\AdminControllers\Tools\ExportOrderTool());
  114. });
  115. // 默认排序
  116. $grid->model()->orderBy('created_at', 'desc');
  117. }
  118. /**
  119. * 配置Show详情
  120. */
  121. public static function show(Show $show): void
  122. {
  123. $show->field('id', 'ID');
  124. $show->divider('基本信息');
  125. $show->field('transfer_app.title', '应用名称');
  126. $show->field('out_order_id', '外部订单ID');
  127. $show->field('out_user_id', '外部用户ID');
  128. $show->field('user_id', '内部用户ID');
  129. $show->field('type', '订单类型')->using([
  130. 1 => '转入',
  131. 2 => '转出',
  132. ]);
  133. $show->field('status', '订单状态')->using([
  134. 1 => '已创建',
  135. 20 => '处理中',
  136. 30 => '已回调',
  137. 100 => '已完成',
  138. -1 => '失败',
  139. ]);
  140. $show->divider('金额信息');
  141. $show->field('out_amount', '外部金额')->as(function ($value) {
  142. return number_format($value, 10);
  143. });
  144. $show->field('amount', '内部金额')->as(function ($value) {
  145. return number_format($value, 10);
  146. });
  147. $show->field('exchange_rate', '使用汇率')->as(function ($value) {
  148. return number_format($value, 4);
  149. });
  150. $show->divider('配置信息');
  151. $show->field('currency_id', '货币类型')->using([
  152. 1 => '金币',
  153. 2 => '钻石',
  154. 3 => '人民币',
  155. 4 => '美元',
  156. ]);
  157. $show->field('fund_id', '资金账户类型');
  158. $show->divider('回调数据');
  159. $show->field('callback_data', '回调数据')->json();
  160. $show->divider('其他信息');
  161. $show->field('error_message', '错误信息');
  162. $show->field('remark', '备注信息');
  163. $show->divider('时间信息');
  164. $show->field('created_at', '创建时间');
  165. $show->field('processed_at', '处理时间');
  166. $show->field('callback_at', '回调时间');
  167. $show->field('completed_at', '完成时间');
  168. $show->field('updated_at', '更新时间');
  169. // 处理时长
  170. $show->field('processing_duration', '处理时长')->as(function () {
  171. if ($this->completed_at && $this->created_at) {
  172. return $this->completed_at->diffForHumans($this->created_at, true);
  173. }
  174. return '未完成';
  175. });
  176. }
  177. /**
  178. * 获取状态统计
  179. */
  180. public static function getStatusStats(): array
  181. {
  182. $stats = TransferOrder::selectRaw('status, COUNT(*) as count, SUM(amount) as amount')
  183. ->groupBy('status')
  184. ->get()
  185. ->keyBy('status');
  186. $result = [];
  187. foreach (TransferStatus::cases() as $status) {
  188. $stat = $stats->get($status->value);
  189. $result[] = [
  190. 'status' => $status->getDescription(),
  191. 'count' => $stat ? number_format($stat->count) : 0,
  192. 'amount' => $stat ? number_format($stat->amount, 2) : '0.00',
  193. 'color' => $status->getColor(),
  194. ];
  195. }
  196. return $result;
  197. }
  198. /**
  199. * 获取类型统计
  200. */
  201. public static function getTypeStats(): array
  202. {
  203. $stats = TransferOrder::selectRaw('type, COUNT(*) as count, SUM(amount) as amount')
  204. ->groupBy('type')
  205. ->get()
  206. ->keyBy('type');
  207. $result = [];
  208. foreach (TransferType::cases() as $type) {
  209. $stat = $stats->get($type->value);
  210. $result[] = [
  211. 'type' => $type->getDescription(),
  212. 'count' => $stat ? number_format($stat->count) : 0,
  213. 'amount' => $stat ? number_format($stat->amount, 2) : '0.00',
  214. 'color' => $type->getColor(),
  215. ];
  216. }
  217. return $result;
  218. }
  219. /**
  220. * 获取今日统计
  221. */
  222. public static function getTodayStats(): array
  223. {
  224. $today = now()->startOfDay();
  225. $total = TransferOrder::where('created_at', '>=', $today)->count();
  226. $completed = TransferOrder::where('created_at', '>=', $today)
  227. ->where('status', TransferStatus::COMPLETED)->count();
  228. $failed = TransferOrder::where('created_at', '>=', $today)
  229. ->where('status', TransferStatus::FAILED)->count();
  230. $amount = TransferOrder::where('created_at', '>=', $today)->sum('amount');
  231. return [
  232. 'total' => number_format($total),
  233. 'completed' => number_format($completed),
  234. 'failed' => number_format($failed),
  235. 'success_rate' => $total > 0 ? number_format($completed / $total * 100, 1) . '%' : '0%',
  236. 'amount' => number_format($amount, 2),
  237. ];
  238. }
  239. }