feat(采购需求): 新增采购需求功能模块

添加采购需求管理功能,包括需求列表展示、新增、编辑、删除和完成操作
新增需求列表页面和API接口
添加剩余时间显示组件用于展示需求截止时间
更新工作台页面添加采购需求入口
This commit is contained in:
砂糖
2025-11-07 14:21:27 +08:00
parent 750f66441b
commit b569e4fef8
7 changed files with 779 additions and 2 deletions

View File

@@ -17,7 +17,7 @@
mode="cover"></image>
<view class="user-meta">
<view class="user-name">
{{ article.author || '未知用户' }}
{{ article.createBy || '未知用户' }}
<!-- <view class="user-level">Lv.1</view> -->
</view>
<view class="user-time-area">
@@ -81,7 +81,6 @@
.then(res => {
this.article = {
...res.data,
author: res.data.createBy,
title: res.data.noticeTitle,
content: res.data.noticeContent,
};

View File

@@ -103,6 +103,12 @@ export default {
url: '/pages/workbench/feedback/feedback',
category: "信息中心"
},
{
text: '采购需求',
icon: '/static/images/requirement.png',
url: '/pages/workbench/requirement/requirement',
category: "信息中心"
},
{
text: '客户管理',
icon: '/static/images/customer.png',

View File

@@ -0,0 +1,619 @@
<template>
<view class="report-schedule">
<!-- header始终显示 -->
<view class="search-bar">
<view class="search-container">
<view class="task-type-button-container">
<view class="task-type-button" @click="openDrawer">
<uni-icons type="settings" color="#2979ff" size="22"></uni-icons>
</view>
</view>
<view class="search-input custom-search-input">
<input v-model="queryParams.title" class="input" type="text" placeholder="搜索采购需求" @confirm="onSearch"
@keyup.enter="onSearch" />
<view class="search-icon" @click="onSearch">
<uni-icons type="search" color="#bbb" size="20"></uni-icons>
</view>
<view v-if="searchName" class="clear-icon" @click="onClearSearch">
<uni-icons type="closeempty" color="#bbb" size="18"></uni-icons>
</view>
</view>
<view class="add-button" @click="handleAdd">
<uni-icons type="plusempty" color="#2979ff" size="22"></uni-icons>
</view>
</view>
</view>
<!-- 筛选抽屉 -->
<uni-drawer ref="drawerRef" mode="left" :width="320">
<view class="drawer-content">
<view class="drawer-title">筛选</view>
<view class="drawer-form">
<view class="drawer-form-item">
<text class="drawer-label">需求方</text>
<oa-user-select v-model="queryParams.requesterId" placeholder="请选择需求方" />
</view>
<view class="drawer-form-item">
<text class="drawer-label">负责人</text>
<oa-user-select v-model="queryParams.ownerId" placeholder="请选择负责人" />
</view>
<view class="drawer-form-item">
<text class="drawer-label">关联项目</text>
<oa-project-select v-model="queryParams.projectId" placeholder="请选择关联项目" />
</view>
</view>
<view class="drawer-btns">
<button class="drawer-btn-primary" @click="applyFilterAndClose">确定</button>
<button class="drawer-btn" @click="resetFilterAndClose">重置</button>
<button class="drawer-btn" @click="closeDrawer">关闭</button>
</view>
</view>
</uni-drawer>
<view>
<!-- 自定义列表右滑菜单uni-ui实现 -->
<scroll-view scroll-y style="height: 100vh;" @scrolltolower="loadMore">
<view v-if="customerList.length">
<uni-swipe-action>
<block v-for="(item, index) in customerList" :key="item.requirementId">
<uni-swipe-action-item :right-options="getSwipeOptions(item)" @click="swipeActionClick($event, item)"
style="margin-bottom: 16rpx;">
<view class="card">
<view class="card-title">
<text class="project">{{ item.title }}</text>
<uni-tag v-if="item.status == 2" type="success" text="已完成"></uni-tag>
<oa-remind-time v-else :expireDate="item.deadline" :thresholdDays="3"></oa-remind-time>
</view>
<view class="card-content">
<view>需求方({{ item.requesterNickName }}) -> 负责人({{ item.ownerNickName }})</view>
<view>关联项目{{ item.projectName }}</view>
<view>需求描述{{ item.description }}</view>
<view>截止日期{{ item.deadline }}</view>
</view>
</view>
</uni-swipe-action-item>
</block>
</uni-swipe-action>
</view>
<view v-else class="empty">暂无数据</view>
<view class="load-more-tips">
<u-loading-icon v-if="loadingMore" text="加载中..." size="20" textSize="14" />
<text v-else-if="!hasMore && customerList.length">没有更多了</text>
</view>
</scroll-view>
<!-- 新增/编辑弹窗 -->
<uni-popup ref="popupRef" type="bottom">
<view class="popup-content">
<view class="uni-form">
<view class="uni-form-item">
<text class="uni-form-label">需求标题</text>
<u-input v-model="form.title" placeholder="请输入客户名称" />
</view>
<view class="uni-form-item">
<text class="uni-form-label">需求描述</text>
<uni-easyinput v-model="form.description" type="textarea" placeholder="请输入需求描述" />
</view>
<view class="uni-form-item">
<text class="uni-form-label">负责人</text>
<oa-user-select v-model="form.ownerId" placeholder="请选择负责人" />
</view>
<view class="uni-form-item">
<text class="uni-form-label">关联项目</text>
<oa-project-select v-model="form.projectId" placeholder="请选择关联项目" />
</view>
<view class="uni-form-item">
<text class="uni-form-label">截至日期</text>
<uni-datetime-picker v-model="form.deadline" type="datetime" placeholder="请选择截止日期"
style="width:100%" />
</view>
<view class="uni-form-item">
<text class="uni-form-label">备注</text>
<u-input v-model="form.remark" type="textarea" placeholder="请输入备注" />
</view>
</view>
<view class="popup-btns">
<u-button type="primary" @click="submitForm">确定</u-button>
<u-button @click="closePopup">取消</u-button>
</view>
</view>
</uni-popup>
</view>
</view>
</template>
<script>
import {
addRequirements,
delRequirements,
updateRequirements,
getRequirements,
listRequirements
} from '@/api/oa/requirement.js'
export default {
data() {
return {
queryParams: {
pageNum: 1,
pageSize: 10,
expressId: undefined,
description: undefined,
reportTime: undefined,
reportBy: undefined,
status: undefined,
},
projectOptions: [],
headerOptions: [],
customerList: [],
total: 0,
single: true,
multiple: true,
form: {},
showStartDate: false,
showEndDate: false,
swipeOptions: [{
text: '编辑',
style: {
backgroundColor: '#2979ff',
color: '#fff'
}
},
{
text: '删除',
style: {
backgroundColor: '#fa3534',
color: '#fff'
}
}
],
// 新增:筛选相关
searchName: '',
filterProject: '',
filterHeader: '',
filterDateRange: [],
pageNum: 1,
pageSize: 10,
loadingMore: false,
hasMore: true,
startLoading: false,
startPopupOpen: false,
_startScheduleId: null,
showGanttView: false,
ganttChartData: {
categories: [],
series: []
}
}
},
onLoad() {
this.pageNum = 1
this.hasMore = true
this.handleQuery()
},
methods: {
openDrawer() {
this.$refs.drawerRef.open();
},
closeDrawer() {
this.$refs.drawerRef.close();
},
applyFilterAndClose() {
this.applyFilter();
this.closeDrawer();
},
resetFilterAndClose() {
this.resetFilter();
this.closeDrawer();
},
// 搜索框检索
onSearch() {
this.searchName = this.searchName
this.handleQuery()
},
onClearSearch() {
this.searchName = ''
this.queryParams.scheduleName = ''
this.handleQuery()
},
// 筛选抽屉应用
applyFilter() {
this.queryParams.projectId = this.filterProject
this.queryParams.header = this.filterHeader
if (this.filterDateRange && this.filterDateRange.length === 2) {
this.queryParams.dateRange = this.filterDateRange
} else {
this.queryParams.dateRange = []
}
this.handleQuery()
},
// 筛选抽屉重置
resetFilter() {
this.queryParams = {
...this.queryParams,
level: undefined,
source: undefined,
industryId: undefined,
}
},
handleQuery() {
this.pageNum = 1
this.hasMore = true
this.queryParams.pageNum = 1
if (this.queryParams.dateRange && this.queryParams.dateRange.length === 2) {
this.queryParams.startDate = this.queryParams.dateRange[0]
this.queryParams.endDate = this.queryParams.dateRange[1]
} else {
this.queryParams.startDate = ''
this.queryParams.endDate = ''
}
this.getList()
},
getList() {
this.loadingMore = true
this.queryParams.pageNum = this.pageNum
this.queryParams.pageSize = this.pageSize
listRequirements(this.queryParams).then(res => {
const rows = res.rows || []
if (this.pageNum === 1) {
this.customerList = rows
} else {
this.customerList = this.customerList.concat(rows)
}
// 判断是否还有更多
this.hasMore = rows.length === this.pageSize
this.total = res.total || 0
}).finally(() => {
this.loadingMore = false
})
},
loadMore() {
if (!this.hasMore || this.loadingMore) return
this.pageNum++
this.getList()
},
handleAdd() {
this.form = {
requirementId: this.$store.state.user.id,
title: undefined,
requesterId: undefined,
ownerId: undefined,
projectId: undefined,
description: undefined,
deadline: undefined,
status: 0,
remark: undefined,
accessory: undefined,
createBy: undefined,
createTime: undefined,
updateBy: undefined,
updateTime: undefined,
delFlag: undefined
}
this.$refs.popupRef.open('bottom')
},
handleUpdate(row) {
getRequirements(row.requirementId).then(res => {
this.form = res.data || {}
this.$refs.popupRef.open('bottom')
})
},
handleDelete(item) {
uni.showModal({
title: '确认删除',
content: `确定要删除排产“${item.scheduleName}”吗?`,
success: (res) => {
if (res.confirm) {
const ids = item ? [item.requirementId] : this.selectedIds
delRequirements(ids).then(() => {
uni.showToast({
title: '删除成功',
icon: 'success'
})
this.getList()
})
}
}
})
},
handleExport() {
// 导出逻辑可根据实际API实现
uni.showToast({
title: '导出功能开发中',
icon: 'none'
})
},
submitForm() {
if (this.form.requirementId) {
updateRequirements(this.form).then(() => {
uni.showToast({
title: '修改成功',
icon: 'success'
})
this.closePopup()
this.getList()
})
} else {
addRequirements(this.form).then(() => {
uni.showToast({
title: '新增成功',
icon: 'success'
})
this.closePopup()
this.getList()
})
}
},
closePopup() {
this.$refs.popupRef.close()
},
getSwipeOptions(item) {
const options = [{
text: '编辑',
style: {
backgroundColor: '#2979ff',
color: '#fff'
}
},
{
text: '删除',
style: {
backgroundColor: '#fa3534',
color: '#fff'
}
}
];
if (!item.status || item.status != 2) {
options.push({
text: '完成',
style: { backgroundColor: '#19be6b', color: '#fff' }
});
}
return options;
},
swipeActionClick(e, item) {
const text = e.content.text;
if (text === '编辑') {
this.handleUpdate(item);
} else if (text === '删除') {
this.handleDelete(item);
} else if (text === '完成') {
this.handleFinish(item);
}
},
handleFinish(row) {
updateRequirements({
...row,
status: 2
}).then(_ => {
uni.showToast({
title: "采购完成"
})
this.getList()
})
}
}
}
</script>
<style scoped>
.report-schedule {
min-height: 100vh;
background-color: #f5f5f5;
padding-bottom: 120rpx;
}
.search-bar {
padding: 20rpx;
position: sticky;
top: 0;
z-index: 100;
background: #fff;
}
.search-container {
display: flex;
align-items: center;
gap: 20rpx;
}
.task-type-button-container {
display: flex;
gap: 12rpx;
}
.task-type-button {
width: 60rpx;
height: 60rpx;
background-color: transparent;
border-radius: 50%;
display: flex;
align-items: center;
justify-content: center;
}
.search-input {
flex: 1;
position: relative;
display: flex;
align-items: center;
background: #f5f5f5;
border-radius: 100rpx;
padding: 0 24rpx;
height: 60rpx;
}
.input {
flex: 1;
border: none;
background: transparent;
font-size: 30rpx;
outline: none;
height: 60rpx;
line-height: 60rpx;
}
.search-icon {
margin-left: 8rpx;
display: flex;
align-items: center;
}
.clear-icon {
margin-left: 8rpx;
display: flex;
align-items: center;
}
.drawer-content {
padding: 32rpx 24rpx 24rpx 24rpx;
}
.drawer-title {
font-size: 32rpx;
font-weight: bold;
margin-bottom: 24rpx;
}
.drawer-form {
margin-bottom: 32rpx;
}
.drawer-form-item {
margin-bottom: 24rpx;
}
.drawer-label {
display: block;
font-size: 28rpx;
color: #333;
margin-bottom: 8rpx;
}
.drawer-btns {
display: flex;
gap: 16rpx;
margin-top: 24rpx;
}
.drawer-btn-primary {
flex: 1;
background: #2979ff;
color: #fff;
border: none;
border-radius: 8rpx;
padding: 16rpx 0;
font-size: 28rpx;
}
.drawer-btn {
flex: 1;
background: #f5f5f5;
color: #333;
border: none;
border-radius: 8rpx;
padding: 16rpx 0;
font-size: 28rpx;
}
.card {
background: #fff;
border-radius: 16rpx;
box-shadow: 0 2rpx 8rpx rgba(0, 0, 0, 0.05);
padding: 24rpx;
}
.card-title {
display: flex;
justify-content: space-between;
align-items: center;
font-weight: bold;
margin-bottom: 12rpx;
}
.project {
color: #2979ff;
}
.status {
font-size: 24rpx;
padding: 0 12rpx;
border-radius: 8rpx;
}
.status-1 {
background: #e0f7fa;
color: #009688;
}
.status-0,
.status-undefined {
background: #fffbe6;
color: #faad14;
}
.status-other {
background: #fbeff2;
color: #e91e63;
}
.card-content view {
margin-bottom: 8rpx;
color: #666;
font-size: 26rpx;
}
.empty {
text-align: center;
color: #bbb;
margin: 40rpx 0;
}
.popup-content {
padding: 24rpx;
background: #fff;
border-radius: 16rpx 16rpx 0 0;
}
.uni-form {
margin-bottom: 32rpx;
max-height: 50vh;
overflow-y: scroll;
}
.uni-form-item {
margin-bottom: 32rpx;
}
.uni-form-label {
display: block;
font-size: 28rpx;
color: #333;
margin-bottom: 8rpx;
}
.load-more-tips {
text-align: center;
color: #bbb;
padding: 24rpx 0 32rpx 0;
font-size: 28rpx;
}
.card-ops {
margin-top: 16rpx;
display: flex;
gap: 16rpx;
}
.gantt-toggle-bar {
display: flex;
gap: 16rpx;
padding: 16rpx 0 8rpx 0;
background: #fff;
justify-content: flex-end;
}
</style>