OfficeSystem/pages/task/list/index.vue

496 lines
15 KiB
Vue
Raw Normal View History

2025-11-05 15:06:23 +08:00
<template>
2025-11-14 15:28:48 +08:00
<view class="task-list-page">
2025-11-13 09:48:26 +08:00
<scroll-view
class="task-scroll"
scroll-y
@scrolltolower="handleScrollToLower"
>
2025-11-05 15:06:23 +08:00
<view class="task-container">
<!-- 任务卡片列表 -->
2025-11-13 09:48:26 +08:00
<view
class="task-card"
v-for="task in tasks"
:key="task.id"
:class="getTaskCardClass(task.status)"
@click="goToTaskDetail(task)"
>
2025-11-05 15:06:23 +08:00
<!-- 状态标签和日期 -->
<view class="task-header">
2025-11-12 15:15:04 +08:00
<view style="display: flex;align-items: center;gap: 12px">
<view class="task-badge-wrapper">
<uv-tags
:text="getStatusText(task.status)"
:type="getTaskStatusType(task.status)"
size="mini"
:plain="false"
:custom-style="getTagCustomStyle(task.status)"
></uv-tags>
</view>
<view class="task-date-wrapper">
<text class="task-date">{{ task.date }}</text>
</view>
2025-11-05 15:06:23 +08:00
</view>
2025-11-12 15:15:04 +08:00
<!-- 立即处理按钮 -->
<view class="task-action" v-if="task.status !== 'completed'">
<uv-button
:type="getButtonType(task.status)"
size="small"
@click.stop="handleTask(task)"
>
立即处理
</uv-button>
2025-11-05 15:06:23 +08:00
</view>
</view>
<!-- 任务内容 -->
<view class="task-content">
<text class="task-project">所属项目: {{ task.project }}</text>
2025-11-12 15:15:04 +08:00
<text class="task-description">{{truncateText(task.description)}}</text>
2025-11-05 15:06:23 +08:00
<view class="task-meta">
<text class="task-owner">负责人: {{ task.owner }}</text>
2025-11-06 14:33:07 +08:00
<text class="task-owner">创建人: {{ task.createName }}</text>
2025-11-05 15:06:23 +08:00
<view class="task-time-row">
<text class="task-time">发布时间: {{ task.releaseTime }}</text>
2025-11-06 13:50:29 +08:00
<view class="task-countdown" v-if="task.status !== 'completed' && task.remainingDays !== null">
2025-11-05 15:06:23 +08:00
<text class="countdown-icon">🕐</text>
<text class="countdown-text" :class="getCountdownClass(task.status)">
{{ task.remainingDays < 0 ? `已逾期${Math.abs(task.remainingDays)}` : `剩余${task.remainingDays}` }}
2025-11-05 15:06:23 +08:00
</text>
</view>
</view>
</view>
</view>
2025-11-13 09:48:26 +08:00
</view>
2025-11-06 13:50:29 +08:00
<!-- 加载状态 -->
<view class="empty-state" v-if="loading">
<text class="empty-text">加载中...</text>
2025-11-05 15:06:23 +08:00
</view>
<!-- 空状态 -->
2025-11-13 09:48:26 +08:00
<view class="empty-state" v-else-if="isEmpty">
2025-11-06 13:50:29 +08:00
<text class="empty-text">暂无{{ getStatusText(statusFilter) || '' }}任务</text>
2025-11-05 15:06:23 +08:00
</view>
2025-11-13 09:48:26 +08:00
<!-- 加载更多提示 -->
<view class="load-more-tip" v-if="!isEmpty && !loading && !noMore">
<text class="load-more-text">上拉加载更多</text>
</view>
<view class="load-more-tip" v-if="!isEmpty && noMore">
<text class="load-more-text">没有更多数据了</text>
</view>
2025-11-05 15:06:23 +08:00
</view>
</scroll-view>
2025-11-14 15:28:48 +08:00
<FabPlus @click="goToCreateTask" />
2025-11-05 15:06:23 +08:00
</view>
</template>
<script setup>
import { ref, computed, watch, onMounted, onUnmounted, nextTick } from 'vue';
2025-11-05 15:06:23 +08:00
import { onLoad } from '@dcloudio/uni-app';
2025-11-05 16:20:12 +08:00
import { getStatusText, getTaskStatusType, getTaskStatusStyle } from '@/utils/taskConfig.js';
2025-11-13 09:48:26 +08:00
import {getTaskList} from '@/api';
2025-11-06 14:33:07 +08:00
import { useTaskStore } from '@/store/task';
2025-11-12 15:15:04 +08:00
import {truncateText} from "@/utils/textSolve/truncateText";
2025-11-12 17:43:07 +08:00
import {useUserStore} from "@/store/user";
2025-11-13 09:48:26 +08:00
import {usePagination} from "@/composables";
2025-11-14 15:28:48 +08:00
import FabPlus from '@/components/FabPlus.vue';
2025-11-13 09:48:26 +08:00
const userStore = useUserStore();
const {
list,
noMore,
isEmpty,
loading,
getList,
loadMore,
updateParams,
refresh,
queryParams,
reset
} = usePagination({
fetchData: getTaskList,
mode: 'loadMore',
pageSize: 10,
defaultParams: {}
});
2025-11-05 15:06:23 +08:00
// 获取页面参数
const statusFilter = ref('');
// 任务状态映射
const statusMap = {
'completed': '完成任务',
'pending': '待完成任务',
'imminent': '即将预期',
'overdue': '逾期任务'
};
// 反向映射(从中文到英文)
const statusReverseMap = {
'完成任务': 'completed',
'待完成任务': 'pending',
'即将预期': 'imminent',
'逾期任务': 'overdue'
};
2025-11-13 09:48:26 +08:00
// 将分页器的 list 转换为任务列表格式
const tasks = computed(() => {
let transformedTasks = list.value.map(item => transformTaskData(item));
// 如果是逾期任务列表过滤掉已完成的任务status === 4
if (statusFilter.value === 'overdue') {
transformedTasks = transformedTasks.filter(task => task.status !== 'completed');
}
return transformedTasks;
});
2025-11-05 15:06:23 +08:00
2025-11-05 16:20:12 +08:00
// 使用全局配置获取标签自定义样式
const getTagCustomStyle = (status) => {
const styleConfig = getTaskStatusStyle(status);
return {
backgroundColor: styleConfig.backgroundColor,
color: styleConfig.color,
borderColor: styleConfig.borderColor
2025-11-05 15:06:23 +08:00
};
};
// 获取卡片样式类
const getTaskCardClass = (status) => {
return {
'task-card-imminent': status === 'imminent',
'task-card-pending': status === 'pending',
'task-card-completed': status === 'completed',
'task-card-overdue': status === 'overdue'
};
};
// 获取按钮类型
const getButtonType = (status) => {
const typeMap = {
'imminent': 'warning',
'pending': 'primary',
'overdue': 'error'
};
return typeMap[status] || 'primary';
};
// 获取倒计时样式类
const getCountdownClass = (status) => {
return {
'countdown-warning': status === 'imminent',
'countdown-primary': status === 'pending',
'countdown-error': status === 'overdue'
};
};
// 处理任务
const handleTask = (task) => {
goToTaskDetail(task);
};
// 跳转到任务详情页
const goToTaskDetail = (task) => {
2025-11-06 14:33:07 +08:00
// 使用 Pinia store 存储任务详情数据
const taskStore = useTaskStore();
taskStore.setTaskDetail({
2025-11-05 15:06:23 +08:00
id: task.id,
name: task.description || '待办任务名称',
project: task.project || '所属项目',
statusTags: task.status === 'overdue' ? ['已逾期', '紧急'] :
task.status === 'imminent' ? ['即将逾期'] :
task.status === 'pending' ? ['待完成'] :
['已完成'],
deadline: task.date || '2025-10-14 18:00',
2025-11-06 14:33:07 +08:00
creator: task.createName,
2025-11-05 15:06:23 +08:00
responsible: task.owner || '张珊珊、李志',
publishTime: task.releaseTime || '2025-10-17',
content: task.description || '任务内容任务。这里是详细的任务描述,可以包含多行文本。根据实际需求,这里可以展示任务的详细要求、步骤说明、注意事项等。任务内容应该清晰明了,便于负责人理解和执行。',
submitRecords: []
});
uni.navigateTo({
2025-11-07 11:40:13 +08:00
url: `/pages/task/detail/index?id=${task.id}`
2025-11-05 15:06:23 +08:00
});
};
2025-11-06 13:50:29 +08:00
// 格式化日期:将 "2024-10-31 23:59:59" 转换为 "2024-10-31"
const formatDate = (dateStr) => {
if (!dateStr) return '';
// 如果包含空格,取日期部分
return dateStr.split(' ')[0];
};
// 格式化日期时间:格式化为 yyyy-MM-dd HH:mm:ss
const formatDateTime = (date) => {
if (!date) return '';
const d = new Date(date);
const year = d.getFullYear();
const month = String(d.getMonth() + 1).padStart(2, '0');
const day = String(d.getDate()).padStart(2, '0');
const hours = String(d.getHours()).padStart(2, '0');
const minutes = String(d.getMinutes()).padStart(2, '0');
const seconds = String(d.getSeconds()).padStart(2, '0');
return `${year}-${month}-${day} ${hours}:${minutes}:${seconds}`;
};
// 获取即将逾期的时间范围当前时间到3天后
const getImminentDateRange = () => {
const now = new Date();
const endDate = new Date(now.getTime() + 3 * 24 * 60 * 60 * 1000);
return {
expireTimeStart: formatDateTime(now),
expireTimeEnd: formatDateTime(endDate)
};
};
// 计算剩余天数
const calculateRemainingDays = (expireTime) => {
if (!expireTime) return null;
const expireDate = new Date(expireTime);
const now = new Date();
now.setHours(0, 0, 0, 0);
expireDate.setHours(0, 0, 0, 0);
const diffTime = expireDate.getTime() - now.getTime();
const diffDays = Math.ceil(diffTime / (1000 * 60 * 60 * 24));
return diffDays;
};
// 提取负责人:从 memberList 中提取所有成员的名称
const getOwnerNames = (memberList) => {
if (!Array.isArray(memberList) || memberList.length === 0) return '';
return memberList.map(member => member.userName || member.name || '').filter(name => name).join('、');
};
// 根据过期时间判断任务状态
const determineTaskStatus = (item, expireTime) => {
// 如果任务已完成状态为4直接返回 completed
const taskStatusFromBackend = item.status;
if (taskStatusFromBackend === 4 || taskStatusFromBackend === 'completed') {
return 'completed';
}
// 如果没有过期时间,使用过滤状态或默认 pending
if (!expireTime) {
return statusFilter.value || 'pending';
}
const expireDate = new Date(expireTime);
const now = new Date();
// 设置时间到当天0点便于日期比较
// now.setHours(0, 0, 0, 0);
// expireDate.setHours(23, 59, 59, 999);
// 如果已过期,标记为逾期
if (expireDate.getTime() < now.getTime()) {
return 'overdue';
}
// 计算距离过期的天数
const diffTime = expireDate.getTime() - now.getTime();
const diffDays = Math.ceil(diffTime / (1000 * 60 * 60 * 24));
// 如果3天内到期标记为即将逾期
if (diffDays <= 3 && diffDays >= 0) {
return 'imminent';
}
// 否则返回待完成状态
return 'pending';
};
2025-11-06 13:50:29 +08:00
// 将接口数据转换为页面需要的格式
const transformTaskData = (item) => {
2025-11-06 13:50:29 +08:00
const expireTime = item.expireTime || item.expire_time || '';
// 最高优先级:判断任务状态 - status===4 或 status==='4' 直接返回completed不做任何其他校验
// 支持多种字段名和数据类型status、taskStatus、statusId 等
const taskStatus = item.status !== undefined ? item.status :
item.taskStatus !== undefined ? item.taskStatus :
item.statusId !== undefined ? item.statusId : null;
// 检查是否已完成支持数字4、字符串'4'、字符串'completed'等多种格式)
const isCompleted = taskStatus === 4 ||
taskStatus === '4' ||
taskStatus === 'completed' ||
String(taskStatus) === '4';
// 如果已完成,直接返回完成状态,不做过期校验,不计算剩余天数
if (isCompleted) {
return {
id: item.id || '',
status: 'completed', // 固定为completed确保显示灰色样式
createName: item.createName || '',
date: formatDate(expireTime) || '',
project: item.projectName || item.project_name || '',
description: item.description || item.task_name || '',
owner: getOwnerNames(item.memberList || item.member_list || []),
releaseTime: formatDate(item.createTime || item.create_time) || '',
remainingDays: null // 已完成任务不计算剩余天数
};
}
// 未完成的任务才计算剩余天数并进行过期校验
const remainingDays = calculateRemainingDays(expireTime);
const finalStatus = determineTaskStatus(item, expireTime);
2025-11-06 13:50:29 +08:00
return {
id: item.id || '',
status: finalStatus,
createName: item.createName || '',
2025-11-06 13:50:29 +08:00
date: formatDate(expireTime) || '',
project: item.projectName || item.project_name || '',
description: item.description || item.task_name || '',
owner: getOwnerNames(item.memberList || item.member_list || []),
releaseTime: formatDate(item.createTime || item.create_time) || '',
remainingDays: remainingDays
};
};
2025-11-13 09:48:26 +08:00
// 获取查询参数
const getQueryParams = () => {
const userId = userStore.getUserInfo?.user?.userId || userStore.getUserInfo?.userId;
const privateView = userStore.privateView;
const ownerId = userId && privateView ? userId : '';
2025-11-24 10:11:47 +08:00
// 排序参数
const sortParams = {
orderByColumn: 'expireTime',
isAsc: 'ascending'
};
2025-11-13 09:48:26 +08:00
// 根据状态类型构建不同的查询参数
if (statusFilter.value === 'completed') {
2025-11-24 10:11:47 +08:00
return {
statusList: [4],
ownerId: ownerId,
...sortParams
};
2025-11-13 09:48:26 +08:00
} else if (statusFilter.value === 'overdue') {
2025-11-24 10:11:47 +08:00
return {
statusList: [2],
overdue: true,
ownerId: ownerId,
...sortParams
};
2025-11-13 09:48:26 +08:00
} else if (statusFilter.value === 'pending') {
2025-11-24 10:11:47 +08:00
return {
statusList: [2],
ownerId: ownerId,
...sortParams
};
2025-11-13 09:48:26 +08:00
} else if (statusFilter.value === 'imminent') {
const dateRange = getImminentDateRange();
return {
ownerId: ownerId,
statusList: [2],
expireTimeStart: dateRange.expireTimeStart,
2025-11-24 10:11:47 +08:00
expireTimeEnd: dateRange.expireTimeEnd,
...sortParams
2025-11-13 09:48:26 +08:00
};
} else {
2025-11-24 10:11:47 +08:00
return {
ownerId: ownerId,
...sortParams
};
2025-11-13 09:48:26 +08:00
}
};
2025-11-06 13:50:29 +08:00
// 加载任务列表数据
const loadTaskList = async () => {
2025-11-13 09:48:26 +08:00
const params = getQueryParams();
updateParams(params);
};
2025-11-14 15:28:48 +08:00
const refreshTaskList = () => {
reset();
loadTaskList();
};
const goToCreateTask = () => {
uni.navigateTo({
url: '/pages/task/add/index'
});
};
2025-11-13 09:48:26 +08:00
// 处理滚动到底部
const handleScrollToLower = () => {
if (!noMore.value && !loading.value) {
loadMore();
2025-11-06 13:50:29 +08:00
}
};
2025-11-13 09:48:26 +08:00
// 页面初始化标志,避免 watch 在 onLoad 之前触发
const isInitialized = ref(false);
2025-11-14 15:28:48 +08:00
onMounted(() => {
uni.$on('taskListRefresh', refreshTaskList);
});
onUnmounted(() => {
uni.$off('taskListRefresh', refreshTaskList);
});
2025-11-13 09:48:26 +08:00
// 监听 statusFilter 变化,自动刷新数据
watch(statusFilter, () => {
if (isInitialized.value) {
loadTaskList();
}
});
// 监听用户私有视角变化,自动刷新数据
watch(() => userStore.privateView, () => {
if (isInitialized.value) {
loadTaskList();
}
});
2025-11-05 15:06:23 +08:00
// 页面加载时获取参数
onLoad((options) => {
// 获取状态参数(此时 isInitialized 为 falsewatch 不会触发)
2025-11-05 15:06:23 +08:00
if (options.status) {
statusFilter.value = options.status;
} else if (options.label) {
// 如果传入的是中文标签,转换为英文状态
statusFilter.value = statusReverseMap[decodeURIComponent(options.label)] || '';
}
// 设置页面标题
if (statusFilter.value && statusMap[statusFilter.value]) {
uni.setNavigationBarTitle({
title: statusMap[statusFilter.value]
});
}
2025-11-06 13:50:29 +08:00
// 在下一个 tick 中标记为已初始化并加载数据
// 这样可以确保 watch 的回调(如果触发)在 isInitialized 还是 false 时不执行
nextTick(() => {
isInitialized.value = true;
loadTaskList();
});
2025-11-05 15:06:23 +08:00
});
</script>
<style lang="scss" scoped>
.task-list-page {
width: 100%;
height: 100vh;
background: #f5f5f5;
}
.task-scroll {
width: 100%;
height: 100%;
}
2025-11-22 16:17:08 +08:00
@import '@/styles/task-card.scss';
2025-11-05 15:06:23 +08:00
</style>