任务管理0.7

This commit is contained in:
WindowBird 2025-11-22 16:35:11 +08:00
parent 04a897a22a
commit 752a754d45
2 changed files with 162 additions and 141 deletions

View File

@ -168,12 +168,13 @@
@scrolltolower="handleScrollToLower"
>
<view class="task-container">
<view
class="task-card"
v-for="task in tasks"
:key="task.id"
:class="getTaskCardClass(task)"
@click="goToTaskDetail(task)"
<!-- 任务卡片列表 -->
<view
class="task-card"
v-for="task in tasks"
:key="task.id"
:class="getTaskCardClass(task.status)"
@click="goToTaskDetail(task)"
>
<!-- 状态标签和日期 -->
<view class="task-header">
@ -186,37 +187,40 @@
:plain="false"
:custom-style="getTagCustomStyle(task.status)"
></uv-tags>
<uv-tags
v-if="task.overdue && task.status !== '4' && task.status !== 4"
text="逾期"
type="error"
size="mini"
:plain="false"
></uv-tags>
</view>
<view class="task-date-wrapper">
<text class="task-date">{{ formatDate(task.expireTime) }}</text>
<text class="task-date">{{ task.date }}</text>
</view>
</view>
<!-- 立即处理按钮 -->
<view class="task-action" v-if="task.status !== 'completed'">
<uv-button
:type="getButtonType(task.status)"
size="small"
@click.stop="handleTask(task)"
>
立即处理
</uv-button>
</view>
</view>
<!-- 任务内容 -->
<view class="task-content">
<text class="task-project">所属项目: {{ task.projectName || '未分配项目' }}</text>
<text class="task-description">{{ truncateText(task.description, 80) }}</text>
<text class="task-project">所属项目: {{ task.project }}</text>
<text class="task-description">{{truncateText(task.description)}}</text>
<view class="task-meta">
<text class="task-owner">负责人: {{ getOwnerNames(task.memberList) || '未分配' }}</text>
<text class="task-owner">创建人: {{ task.createName || '未知' }}</text>
<text class="task-owner">负责人: {{ task.owner }}</text>
<text class="task-owner">创建人: {{ task.createName }}</text>
<view class="task-time-row">
<text class="task-time">发布时间: {{ formatDate(task.createTime) }}</text>
<view class="task-countdown" v-if="task.cardStatus !== 'completed' && task.remainingDays !== null">
<text class="task-time">发布时间: {{ task.releaseTime }}</text>
<view class="task-countdown" v-if="task.status !== 'completed' && task.remainingDays !== null">
<text class="countdown-icon">🕐</text>
<text class="countdown-text" :class="getCountdownClass(task.cardStatus)">
<text class="countdown-text" :class="getCountdownClass(task.status)">
{{ task.remainingDays < 0 ? `已逾期${Math.abs(task.remainingDays)}` : `剩余${task.remainingDays}` }}
</text>
</view>
</view>
<text v-if="task.passTime" class="task-time">通过时间: {{ formatDate(task.passTime) }}</text>
</view>
</view>
</view>
@ -319,9 +323,11 @@
import { ref, computed, onMounted } from 'vue';
import { getTaskList, getProjectListAll, getUserList } from '@/api';
import { useDictStore } from '@/store/dict';
import { useTaskStore } from '@/store/task';
import { usePagination } from '@/composables';
import { getDictLabel } from '@/utils/dict';
import { truncateText } from '@/utils/textSolve/truncateText';
import { getStatusText, getTaskStatusType, getTaskStatusStyle } from '@/utils/taskConfig.js';
import FabPlus from '@/components/FabPlus.vue';
const dictStore = useDictStore();
@ -448,6 +454,13 @@ const {
defaultParams: {}
});
// "2024-10-31 23:59:59" "2024-10-31"
const formatDate = (dateStr) => {
if (!dateStr) return '';
//
return dateStr.split(' ')[0];
};
//
const calculateRemainingDays = (expireTime) => {
if (!expireTime) return null;
@ -460,16 +473,21 @@ const calculateRemainingDays = (expireTime) => {
return diffDays;
};
//
const determineTaskStatusForCard = (task) => {
// 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 taskStatus = task.status;
if (taskStatus === 4 || taskStatus === '4') {
const taskStatusFromBackend = item.status;
if (taskStatusFromBackend === 4 || taskStatusFromBackend === 'completed') {
return 'completed';
}
// pending
const expireTime = task.expireTime;
// 使 pending
if (!expireTime) {
return 'pending';
}
@ -495,130 +513,101 @@ const determineTaskStatusForCard = (task) => {
return 'pending';
};
//
const tasks = computed(() => {
return list.value.map(task => {
const expireTime = task.expireTime || '';
const remainingDays = calculateRemainingDays(expireTime);
const cardStatus = determineTaskStatusForCard(task);
//
const transformTaskData = (item) => {
const expireTime = item.expireTime || item.expire_time || '';
// - status===4 status==='4' completed
// statustaskStatusstatusId
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 {
...task,
remainingDays,
cardStatus
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);
return {
id: item.id || '',
status: finalStatus,
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: remainingDays
};
};
// list
const tasks = computed(() => {
return list.value.map(item => transformTaskData(item));
});
//
const getStatusText = (status) => {
if (!status && status !== 0) return '未知';
//
const dictLabel = getDictLabel('task_status', String(status));
if (dictLabel && dictLabel !== String(status)) {
return dictLabel;
}
//
const statusMap = {
'1': '待接收',
'2': '进行中',
'3': '已提交',
'4': '已完成',
'5': '已驳回',
'6': '已取消',
'7': '逾期完成',
'10': '待完成'
};
return statusMap[String(status)] || `状态${status}`;
};
//
const getTaskStatusType = (status) => {
if (!status && status !== 0) return 'primary';
// listClassuv-tagstype
const statusDict = dictStore.getDictByType('task_status');
const statusItem = statusDict.find(item => item.dictValue === String(status));
if (statusItem) {
const listClassMap = {
'primary': 'primary',
'success': 'success',
'warning': 'warning',
'danger': 'error',
'info': 'info'
};
return listClassMap[statusItem.listClass] || 'primary';
}
//
const typeMap = {
'1': 'info', //
'2': 'warning', //
'3': 'primary', //
'4': 'success', //
'5': 'error', //
'6': 'error', //
'7': 'warning', //
'10': 'primary' //
};
return typeMap[String(status)] || 'primary';
};
//
// 使
const getTagCustomStyle = (status) => {
const statusDict = dictStore.getDictByType('task_status');
const statusItem = statusDict.find(item => item.dictValue === String(status));
//
const defaultStyle = {
backgroundColor: '#909399',
color: '#fff',
borderColor: '#909399'
const styleConfig = getTaskStatusStyle(status);
return {
backgroundColor: styleConfig.backgroundColor,
color: styleConfig.color,
borderColor: styleConfig.borderColor
};
if (!statusItem) {
return defaultStyle;
}
// listClass
const colorMap = {
'primary': { backgroundColor: '#2885ff', color: '#fff', borderColor: '#2885ff' },
'success': { backgroundColor: '#67c23a', color: '#fff', borderColor: '#67c23a' },
'warning': { backgroundColor: '#ff9800', color: '#fff', borderColor: '#ff9800' },
'danger': { backgroundColor: '#f56c6c', color: '#fff', borderColor: '#f56c6c' },
'info': { backgroundColor: '#909399', color: '#fff', borderColor: '#909399' }
};
return colorMap[statusItem.listClass] || defaultStyle;
};
//
const getTaskCardClass = (task) => {
const getTaskCardClass = (status) => {
return {
'task-card-imminent': task.cardStatus === 'imminent',
'task-card-pending': task.cardStatus === 'pending',
'task-card-completed': task.cardStatus === 'completed',
'task-card-overdue': task.cardStatus === 'overdue'
'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 = (cardStatus) => {
const getCountdownClass = (status) => {
return {
'countdown-warning': cardStatus === 'imminent',
'countdown-primary': cardStatus === 'pending',
'countdown-error': cardStatus === 'overdue'
'countdown-warning': status === 'imminent',
'countdown-primary': status === 'pending',
'countdown-error': status === 'overdue'
};
};
//
const formatDate = (dateStr) => {
if (!dateStr) return '';
return dateStr.split(' ')[0];
};
//
const getOwnerNames = (memberList) => {
if (!Array.isArray(memberList) || memberList.length === 0) return '';
return memberList.map(member => member.userName || member.name || '').filter(name => name).join('、');
//
const handleTask = (task) => {
goToTaskDetail(task);
};
//
@ -791,8 +780,26 @@ const handleScrollToLower = () => {
}
};
//
//
const goToTaskDetail = (task) => {
// 使 Pinia store
const taskStore = useTaskStore();
taskStore.setTaskDetail({
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',
creator: task.createName,
responsible: task.owner || '张珊珊、李志',
publishTime: task.releaseTime || '2025-10-17',
content: task.description || '任务内容任务。这里是详细的任务描述,可以包含多行文本。根据实际需求,这里可以展示任务的详细要求、步骤说明、注意事项等。任务内容应该清晰明了,便于负责人理解和执行。',
submitRecords: []
});
uni.navigateTo({
url: `/pages/task/detail/index?id=${task.id}`
});
@ -1049,21 +1056,34 @@ onMounted(() => {
align-items: center;
gap: 4px;
padding: 4px 12px;
background: #f5f5f5;
border-radius: 16px;
font-size: 13px;
background: transparent;
border-radius: 0;
font-size: 14px;
color: #666;
transition: all 0.2s;
white-space: nowrap;
cursor: pointer;
position: relative;
&:active {
opacity: 0.7;
}
&.active {
background: #2885ff;
color: #fff;
background: transparent;
color: #2885ff;
font-weight: 500;
&::after {
content: '';
position: absolute;
bottom: 0;
left: 12px;
right: 12px;
height: 2px;
background-color: #2885ff;
border-radius: 1px;
}
}
.count {
@ -1107,3 +1127,4 @@ onMounted(() => {
@import '@/styles/task-card.scss';
</style>

View File

@ -11,7 +11,7 @@ export const Request = () => {
uni.$uv.http.setConfig((config) => {
/* config 为默认全局配置*/
config.baseURL = 'http://192.168.1.4:4001'; /* 根域名 */
config.baseURL = 'https://pm.ccttiot.com/prod-api'; /* 根域名 */
// config.baseURL = 'https://pm.ccttiot.com/prod-api'; /* 根域名 */
return config
})