Files
erp-next/ruoyi-ui/src/views/bid/order/closeDate.vue
王文昊 7b71822a32 feat: 完成履约管理模块全量功能迭代
本次迭代包含以下核心功能:
1. 新增履约时效总览可视化页面,支持多维度数据统计
2. 实现物料/客户/供应商的Excel批量导入导出功能
3. 新增订单批量结单功能,优化结单流程校验
4. 完善日志配置,新增文件日志落地
5. 修复分类查询逻辑,优化多租户数据隔离
6. 新增甲方履约结单管理页面与权限控制
7. 重构部分Mapper与Service接口,增强代码健壮性
2026-06-18 11:10:36 +08:00

221 lines
11 KiB
Vue

<template>
<div class="jd-cd-page">
<!-- JD 统计卡片 -->
<el-row :gutter="12" class="stat-row">
<el-col :span="6" v-for="c in statCards" :key="c.key">
<div class="stat-card">
<div class="stat-body"><div class="stat-num">{{ stats[c.key] != null ? stats[c.key] : '-' }}</div><div class="stat-lbl">{{ c.label }}</div></div>
<i :class="c.icon" class="stat-icon"></i>
</div>
</el-col>
</el-row>
<!-- JD 筛选栏 -->
<div class="jd-filter">
<div class="filter-left">
<el-input v-model="q.doNo" placeholder="搜索单号" clearable size="small" class="filter-input" @keyup.enter.native="handleSearch" />
<el-select v-model="q.deliveryStatus" placeholder="状态" clearable size="small" style="width:100px" @change="getList">
<el-option label="已签收(待结单)" value="history" />
<el-option label="已结单" value="closed" />
</el-select>
<el-button type="primary" size="small" @click="handleSearch">搜索</el-button>
<el-button size="small" @click="resetSearch">重置</el-button>
</div>
<div class="filter-right">
<el-button size="small" icon="el-icon-refresh" @click="getList">刷新</el-button>
</div>
</div>
<div class="cd-body">
<!-- 左侧列表 -->
<div class="cd-left">
<div class="left-header">
<span class="left-title">已签收待结单</span>
</div>
<el-table ref="table" v-loading="loading" :data="list" border stripe size="small"
@selection-change="onSelectionChange" class="jd-table"
style="width:100%" :row-class-name="rowClass">
<el-table-column type="selection" width="38" align="center" />
<el-table-column label="单号" width="130">
<template slot-scope="s">
<span class="order-link">{{ s.row.doNo }}</span>
</template>
</el-table-column>
<el-table-column label="供应商" prop="supplierName" min-width="120" show-overflow-tooltip />
<el-table-column label="交货期" prop="deliveryDate" width="85" align="center" />
<el-table-column label="收货日期" width="115" align="center">
<template slot-scope="s">
<el-date-picker v-model="s.row._editDate" type="date" value-format="yyyy-MM-dd"
size="mini" style="width:105px" placeholder="选择日期" :clearable="true"
@change="onDateChange(s.row)" />
</template>
</el-table-column>
<el-table-column label="周期" width="75" align="center">
<template slot-scope="s">
<span :class="cycleClass(s.row._cycleDays)">{{ s.row._cycleDays != null ? s.row._cycleDays + '天' : '-' }}</span>
</template>
</el-table-column>
<el-table-column label="差异" width="75" align="center">
<template slot-scope="s">
<span :class="diffClass(s.row._diffDays)">{{ s.row._diffDays != null ? diffLabel(s.row._diffDays) : '-' }}</span>
</template>
</el-table-column>
</el-table>
<el-empty v-if="!loading && list.length === 0" description="暂无数据" style="padding:40px 0" />
<pagination v-show="total>0" :total="total" :page.sync="q.pageNum" :limit.sync="q.pageSize" @pagination="getList" />
</div>
<!-- 右侧操作区 -->
<div class="cd-right">
<div class="right-panel">
<div class="right-title">批量操作</div>
<div class="right-section">
<div class="rs-header">已选择 <b>{{ selected.length }}</b> </div>
<div v-if="selected.length" class="rs-list">
<div v-for="r in selected" :key="r.doId" class="rs-item">{{ r.doNo }}</div>
</div>
<div v-else class="rs-empty">请在左侧勾选订单</div>
</div>
<div class="right-section">
<div class="rs-header">批量设置收货日期</div>
<div class="rs-date-row">
<el-date-picker v-model="batchDate" type="date" value-format="yyyy-MM-dd" size="small" style="width:140px" placeholder="选择日期" />
<el-button size="small" @click="applyBatchDate" :disabled="!selected.length || !batchDate">应用到选中</el-button>
</div>
<div class="rs-quick">
<el-button size="mini" @click="batchDate = todayStr()">今天</el-button>
<el-button size="mini" @click="batchDate = yesterdayStr()">昨天</el-button>
</div>
</div>
<div class="right-section">
<div class="rs-header">批量确认结单</div>
<el-button type="primary" size="small" style="width:100%" @click="batchConfirm"
:disabled="!selected.length || !allHaveDate">确认结单 ({{ selected.length }})</el-button>
<div v-if="selected.length && !allHaveDate" class="rs-warn">有订单未设置收货日期</div>
</div>
</div>
</div>
</div>
</div>
</template>
<script>
import { listDelivery, setCloseDate, batchSetCloseDate } from "@/api/bid/delivery"
import request from '@/utils/request'
export default {
name: "CloseDate",
data() {
return {
loading: false, list: [], total: 0, stats: {},
selected: [],
batchDate: null,
q: { pageNum: 1, pageSize: 50, doNo: "", deliveryStatus: "history" },
statCards: [
{ key: "pendingClose", label: "已收货未结单", icon: "el-icon-document", color: "#e4393c" },
{ key: "todayClosed", label: "今日结单", icon: "el-icon-circle-check", color: "#67c23a" },
{ key: "weekClosed", label: "本周结单", icon: "el-icon-data-line", color: "#e6a23c" },
{ key: "avgCycleDays", label: "平均周期(天)", icon: "el-icon-time", color: "#8e44ad" }
]
}
},
computed: {
allHaveDate() { return this.selected.every(r => r._editDate) }
},
created() { this.getList(); this.getStats() },
methods: {
getList() {
this.loading = true
listDelivery(this.q).then(r => {
this.list = (r.rows || []).map(d => ({
...d,
deliveryDate: d.deliveryDate ? d.deliveryDate.substring(0, 10) : '',
_editDate: d.actualCloseDate ? d.actualCloseDate.substring(0, 10) : '',
_cycleDays: null,
_diffDays: null
})).map(d => { this.calcRow(d); return d })
this.total = r.total || 0; this.loading = false
}).catch(() => { this.loading = false })
},
getStats() {
request({ url: '/bid/delivery/closeDate/stats', method: 'get' }).then(r => { this.stats = r.data || {} }).catch(() => {})
},
handleSearch() { this.q.pageNum = 1; this.getList() },
resetSearch() { this.q.doNo = ""; this.q.deliveryStatus = ""; this.handleSearch() },
onSelectionChange(rows) { this.selected = rows },
rowClass({ row }) { return this.selected.includes(row) ? 'selected-row' : '' },
onDateChange(row) { this.calcRow(row) },
calcRow(row) {
if (!row.deliveryDate || !row._editDate) { row._cycleDays = null; row._diffDays = null; return }
const cd = new Date(row._editDate)
// 履约周期以约定交货日为起点,存在延期日期时以延期日期替代计算差异
const baseDate = row.delayDate || row.deliveryDate
const dd = new Date(baseDate)
row._cycleDays = Math.round((cd - dd) / 86400000)
row._diffDays = row._cycleDays
},
cycleClass(d) { if (d === null) return ''; return d <= 0 ? 'diff-early' : 'diff-late' },
diffClass(d) { if (d === null) return ''; return d <= 0 ? 'diff-early' : 'diff-late' },
diffLabel(d) { if (d === 0) return '准时'; if (d < 0) return '提前' + Math.abs(d) + '天'; return '延期' + d + '天' },
todayStr() { const d = new Date(); return d.toISOString().slice(0, 10) },
yesterdayStr() { const d = new Date(); d.setDate(d.getDate() - 1); return d.toISOString().slice(0, 10) },
applyBatchDate() {
if (!this.batchDate || !this.selected.length) return
this.selected.forEach(r => { r._editDate = this.batchDate; this.calcRow(r) })
this.$modal.msgSuccess("已应用到 " + this.selected.length + " 条")
},
batchConfirm() {
if (!this.selected.length) return
if (!this.allHaveDate) { this.$modal.msgError("有订单未设置收货日期"); return }
this.$modal.confirm("确认批量结单 " + this.selected.length + " 条?").then(() => {
const ids = this.selected.map(r => r.doId)
return batchSetCloseDate(ids, this.selected[0]._editDate)
}).then(() => {
this.$modal.msgSuccess("批量结单成功"); this.getList(); this.getStats(); this.selected = []
}).catch(() => {})
}
}
}
</script>
<style scoped>
.jd-cd-page { padding: 12px; min-height: calc(100vh - 84px); }
.stat-row { margin-bottom: 12px !important; }
.jd-filter { display: flex; align-items: center; background: #ffffff; padding: 10px 16px; border-radius: 2px; margin-bottom: 12px; flex-wrap: wrap; gap: 8px; }
.filter-left { display: flex; align-items: center; gap: 8px; flex-wrap: wrap; }
.filter-right { margin-left: auto; }
.filter-input { width: 130px; }
.cd-body { display: flex; gap: 12px; align-items: flex-start; }
.cd-left { flex: 1; background: #fff; border-radius: 2px; border: 1px solid #e5e5e5; padding: 14px; }
.left-header { display: flex; align-items: center; margin-bottom: 12px; }
.left-title { font-size: 14px; font-weight: 700; color: #333; }
.jd-table { border: none !important; }
.cd-right { width: 320px; flex-shrink: 0; background: #fff; border-radius: 2px; border: 1px solid #e5e5e5; padding: 16px; }
.right-title { font-size: 14px; font-weight: 700; color: #333; margin-bottom: 16px; padding-bottom: 8px; border-bottom: 2px solid #e4393c; }
.right-section { margin-bottom: 20px; }
.rs-header { font-size: 12px; color: #666; margin-bottom: 8px; }
.rs-list { max-height: 150px; overflow-y: auto; border: 1px solid #e5e5e5; border-radius: 2px; padding: 4px; }
.rs-item { padding: 4px 8px; font-size: 12px; color: #333; border-bottom: 1px solid #ffffff; }
.rs-item:last-child { border-bottom: none; }
.rs-empty { text-align: center; padding: 20px; color: #999; font-size: 12px; }
.rs-date-row { display: flex; gap: 6px; margin-bottom: 8px; }
.rs-quick { display: flex; gap: 6px; }
.rs-warn { font-size: 11px; color: #f56c6c; margin-top: 6px; }
.order-link { color: #e4393c; cursor: pointer; }
.order-link:hover { color: #e4393c; }
::v-deep .selected-row td { background: #fafafa !important; }
.diff-early { color: #67c23a; font-weight: 600; }
.diff-late { color: #f56c6c; font-weight: 600; }
</style>