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.

195 lines
12 KiB

1 month ago
1 month ago
4 days ago
1 month ago
1 month ago
4 weeks ago
1 month ago
4 weeks ago
1 month ago
1 month ago
4 days ago
1 month ago
1 month ago
4 days ago
1 month ago
4 days ago
4 days ago
4 days ago
4 days ago
4 days ago
4 days ago
4 days ago
4 days ago
  1. <template>
  2. <div class="screen-wrap">
  3. <div class="top-bar">
  4. <div class="header-left"><img class="site-navbar__brand-logo" src="~@/assets/img/lc.png" alt="龙闯电梯"></div>
  5. <div class="header-center">
  6. <div class="title">生产进度管理看板 · 家用电梯</div>
  7. </div>
  8. <div class="tools">
  9. <span class="time">{{ currentTime }}</span>
  10. </div>
  11. </div>
  12. <div class="kpi-row">
  13. <div class="kpi-card"><div class="kpi-label">排产订单</div><div class="kpi-value">{{ kpi.total }}</div></div>
  14. <div class="kpi-card"><div class="kpi-label">进行中</div><div class="kpi-value warning">{{ kpi.processing }}</div></div>
  15. <div class="kpi-card"><div class="kpi-label">已完成</div><div class="kpi-value success">{{ kpi.finished }}</div></div>
  16. <div class="kpi-card"><div class="kpi-label">完工达成率</div><div class="kpi-value highlight">{{ kpi.finishRate }}%</div></div>
  17. </div>
  18. <div class="legend-row">
  19. <span class="legend-item"><i class="dot done"></i>已完成</span>
  20. <span class="legend-item"><i class="dot todo"></i>未开始</span>
  21. </div>
  22. <el-table class="board-table" :data="boardList" :height="tableHeight" border stripe>
  23. <el-table-column type="index" label="序号" width="60" align="center"></el-table-column>
  24. <el-table-column prop="projectNo" label="项目号" width="140" align="center"></el-table-column>
  25. <el-table-column prop="modelNo" label="型号" width="130" align="center"></el-table-column>
  26. <el-table-column prop="color" label="颜色" width="90" align="center"></el-table-column>
  27. <el-table-column prop="floorCount" label="层数" width="80" align="center"></el-table-column>
  28. <el-table-column prop="specialRequirement" label="特殊要求" min-width="160" show-overflow-tooltip></el-table-column>
  29. <el-table-column prop="planDeliveryDate" label="计划发货日期" width="130" align="center"></el-table-column>
  30. <el-table-column label="仓库配料" width="105" align="center"><template slot-scope="scope"><span :class="getNodeCellClass(scope.row, 'stocking')"></span></template></el-table-column>
  31. <el-table-column label="平台组装/调试" width="130" align="center"><template slot-scope="scope"><span :class="getNodeCellClass(scope.row, 'platformDebug')"></span></template></el-table-column>
  32. <el-table-column label="背景墙/吊顶组装" width="140" align="center"><template slot-scope="scope"><span :class="getNodeCellClass(scope.row, 'bgCeiling')"></span></template></el-table-column>
  33. <el-table-column label="门组装" width="95" align="center"><template slot-scope="scope"><span :class="getNodeCellClass(scope.row, 'doorAssy')"></span></template></el-table-column>
  34. <el-table-column label="打包" width="90" align="center"><template slot-scope="scope"><span :class="getNodeCellClass(scope.row, 'pack')"></span></template></el-table-column>
  35. <el-table-column prop="status" label="订单状态" width="100" align="center"><template slot-scope="scope"><el-tag class="board-tag" :class="getOrderStatusClass(scope.row.status)" size="small">{{ scope.row.status }}</el-tag></template></el-table-column>
  36. <el-table-column prop="finishDate" label="完工时间" width="120" align="center"><template slot-scope="scope">{{ scope.row.finishDate || '-' }}</template></el-table-column>
  37. </el-table>
  38. </div>
  39. </template>
  40. <script>
  41. import { getHomeLiftOrderList } from '@/api/longchuang/productionPlan'
  42. import { getScreenInnerHeight } from '@/utils/screenAdapt'
  43. const NODE_TEMPLATE = ['stocking', 'platformDebug', 'bgCeiling', 'doorAssy', 'pack']
  44. export default {
  45. name: 'ScreenWholeLiftProgress',
  46. data() {
  47. return {
  48. loading: false,
  49. tableHeight: 640,
  50. currentTime: '',
  51. timerId: null,
  52. boardTimerId: null,
  53. boardList: [],
  54. kpi: { total: 0, processing: 0, finished: 0, finishRate: 0 }
  55. }
  56. },
  57. mounted() {
  58. this.setTableHeight()
  59. this.loadBoardData()
  60. this.updateTime()
  61. this.timerId = setInterval(this.updateTime, 1000)
  62. // 看板数据60秒刷新一次
  63. this.boardTimerId = setInterval(() => {
  64. this.loadBoardData()
  65. }, 60000)
  66. window.addEventListener('resize', this.setTableHeight)
  67. },
  68. beforeDestroy() {
  69. if (this.timerId) {
  70. clearInterval(this.timerId)
  71. }
  72. if (this.boardTimerId) {
  73. clearInterval(this.boardTimerId)
  74. }
  75. window.removeEventListener('resize', this.setTableHeight)
  76. },
  77. methods: {
  78. setTableHeight() {
  79. this.tableHeight = Math.max(420, getScreenInnerHeight() - 280)
  80. },
  81. loadBoardData() {
  82. if (this.loading) return
  83. this.loading = true
  84. let statusList = ['已排产', '进行中']
  85. getHomeLiftOrderList({ page: 1, limit: 300,statusList: statusList}).then(({data}) => {
  86. this.loading = false
  87. const source = (data && data.code === 0 && data.page && data.page.list) ? data.page.list : this.getMockList()
  88. this.boardList = this.buildBoardList(source)
  89. this.buildKpi()
  90. }).catch(() => {
  91. this.loading = false
  92. this.boardList = this.buildBoardList(this.getMockList())
  93. this.buildKpi()
  94. })
  95. },
  96. buildBoardList(sourceList) {
  97. const statusAllow = ['已排产', '进行中', '已完成']
  98. return sourceList.filter(item => statusAllow.includes(item.status)).map(item => ({
  99. projectNo: item.projectNo || '',
  100. modelNo: item.modelNo || '',
  101. color: item.color || '',
  102. floorCount: item.floorCount || '',
  103. specialRequirement: item.specialRequirement || '',
  104. planDeliveryDate: item.planDeliveryDate || '',
  105. status: item.status || '',
  106. finishDate: item.finishDate || '',
  107. nodeStatusMap: this.toNodeStatusMap(item.nodeList)
  108. }))
  109. },
  110. toNodeStatusMap(nodeList) {
  111. const map = {}
  112. NODE_TEMPLATE.forEach(code => { map[code] = '未开始' })
  113. ;(nodeList || []).forEach(node => { map[node.nodeCode] = node.status || '未开始' })
  114. return map
  115. },
  116. buildKpi() {
  117. const total = this.boardList.length
  118. const processing = this.boardList.filter(item => item.status === '进行中').length
  119. const finished = this.boardList.filter(item => item.status === '已完成').length
  120. this.kpi = { total, processing, finished, finishRate: total ? Math.round((finished / total) * 100) : 0 }
  121. },
  122. getNodeCellText(row, code) {
  123. return row.nodeStatusMap[code] || '未开始'
  124. },
  125. getNodeCellClass(row, code) {
  126. const status = this.getNodeCellText(row, code)
  127. if (status === '已完成') return 'node-chip done'
  128. return 'node-chip todo'
  129. },
  130. getOrderStatusClass(status) {
  131. if (status === '已完成') return 'status-done'
  132. if (status === '进行中') return 'status-doing'
  133. return 'status-planned'
  134. },
  135. updateTime() {
  136. const now = new Date()
  137. const weekList = ['星期日', '星期一', '星期二', '星期三', '星期四', '星期五', '星期六']
  138. const d = this.dayjs(now)
  139. this.currentTime = `${d.format('YYYY/MM/DD')} ${weekList[now.getDay()]} ${d.format('HH:mm:ss')}`
  140. },
  141. getMockList() {
  142. return []
  143. }
  144. }
  145. }
  146. </script>
  147. <style>
  148. .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}
  149. .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)}
  150. .top-bar{position:relative}
  151. .header-left{display:flex;align-items:center;min-width:110px;z-index:2}
  152. .logo-box{width:86px;height:30px;border-radius:6px;border:1px solid rgba(128,198,255,.45);color:#d8edff;font-size:13px;font-weight:700;display:flex;align-items:center;justify-content:center;background:rgba(36,82,122,.35)}
  153. .header-center{position:absolute;left:50%;top:50%;transform:translate(-50%,-50%);text-align:center;pointer-events:none;max-width:70%;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))}
  154. .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))}
  155. .top-bar .tools{margin-left:auto;z-index:2}
  156. .title{color:#8fe7ff;font-size:32px;font-weight:800;letter-spacing:3px;line-height:1.05;text-shadow:0 0 14px rgba(79,179,255,.36)}
  157. .subtitle{margin-top:4px;color:#c7e8ff;font-size:11px;letter-spacing:1px}
  158. .tools{display:flex;align-items:center;gap:12px}.time{color:#89d7ff;font-size:22px;font-weight:600}
  159. .kpi-row{display:grid;grid-template-columns:repeat(4,1fr);gap:12px;margin:12px 0 8px}
  160. .kpi-card{border-radius:10px;padding:12px 14px;background:rgba(14,43,70,.82);border:1px solid rgba(101,157,209,.22)}
  161. .kpi-label{color:#b7d3ee;font-size:14px}.kpi-value{margin-top:6px;font-size:32px;font-weight:700;color:#edf6ff}
  162. .kpi-value.warning{color:#ffd166}.kpi-value.success{color:#74dfa3}.kpi-value.highlight{color:#79d5ff}
  163. .legend-row{display:flex;gap:16px;margin:0 0 10px 2px}.legend-item{display:inline-flex;align-items:center;color:#c7dff4;font-size:13px}
  164. .dot{width:10px;height:10px;border-radius:50%;margin-right:6px;display:inline-block}.dot.done{background:#69e4a4}.dot.doing{background:#ffd35d}.dot.todo{background:#dbe3ee}
  165. .board-table{border-radius:10px;overflow:hidden;border:1px solid rgba(86,140,190,.35);flex:1;min-height:0}
  166. .board-table .cell {
  167. line-height: 36px;
  168. font-size: 17px;
  169. height: 36px;
  170. }
  171. .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}
  172. .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:15px!important;padding:14px 0!important}
  173. .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:55px!important;vertical-align:middle!important}
  174. .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}
  175. .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}
  176. .screen-wrap .board-table .el-table .cell{color:#fff!important;line-height:29px!important;overflow:visible!important}
  177. .screen-wrap .board-table .el-table__empty-text{color:#9ac2e7!important}
  178. .node-chip{display:inline-block!important;width:46px;height:14px;margin:0 auto;border-radius:2px}
  179. .node-chip.done{background:#6de3a0}.node-chip.todo{background:#dce4ee}
  180. .board-tag{display:inline-flex!important;align-items:center!important;justify-content:center!important;min-width:80px!important;height:29px!important;padding:0 10px!important;border-radius:14px!important;border:1px solid transparent!important;font-size:13px!important;font-weight:600!important;line-height:27px!important;box-sizing:border-box!important}
  181. .board-tag.status-planned{color:#b9c7d8!important;background:rgba(96,118,141,.28)!important;border-color:rgba(185,199,216,.34)!important}
  182. .board-tag.status-doing{color:#ffd774!important;background:rgba(120,92,27,.32)!important;border-color:rgba(255,215,116,.45)!important}
  183. .board-tag.status-done{color:#83f3be!important;background:rgba(31,110,84,.32)!important;border-color:rgba(131,243,190,.45)!important}
  184. </style>