You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
 
 
 
 
 

398 lines
23 KiB

<template>
<div class="screen-wrap">
<div class="top-bar">
<div class="header-left"><img class="site-navbar__brand-logo" src="~@/assets/img/lc.png" alt="龙闯电梯"></div>
<div class="header-center">
<div class="title">生产进度管理看板 · VL2.5升级/线缆COP/机加工</div>
</div>
<div class="tools">
<span class="time">{{ currentTime }}</span>
</div>
</div>
<div class="summary-row">
<div class="summary-card">
<div class="summary-title">VL2.5升级订单</div>
<div class="summary-main">
<span class="summary-rate">{{ renovationKpi.finishRate }}%</span>
<span class="summary-rate-label">完工达成率</span>
</div>
<div class="summary-meta">排产 {{ renovationKpi.total }} · 进行中 {{ renovationKpi.processing }} · 已完成 {{ renovationKpi.finished }}</div>
</div>
<div class="summary-card">
<div class="summary-title">线缆/COP任务</div>
<div class="summary-main">
<span class="summary-rate">{{ cableCopKpi.finishRate }}%</span>
<span class="summary-rate-label">完工达成率</span>
</div>
<div class="summary-meta">排产 {{ cableCopKpi.total }} · 进行中 {{ cableCopKpi.processing }} · 已完成 {{ cableCopKpi.finished }}</div>
</div>
<div class="summary-card">
<div class="summary-title">机加工任务</div>
<div class="summary-main">
<span class="summary-rate">{{ machiningKpi.finishRate }}%</span>
<span class="summary-rate-label">完工达成率</span>
</div>
<div class="summary-meta">排产 {{ machiningKpi.total }} · 进行中 {{ machiningKpi.processing }} · 已完成 {{ machiningKpi.finished }}</div>
</div>
</div>
<div class="legend-row">
<span class="legend-item"><i class="dot done"></i>已完成</span>
<span class="legend-item"><i class="dot todo"></i>未开始</span>
</div>
<div class="board-section">
<div class="section-card section-top">
<div class="section-header">
<span class="section-title">VL2.5升级订单进度</span>
<span class="section-count"> {{ renovationPage + 1 }}/{{ renovationPageCount }} · 显示 {{ displayRenovationList.length }} / {{ renovationList.length }} </span>
</div>
<el-table class="board-table" :data="displayRenovationList" :height="tableHeightTop" border stripe>
<el-table-column type="index" label="序号" width="54" align="center"></el-table-column>
<el-table-column prop="projectNo" label="项目号" min-width="120" align="center"></el-table-column>
<el-table-column prop="floors" label="楼层" min-width="70" align="center"></el-table-column>
<el-table-column prop="planFinishDate" label="计划完工日期" min-width="120" align="center"></el-table-column>
<el-table-column label="仓库配料" width="90" align="center"><template slot-scope="scope"><span :class="getNodeCellClass(scope.row, 'stocking')"></span></template></el-table-column>
<el-table-column label="组装" width="84" align="center"><template slot-scope="scope"><span :class="getNodeCellClass(scope.row, 'assy')"></span></template></el-table-column>
<el-table-column label="检验" width="84" align="center"><template slot-scope="scope"><span :class="getNodeCellClass(scope.row, 'inspect')"></span></template></el-table-column>
<el-table-column label="打包" width="84" align="center"><template slot-scope="scope"><span :class="getNodeCellClass(scope.row, 'pack')"></span></template></el-table-column>
<el-table-column prop="status" label="订单状态" width="96" align="center"><template slot-scope="scope"><el-tag class="board-tag" :class="getStatusTagClass(scope.row.status)" size="small">{{ scope.row.status }}</el-tag></template></el-table-column>
<el-table-column prop="finishDate" label="完工时间" width="118" align="center"><template slot-scope="scope">{{ scope.row.finishDate || '-' }}</template></el-table-column>
</el-table>
</div>
<div class="section-card section-bottom-left">
<div class="section-header">
<span class="section-title">线缆/COP任务进度</span>
<span class="section-count"> {{ cableCopPage + 1 }}/{{ cableCopPageCount }} · 显示 {{ displayCableCopList.length }} / {{ cableCopList.length }} </span>
</div>
<el-table class="board-table" :data="displayCableCopList" :height="tableHeightBottom" border stripe>
<el-table-column type="index" label="序号" width="54" align="center"></el-table-column>
<el-table-column prop="taskNo" label="任务单号" min-width="126" align="center"></el-table-column>
<el-table-column prop="taskType" label="类别" min-width="96" align="center">
<template slot-scope="scope">
<el-tag class="board-tag" :class="getTaskTypeClass(scope.row.taskType)" size="small">{{ scope.row.taskType || '-' }}</el-tag>
</template>
</el-table-column>
<el-table-column prop="planFinishDate" label="计划完工日期" min-width="120" align="center"></el-table-column>
<el-table-column prop="taskQty" label="任务数量" min-width="84" align="center"></el-table-column>
<el-table-column prop="reportQty" label="已报工数量" min-width="84" align="center"></el-table-column>
<el-table-column label="线缆生产" width="86" align="center"><template slot-scope="scope"><span :class="getNodeCellClass(scope.row, 'lineProduction')"></span></template></el-table-column>
<el-table-column label="COP生产" width="86" align="center"><template slot-scope="scope"><span :class="getNodeCellClass(scope.row, 'copProduction')"></span></template></el-table-column>
<el-table-column prop="status" label="任务状态" width="96" align="center">
<template slot-scope="scope">
<el-tag class="board-tag" :class="getStatusTagClass(scope.row.status)" size="small">{{ scope.row.status }}</el-tag>
</template>
</el-table-column>
<el-table-column prop="finishDate" label="完工时间" width="118" align="center"><template slot-scope="scope">{{ scope.row.finishDate || '-' }}</template></el-table-column>
</el-table>
</div>
<div class="section-card section-bottom-right">
<div class="section-header">
<span class="section-title">机加工任务进度</span>
<span class="section-count"> {{ machiningPage + 1 }}/{{ machiningPageCount }} · 显示 {{ displayMachiningList.length }} / {{ machiningList.length }} </span>
</div>
<el-table class="board-table" :data="displayMachiningList" :height="tableHeightBottom" border stripe>
<el-table-column type="index" label="序号" width="54" align="center"></el-table-column>
<el-table-column prop="modelNo" label="物料号" min-width="170" align="center"></el-table-column>
<el-table-column prop="taskQty" label="计划数量" min-width="88" align="center"></el-table-column>
<el-table-column prop="reportQty" label="实际数量" min-width="88" align="center"></el-table-column>
<el-table-column prop="planFinishDate" label="计划完工日期" min-width="120" align="center"></el-table-column>
<el-table-column prop="status" label="任务状态" width="96" align="center">
<template slot-scope="scope">
<el-tag class="board-tag" :class="getStatusTagClass(scope.row.status)" size="small">{{ scope.row.status }}</el-tag>
</template>
</el-table-column>
<el-table-column prop="finishDate" label="实际完工日期" min-width="120" align="center">
<template slot-scope="scope">{{ scope.row.finishDate || '-' }}</template>
</el-table-column>
</el-table>
</div>
</div>
</div>
</template>
<script>
import { getCableCopTaskList, getMachiningTaskList, getRenovationOrderList } from '@/api/longchuang/productionPlan'
import { getScreenInnerHeight } from '@/utils/screenAdapt'
const STATUS_ALLOW_LIST = ['已排产', '进行中', '已完成']
const RENOVATION_NODE_TEMPLATE = ['stocking', 'assy', 'inspect', 'pack']
const CABLE_COP_NODE_TEMPLATE = ['lineProduction', 'copProduction']
const BOARD_PAGE_SIZE = 6
export default {
name: 'ScreenRenovationProgress',
data() {
return {
loading: false,
boardTimerId: null,
tableHeightTop: 220,
tableHeightBottom: 180,
topRowLimit: BOARD_PAGE_SIZE,
bottomRowLimit: BOARD_PAGE_SIZE,
carouselTimerId: null,
carouselIntervalMs: 8000,
renovationPage: 0,
cableCopPage: 0,
machiningPage: 0,
currentTime: '',
timerId: null,
renovationList: [],
cableCopList: [],
machiningList: [],
renovationKpi: { total: 0, processing: 0, finished: 0, finishRate: 0 },
cableCopKpi: { total: 0, processing: 0, finished: 0, finishRate: 0 },
machiningKpi: { total: 0, processing: 0, finished: 0, finishRate: 0 }
}
},
computed: {
renovationPageCount() {
return this.getPageCount(this.renovationList.length, this.topRowLimit)
},
cableCopPageCount() {
return this.getPageCount(this.cableCopList.length, this.bottomRowLimit)
},
machiningPageCount() {
return this.getPageCount(this.machiningList.length, this.bottomRowLimit)
},
displayRenovationList() {
return this.getPageList(this.renovationList, this.renovationPage, this.topRowLimit)
},
displayCableCopList() {
return this.getPageList(this.cableCopList, this.cableCopPage, this.bottomRowLimit)
},
displayMachiningList() {
return this.getPageList(this.machiningList, this.machiningPage, this.bottomRowLimit)
}
},
mounted() {
this.setTableHeight()
this.loadBoardData()
this.updateTime()
this.timerId = setInterval(this.updateTime, 1000)
this.boardTimerId = setInterval(() => {
this.loadBoardData()
}, 60000)
this.startCarousel()
window.addEventListener('resize', this.setTableHeight)
},
beforeDestroy() {
if (this.timerId) clearInterval(this.timerId)
if (this.boardTimerId) clearInterval(this.boardTimerId)
this.stopCarousel()
window.removeEventListener('resize', this.setTableHeight)
},
methods: {
setTableHeight() {
const boardGap = 10
const topRatio = 0.56
const reservedHeight = 286
const sectionHeaderHeight = 46
const boardHeight = Math.max(420, getScreenInnerHeight() - reservedHeight)
const topSectionHeight = Math.floor((boardHeight - boardGap) * topRatio)
const bottomSectionHeight = boardHeight - boardGap - topSectionHeight
this.tableHeightTop = Math.max(120, topSectionHeight - sectionHeaderHeight)
this.tableHeightBottom = Math.max(120, bottomSectionHeight - sectionHeaderHeight)
this.topRowLimit = BOARD_PAGE_SIZE
this.bottomRowLimit = BOARD_PAGE_SIZE
this.normalizeCarouselPage()
},
loadBoardData() {
if (this.loading) return
this.loading = true
const params = { page: 1, limit: 300, statusList: STATUS_ALLOW_LIST }
Promise.all([
getRenovationOrderList(params).catch(() => null),
getCableCopTaskList(params).catch(() => null),
getMachiningTaskList(params).catch(() => null)
]).then(([renovationRes, cableCopRes, machiningRes]) => {
this.renovationList = this.buildRenovationList(this.getListFromResponse(renovationRes))
this.cableCopList = this.buildCableCopList(this.getListFromResponse(cableCopRes))
this.machiningList = this.buildMachiningList(this.getListFromResponse(machiningRes))
this.renovationKpi = this.buildKpi(this.renovationList)
this.cableCopKpi = this.buildKpi(this.cableCopList)
this.machiningKpi = this.buildKpi(this.machiningList)
this.normalizeCarouselPage()
}).finally(() => {
this.loading = false
})
},
getPageCount(total, pageSize) {
if (!pageSize) return 1
return Math.max(1, Math.ceil(total / pageSize))
},
getPageList(list, pageIndex, pageSize) {
const safeSize = Math.max(1, pageSize || 1)
const count = this.getPageCount((list || []).length, safeSize)
const safePage = Math.min(Math.max(pageIndex, 0), count - 1)
const start = safePage * safeSize
return (list || []).slice(start, start + safeSize)
},
normalizeCarouselPage() {
this.renovationPage = Math.min(this.renovationPage, this.renovationPageCount - 1)
this.cableCopPage = Math.min(this.cableCopPage, this.cableCopPageCount - 1)
this.machiningPage = Math.min(this.machiningPage, this.machiningPageCount - 1)
this.renovationPage = Math.max(this.renovationPage, 0)
this.cableCopPage = Math.max(this.cableCopPage, 0)
this.machiningPage = Math.max(this.machiningPage, 0)
},
nextPage(page, pageCount) {
if (pageCount <= 1) return 0
return (page + 1) % pageCount
},
startCarousel() {
this.stopCarousel()
this.carouselTimerId = setInterval(() => {
this.renovationPage = this.nextPage(this.renovationPage, this.renovationPageCount)
this.cableCopPage = this.nextPage(this.cableCopPage, this.cableCopPageCount)
this.machiningPage = this.nextPage(this.machiningPage, this.machiningPageCount)
}, this.carouselIntervalMs)
},
stopCarousel() {
if (!this.carouselTimerId) return
clearInterval(this.carouselTimerId)
this.carouselTimerId = null
},
getListFromResponse(response) {
const data = response && response.data
const list = data && data.code === 0 && data.page && Array.isArray(data.page.list) ? data.page.list : null
return list || this.getMockList()
},
buildRenovationList(sourceList) {
return (sourceList || [])
.filter(item => STATUS_ALLOW_LIST.includes(item.status))
.map(item => ({
projectNo: item.projectNo || '',
floors: item.floors || '',
planFinishDate: item.planFinishDate || '',
status: item.status || '',
finishDate: item.finishDate || '',
nodeStatusMap: this.toNodeStatusMap(item.nodeList, RENOVATION_NODE_TEMPLATE)
}))
},
buildCableCopList(sourceList) {
return (sourceList || [])
.filter(item => STATUS_ALLOW_LIST.includes(item.status))
.map(item => ({
taskNo: item.taskNo || '',
sourceProjectNo: item.sourceProjectNo || '',
taskType: item.taskType || '',
planFinishDate: item.planFinishDate || '',
status: item.status || '',
finishDate: item.finishDate || '',
taskQty: item.taskQty == null ? 0 : item.taskQty,
reportQty: item.reportQty == null ? 0 : item.reportQty,
nodeStatusMap: this.toNodeStatusMap(item.nodeList, CABLE_COP_NODE_TEMPLATE)
}))
},
buildMachiningList(sourceList) {
return (sourceList || [])
.filter(item => STATUS_ALLOW_LIST.includes(item.status))
.map(item => ({
modelNo: item.modelNo || '',
planFinishDate: item.planFinishDate || '',
status: item.status || '',
finishDate: item.finishDate || '',
taskQty: item.taskQty == null ? 0 : item.taskQty,
reportQty: item.reportQty == null ? 0 : item.reportQty
}))
},
toNodeStatusMap(nodeList, template) {
const map = {}
;(template || []).forEach(code => { map[code] = '未开始' })
;(nodeList || []).forEach(node => { map[node.nodeCode] = node.status || '未开始' })
return map
},
buildKpi(list) {
const total = list.length
const processing = list.filter(item => item.status === '进行中').length
const finished = list.filter(item => item.status === '已完成').length
return { total, processing, finished, finishRate: total ? Math.round((finished / total) * 100) : 0 }
},
getNodeCellText(row, code) {
return (row.nodeStatusMap && row.nodeStatusMap[code]) || '未开始'
},
getNodeCellClass(row, code) {
const status = this.getNodeCellText(row, code)
if (status === '已完成') return 'node-chip done'
return 'node-chip todo'
},
getTaskTypeClass(taskType) {
if (taskType === '线缆自制') return 'type-cable'
if (taskType === 'COP自制') return 'type-cop'
return 'type-default'
},
getStatusTagClass(status) {
if (status === '已完成') return 'status-done'
if (status === '进行中') return 'status-doing'
return 'status-planned'
},
updateTime() {
const now = new Date()
const weekList = ['星期日', '星期一', '星期二', '星期三', '星期四', '星期五', '星期六']
const d = this.dayjs(now)
this.currentTime = `${d.format('YYYY/MM/DD')} ${weekList[now.getDay()]} ${d.format('HH:mm:ss')}`
},
getMockList() {
return []
}
}
}
</script>
<style>
.screen-wrap{height:100vh;padding:16px;background:linear-gradient(165deg,#0a1f36 0%,#0f2b47 50%,#0a2037 100%);display:flex;flex-direction:column;box-sizing:border-box;overflow:hidden}
.top-bar{display:flex;justify-content:space-between;align-items:center;padding:14px 18px;border-radius:12px;border:1px solid rgba(109,167,219,.24);background:rgba(9,29,49,.82);position:relative}
.header-left{display:flex;align-items:center;min-width:110px;z-index:2}
.header-center{position:absolute;left:50%;top:50%;transform:translate(-50%,-50%);text-align:center;pointer-events:none;max-width:72%;width:auto;padding:6px 24px 8px;border-radius:8px;border:1px solid rgba(96,170,232,.34);background:linear-gradient(180deg,rgba(33,73,116,.52),rgba(16,44,77,.42))}
.header-center::before{content:'';position:absolute;left:50%;transform:translateX(-50%);top:-8px;width:68%;height:1px;background:linear-gradient(90deg,rgba(87,164,230,0),rgba(87,164,230,.9),rgba(87,164,230,0))}
.top-bar .tools{margin-left:auto;z-index:2}
.title{color:#8fe7ff;font-size:37px;font-weight:800;letter-spacing:2px;line-height:1.1;text-shadow:0 0 14px rgba(79,179,255,.36)}
.tools{display:flex;align-items:center;gap:12px}
.time{color:#89d7ff;font-size:26px;font-weight:600}
.summary-row{display:grid;grid-template-columns:repeat(3,minmax(0,1fr));gap:12px;margin:12px 0 8px}
.summary-card{border-radius:10px;padding:10px 14px;background:rgba(14,43,70,.82);border:1px solid rgba(101,157,209,.22)}
.summary-title{color:#b7d3ee;font-size:16px}
.summary-main{display:flex;align-items:baseline;gap:8px;margin-top:6px}
.summary-rate{font-size:39px;font-weight:700;color:#79d5ff;line-height:1}
.summary-rate-label{color:#9dc5e8;font-size:15px}
.summary-meta{margin-top:4px;color:#d4e6f9;font-size:15px;line-height:1.3}
.legend-row{display:flex;gap:16px;margin:0 0 10px 2px}
.legend-item{display:inline-flex;align-items:center;color:#c7dff4;font-size:15px}
.dot{width:10px;height:10px;border-radius:50%;margin-right:6px;display:inline-block}
.dot.done{background:#69e4a4}
.dot.todo{background:#dbe3ee}
.board-section{display:grid;grid-template-columns:repeat(2,minmax(0,1fr));grid-template-rows:minmax(0,1.1fr) minmax(0,1fr);gap:10px;flex:1;min-height:0}
.section-card{display:flex;flex-direction:column;min-height:0;border-radius:10px;border:1px solid rgba(86,140,190,.35);background:rgba(12,39,64,.96)}
.section-top{grid-column:1 / span 2}
.section-bottom-left{grid-column:1}
.section-bottom-right{grid-column:2}
.section-header{display:flex;align-items:center;justify-content:space-between;padding:10px 12px;border-bottom:1px solid rgba(80,133,181,.45)}
.section-title{color:#d7ebff;font-size:17px;font-weight:600}
.section-count{color:#9ec6e9;font-size:15px}
.board-table{border-radius:0;overflow:hidden;flex:1;min-height:0}
.board-table .cell{line-height:36px;font-size:16px;height:36px}
.screen-wrap .board-table .el-table,.screen-wrap .board-table .el-table__expanded-cell,.screen-wrap .board-table .el-table__body-wrapper,.screen-wrap .board-table .el-table__empty-block,.screen-wrap .board-table .el-table__fixed-body-wrapper{background:rgba(12,39,64,.96)!important;color:#fff!important}
.screen-wrap .board-table .el-table__header-wrapper th,.screen-wrap .board-table .el-table__fixed-header-wrapper th{background:#123a5e!important;color:#d9e9f8!important;border-color:rgba(80,133,181,.6)!important;font-size:16px!important;padding:12px 0!important}
.screen-wrap .board-table .el-table__body tr>td,.screen-wrap .board-table .el-table__fixed-body-wrapper tr>td,.screen-wrap .board-table .el-table__fixed-right .el-table__fixed-body-wrapper tr>td{background:rgba(20,52,83,.96)!important;border-color:rgba(88,139,187,.4)!important;color:#fff!important;height:48px!important;vertical-align:middle!important}
.screen-wrap .board-table .el-table--striped .el-table__body tr.el-table__row--striped>td,.screen-wrap .board-table .el-table__fixed-body-wrapper tr.el-table__row--striped>td{background:rgba(29,66,102,.96)!important}
.screen-wrap .board-table .el-table--enable-row-hover .el-table__body tr:hover>td,.screen-wrap .board-table .el-table__fixed-body-wrapper tr:hover>td{background:rgba(39,81,123,.96)!important}
.screen-wrap .board-table .el-table .cell{color:#fff!important;line-height:31px!important;overflow:visible!important}
.screen-wrap .board-table .el-table__body-wrapper{scrollbar-width:none;-ms-overflow-style:none}
.screen-wrap .board-table .el-table__body-wrapper::-webkit-scrollbar{width:0;height:0}
.screen-wrap .board-table .el-table__empty-text{color:#9ac2e7!important}
.node-chip{display:inline-block!important;width:46px;height:14px;margin:0 auto;border-radius:2px}
.node-chip.done{background:#6de3a0}
.node-chip.todo{background:#dce4ee}
.board-tag{display:inline-flex!important;align-items:center!important;justify-content:center!important;min-width:88px!important;height:29px!important;padding:0 10px!important;border-radius:14px!important;border:1px solid transparent!important;font-size:14px!important;font-weight:600!important;line-height:27px!important;box-sizing:border-box!important}
.board-tag.type-cable{color:#73cfff!important;background:rgba(28,88,136,.32)!important;border-color:rgba(115,207,255,.45)!important}
.board-tag.type-cop{color:#7ef0c0!important;background:rgba(31,108,89,.32)!important;border-color:rgba(126,240,192,.45)!important}
.board-tag.type-default{color:#d0e5f9!important;background:rgba(80,118,149,.28)!important;border-color:rgba(208,229,249,.34)!important}
.board-tag.status-planned{color:#b9c7d8!important;background:rgba(96,118,141,.28)!important;border-color:rgba(185,199,216,.34)!important}
.board-tag.status-doing{color:#ffd774!important;background:rgba(120,92,27,.32)!important;border-color:rgba(255,215,116,.45)!important}
.board-tag.status-done{color:#83f3be!important;background:rgba(31,110,84,.32)!important;border-color:rgba(131,243,190,.45)!important}
</style>