| | |
| | | font-weight: 700; |
| | | } |
| | | |
| | | .panel-actions { |
| | | display: flex; |
| | | align-items: center; |
| | | justify-content: flex-end; |
| | | flex-wrap: wrap; |
| | | gap: 8px; |
| | | } |
| | | |
| | | .mini-grid { |
| | | display: grid; |
| | | grid-template-columns: repeat(4, minmax(0, 1fr)); |
| | |
| | | border-color: rgba(151, 110, 204, 0.18); |
| | | } |
| | | |
| | | .network-mini-ok { |
| | | background: linear-gradient(180deg, rgba(47, 163, 142, 0.11) 0%, rgba(47, 163, 142, 0.03) 100%); |
| | | border-color: rgba(47, 163, 142, 0.18); |
| | | } |
| | | |
| | | .network-mini-warning { |
| | | background: linear-gradient(180deg, rgba(245, 154, 74, 0.12) 0%, rgba(245, 154, 74, 0.03) 100%); |
| | | border-color: rgba(245, 154, 74, 0.18); |
| | | } |
| | | |
| | | .network-mini-offline { |
| | | background: linear-gradient(180deg, rgba(222, 92, 92, 0.10) 0%, rgba(222, 92, 92, 0.03) 100%); |
| | | border-color: rgba(222, 92, 92, 0.18); |
| | | } |
| | | |
| | | .network-mini-latency { |
| | | background: linear-gradient(180deg, rgba(31, 111, 178, 0.09) 0%, rgba(31, 111, 178, 0.02) 100%); |
| | | border-color: rgba(31, 111, 178, 0.16); |
| | | } |
| | | |
| | | .chart-grid { |
| | | display: grid; |
| | | grid-template-columns: repeat(2, minmax(0, 1fr)); |
| | |
| | | margin-bottom: 8px; |
| | | } |
| | | |
| | | .chart-subtitle { |
| | | margin-bottom: 10px; |
| | | font-size: 12px; |
| | | color: #7d90a4; |
| | | line-height: 1.6; |
| | | } |
| | | |
| | | .chart-box { |
| | | width: 100%; |
| | | height: 280px; |
| | | } |
| | | |
| | | .panel-device .mini-grid, |
| | | .panel-network .mini-grid, |
| | | .panel-ai .mini-grid { |
| | | grid-template-columns: repeat(2, minmax(0, 1fr)); |
| | | } |
| | |
| | | } |
| | | |
| | | .device-chart-box, |
| | | .network-chart-box, |
| | | .ai-chart-box { |
| | | width: 100%; |
| | | height: 250px; |
| | |
| | | padding: 8px 10px; |
| | | } |
| | | |
| | | .network-note { |
| | | margin-top: 6px; |
| | | font-size: 12px; |
| | | line-height: 1.5; |
| | | border-radius: 12px; |
| | | padding: 8px 10px; |
| | | } |
| | | |
| | | .network-note-danger { |
| | | color: #c15b5b; |
| | | background: rgba(222, 92, 92, 0.08); |
| | | } |
| | | |
| | | .network-note-warning { |
| | | color: #b67632; |
| | | background: rgba(245, 154, 74, 0.12); |
| | | } |
| | | |
| | | .network-note-info { |
| | | color: #657d95; |
| | | background: rgba(125, 144, 164, 0.10); |
| | | } |
| | | |
| | | .network-healthy-state { |
| | | margin-top: 14px; |
| | | padding: 14px 16px; |
| | | border-radius: 16px; |
| | | border: 1px solid rgba(87, 186, 128, 0.20); |
| | | background: linear-gradient(135deg, rgba(87, 186, 128, 0.10) 0%, rgba(87, 186, 128, 0.03) 100%); |
| | | display: flex; |
| | | align-items: center; |
| | | justify-content: space-between; |
| | | gap: 14px; |
| | | } |
| | | |
| | | .network-healthy-main { |
| | | min-width: 0; |
| | | flex: 1; |
| | | } |
| | | |
| | | .network-healthy-title { |
| | | font-size: 14px; |
| | | font-weight: 700; |
| | | color: #2d7f56; |
| | | line-height: 1.5; |
| | | } |
| | | |
| | | .network-healthy-desc { |
| | | margin-top: 4px; |
| | | font-size: 12px; |
| | | color: #698399; |
| | | line-height: 1.6; |
| | | } |
| | | |
| | | .network-healthy-tags { |
| | | display: flex; |
| | | align-items: center; |
| | | justify-content: flex-end; |
| | | flex-wrap: wrap; |
| | | gap: 8px; |
| | | } |
| | | |
| | | .network-healthy-tag { |
| | | padding: 6px 10px; |
| | | border-radius: 999px; |
| | | background: rgba(255, 255, 255, 0.72); |
| | | border: 1px solid rgba(87, 186, 128, 0.18); |
| | | font-size: 12px; |
| | | color: #557160; |
| | | white-space: nowrap; |
| | | } |
| | | |
| | | .recent-panel { |
| | | min-height: 100%; |
| | | min-height: 0; |
| | | } |
| | | |
| | | .recent-table { |
| | |
| | | } |
| | | |
| | | .panel-device .mini-grid, |
| | | .panel-network .mini-grid, |
| | | .panel-ai .mini-grid { |
| | | grid-template-columns: 1fr; |
| | | } |
| | |
| | | .route-row-side { |
| | | align-items: flex-start; |
| | | text-align: left; |
| | | } |
| | | |
| | | .network-healthy-state { |
| | | flex-direction: column; |
| | | align-items: flex-start; |
| | | } |
| | | |
| | | .network-healthy-tags { |
| | | justify-content: flex-start; |
| | | } |
| | | |
| | | .panel-actions { |
| | | width: 100%; |
| | | justify-content: flex-start; |
| | | } |
| | | } |
| | | </style> |
| | |
| | | </el-table> |
| | | </div> |
| | | </section> |
| | | </div> |
| | | |
| | | <div class="dashboard-column"> |
| | | <section class="panel panel-device"> |
| | | <div class="panel-header"> |
| | | <div> |
| | | <div class="panel-kicker">Devices</div> |
| | | <h2 class="panel-title">设备态势</h2> |
| | | <div class="panel-desc">汇总输送站点、堆垛机、双工位堆垛机与 RGV 的在线、忙碌和告警情况。</div> |
| | | </div> |
| | | <el-tag size="small" type="info">在线率 {{ devices.overview.onlineRate || 0 }}%</el-tag> |
| | | </div> |
| | | |
| | | <div class="mini-grid"> |
| | | <div class="mini-card"> |
| | | <div class="mini-label">设备总数</div> |
| | | <div class="mini-value">{{ formatNumber(devices.overview.total) }}</div> |
| | | <div class="mini-hint">已启用配置设备</div> |
| | | </div> |
| | | <div class="mini-card"> |
| | | <div class="mini-label">在线设备</div> |
| | | <div class="mini-value">{{ formatNumber(devices.overview.online) }}</div> |
| | | <div class="mini-hint">实时连通设备数量</div> |
| | | </div> |
| | | <div class="mini-card"> |
| | | <div class="mini-label">忙碌设备</div> |
| | | <div class="mini-value">{{ formatNumber(devices.overview.busy) }}</div> |
| | | <div class="mini-hint">当前承载任务的设备</div> |
| | | </div> |
| | | <div class="mini-card"> |
| | | <div class="mini-label">告警设备</div> |
| | | <div class="mini-value">{{ formatNumber(devices.overview.alarm) }}</div> |
| | | <div class="mini-hint">含阻塞或报警状态</div> |
| | | </div> |
| | | </div> |
| | | |
| | | <div class="chart-card"> |
| | | <div class="chart-title">设备在线分布</div> |
| | | <div ref="deviceTypeChart" class="device-chart-box"></div> |
| | | </div> |
| | | |
| | | <div class="type-list"> |
| | | <div v-for="item in devices.typeStats" :key="item.name" class="type-row"> |
| | | <div class="type-row-main"> |
| | | <div class="type-row-name">{{ item.name }}</div> |
| | | <div class="type-row-desc">在线 {{ formatNumber(item.online) }} / 总数 {{ formatNumber(item.total) }},离线 {{ formatNumber(item.offline) }}</div> |
| | | </div> |
| | | <div class="type-row-side"> |
| | | <el-tag size="mini" type="success">忙碌 {{ formatNumber(item.busy) }}</el-tag> |
| | | <el-tag size="mini" :type="item.alarm > 0 ? 'danger' : 'info'">告警 {{ formatNumber(item.alarm) }}</el-tag> |
| | | </div> |
| | | </div> |
| | | </div> |
| | | </section> |
| | | |
| | | <section class="panel panel-ai"> |
| | | <div class="panel-header"> |
| | |
| | | <el-empty v-else description="暂无 AI 路由数据"></el-empty> |
| | | </section> |
| | | </div> |
| | | |
| | | <div class="dashboard-column"> |
| | | <section class="panel panel-device"> |
| | | <div class="panel-header"> |
| | | <div> |
| | | <div class="panel-kicker">Devices</div> |
| | | <h2 class="panel-title">设备态势</h2> |
| | | <div class="panel-desc">汇总输送站点、堆垛机、双工位堆垛机与 RGV 的在线、忙碌和告警情况。</div> |
| | | </div> |
| | | <el-tag size="small" type="info">在线率 {{ devices.overview.onlineRate || 0 }}%</el-tag> |
| | | </div> |
| | | |
| | | <div class="mini-grid"> |
| | | <div class="mini-card"> |
| | | <div class="mini-label">设备总数</div> |
| | | <div class="mini-value">{{ formatNumber(devices.overview.total) }}</div> |
| | | <div class="mini-hint">已启用配置设备</div> |
| | | </div> |
| | | <div class="mini-card"> |
| | | <div class="mini-label">在线设备</div> |
| | | <div class="mini-value">{{ formatNumber(devices.overview.online) }}</div> |
| | | <div class="mini-hint">实时连通设备数量</div> |
| | | </div> |
| | | <div class="mini-card"> |
| | | <div class="mini-label">忙碌设备</div> |
| | | <div class="mini-value">{{ formatNumber(devices.overview.busy) }}</div> |
| | | <div class="mini-hint">当前承载任务的设备</div> |
| | | </div> |
| | | <div class="mini-card"> |
| | | <div class="mini-label">告警设备</div> |
| | | <div class="mini-value">{{ formatNumber(devices.overview.alarm) }}</div> |
| | | <div class="mini-hint">含阻塞或报警状态</div> |
| | | </div> |
| | | </div> |
| | | |
| | | <div class="chart-card"> |
| | | <div class="chart-title">设备在线分布</div> |
| | | <div ref="deviceTypeChart" class="device-chart-box"></div> |
| | | </div> |
| | | |
| | | <div class="type-list"> |
| | | <div v-for="item in devices.typeStats" :key="item.name" class="type-row"> |
| | | <div class="type-row-main"> |
| | | <div class="type-row-name">{{ item.name }}</div> |
| | | <div class="type-row-desc">在线 {{ formatNumber(item.online) }} / 总数 {{ formatNumber(item.total) }},离线 {{ formatNumber(item.offline) }}</div> |
| | | </div> |
| | | <div class="type-row-side"> |
| | | <el-tag size="mini" type="success">忙碌 {{ formatNumber(item.busy) }}</el-tag> |
| | | <el-tag size="mini" :type="item.alarm > 0 ? 'danger' : 'info'">告警 {{ formatNumber(item.alarm) }}</el-tag> |
| | | </div> |
| | | </div> |
| | | </div> |
| | | </section> |
| | | |
| | | <section class="panel panel-network"> |
| | | <div class="panel-header"> |
| | | <div> |
| | | <div class="panel-kicker">Network</div> |
| | | <h2 class="panel-title">设备网络分析</h2> |
| | | <div class="panel-desc">汇总最新 Ping 样本的连通性、延迟与异常设备,帮助快速发现网络波动。</div> |
| | | </div> |
| | | <div class="panel-actions"> |
| | | <el-tag size="small" :type="network.overview.attentionDevices > 0 ? 'warning' : 'success'"> |
| | | 需关注 {{ formatNumber(network.overview.attentionDevices) }} |
| | | </el-tag> |
| | | <el-button size="mini" plain @click="openDevicePingAnalysis">查看明细</el-button> |
| | | </div> |
| | | </div> |
| | | |
| | | <div class="mini-grid"> |
| | | <div class="mini-card network-mini-ok"> |
| | | <div class="mini-label">正常</div> |
| | | <div class="mini-value">{{ formatNumber(network.overview.okDevices) }}</div> |
| | | <div class="mini-hint">最新样本状态 OK</div> |
| | | </div> |
| | | <div class="mini-card network-mini-warning"> |
| | | <div class="mini-label">波动</div> |
| | | <div class="mini-value">{{ formatNumber(network.overview.unstableDevices) }}</div> |
| | | <div class="mini-hint">部分探测成功</div> |
| | | </div> |
| | | <div class="mini-card network-mini-offline"> |
| | | <div class="mini-label">超时/异常</div> |
| | | <div class="mini-value">{{ formatNumber(network.overview.offlineDevices) }}</div> |
| | | <div class="mini-hint">暂无数据 {{ formatNumber(network.overview.noDataDevices) }}</div> |
| | | </div> |
| | | <div class="mini-card network-mini-latency"> |
| | | <div class="mini-label">平均延迟</div> |
| | | <div class="mini-value">{{ formatLatency(network.overview.avgLatencyMs) }}</div> |
| | | <div class="mini-hint">峰值 {{ formatLatency(network.overview.maxLatencyMs) }}</div> |
| | | </div> |
| | | </div> |
| | | |
| | | <div class="chart-card"> |
| | | <div class="chart-title">连通状态分布</div> |
| | | <div class="chart-subtitle">{{ networkSamplingText() }}</div> |
| | | <div ref="networkStatusChart" class="network-chart-box"></div> |
| | | </div> |
| | | |
| | | <div class="route-list" v-if="network.focusDevices.length"> |
| | | <div v-for="item in network.focusDevices" :key="item.name + '-' + item.ip" class="route-row"> |
| | | <div class="route-row-main"> |
| | | <div class="route-row-name">{{ item.name }}</div> |
| | | <div class="route-row-desc">{{ displayText(item.ip, '-') }}</div> |
| | | <div v-if="item.message" :class="['network-note', 'network-note-' + (item.statusType || 'info')]">{{ item.message }}</div> |
| | | </div> |
| | | <div class="route-row-side"> |
| | | <el-tag size="mini" :type="item.statusType">{{ item.statusText }}</el-tag> |
| | | <div class="route-extra">平均 {{ formatLatency(item.avgLatencyMs) }}</div> |
| | | <div class="route-extra">最近样本 {{ displayText(item.latestTimeLabel, '-') }}</div> |
| | | </div> |
| | | </div> |
| | | </div> |
| | | <div v-else-if="network.overview.totalDevices > 0" class="network-healthy-state"> |
| | | <div class="network-healthy-main"> |
| | | <div class="network-healthy-title">当前网络探测稳定</div> |
| | | <div class="network-healthy-desc">已纳入 {{ formatNumber(network.overview.totalDevices) }} 台设备,最近一轮未发现超时或波动。</div> |
| | | </div> |
| | | <div class="network-healthy-tags"> |
| | | <div class="network-healthy-tag">正常 {{ formatNumber(network.overview.okDevices) }}</div> |
| | | <div class="network-healthy-tag">平均 {{ formatLatency(network.overview.avgLatencyMs) }}</div> |
| | | <div class="network-healthy-tag">峰值 {{ formatLatency(network.overview.maxLatencyMs) }}</div> |
| | | </div> |
| | | </div> |
| | | <el-empty v-else description="暂无设备网络样本"></el-empty> |
| | | </section> |
| | | </div> |
| | | </div> |
| | | |
| | | <div v-if="loading" class="loading-mask"> |
| | |
| | | <script type="text/javascript" src="../../static/vue/js/vue.min.js"></script> |
| | | <script type="text/javascript" src="../../static/vue/element/element.js"></script> |
| | | <script type="text/javascript" src="../../static/js/echarts/echarts.min.js"></script> |
| | | <script type="text/javascript" src="../../static/js/dashboard/dashboard.js"></script> |
| | | <script type="text/javascript" src="../../static/js/dashboard/dashboard.js?v=20260317-dashboard-network-focus"></script> |
| | | </body> |
| | | </html> |