| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589590591592593594595596597598599600601602603604605606607608609610611612613614615616617618619620621622623624625626627628629630631632633634635636637638639640641642643644645646647648649650651652653654655656657658659660661662663664665666667668669670671672673674675676677678679680681682683684685686687688689690691692693694695696697698699700701702703704705706707708709710711712713714715716717718719720721722723724725726727728729730731732733734735736737738739740741742743744745746747748749750751752753754755756757758759760761762763764765766767768769770771772773774775776777778779780781782783784785786787788789790791792793794795796797798799800801802803804805806807808809810811812813814815816817818819820821822823824825826827828829830831832833834835836837838839840841842843844845846847 |
- <template>
- <view class="container">
- <!-- 搜索+筛选栏 -->
- <view class="top-box">
- <view class="input-item">
- <image class="icon search-icon" src="@/static/image/search.png" mode="widthFix"></image>
- <input v-model="searchKeywords" @input="handleSearchInput" placeholder="请输入发起人姓名、手机号" placeholder-class="placeholder" />
- </view>
- </view>
- <!-- 顶部选项卡 -->
- <view class="top-tabs">
- <view class="top-tab-item" :class="{ active: currentTopTab === index }" @click="switchTopTab(index)"
- v-for="(item, index) in topTabs" :key="index">
- {{ item.label }}·{{ item.badge }}
- </view>
- </view>
- <!-- 核心修改:新增子标签容器,作为popup-box的定位参考 -->
- <view class="sub-tabs-wrapper">
- <!-- 动态子标签 -->
- <view class="sub-tabs">
- <view class="sub-tab-item" :class="{ active: currentSubTab === item.value }" @click="popShow = !popShow"
- v-for="(item, index) in currentSubTabsList" :key="index">
- <text>{{ item.label }}</text>
- <image class="icon" src="/static/image/icon_expand.png" mode="widthFix"></image>
- </view>
- </view>
- <!-- 使用tab-popup组件 -->
- <tab-popup v-model:show="popShow" :items="tabsList" :selected-item="selectedTabItem" @select="selectTabItem" />
- </view>
- <!-- 列表内容 -->
- <scroll-view class="content" scroll-y>
- <view class="task-card" v-for="(item, index) in currentList" :key="index" @click="goDetails(item)">
- <!-- 任务/审核标题+状态 -->
- <view class="card-top">
- <text class="card-task-title">{{ item.auditName || '未设置' }}</text>
- <view class="status-tag" :class="item.statusClass">{{ item.statusText }}</view>
- </view>
- <view class="card-content">
- <!-- 时间信息 -->
- <view class="time-info">
- <view class="time-item">
- <text class="title">任务名称:</text>
- <text>{{ item.auditName || '未设置' }}</text>
- </view>
- <view class="time-item">
- <text class="title">任务类型:</text>
- <text>{{ getBusinessTypeLabel(item.businessType) }}</text>
- </view>
- </view>
- <!-- 操作按钮 -->
- <view class="operate-btn-group">
- <view class="share-btn" @click="handleShare(item)">
- <image class="share-icon" src="/static/image/icon_user.png" mode="widthFix"></image>
- <text>{{ item.initiatorName || '未命名' }}</text>
- </view>
- <view class="date">
- {{ formatTime(item.createTime) }}
- </view>
- </view>
- </view>
- </view>
- </scroll-view>
- <!-- 筛选弹窗 -->
- <view class="filter-popup" v-if="showFilter" @click="closeFilter">
- <view class="filter-content" @click.stop>
- <view class="filter-header">
- <view class="filter-title">筛选</view>
- <image class="filter-close-btn" src="@/static/image/icon_close.png" @click="closeFilter"
- mode="widthFix"></image>
- </view>
- <view class="filter-form">
- <!-- 任务申请时间 -->
- <view class="filter-section">
- <view class="section-label">任务申请时间</view>
- <view class="time-range">
- <input class="time-input" placeholder="开始时间" v-model="filters.applyTimeStart" />
- <view class="time-separator">-</view>
- <input class="time-input" placeholder="结束时间" v-model="filters.applyTimeEnd" />
- </view>
- </view>
- <!-- 任务完成时间:改为判断index:1对应已办 -->
- <view class="filter-section" v-if="currentTopTab === 1">
- <view class="section-label">任务完成时间</view>
- <view class="time-range">
- <input class="time-input" placeholder="开始时间" v-model="filters.finishTimeStart" />
- <view class="time-separator">-</view>
- <input class="time-input" placeholder="结束时间" v-model="filters.finishTimeEnd" />
- </view>
- </view>
- <!-- 完成审核时间:改为判断index:1对应已办 -->
- <view class="filter-section" v-if="currentTopTab === 1">
- <view class="section-label">完成审核时间</view>
- <view class="time-range">
- <input class="time-input" placeholder="开始时间" v-model="filters.auditTimeStart" />
- <view class="time-separator">-</view>
- <input class="time-input" placeholder="结束时间" v-model="filters.auditTimeEnd" />
- </view>
- </view>
- <!-- 任务归属 -->
- <view class="filter-section">
- <view class="section-label">任务归属</view>
- <view class="btn-group">
- <view class="filter-btn" :class="{ active: filters.taskBelong === 'my' }"
- @click="filters.taskBelong = 'my'">我的任务</view>
- <view class="filter-btn" :class="{ active: filters.taskBelong === 'dept' }"
- @click="filters.taskBelong = 'dept'">部门任务</view>
- </view>
- </view>
- <!-- 完结状态:改为判断index:1对应已办 -->
- <view class="filter-section" v-if="currentTopTab === 1">
- <view class="section-label">完结状态</view>
- <view class="btn-group">
- <view class="filter-btn" :class="{ active: filters.finishStatus === 'unfinished' }"
- @click="filters.finishStatus = 'unfinished'">未完结</view>
- <view class="filter-btn" :class="{ active: filters.finishStatus === 'finished' }"
- @click="filters.finishStatus = 'finished'">已完结</view>
- </view>
- </view>
- <!-- 归属类型 -->
- <view class="filter-section">
- <view class="section-label">归属类型</view>
- <view class="btn-group">
- <view class="filter-btn" :class="{ active: filters.belongType === 'inner' }"
- @click="filters.belongType = 'inner'">院内</view>
- <view class="filter-btn" :class="{ active: filters.belongType === 'outer' }"
- @click="filters.belongType = 'outer'">院外</view>
- </view>
- </view>
- </view>
- <view class="filter-actions">
- <view class="reset-btn" @click="resetFilters">重置</view>
- <view class="confirm-btn" @click="confirmFilters">确定</view>
- </view>
- </view>
- </view>
- </view>
- </template>
- <script>
- import utils from '@/utils/common.js'
- import TabPopup from '@/components/tab-popup.vue'
- import { getPendingAuditList, searchCompanyUser } from '@/api/audit.js'
-
- export default {
- components: {
- TabPopup
- },
- data() {
- return {
- processTemplate: null,
- popShow: false,
- showFilter: false,
- selectedTabItem: '',
- searchKeywords: '',
- searchTimer: null,
- filters: {
- applyTimeStart: '',
- applyTimeEnd: '',
- finishTimeStart: '',
- finishTimeEnd: '',
- auditTimeStart: '',
- auditTimeEnd: '',
- taskBelong: 'my',
- finishStatus: '',
- belongType: ''
- },
- // 顶部选项卡数组
- topTabs: [{
- label: '待办',
- value: 'todo',
- badge: 0
- },
- {
- label: '已办',
- value: 'done',
- badge: 0
- },
- {
- label: '我发起的',
- value: 'myInitiate',
- badge: 0
- }
- ],
- currentTopTab: 0,
- currentSubTab: 'all',
- // 子标签数组
- taskSubTabs: [{
- label: '全部审批',
- value: 'all'
- },
- {
- label: '最新发起',
- value: 'unfinished'
- }
- ],
- tabsList: ['全部审批', '任务创建'],
- tabsList2: ['最新发起', '最早发起'],
- // auditSubTabs: [{
- // label: '全部',
- // value: 'all'
- // },
- // {
- // label: '驳回',
- // value: 'rejected'
- // }
- // ],
- // 加载状态
- loading: {
- todo: false,
- done: false,
- myInitiate: false
- },
- // 列表数据
- todoList: [], // 待办列表(index=0)
- doneList: [], // 已办列表(index=1)
- myInitiateList: [] // 我发起的列表(index=2)
- }
- },
- computed: {
- currentSubTabsList() {
- return this.taskSubTabs;
- // if (this.currentTopTab === 0) return this.taskSubTabs;
- // if (this.currentTopTab === 1) return this.auditSubTabs;
- // if (this.currentTopTab === 2) return this.myInitiateList;
- },
- currentList() {
- if (this.currentTopTab === 0) return this.todoList;
- if (this.currentTopTab === 1) return this.doneList;
- if (this.currentTopTab === 2) return this.myInitiateList;
- }
- },
- onLoad() {
- // 页面加载时的初始化操作
- this.loadData();
- utils.getDicts("FLOW_TEMPLATE").then(res => {
- this.processTemplate = res;
- console.log("流程模板", this.processTemplate)
- });
- },
- methods: {
- selectTabItem(item) {
- this.selectedTabItem = item;
- this.popShow = false;
- },
- goDetails(item) {
- console.log("跳转参数",item)
- uni.navigateTo({
- url: `/pages_task/approvalTaskDetail?taskId=${item.id}&businessType=${item.businessType}`
- })
- },
- switchTopTab(index) {
- this.currentTopTab = index;
- this.currentSubTab = 'all';
- this.popShow = false;
- // 切换标签时重新加载数据
- this.loadData();
- },
- closeFilter() {
- this.showFilter = false
- },
- resetFilters() {
- this.filters = {
- applyTimeStart: '',
- applyTimeEnd: '',
- finishTimeStart: '',
- finishTimeEnd: '',
- auditTimeStart: '',
- auditTimeEnd: '',
- taskBelong: 'my',
- finishStatus: '',
- belongType: ''
- }
- },
- confirmFilters() {
- this.showFilter = false
- console.log('筛选条件:', this.filters)
- // 筛选后重新加载数据
- this.loadData()
- },
- handleShare(item) {
- uni.showToast({
- title: '分享功能待实现',
- icon: 'none'
- })
- },
- handleSearchInput() {
- // 防抖处理,避免频繁请求
- if (this.searchTimer) {
- clearTimeout(this.searchTimer)
- }
-
- this.searchTimer = setTimeout(async () => {
- if (this.searchKeywords.trim()) {
- try {
- // 调用搜索发起人接口
- const res = await searchCompanyUser({
- keywords: this.searchKeywords.trim(),
- pageNum: 1,
- pageSize: 10
- })
-
- if (res.code === 200 && res.rows) {
- // 根据当前标签页更新对应列表
- if (this.currentTopTab === 0) {
- this.todoList = res.rows.map(item => ({
- auditName: item.auditName || '未知审核',
- statusText: '待审核',
- statusClass: 'status-createPending',
- id: item.id,
- status: item.status,
- auditType: item.auditType,
- businessId: item.businessId,
- businessType: item.businessType,
- createTime: item.createTime,
- initiatorName: item.initiatorName
- }))
- } else if (this.currentTopTab === 1) {
- this.doneList = res.rows.map(item => ({
- auditName: item.auditName || '未知审核',
- statusText: item.status === 1 ? '已通过' : '已驳回',
- statusClass: item.status === 1 ? 'status-finish' : 'status-rejected',
- id: item.id,
- status: item.status,
- auditType: item.auditType,
- businessId: item.businessId,
- businessType: item.businessType,
- createTime: item.createTime,
- initiatorName: item.initiatorName
- }))
- } else if (this.currentTopTab === 2) {
- this.myInitiateList = res.rows.map(item => ({
- auditName: item.auditName || '未知审核',
- statusText: '待审核',
- statusClass: 'status-createPending',
- id: item.id,
- status: item.status,
- auditType: item.auditType,
- businessId: item.businessId,
- businessType: item.businessType,
- createTime: item.createTime,
- initiatorName: item.initiatorName
- }))
- }
- } else {
- // 搜索结果为空,清空对应列表
- if (this.currentTopTab === 0) {
- this.todoList = []
- } else if (this.currentTopTab === 1) {
- this.doneList = []
- } else if (this.currentTopTab === 2) {
- this.myInitiateList = []
- }
- }
- } catch (e) {
- console.error('搜索失败', e)
- uni.showToast({
- title: '搜索失败',
- icon: 'none'
- })
- }
- } else {
- // 搜索框为空,重新加载默认数据
- this.loadData()
- }
- }, 300)
- },
- // 加载审批数据
- async loadData() {
- try {
- // 无论当前标签是什么,都调用getPendingAuditList请求
- const userInfo = JSON.parse(uni.getStorageSync('userInfo') || '{}')
- const params = {
- initiatorName: userInfo.nickName || '',
- initiatorPhone: userInfo.phone || '',
- // initiatorId: 0,
- status: userInfo.status|| '',
- // tempCode: userInfo.productCode || '',
- sort: 0,
- userId:userInfo.userId|| '',
- companyId: userInfo.companyId || 0
- }
- const res = await getPendingAuditList(params)
- if (res.code === 200) {
- // 处理返回的数据
- const responseData = res.rows|| [];
-
- // 根据status分类数据
- const todoData = responseData.filter(item => item.status === 1);
- const doneData = responseData.filter(item => item.status === 2);
- const myInitiateData = responseData.filter(item => item.status === 3);
-
- // 更新对应列表
- if (this.currentTopTab === 0) {
- // 待办列表
- this.loading.todo = true
- this.todoList = todoData.map(item => ({
- auditName: item.auditName || '未知审核',
- statusText: '待审核',
- statusClass: 'status-createPending',
- id: item.id,
- status: item.status,
- auditType: item.auditType,
- businessId: item.businessId,
- businessType: item.businessType,
- createTime: item.createTime,
- initiatorName:item.initiatorName
- }))
- this.loading.todo = false
- } else if (this.currentTopTab === 1) {
- // 已办列表
- this.loading.done = true
- this.doneList = doneData.map(item => ({
- auditName: item.auditName || '未知审核',
- statusText: item.status === 1 ? '已通过' : '已驳回',
- statusClass: item.status === 1 ? 'status-finish' : 'status-rejected',
- id: item.id,
- status: item.status,
- auditType: item.auditType,
- businessId: item.businessId,
- businessType: item.businessType,
- createTime: item.createTime,
- initiatorName:item.initiatorName
- }))
- this.loading.done = false
- } else if (this.currentTopTab === 2) {
- // 我发起的列表
- this.loading.myInitiate = true
- this.myInitiateList = myInitiateData.map(item => ({
- auditName: item.auditName || '未知审核',
- statusText: '待审核',
- statusClass: 'status-createPending',
- id: item.id,
- status: item.status,
- auditType: item.auditType,
- businessId: item.businessId,
- businessType: item.businessType,
- createTime: item.createTime,
- initiatorName:item.initiatorName
- }))
- this.loading.myInitiate = false
- }
-
- // 更新badge数量
- this.topTabs[0].badge = todoData.length
- this.topTabs[1].badge = doneData.length
- this.topTabs[2].badge = myInitiateData.length
- } else {
- uni.showToast({
- title: '获取审核列表失败',
- icon: 'none'
- })
- }
- } catch (e) {
- console.error('加载数据失败', e)
- uni.showToast({
- title: '加载数据失败',
- icon: 'none'
- })
- // 重置加载状态
- this.loading.todo = false
- this.loading.done = false
- this.loading.myInitiate = false
- }
- },
- // 格式化时间
- formatTime(timeStr) {
- if (!timeStr) return ''
- const date = new Date(timeStr)
- const year = date.getFullYear()
- const month = String(date.getMonth() + 1).padStart(2, '0')
- const day = String(date.getDate()).padStart(2, '0')
- const hours = String(date.getHours()).padStart(2, '0')
- const minutes = String(date.getMinutes()).padStart(2, '0')
- return `${year}-${month}-${day} ${hours}:${minutes}`
- },
- getBusinessTypeLabel(businessType) {
- if (!businessType || !this.processTemplate) {
- return '未设置'
- }
- const dictItem = this.processTemplate.find(item => item.dictValue === businessType)
- return dictItem ? dictItem.dictLabel : businessType
- }
- }
- }
- </script>
- <style lang="scss" scoped>
- .container {
- min-height: 100vh;
- background: #F7F8FA;
- position: relative;
- // 搜索栏
- .top-box {
- padding: 20rpx 32rpx;
- display: flex;
- align-items: center;
- background: #fff;
- position: relative;
- z-index: 101;
- .input-item {
- display: flex;
- align-items: center;
- flex: 1;
- height: 72rpx;
- background: #F7F8FA;
- border-radius: 38rpx;
- .search-icon {
- width: 26rpx;
- height: 26rpx;
- margin: 0 10rpx 0 28rpx;
- }
- input {
- flex: 1;
- font-size: 28rpx;
- color: #333;
- }
- }
- }
- // 顶部选项卡
- .top-tabs {
- display: flex;
- background: #fff;
- position: relative;
- z-index: 101;
- .top-tab-item {
- flex: 1;
- text-align: center;
- padding: 24rpx 0;
- font-size: 28rpx;
- color: #999;
- transition: all 0.2s;
- &.active {
- color: #333;
- font-weight: 500;
- position: relative;
- &::after {
- content: '';
- position: absolute;
- bottom: 0;
- left: 50%;
- transform: translateX(-50%);
- border-radius: 3rpx;
- width: 80rpx;
- height: 6rpx;
- background: #388BFF;
- }
- }
- }
- }
- // 核心修改:子标签容器(定位参考)
- .sub-tabs-wrapper {
- position: relative; // 作为popup-box的定位参考
- z-index: 101; // 确保在遮罩层之上
- // 子标签栏
- .sub-tabs {
- display: flex;
- justify-content: space-between;
- background: #fff;
- padding: 16rpx 32rpx;
- gap: 24rpx;
- .sub-tab-item {
- flex: 1;
- height: 64rpx;
- display: flex;
- border-radius: 70rpx;
- border: 2rpx solid #F2F2F2;
- justify-content: center;
- align-items: center;
- font-size: 28rpx;
- color: #666;
- transition: all 0.2s;
- .icon {
- width: 28rpx;
- height: 28rpx;
- margin-left: 12rpx;
- }
- }
- }
- }
- // 列表区域
- .content {
- padding: 24rpx;
- box-sizing: border-box;
- min-height: calc(100vh - 300rpx);
- position: relative;
- z-index: 98; // 低于遮罩层
- .task-card {
- background: #fff;
- border-radius: 24rpx;
- margin-bottom: 20rpx;
- border: 2rpx solid #E9F2FF;
- position: relative;
- padding-bottom: 24rpx;
- .card-top {
- display: flex;
- justify-content: space-between;
- align-items: center;
- background: linear-gradient(90deg, #E8F1FF 0%, #FFFFFF 100%);
- border-radius: 24rpx 24rpx 0 0;
- padding: 26rpx 24rpx;
- margin-bottom: 16rpx;
- .card-task-title {
- font-size: 32rpx;
- font-weight: 600;
- color: #333;
- }
- .status-tag {
- padding: 8rpx 16rpx;
- border-radius: 20rpx;
- font-size: 24rpx;
- &.status-wait {
- background: #E3F2FD;
- color: #2196F3;
- }
- &.status-finish {
- background: #E6FAEF;
- color: #07C160;
- }
- &.status-rejected {
- background: #FFF4F5;
- color: #CF3546;
- }
- &.status-createPending {
- background: #FFF8E6;
- color: #FF9500;
- }
- }
- }
- .card-content {
- padding: 0 24rpx;
- .time-info {
- font-size: 28rpx;
- color: #666;
- display: flex;
- flex-direction: column;
- gap: 8rpx;
- margin-bottom: 24rpx;
- .time-item {
- display: flex;
- font-size: 28rpx;
- color: #333;
- .title {
- color: #666;
- }
- }
- }
- .operate-btn-group {
- display: flex;
- align-items: center;
- justify-content: space-between;
- .share-btn {
- display: flex;
- align-items: center;
- font-size: 28rpx;
- color: #666;
- .share-icon {
- width: 36rpx;
- height: 36rpx;
- margin-right: 8rpx;
- }
- }
- .date {
- font-size: 24rpx;
- color: #999;
- }
- }
- }
- }
- }
- // 筛选弹窗
- .filter-popup {
- position: fixed;
- top: 0;
- left: 0;
- right: 0;
- bottom: 0;
- background: rgba(0, 0, 0, 0.5);
- z-index: 999;
- display: flex;
- align-items: flex-end;
- .filter-content {
- width: 100%;
- background: #fff;
- border-radius: 24rpx 24rpx 0 0;
- padding: 32rpx;
- .filter-header {
- display: flex;
- align-items: center;
- justify-content: center;
- margin-bottom: 32rpx;
- position: relative;
- .filter-title {
- font-size: 32rpx;
- font-weight: bold;
- color: #333;
- }
- .filter-close-btn {
- position: absolute;
- right: 0;
- width: 44rpx;
- height: 44rpx;
- }
- }
- .filter-form {
- padding: 0 0 24rpx 0;
- .filter-section {
- margin-bottom: 32rpx;
- .section-label {
- font-weight: 500;
- font-size: 28rpx;
- color: #333;
- margin-bottom: 24rpx;
- }
- .time-range {
- display: flex;
- align-items: center;
- gap: 16rpx;
- .time-input {
- flex: 1;
- height: 72rpx;
- text-align: center;
- background: #F7F8FA;
- border-radius: 8rpx;
- padding: 0 16rpx;
- font-size: 26rpx;
- color: #333;
- }
- .time-separator {
- font-size: 24rpx;
- color: #999;
- }
- }
- .btn-group {
- display: flex;
- gap: 16rpx;
- .filter-btn {
- width: 214rpx;
- height: 72rpx;
- background: #F7F8FA;
- border-radius: 70rpx;
- line-height: 72rpx;
- text-align: center;
- font-size: 28rpx;
- transition: all 0.2s;
- color: #333;
- &.active {
- background: rgba(56, 139, 255, 0.15);
- color: #388BFF;
- }
- }
- }
- }
- }
- .filter-actions {
- display: flex;
- gap: 24rpx;
- margin-top: 40rpx;
- .reset-btn,
- .confirm-btn {
- flex: 1;
- height: 80rpx;
- line-height: 80rpx;
- text-align: center;
- border-radius: 200rpx;
- font-size: 28rpx;
- transition: all 0.2s;
- }
- .reset-btn {
- background: #fff;
- color: #388BFF;
- border: 2rpx solid #388BFF;
- &:active {
- background: #EBF3FF;
- }
- }
- .confirm-btn {
- background: #388BFF;
- color: #fff;
- &:active {
- background: #2A78E5;
- }
- }
- }
- }
- }
- }
- </style>
|