| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322 |
- <?php
- namespace App\Module\Mex\AdminControllers;
- use App\Module\Fund\Enums\FUND_CURRENCY_TYPE;
- use App\Module\Mex\AdminControllers\Helper\GridHelper;
- use App\Module\Mex\Metrics\PriceTrendChart;
- use App\Module\Mex\Models\MexDailyPriceTrend;
- use App\Module\Mex\Repositories\MexDailyPriceTrendRepository;
- use App\Module\Mex\Service\MexDailyPriceTrendService;
- use Spatie\RouteAttributes\Attributes\Get;
- use Spatie\RouteAttributes\Attributes\Resource;
- use UCore\DcatAdmin\AdminController;
- use Dcat\Admin\Grid;
- use Dcat\Admin\Show;
- use Dcat\Admin\Layout\Content;
- /**
- * 农贸市场每日价格趋势管理
- *
- * 路由:/admin/mex-daily-price-trends
- */
- #[Resource('mex-daily-price-trends', names: 'dcat.admin.mex-daily-price-trends')]
- class MexDailyPriceTrendController extends AdminController
- {
- /**
- * 页面标题
- */
- protected $title = '农贸市场每日价格趋势';
- /**
- * 列表页面
- */
- public function index(Content $content)
- {
- return $content
- ->title('农贸市场每日价格趋势')
- ->description('价格趋势分析与数据管理')
- ->row(function ($row) {
- // 添加价格趋势统计卡片
- $row->column(4, $this->createPriceStatsCard());
- $row->column(4, $this->createVolumeStatsCard());
- $row->column(4, $this->createTrendStatsCard());
- })
- ->body($this->grid());
- }
- /**
- * 数据表格
- */
- protected function grid()
- {
- return Grid::make(new MexDailyPriceTrendRepository(['item']), function (Grid $grid) {
- $helper = new GridHelper($grid, $this);
-
- $grid->column('id', 'ID')->sortable();
- $grid->column('item_id', '商品ID')->link(function ($value) {
- return admin_url("game-items/{$value}");
- });
- $grid->column('item.name', '商品名称');
- $grid->column('currency_type', '币种')->display(function ($value) {
- if (is_string($value)) {
- return FUND_CURRENCY_TYPE::from($value)->name;
- }
- return $value->name;
- });
- $grid->column('trade_date', '交易日期')->sortable();
-
- // 价格信息
- $grid->column('open_price', '开盘价')->display(function ($value) {
- return $value ? number_format($value, 5) : '-';
- });
- $grid->column('close_price', '收盘价')->display(function ($value) {
- return $value ? number_format($value, 5) : '-';
- });
- $grid->column('high_price', '最高价')->display(function ($value) {
- return $value ? number_format($value, 5) : '-';
- });
- $grid->column('low_price', '最低价')->display(function ($value) {
- return $value ? number_format($value, 5) : '-';
- });
-
- // 价格变化
- $grid->column('price_change_percent', '涨跌幅')->display(function ($value) {
- if ($value === null) return '-';
- $color = $value > 0 ? 'success' : ($value < 0 ? 'danger' : 'secondary');
- $symbol = $value > 0 ? '+' : '';
- return "<span class='text-{$color}'>{$symbol}" . number_format($value, 2) . "%</span>";
- });
-
- // 交易统计
- $grid->column('total_volume', '成交量')->sortable();
- $grid->column('total_amount', '成交额')->display(function ($value) {
- return number_format($value, 5);
- })->sortable();
- $grid->column('transaction_count', '成交笔数');
-
- // 波动率
- $grid->column('volatility', '波动率')->display(function ($value) {
- if ($value === null) return '-';
- $level = $this->volatility_level;
- $color = match($level) {
- '低波动' => 'success',
- '中等波动' => 'warning',
- '高波动' => 'danger',
- '极高波动' => 'dark',
- default => 'secondary'
- };
- return "<span class='badge badge-{$color}'>{$level} (" . number_format($value, 2) . "%)</span>";
- });
-
- $helper->columnUpdatedAt();
- // 筛选器
- $grid->filter(function (Grid\Filter $filter) {
- $filter->equal('item_id', '商品ID');
- $filter->equal('currency_type', '币种')->select([
- FUND_CURRENCY_TYPE::ZUANSHI->value => '钻石',
- FUND_CURRENCY_TYPE::JINBI->value => '金币',
- ]);
- $filter->between('trade_date', '交易日期')->date();
- $filter->between('total_volume', '成交量范围');
- $filter->between('total_amount', '成交额范围');
- $filter->between('price_change_percent', '涨跌幅范围');
- });
- // 默认排序
- $grid->model()->orderBy('trade_date', 'desc')->orderBy('item_id');
- // 禁用新增、编辑、删除操作
- $grid->disableCreateButton();
- $grid->disableActions();
- $grid->disableBatchActions();
-
- // 添加工具栏按钮
- $grid->tools(function (Grid\Tools $tools) {
- $tools->append('<a href="' . admin_url('mex-daily-price-trends/generate') . '" class="btn btn-primary btn-sm">
- <i class="fa fa-refresh"></i> 生成趋势数据
- </a>');
- });
- });
- }
- /**
- * 详情页面
- */
- protected function detail($id)
- {
- return Show::make($id, new MexDailyPriceTrendRepository(['item']), function (Show $show) {
- $show->field('id', 'ID');
- $show->field('item_id', '商品ID');
- $show->field('item.name', '商品名称');
- $show->field('currency_type', '币种')->as(function ($value) {
- if (is_string($value)) {
- return FUND_CURRENCY_TYPE::from($value)->name;
- }
- return $value->name;
- });
- $show->field('trade_date', '交易日期');
- $show->divider('价格信息');
-
- $show->field('open_price', '开盘价')->as(function ($value) {
- return $value ? number_format($value, 5) : '-';
- });
- $show->field('close_price', '收盘价')->as(function ($value) {
- return $value ? number_format($value, 5) : '-';
- });
- $show->field('high_price', '最高价')->as(function ($value) {
- return $value ? number_format($value, 5) : '-';
- });
- $show->field('low_price', '最低价')->as(function ($value) {
- return $value ? number_format($value, 5) : '-';
- });
- $show->field('avg_price', '平均价')->as(function ($value) {
- return $value ? number_format($value, 5) : '-';
- });
- $show->divider('价格变化');
-
- $show->field('price_change', '价格变化')->as(function ($value) {
- return $value ? number_format($value, 5) : '-';
- });
- $show->field('price_change_percent', '涨跌幅')->as(function ($value) {
- return $value ? number_format($value, 2) . '%' : '-';
- });
- $show->field('volatility', '波动率')->as(function ($value) {
- return $value ? number_format($value, 2) . '%' : '-';
- });
- $show->divider('交易统计');
-
- $show->field('total_volume', '总成交量');
- $show->field('total_amount', '总成交额')->as(function ($value) {
- return number_format($value, 5);
- });
- $show->field('transaction_count', '成交笔数');
- $show->field('buy_volume', '买入量');
- $show->field('sell_volume', '卖出量');
- $show->field('buy_amount', '买入额')->as(function ($value) {
- return number_format($value, 5);
- });
- $show->field('sell_amount', '卖出额')->as(function ($value) {
- return number_format($value, 5);
- });
- $show->divider('管理员操作');
-
- $show->field('admin_inject_volume', '管理员注入量');
- $show->field('admin_recycle_volume', '管理员回收量');
- $show->field('admin_inject_amount', '管理员注入额')->as(function ($value) {
- return number_format($value, 5);
- });
- $show->field('admin_recycle_amount', '管理员回收额')->as(function ($value) {
- return number_format($value, 5);
- });
- $show->field('created_at', '创建时间');
- $show->field('updated_at', '更新时间');
- // 禁用编辑和删除按钮
- $show->disableEditButton();
- $show->disableDeleteButton();
- });
- }
- /**
- * 生成趋势数据
- */
- #[Get('mex-daily-price-trends/generate', name: 'admin.mex-daily-price-trends.generate')]
- public function generate()
- {
- try {
- // 调用服务生成今日价格趋势数据
- $trends = MexDailyPriceTrendService::generateTodayTrends();
- return redirect()->back()->with('success', "价格趋势数据生成成功!共生成 {$trends->count()} 条记录。");
- } catch (\Exception $e) {
- return redirect()->back()->with('error', '生成失败:' . $e->getMessage());
- }
- }
- /**
- * 创建价格统计卡片
- */
- protected function createPriceStatsCard()
- {
- // 获取最近7天的价格统计
- $trends = MexDailyPriceTrend::where('trade_date', '>=', now()->subDays(7)->toDateString())
- ->selectRaw('AVG(close_price) as avg_price, MAX(high_price) as max_price, MIN(low_price) as min_price')
- ->first();
- $avgPrice = $trends->avg_price ?? 0;
- $maxPrice = $trends->max_price ?? 0;
- $minPrice = $trends->min_price ?? 0;
- return '<div class="info-box bg-info">
- <span class="info-box-icon"><i class="fa fa-chart-line"></i></span>
- <div class="info-box-content">
- <span class="info-box-text">价格统计</span>
- <span class="info-box-number">' . number_format($avgPrice, 5) . '</span>
- <div class="progress">
- <div class="progress-bar" style="width: 70%"></div>
- </div>
- <span class="progress-description">
- 最高: ' . number_format($maxPrice, 5) . ' | 最低: ' . number_format($minPrice, 5) . '
- </span>
- </div>
- </div>';
- }
- /**
- * 创建成交量统计卡片
- */
- protected function createVolumeStatsCard()
- {
- // 获取最近7天的成交量统计
- $totalVolume = MexDailyPriceTrend::where('trade_date', '>=', now()->subDays(7)->toDateString())
- ->sum('total_volume');
- $todayVolume = MexDailyPriceTrend::where('trade_date', now()->toDateString())
- ->sum('total_volume');
- return '<div class="info-box bg-success">
- <span class="info-box-icon"><i class="fa fa-chart-bar"></i></span>
- <div class="info-box-content">
- <span class="info-box-text">成交量统计</span>
- <span class="info-box-number">' . number_format($totalVolume) . '</span>
- <div class="progress">
- <div class="progress-bar" style="width: 50%"></div>
- </div>
- <span class="progress-description">
- 今日: ' . number_format($todayVolume) . '
- </span>
- </div>
- </div>';
- }
- /**
- * 创建趋势统计卡片
- */
- protected function createTrendStatsCard()
- {
- // 获取活跃商品数量和数据天数
- $activeItems = MexDailyPriceTrend::distinct('item_id')->count();
- $dataDays = MexDailyPriceTrend::distinct('trade_date')->count();
- return '<div class="info-box bg-warning">
- <span class="info-box-icon"><i class="fa fa-trending-up"></i></span>
- <div class="info-box-content">
- <span class="info-box-text">趋势统计</span>
- <span class="info-box-number">' . $activeItems . ' 个</span>
- <div class="progress">
- <div class="progress-bar" style="width: 80%"></div>
- </div>
- <span class="progress-description">
- 数据天数: ' . $dataDays . ' 天
- </span>
- </div>
- </div>';
- }
- }
|