|
@@ -7,188 +7,50 @@
|
|
|
</div>
|
|
|
|
|
|
<!-- 顶部统计卡片 -->
|
|
|
- <div class="statistics-panel">
|
|
|
- <div class="stat-item">
|
|
|
- <div class="icon-wrapper">
|
|
|
- <svg-icon icon-class="order-total" class="icon" />
|
|
|
- </div>
|
|
|
- <div class="stat-info">
|
|
|
- <count-to :start-val="0" :end-val="20" :duration="2000" class="num"/>
|
|
|
- <div class="text">本月总订单</div>
|
|
|
- </div>
|
|
|
- </div>
|
|
|
- <div class="stat-item">
|
|
|
- <div class="icon-wrapper">
|
|
|
- <svg-icon icon-class="order-produced" class="icon" />
|
|
|
- </div>
|
|
|
- <div class="stat-info">
|
|
|
- <count-to :start-val="0" :end-val="5" :duration="2000" class="num"/>
|
|
|
- <div class="text">本月已生产订单</div>
|
|
|
- </div>
|
|
|
- </div>
|
|
|
- <div class="stat-item">
|
|
|
- <div class="icon-wrapper">
|
|
|
- <svg-icon icon-class="order-producing" class="icon" />
|
|
|
- </div>
|
|
|
- <div class="stat-info">
|
|
|
- <count-to :start-val="0" :end-val="8" :duration="2000" class="num"/>
|
|
|
- <div class="text">正在生产订单</div>
|
|
|
- </div>
|
|
|
- </div>
|
|
|
- <div class="stat-item">
|
|
|
- <div class="icon-wrapper">
|
|
|
- <svg-icon icon-class="order-unproduced" class="icon" />
|
|
|
- </div>
|
|
|
- <div class="stat-info">
|
|
|
- <count-to :start-val="0" :end-val="7" :duration="2000" class="num"/>
|
|
|
- <div class="text">未生产</div>
|
|
|
- </div>
|
|
|
- </div>
|
|
|
- <div class="stat-item">
|
|
|
- <div class="icon-wrapper">
|
|
|
- <svg-icon icon-class="order-total-count" class="icon" />
|
|
|
- </div>
|
|
|
- <div class="stat-info">
|
|
|
- <count-to :start-val="0" :end-val="1000" :duration="2000" class="num"/>
|
|
|
- <div class="text">订单总量</div>
|
|
|
- </div>
|
|
|
- </div>
|
|
|
- <div class="stat-item">
|
|
|
- <div class="icon-wrapper">
|
|
|
- <svg-icon icon-class="order-completed" class="icon" />
|
|
|
- </div>
|
|
|
- <div class="stat-info">
|
|
|
- <count-to :start-val="0" :end-val="5000" :duration="2000" class="num"/>
|
|
|
- <div class="text">已完成量</div>
|
|
|
- </div>
|
|
|
- </div>
|
|
|
- <div class="stat-item">
|
|
|
- <div class="icon-wrapper">
|
|
|
- <svg-icon icon-class="order-uncompleted" class="icon" />
|
|
|
- </div>
|
|
|
- <div class="stat-info">
|
|
|
- <count-to :start-val="0" :end-val="5000" :duration="2000" class="num"/>
|
|
|
- <div class="text">未完成量</div>
|
|
|
- </div>
|
|
|
- </div>
|
|
|
- </div>
|
|
|
+ <statistics-panel />
|
|
|
|
|
|
<!-- 生产概况表格 -->
|
|
|
- <el-card class="box-card" style="margin-top: 20px;">
|
|
|
- <div slot="header" class="clearfix">
|
|
|
- <div style="display: flex; justify-content: space-between; align-items: center;">
|
|
|
- <div>
|
|
|
- <span style="font-size: 14px;">生产概况</span>
|
|
|
- <span style="margin-left: 10px; font-size: 14px; color: #666;">{{ updateTime }}</span>
|
|
|
- </div>
|
|
|
- <el-tag type="primary" style="cursor: pointer" @click="handleUnassignedOrders">未接单:7条</el-tag>
|
|
|
- </div>
|
|
|
- </div>
|
|
|
- <el-table :data="tableData" style="width: 100%" border>
|
|
|
- <el-table-column prop="orderNo" label="订单编号" width="120" align="center"/>
|
|
|
- <el-table-column prop="productName" label="货品名称" min-width="120" align="center"/>
|
|
|
- <el-table-column prop="status" label="单据状态" width="100" align="center">
|
|
|
- <template slot-scope="scope">
|
|
|
- <el-tag :type="getStatusType(scope.row.status)" size="medium" effect="plain">
|
|
|
- {{ scope.row.status }}
|
|
|
- </el-tag>
|
|
|
- </template>
|
|
|
- </el-table-column>
|
|
|
- <el-table-column prop="progress" label="进度" width="350" align="center">
|
|
|
- <template slot-scope="scope">
|
|
|
- <div class="progress-wrapper">
|
|
|
- <div class="progress-container">
|
|
|
- <div class="progress-bar">
|
|
|
- <el-progress
|
|
|
- :percentage="(scope.row.progress / scope.row.total) * 100"
|
|
|
- :stroke-width="12"
|
|
|
- :show-text="false"
|
|
|
- :color="getProgressColor(scope.row.status)"/>
|
|
|
- </div>
|
|
|
- <div class="progress-info">
|
|
|
- <span
|
|
|
- class="finished-number"
|
|
|
- @click="handleProgressClick(scope.row)"
|
|
|
- style="cursor: pointer; color: #409EFF; margin-right: 4px;">
|
|
|
- {{ scope.row.progress }}
|
|
|
- </span>
|
|
|
- <span style="color: #909399;">/</span>
|
|
|
- <span class="total-number" style="color: #606266; margin-left: 4px;">{{ scope.row.total }}</span>
|
|
|
- </div>
|
|
|
- </div>
|
|
|
- </div>
|
|
|
- </template>
|
|
|
- </el-table-column>
|
|
|
- <el-table-column prop="total" label="计划量" width="100" align="center"/>
|
|
|
- <el-table-column prop="finished" label="已完成量" width="100" align="center"/>
|
|
|
- <el-table-column prop="unfinished" label="未完成量" width="100" align="center"/>
|
|
|
- <el-table-column prop="rate" label="完成率" width="100" align="center">
|
|
|
- <template slot-scope="scope">
|
|
|
- {{ (scope.row.finished / scope.row.total * 100).toFixed(1) + '%' }}
|
|
|
- </template>
|
|
|
- </el-table-column>
|
|
|
- <el-table-column prop="remainingDays" label="距离完成时间还剩" width="150" align="center">
|
|
|
- <template slot-scope="scope">
|
|
|
- <span :style="{ color: scope.row.remainingDays < 0 ? '#F56C6C' : '' }">
|
|
|
- {{ scope.row.remainingDays > 0 ? scope.row.remainingDays + '天' : '超期' + Math.abs(scope.row.remainingDays) + '天' }}
|
|
|
- </span>
|
|
|
- </template>
|
|
|
- </el-table-column>
|
|
|
- </el-table>
|
|
|
- </el-card>
|
|
|
+ <production-table
|
|
|
+ :table-data="tableData"
|
|
|
+ :update-time="updateTime"
|
|
|
+ @unassigned-click="handleUnassignedOrders"
|
|
|
+ @progress-click="handleProgressClick"
|
|
|
+ />
|
|
|
|
|
|
<!-- 未接单弹窗 -->
|
|
|
- <el-dialog
|
|
|
+ <unassigned-dialog
|
|
|
:visible.sync="dialogVisible"
|
|
|
- width="80%"
|
|
|
- custom-class="unassigned-dialog"
|
|
|
- :before-close="handleClose">
|
|
|
- <div slot="title" class="dialog-title">
|
|
|
- <div class="title-content">
|
|
|
- <span>生产未接单</span>
|
|
|
- <span class="dialog-update-time">{{ updateTime }}</span>
|
|
|
- </div>
|
|
|
- </div>
|
|
|
- <el-table :data="unassignedOrders" style="width: 100%" border>
|
|
|
- <el-table-column prop="orderNo" label="订单编号" width="120" align="center"/>
|
|
|
- <el-table-column prop="planNum" label="计划量" width="120" align="center"/>
|
|
|
- <el-table-column prop="goods" label="货品" min-width="120" align="center"/>
|
|
|
- <el-table-column prop="orderPerson" label="下单人" width="120" align="center"/>
|
|
|
- <el-table-column prop="orderTime" label="下单时间" width="120" align="center"/>
|
|
|
- <el-table-column prop="priority" label="优先级" width="100" align="center"/>
|
|
|
- <el-table-column prop="contract" label="合同" min-width="200" align="center"/>
|
|
|
- </el-table>
|
|
|
- <div slot="footer" class="dialog-footer">
|
|
|
- <el-button type="primary" size="medium" @click="dialogVisible = false">关 闭</el-button>
|
|
|
- </div>
|
|
|
- </el-dialog>
|
|
|
+ :update-time="updateTime"
|
|
|
+ :unassigned-orders="unassignedOrders"
|
|
|
+ />
|
|
|
|
|
|
- <!-- 月度生产概况图表 -->
|
|
|
- <el-card class="box-card" style="margin-top: 20px;">
|
|
|
- <div slot="header" class="clearfix">
|
|
|
- <span>月度生产概况</span>
|
|
|
- </div>
|
|
|
- <div class="chart-wrapper">
|
|
|
- <bar-chart :chart-data="chartData"/>
|
|
|
- </div>
|
|
|
- </el-card>
|
|
|
+ <!-- 货品生产情况 -->
|
|
|
+ <production-status-chart
|
|
|
+ :chart-data="chartData"
|
|
|
+ @filter-change="handleFilterChange"
|
|
|
+ />
|
|
|
</div>
|
|
|
</template>
|
|
|
|
|
|
<script>
|
|
|
-import CountTo from 'vue-count-to'
|
|
|
-import BarChart from './components/BarChart'
|
|
|
+import StatisticsPanel from './components/StatisticsPanel'
|
|
|
+import ProductionTable from './components/ProductionTable'
|
|
|
+import UnassignedDialog from './components/UnassignedDialog'
|
|
|
+import ProductionStatusChart from './components/ProductionStatusChart'
|
|
|
|
|
|
export default {
|
|
|
name: 'ProductionSummary',
|
|
|
components: {
|
|
|
- CountTo,
|
|
|
- BarChart
|
|
|
+ StatisticsPanel,
|
|
|
+ ProductionTable,
|
|
|
+ UnassignedDialog,
|
|
|
+ ProductionStatusChart
|
|
|
},
|
|
|
data() {
|
|
|
return {
|
|
|
currentTime: '',
|
|
|
- updateTime: '', // 后端返回的更新时间
|
|
|
+ updateTime: '',
|
|
|
+ dialogVisible: false,
|
|
|
tableData: [
|
|
|
{
|
|
|
orderNo: 'DD20240001',
|
|
@@ -301,22 +163,6 @@ export default {
|
|
|
remainingDays: 6
|
|
|
}
|
|
|
],
|
|
|
- chartData: {
|
|
|
- labels: ['包装材料', '包装材料', '板材类', '橡胶类型', '塑料类型', '特定订单', '无规格类型'],
|
|
|
- datasets: [
|
|
|
- {
|
|
|
- label: '计划量',
|
|
|
- data: [500, 2000, 600, 1200, 600, 1400, 500, 1600],
|
|
|
- backgroundColor: '#409EFF'
|
|
|
- },
|
|
|
- {
|
|
|
- label: '完成量',
|
|
|
- data: [400, 1600, 500, 900, 400, 1200, 500, 1600],
|
|
|
- backgroundColor: '#67C23A'
|
|
|
- }
|
|
|
- ]
|
|
|
- },
|
|
|
- dialogVisible: false,
|
|
|
unassignedOrders: [
|
|
|
{
|
|
|
orderNo: '00000021',
|
|
@@ -372,23 +218,44 @@ export default {
|
|
|
priority: '一般',
|
|
|
contract: 'HT00000007 海丰牧场'
|
|
|
}
|
|
|
- ]
|
|
|
+ ],
|
|
|
+ chartData: {
|
|
|
+ labels: ['智能喷淋', '智能膜环', '佳沃评分', '精准阉割', '奶牛称重', '车载控制器', '大屏', '耳标'],
|
|
|
+ datasets: [
|
|
|
+ {
|
|
|
+ label: '计划量',
|
|
|
+ data: [600, 2000, 700, 970, 680, 840, 500, 1600],
|
|
|
+ backgroundColor: '#409EFF',
|
|
|
+ order: 2
|
|
|
+ },
|
|
|
+ {
|
|
|
+ label: '完成量',
|
|
|
+ data: [550, 1590, 500, 800, 630, 800, 500, 1600],
|
|
|
+ backgroundColor: '#FF9F43',
|
|
|
+ order: 2
|
|
|
+ },
|
|
|
+ {
|
|
|
+ label: '完成率',
|
|
|
+ data: [91.7, 79.5, 71.4, 82.5, 92.6, 95.2, 100, 100],
|
|
|
+ type: 'line',
|
|
|
+ borderColor: '#67C23A',
|
|
|
+ borderWidth: 2,
|
|
|
+ fill: false,
|
|
|
+ yAxisID: 'percentage',
|
|
|
+ order: 1
|
|
|
+ }
|
|
|
+ ]
|
|
|
+ }
|
|
|
}
|
|
|
},
|
|
|
created() {
|
|
|
- // 初始化并启动生产总览时间更新
|
|
|
this.updateCurrentTime()
|
|
|
setInterval(this.updateCurrentTime, 1000)
|
|
|
-
|
|
|
- // 初始化并启动生产概况更新时间获取
|
|
|
this.getUpdateTime()
|
|
|
setInterval(this.getUpdateTime, 60000)
|
|
|
-
|
|
|
- // 获取表格数据
|
|
|
this.getTableData()
|
|
|
},
|
|
|
methods: {
|
|
|
- // 更新生产总览的当前时间
|
|
|
updateCurrentTime() {
|
|
|
const now = new Date()
|
|
|
const year = now.getFullYear()
|
|
@@ -399,15 +266,8 @@ export default {
|
|
|
const seconds = now.getSeconds().toString().padStart(2, '0')
|
|
|
this.currentTime = `${year}-${month}-${day} ${hours}:${minutes}:${seconds}`
|
|
|
},
|
|
|
-
|
|
|
- // 从后端获取生产概况更新时间
|
|
|
async getUpdateTime() {
|
|
|
try {
|
|
|
- // TODO: 替换为实际的后端API
|
|
|
- // const res = await getProductionUpdateTime()
|
|
|
- // this.updateTime = res.data.updateTime
|
|
|
-
|
|
|
- // 临时使用本地时间模拟
|
|
|
const now = new Date()
|
|
|
const month = (now.getMonth() + 1).toString().padStart(2, '0')
|
|
|
const day = now.getDate().toString().padStart(2, '0')
|
|
@@ -418,83 +278,40 @@ export default {
|
|
|
console.error('获取更新时间失败:', error)
|
|
|
}
|
|
|
},
|
|
|
- // 处理未接单点击事件
|
|
|
handleUnassignedOrders() {
|
|
|
- // 实际开发时调用获取数据的方法
|
|
|
this.getUnassignedOrders()
|
|
|
this.dialogVisible = true
|
|
|
},
|
|
|
- // 处理弹窗关闭前的回调
|
|
|
- handleClose(done) {
|
|
|
- done()
|
|
|
- },
|
|
|
- // 获取未接单数据
|
|
|
async getUnassignedOrders() {
|
|
|
try {
|
|
|
- // TODO: 实际开发时替换为后端API
|
|
|
- // const res = await this.$http.get('/api/orders/unassigned')
|
|
|
- // this.unassignedOrders = res.data.list
|
|
|
-
|
|
|
- // 目前使用示例数据,实际开发时删除这部分
|
|
|
console.log('获取未接单数据')
|
|
|
} catch (error) {
|
|
|
console.error('获取未接单数据失败:', error)
|
|
|
this.$message.error('获取未接单数据失败')
|
|
|
}
|
|
|
},
|
|
|
- getStatusType(status) {
|
|
|
- // 根据状态返回相应的标签类型
|
|
|
- switch (status) {
|
|
|
- case '进行中':
|
|
|
- return 'primary'
|
|
|
- case '已超期':
|
|
|
- return 'danger'
|
|
|
- case '未开始':
|
|
|
- return 'info'
|
|
|
- default:
|
|
|
- return 'info'
|
|
|
- }
|
|
|
- },
|
|
|
- getProgressColor(status) {
|
|
|
- // 根据状态返回相应的进度条颜色
|
|
|
- switch (status) {
|
|
|
- case '进行中':
|
|
|
- return '#409EFF' // 蓝色
|
|
|
- case '已超期':
|
|
|
- return '#F56C6C' // 红色
|
|
|
- case '未开始':
|
|
|
- return '#909399' // 灰色
|
|
|
- default:
|
|
|
- return '#909399'
|
|
|
- }
|
|
|
- },
|
|
|
handleProgressClick(row) {
|
|
|
- // 处理进度条点击事件
|
|
|
console.log('进度条被点击:', row)
|
|
|
},
|
|
|
- formatRemainingDays(days) {
|
|
|
- // 格式化剩余天数
|
|
|
- if (days < 0) {
|
|
|
- return '已逾期'
|
|
|
- } else if (days === 0) {
|
|
|
- return '今日完成'
|
|
|
- } else {
|
|
|
- return `${days}天`
|
|
|
- }
|
|
|
- },
|
|
|
- // 获取表格数据
|
|
|
async getTableData() {
|
|
|
try {
|
|
|
- // TODO: 实际开发时替换为后端API
|
|
|
- // const res = await this.$http.get('/api/production/summary/list')
|
|
|
- // this.tableData = res.data.list
|
|
|
-
|
|
|
- // 目前使用示例数据,实际开发时删除这部分
|
|
|
console.log('获取生产概况表格数据')
|
|
|
} catch (error) {
|
|
|
console.error('获取表格数据失败:', error)
|
|
|
this.$message.error('获取表格数据失败')
|
|
|
}
|
|
|
+ },
|
|
|
+ handleFilterChange({ month, product }) {
|
|
|
+ console.log('筛选条件变更:', month, product)
|
|
|
+ this.getChartData(month, product)
|
|
|
+ },
|
|
|
+ async getChartData(month, product) {
|
|
|
+ try {
|
|
|
+ console.log('获取图表数据:', month, product)
|
|
|
+ } catch (error) {
|
|
|
+ console.error('获取图表数据失败:', error)
|
|
|
+ this.$message.error('获取图表数据失败')
|
|
|
+ }
|
|
|
}
|
|
|
}
|
|
|
}
|
|
@@ -537,170 +354,4 @@ export default {
|
|
|
font-size: 14px;
|
|
|
}
|
|
|
}
|
|
|
-
|
|
|
-.statistics-panel {
|
|
|
- background: #fff;
|
|
|
- border-radius: 8px;
|
|
|
- padding: 24px;
|
|
|
- margin-top: 24px;
|
|
|
- display: flex;
|
|
|
- justify-content: space-between;
|
|
|
- align-items: center;
|
|
|
- box-shadow: 0 2px 12px 0 rgba(0, 0, 0, 0.05);
|
|
|
-
|
|
|
- .stat-item {
|
|
|
- display: flex;
|
|
|
- align-items: center;
|
|
|
- padding: 0 16px;
|
|
|
- cursor: pointer;
|
|
|
- transition: all 0.3s ease;
|
|
|
-
|
|
|
- &:hover {
|
|
|
- transform: translateY(-2px);
|
|
|
- .icon-wrapper {
|
|
|
- box-shadow: 0 4px 12px rgba(0, 0, 0, 0.1);
|
|
|
- .icon {
|
|
|
- color: #409EFF;
|
|
|
- }
|
|
|
- }
|
|
|
- .stat-info {
|
|
|
- .num {
|
|
|
- color: #409EFF;
|
|
|
- }
|
|
|
- }
|
|
|
- }
|
|
|
-
|
|
|
- &:active {
|
|
|
- transform: translateY(0);
|
|
|
- }
|
|
|
-
|
|
|
- .icon-wrapper {
|
|
|
- width: 56px;
|
|
|
- height: 56px;
|
|
|
- border-radius: 50%;
|
|
|
- display: flex;
|
|
|
- justify-content: center;
|
|
|
- align-items: center;
|
|
|
- margin-right: 12px;
|
|
|
- transition: all 0.3s ease;
|
|
|
- background: #f5f7fa;
|
|
|
-
|
|
|
- .icon {
|
|
|
- width: 100%;
|
|
|
- height: 100%;
|
|
|
- color: #333;
|
|
|
- transition: all 0.3s ease;
|
|
|
- }
|
|
|
- }
|
|
|
-
|
|
|
- .stat-info {
|
|
|
- display: flex;
|
|
|
- flex-direction: column;
|
|
|
- justify-content: center;
|
|
|
-
|
|
|
- .num {
|
|
|
- font-size: 20px;
|
|
|
- font-weight: 600;
|
|
|
- color: #1f2d3d;
|
|
|
- line-height: 1;
|
|
|
- margin-bottom: 8px;
|
|
|
- }
|
|
|
-
|
|
|
- .text {
|
|
|
- font-size: 14px;
|
|
|
- color: #606266;
|
|
|
- white-space: nowrap;
|
|
|
- }
|
|
|
- }
|
|
|
- }
|
|
|
-}
|
|
|
-
|
|
|
-.chart-wrapper {
|
|
|
- background: #fff;
|
|
|
- padding: 24px;
|
|
|
- margin-bottom: 32px;
|
|
|
- height: 400px;
|
|
|
- border-radius: 8px;
|
|
|
- box-shadow: 0 2px 12px 0 rgba(0, 0, 0, 0.05);
|
|
|
-}
|
|
|
-
|
|
|
-// 优化表格卡片样式
|
|
|
-.box-card {
|
|
|
- border-radius: 8px;
|
|
|
- box-shadow: 0 2px 12px 0 rgba(0, 0, 0, 0.05);
|
|
|
- transition: all 0.3s ease;
|
|
|
-
|
|
|
- .el-card__header {
|
|
|
- padding: 20px 24px;
|
|
|
- border-bottom: 1px solid #ebeef5;
|
|
|
- }
|
|
|
-}
|
|
|
-
|
|
|
-// 未接单弹窗样式
|
|
|
-:deep(.unassigned-dialog) {
|
|
|
- .el-dialog__header {
|
|
|
- padding: 20px 24px;
|
|
|
- border-bottom: 1px solid #ebeef5;
|
|
|
- margin-right: 0;
|
|
|
-
|
|
|
- .dialog-title {
|
|
|
- .title-content {
|
|
|
- display: inline-flex;
|
|
|
- align-items: center;
|
|
|
- font-size: 16px;
|
|
|
- font-weight: 500;
|
|
|
-
|
|
|
- .dialog-update-time {
|
|
|
- font-size: 14px;
|
|
|
- color: #666;
|
|
|
- font-weight: normal;
|
|
|
- margin-left: 16px;
|
|
|
- }
|
|
|
- }
|
|
|
- }
|
|
|
- }
|
|
|
-
|
|
|
- .el-dialog__body {
|
|
|
- padding: 24px;
|
|
|
- }
|
|
|
-
|
|
|
- .el-table {
|
|
|
- th {
|
|
|
- background-color: #f5f7fa;
|
|
|
- color: #1f2d3d;
|
|
|
- font-weight: 500;
|
|
|
- }
|
|
|
-
|
|
|
- td {
|
|
|
- padding: 12px 0;
|
|
|
- }
|
|
|
- }
|
|
|
-}
|
|
|
-
|
|
|
-.progress-wrapper {
|
|
|
- padding: 10px 0;
|
|
|
-
|
|
|
- .progress-container {
|
|
|
- display: flex;
|
|
|
- flex-direction: column;
|
|
|
- align-items: center;
|
|
|
- gap: 8px;
|
|
|
-
|
|
|
- .progress-bar {
|
|
|
- width: 100%;
|
|
|
- padding: 0 20px;
|
|
|
- }
|
|
|
-
|
|
|
- .progress-info {
|
|
|
- font-size: 13px;
|
|
|
- font-weight: 500;
|
|
|
-
|
|
|
- .finished-number {
|
|
|
- &:hover {
|
|
|
- opacity: 0.8;
|
|
|
- }
|
|
|
- }
|
|
|
- }
|
|
|
- }
|
|
|
-}
|
|
|
</style>
|