FarmCropLogController.php 9.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232
  1. <?php
  2. namespace App\Module\Farm\AdminControllers;
  3. use App\Module\Farm\Models\FarmCropLog;
  4. use App\Module\Farm\Repositories\FarmCropLogRepository;
  5. use UCore\DcatAdmin\AdminController;
  6. use App\Module\Farm\AdminControllers\Helper\GridHelper;
  7. use App\Module\Farm\AdminControllers\Helper\ShowHelper;
  8. use App\Module\Farm\AdminControllers\Helper\FilterHelper;
  9. use Dcat\Admin\Grid;
  10. use Dcat\Admin\Show;
  11. use Dcat\Admin\Http\Controllers\AdminController as BaseAdminController;
  12. use Dcat\Admin\Layout\Content;
  13. use Dcat\Admin\Widgets\Alert;
  14. use Spatie\RouteAttributes\Attributes\Resource;
  15. /**
  16. * 作物事件日志管理控制器
  17. */
  18. #[Resource('farm-crop-logs', names: 'dcat.admin.farm-crop-logs')]
  19. class FarmCropLogController extends AdminController
  20. {
  21. /**
  22. * 页面标题
  23. *
  24. * @var string
  25. */
  26. protected $title = '作物事件日志管理';
  27. /**
  28. * 页面描述
  29. *
  30. * @var string
  31. */
  32. protected $description = '查看作物生长过程中的各种事件记录';
  33. /**
  34. * 构建表格
  35. *
  36. * @return Grid
  37. */
  38. protected function grid()
  39. {
  40. return Grid::make(new FarmCropLogRepository(['crop', 'seed', 'land']), function (Grid $grid) {
  41. $helper = new GridHelper($grid, $this);
  42. $helper->columnId();
  43. $grid->column('user_id', '用户ID')->sortable();
  44. $grid->column('crop.id', '作物ID')->display(function ($value) {
  45. return "<span class='badge badge-primary'>作物#{$value}</span>";
  46. });
  47. $grid->column('seed.name', '种子名称')->display(function ($value) {
  48. return "<span class='badge badge-info'>{$value}</span>";
  49. });
  50. $grid->column('land.id', '土地ID')->display(function ($value) {
  51. return "<span class='badge badge-secondary'>土地#{$value}</span>";
  52. });
  53. $grid->column('event_type', '事件类型')->display(function ($value) {
  54. $colors = [
  55. FarmCropLog::EVENT_FRUIT_CONFIRMED => 'success',
  56. FarmCropLog::EVENT_OUTPUT_CALCULATED => 'info',
  57. FarmCropLog::EVENT_DISASTER_OCCURRED => 'danger',
  58. FarmCropLog::EVENT_DISASTER_CLEARED => 'warning',
  59. FarmCropLog::EVENT_HARVESTED => 'primary',
  60. FarmCropLog::EVENT_FERTILIZED => 'success',
  61. FarmCropLog::EVENT_PESTICIDE_USED => 'warning',
  62. FarmCropLog::EVENT_WEEDICIDE_USED => 'warning',
  63. FarmCropLog::EVENT_WATERING => 'info',
  64. FarmCropLog::EVENT_REMOVED => 'danger',
  65. ];
  66. $color = $colors[$value] ?? 'secondary';
  67. $names = [
  68. FarmCropLog::EVENT_FRUIT_CONFIRMED => '确认果实种类',
  69. FarmCropLog::EVENT_OUTPUT_CALCULATED => '确认产出数量',
  70. FarmCropLog::EVENT_DISASTER_OCCURRED => '灾害产生',
  71. FarmCropLog::EVENT_DISASTER_CLEARED => '灾害清除',
  72. FarmCropLog::EVENT_HARVESTED => '收获',
  73. FarmCropLog::EVENT_FERTILIZED => '施肥',
  74. FarmCropLog::EVENT_PESTICIDE_USED => '使用杀虫剂',
  75. FarmCropLog::EVENT_WEEDICIDE_USED => '使用除草剂',
  76. FarmCropLog::EVENT_WATERING => '浇水',
  77. FarmCropLog::EVENT_REMOVED => '铲除作物',
  78. ];
  79. $name = $names[$value] ?? '未知事件';
  80. return "<span class='badge badge-{$color}'>{$name}</span>";
  81. });
  82. $grid->column('growth_stage_name', '生长阶段')->display(function ($value) {
  83. return "<span class='badge badge-info'>{$value}</span>";
  84. });
  85. $grid->column('event_data', '事件数据')->display(function ($value) {
  86. if (empty($value)) {
  87. return '<span class="text-muted">无数据</span>';
  88. }
  89. // 使用静态方法解析数据
  90. $summary = FarmCropLog::parseEventDataSummary($this->event_type, $value);
  91. return "<small class='text-info'>{$summary}</small>";
  92. });
  93. $helper->columnCreatedAt();
  94. // 筛选器
  95. $grid->filter(function (Grid\Filter $filter) {
  96. $helper = new FilterHelper($filter, $this);
  97. $filter->equal('user_id', '用户ID');
  98. $filter->equal('crop_id', '作物ID');
  99. $filter->equal('event_type', '事件类型')->select([
  100. FarmCropLog::EVENT_FRUIT_CONFIRMED => '确认果实种类',
  101. FarmCropLog::EVENT_OUTPUT_CALCULATED => '确认产出数量',
  102. FarmCropLog::EVENT_DISASTER_OCCURRED => '灾害产生',
  103. FarmCropLog::EVENT_DISASTER_CLEARED => '灾害清除',
  104. FarmCropLog::EVENT_HARVESTED => '收获',
  105. FarmCropLog::EVENT_FERTILIZED => '施肥',
  106. FarmCropLog::EVENT_PESTICIDE_USED => '使用杀虫剂',
  107. FarmCropLog::EVENT_WEEDICIDE_USED => '使用除草剂',
  108. FarmCropLog::EVENT_WATERING => '浇水',
  109. FarmCropLog::EVENT_REMOVED => '铲除作物',
  110. ]);
  111. $filter->equal('growth_stage', '生长阶段')->select(\App\Module\Farm\Enums\GROWTH_STAGE::getAll());
  112. $helper->betweenDatetime('created_at', '事件时间');
  113. });
  114. // 禁用新增、编辑、删除操作(只读日志)
  115. $grid->disableCreateButton();
  116. $grid->disableEditButton();
  117. $grid->disableDeleteButton();
  118. $grid->disableBatchDelete();
  119. // 设置默认排序
  120. $grid->model()->orderBy('created_at', 'desc');
  121. // 设置每页显示数量
  122. $grid->paginate(20);
  123. });
  124. }
  125. /**
  126. * 构建详情页
  127. *
  128. * @param mixed $id
  129. * @return Show
  130. */
  131. protected function detail($id)
  132. {
  133. return Show::make($id, new FarmCropLogRepository(['crop', 'seed', 'land']), function (Show $show) {
  134. $helper = new ShowHelper($show, $this);
  135. $show->field('id', 'ID');
  136. $show->field('user_id', '用户ID');
  137. $show->field('crop.id', '作物ID');
  138. $show->field('seed.name', '种子名称');
  139. $show->field('land.id', '土地ID');
  140. $show->field('event_type_name', '事件类型');
  141. $show->field('growth_stage_name', '生长阶段');
  142. $show->field('land_type', '土地类型');
  143. $show->field('event_data', '事件详细数据')->as(function ($value) {
  144. return FarmCropLog::parseEventDataDetail($this->event_type, $value);
  145. });
  146. $show->field('created_at', '创建时间');
  147. $show->field('updated_at', '更新时间');
  148. // 禁用编辑和删除按钮
  149. $show->disableEditButton();
  150. $show->disableDeleteButton();
  151. });
  152. }
  153. /**
  154. * 重写index方法,添加统计信息
  155. */
  156. public function index(Content $content)
  157. {
  158. return $content
  159. ->title($this->title())
  160. ->description($this->description())
  161. ->body($this->getStatsCards())
  162. ->body($this->grid());
  163. }
  164. /**
  165. * 获取统计卡片
  166. */
  167. protected function getStatsCards()
  168. {
  169. $stats = [
  170. '总事件数' => FarmCropLog::count(),
  171. '今日事件数' => FarmCropLog::whereDate('created_at', today())->count(),
  172. '确认果实事件' => FarmCropLog::byEventType(FarmCropLog::EVENT_FRUIT_CONFIRMED)->count(),
  173. '产出计算事件' => FarmCropLog::byEventType(FarmCropLog::EVENT_OUTPUT_CALCULATED)->count(),
  174. '灾害产生事件' => FarmCropLog::byEventType(FarmCropLog::EVENT_DISASTER_OCCURRED)->count(),
  175. '灾害清除事件' => FarmCropLog::byEventType(FarmCropLog::EVENT_DISASTER_CLEARED)->count(),
  176. '收获事件' => FarmCropLog::byEventType(FarmCropLog::EVENT_HARVESTED)->count(),
  177. '施肥事件' => FarmCropLog::byEventType(FarmCropLog::EVENT_FERTILIZED)->count(),
  178. '除虫事件' => FarmCropLog::byEventType(FarmCropLog::EVENT_PESTICIDE_USED)->count(),
  179. '除草事件' => FarmCropLog::byEventType(FarmCropLog::EVENT_WEEDICIDE_USED)->count(),
  180. '浇水事件' => FarmCropLog::byEventType(FarmCropLog::EVENT_WATERING)->count(),
  181. ];
  182. $cards = '';
  183. foreach ($stats as $title => $count) {
  184. $cards .= "
  185. <div class='col-md-3'>
  186. <div class='small-box bg-info'>
  187. <div class='inner'>
  188. <h3>{$count}</h3>
  189. <p>{$title}</p>
  190. </div>
  191. <div class='icon'>
  192. <i class='fa fa-list'></i>
  193. </div>
  194. </div>
  195. </div>
  196. ";
  197. }
  198. return "<div class='row'>{$cards}</div>";
  199. }
  200. }