MexPriceConfigController.php 9.3 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243
  1. <?php
  2. namespace App\Module\Mex\AdminControllers;
  3. use App\Module\Mex\AdminControllers\Helper\GridHelper;
  4. use App\Module\Mex\Repositories\MexPriceConfigRepository;
  5. use App\Module\Mex\Models\MexPriceConfig;
  6. use App\Module\Mex\Service\MexPriceAdjustmentService;
  7. use Spatie\RouteAttributes\Attributes\Resource;
  8. use UCore\DcatAdmin\AdminController;
  9. use Dcat\Admin\Form;
  10. use Dcat\Admin\Grid;
  11. use Dcat\Admin\Show;
  12. use Dcat\Admin\Admin;
  13. /**
  14. * 农贸市场价格配置管理
  15. *
  16. * 路由:/admin/mex-price-configs
  17. */
  18. #[Resource('mex-price-configs', names: 'dcat.admin.mex-price-configs')]
  19. class MexPriceConfigController extends AdminController
  20. {
  21. /**
  22. * 页面标题
  23. *
  24. * @var string
  25. */
  26. protected $title = '农贸市场价格配置';
  27. /**
  28. * 列表页面
  29. *
  30. * @return Grid
  31. */
  32. protected function grid()
  33. {
  34. return Grid::make(new MexPriceConfigRepository( ['item']), function (Grid $grid) {
  35. $helper = new GridHelper($grid,$this);
  36. $grid->column('id', 'ID')->sortable();
  37. $grid->column('item_id', '商品ID')->link(function ($value) {
  38. return admin_url("game-items/{$value}");
  39. });
  40. $grid->column('item.name', '商品');
  41. $grid->column('min_price', '最低价格')->display(function ($value) {
  42. return number_format($value, 5);
  43. });
  44. $grid->column('max_price', '最高价格')->display(function ($value) {
  45. return number_format($value, 5);
  46. });
  47. $grid->column('protection_threshold', '保护阈值');
  48. $grid->column('is_enabled', '启用状态')->bool();
  49. $helper->columnCreatedAt();
  50. $helper->columnUpdatedAt();
  51. // 筛选器
  52. $grid->filter(function (Grid\Filter $filter) {
  53. $filter->equal('id', 'ID');
  54. $filter->equal('item_id', '商品ID');
  55. $filter->between('min_price', '最低价格范围');
  56. $filter->between('max_price', '最高价格范围');
  57. $filter->between('protection_threshold', '保护阈值范围');
  58. $filter->equal('is_enabled', '启用状态')->select([
  59. 1 => '启用',
  60. 0 => '禁用',
  61. ]);
  62. $filter->between('created_at', '创建时间')->datetime();
  63. });
  64. // 默认排序
  65. $grid->model()->orderBy('created_at', 'desc');
  66. // 行操作
  67. $grid->actions(function (Grid\Displayers\Actions $actions) {
  68. // 可以添加自定义操作
  69. });
  70. });
  71. }
  72. /**
  73. * 详情页面
  74. *
  75. * @param mixed $id
  76. * @return Show
  77. */
  78. protected function detail($id)
  79. {
  80. return Show::make($id, new MexPriceConfigRepository(), function (Show $show) {
  81. $show->field('id', 'ID');
  82. $show->field('item_id', '商品ID');
  83. $show->field('min_price', '最低价格');
  84. $show->field('max_price', '最高价格');
  85. $show->field('protection_threshold', '保护阈值');
  86. $show->field('is_enabled', '启用状态')->as(function ($value) {
  87. return $value ? '启用' : '禁用';
  88. });
  89. $show->field('created_at', '创建时间');
  90. $show->field('updated_at', '更新时间');
  91. });
  92. }
  93. /**
  94. * 表单页面
  95. *
  96. * @return Form
  97. */
  98. protected function form()
  99. {
  100. return Form::make(new MexPriceConfigRepository(), function (Form $form) {
  101. $form->display('id', 'ID');
  102. $form->number('item_id', '商品ID')
  103. ->required()
  104. ->min(1)
  105. ->help('请输入有效的商品ID');
  106. $form->text('min_price', '最低价格')
  107. ->required()
  108. ->rules('numeric|min:0|regex:/^\d+(\.\d{1,5})?$/')
  109. ->placeholder('0.00000')
  110. ->help('商品的最低售价,用户卖出价格不能高于此价格(最多5位小数)');
  111. $form->text('max_price', '最高价格')
  112. ->required()
  113. ->rules('numeric|min:0|regex:/^\d+(\.\d{1,5})?$/')
  114. ->placeholder('0.00000')
  115. ->help('商品的最高售价,用户买入价格不能低于此价格(最多5位小数)');
  116. $form->number('protection_threshold', '保护阈值')
  117. ->required()
  118. ->min(1)
  119. ->help('单次买入订单的最大数量限制');
  120. $form->switch('is_enabled', '启用状态')
  121. ->default(1)
  122. ->help('是否启用此价格配置');
  123. $form->display('created_at', '创建时间');
  124. $form->display('updated_at', '更新时间');
  125. // 添加调整原因字段(仅在编辑时显示,不保存到数据库)
  126. if (!$form->isCreating()) {
  127. $form->textarea('adjustment_reason', '调整原因')
  128. ->placeholder('请说明本次价格调整的原因...')
  129. ->help('记录价格调整的原因,便于后续追踪')
  130. ->customFormat(function ($value) {
  131. // 不保存到数据库
  132. return null;
  133. });
  134. $form->textarea('market_impact_note', '市场影响说明')
  135. ->placeholder('请说明本次调整对市场可能产生的影响...')
  136. ->help('可选:说明价格调整对市场的预期影响')
  137. ->customFormat(function ($value) {
  138. // 不保存到数据库
  139. return null;
  140. });
  141. }
  142. // 表单验证和保存处理
  143. $form->saving(function (Form $form) {
  144. // 转换价格为数值类型进行比较
  145. $minPrice = (float)$form->min_price;
  146. $maxPrice = (float)$form->max_price;
  147. // 验证价格范围
  148. if ($minPrice >= $maxPrice) {
  149. return $form->response()->error('最低价格必须小于最高价格');
  150. }
  151. // 验证价格不能为负数
  152. if ($minPrice < 0 || $maxPrice < 0) {
  153. return $form->response()->error('价格不能为负数');
  154. }
  155. // 检查商品ID是否存在(如果有GameItems模块)
  156. if ($form->isCreating()) {
  157. $itemId = $form->item_id;
  158. // 这里可以添加商品存在性验证
  159. // $item = \App\Module\GameItems\Models\Item::find($itemId);
  160. // if (!$item) {
  161. // return $form->response()->error('商品ID不存在');
  162. // }
  163. // 检查是否已存在配置
  164. $exists = MexPriceConfig::where('item_id', $itemId)->exists();
  165. if ($exists) {
  166. return $form->response()->error('该商品已存在价格配置');
  167. }
  168. }
  169. // 保存调整原因和市场影响说明到变量中,然后移除字段
  170. $adjustmentReason = $form->input('adjustment_reason');
  171. $marketImpactNote = $form->input('market_impact_note');
  172. // 移除不需要保存到数据库的字段
  173. $form->deleteInput('adjustment_reason');
  174. $form->deleteInput('market_impact_note');
  175. });
  176. // 保存前处理 - 获取旧配置数据
  177. $oldConfig = null;
  178. $adjustmentReason = null;
  179. $marketImpactNote = null;
  180. $form->saving(function (Form $form) use (&$oldConfig, &$adjustmentReason, &$marketImpactNote) {
  181. if (!$form->isCreating()) {
  182. // 保存调整前的配置数据
  183. $oldConfig = MexPriceConfig::find($form->model()->getKey());
  184. }
  185. });
  186. // 保存后处理 - 记录价格调整历史
  187. $form->saved(function (Form $form, $result) use (&$oldConfig, &$adjustmentReason, &$marketImpactNote) {
  188. if (!$form->isCreating() && $oldConfig) {
  189. // 重新获取调整后的配置
  190. $newConfig = MexPriceConfig::find($form->model()->getKey());
  191. // 获取当前管理员用户ID
  192. $adminUserId = Admin::user()->id ?? 1;
  193. // 记录价格调整历史
  194. try {
  195. MexPriceAdjustmentService::recordPriceConfigAdjustment(
  196. $oldConfig,
  197. $newConfig,
  198. $adminUserId,
  199. $adjustmentReason,
  200. $marketImpactNote
  201. );
  202. } catch (\Exception $e) {
  203. // 记录日志但不影响主流程
  204. \Log::error('记录价格调整历史失败', [
  205. 'error' => $e->getMessage(),
  206. 'config_id' => $newConfig ? $newConfig->id : 'unknown',
  207. 'admin_user_id' => $adminUserId,
  208. ]);
  209. }
  210. }
  211. });
  212. });
  213. }
  214. }