2025-10-27 23:13:08 +08:00

1492 lines
32 KiB
Vue

<template>
<view class="tasks-page">
<!-- 统一顶部导航 -->
<view class="unified-header">
<view class="header-content">
<view class="header-left">
<i class="fas fa-search header-icon" @click="showSearch = !showSearch"></i>
</view>
<view class="header-title">任务管理</view>
<view class="header-right">
<i class="fas fa-plus header-icon" @click="showAddTask = true"></i>
</view>
</view>
</view>
<!-- 页面内容 -->
<scroll-view scroll-y class="unified-content">
<!-- 搜索栏 -->
<view class="search-container" v-if="showSearch">
<view class="search-input-container">
<i class="fas fa-search search-icon"></i>
<input
v-model="searchKeyword"
placeholder="搜索任务..."
class="search-input"
@input="handleSearch"
/>
<view class="clear-button" @click="clearSearch" v-if="searchKeyword">
<i class="fas fa-times"></i>
</view>
</view>
</view>
<!-- 统计概览 -->
<view class="stats-overview">
<view class="stats-grid">
<view class="stat-card total">
<view class="stat-icon">
<i class="fas fa-list"></i>
</view>
<view class="stat-content">
<text class="stat-number">{{ taskStats.total }}</text>
<text class="stat-label">总任务</text>
</view>
</view>
<view class="stat-card pending">
<view class="stat-icon">
<i class="fas fa-clock"></i>
</view>
<view class="stat-content">
<text class="stat-number">{{ taskStats.pending }}</text>
<text class="stat-label">待完成</text>
</view>
</view>
<view class="stat-card completed">
<view class="stat-icon">
<i class="fas fa-check-circle"></i>
</view>
<view class="stat-content">
<text class="stat-number">{{ taskStats.completed }}</text>
<text class="stat-label">已完成</text>
</view>
</view>
<view class="stat-card overdue">
<view class="stat-icon">
<i class="fas fa-exclamation-triangle"></i>
</view>
<view class="stat-content">
<text class="stat-number">{{ taskStats.overdue }}</text>
<text class="stat-label">已逾期</text>
</view>
</view>
</view>
</view>
<!-- 筛选和排序 -->
<view class="filter-container">
<view class="filter-tabs-wrapper">
<view class="filter-tabs">
<view
v-for="filter in filterOptions"
:key="filter.value"
class="filter-tab"
:class="{ active: currentFilter === filter.value }"
@click="setFilter(filter.value)"
>
<text class="filter-text">{{ filter.label }}</text>
<view class="filter-badge" v-if="getFilterCount(filter.value) > 0">
{{ getFilterCount(filter.value) }}
</view>
</view>
</view>
</view>
<view class="sort-container" @click="showSortOptions = !showSortOptions">
<text class="sort-text">{{ getCurrentSortLabel() }}</text>
<i
class="fas fa-chevron-down sort-icon"
:class="{ rotated: showSortOptions }"
></i>
</view>
</view>
<!-- 排序选项下拉 -->
<view class="sort-dropdown" v-if="showSortOptions">
<view
v-for="sort in sortOptions"
:key="sort.value"
class="sort-option"
:class="{ active: currentSort === sort.value }"
@click="setSort(sort.value)"
>
<text class="sort-option-text">{{ sort.label }}</text>
<i class="fas fa-check" v-if="currentSort === sort.value"></i>
</view>
</view>
<!-- 任务列表容器 -->
<view class="task-list-container">
<scroll-view scroll-y class="task-list" v-if="filteredTasks.length > 0">
<view
v-for="task in filteredTasks"
:key="task.id"
class="task-card"
:class="{ completed: task.completed, overdue: isOverdue(task) }"
@click="toggleTask(task.id)"
>
<view class="task-main">
<view class="task-checkbox-container">
<view
class="custom-checkbox"
:class="{ checked: task.completed }"
>
<i class="fas fa-check" v-if="task.completed"></i>
</view>
</view>
<view class="task-content">
<view class="task-header">
<text class="task-title">{{ task.title }}</text>
<view class="task-priority-indicator" :class="task.priority">
<i class="fas fa-circle"></i>
</view>
</view>
<text class="task-description" v-if="task.description">{{
task.description
}}</text>
<view class="task-footer">
<view class="task-tags" v-if="task.tags.length > 0">
<text
v-for="tag in task.tags.slice(0, 3)"
:key="tag"
class="task-tag"
>
{{ tag }}
</text>
</view>
<view class="task-due-date" v-if="task.dueDate">
<i class="fas fa-calendar-alt"></i>
<text class="due-date-text">{{
formatDate(task.dueDate)
}}</text>
</view>
</view>
</view>
</view>
<view class="task-actions" v-if="!task.completed">
<view class="action-button edit" @click.stop="editTask(task)">
<i class="fas fa-edit"></i>
</view>
<view
class="action-button delete"
@click.stop="deleteTask(task.id)"
>
<i class="fas fa-trash"></i>
</view>
</view>
</view>
</scroll-view>
<!-- 空状态 -->
<view class="empty-state" v-else>
<view class="empty-icon-container">
<i class="fas fa-tasks empty-icon"></i>
</view>
<text class="empty-title">暂无任务</text>
<text class="empty-description">点击右上角 + 号添加新任务</text>
<view class="empty-action" @click="showAddTask = true">
<i class="fas fa-plus"></i>
<text>添加任务</text>
</view>
</view>
</view>
<!-- 添加任务弹窗 -->
<view class="modal-overlay" v-if="showAddTask" @click="closeModal">
<view class="modal-content" @click.stop>
<view class="modal-header">
<text class="modal-title">添加任务</text>
<view class="close-btn" @click="closeModal">
<i class="fas fa-times"></i>
</view>
</view>
<view class="modal-body">
<view class="form-group">
<text class="form-label">任务标题 *</text>
<input
v-model="newTask.title"
placeholder="请输入任务标题"
class="form-input"
/>
</view>
<view class="form-group">
<text class="form-label">任务描述</text>
<textarea
v-model="newTask.description"
placeholder="请输入任务描述"
class="form-textarea"
>
</textarea>
</view>
<view class="form-group">
<text class="form-label">优先级</text>
<view class="priority-options">
<view
v-for="priority in priorityOptions"
:key="priority.value"
class="priority-option"
:class="{ active: newTask.priority === priority.value }"
@click="newTask.priority = priority.value"
>
<view class="priority-dot" :class="priority.value"></view>
<text class="priority-text">{{ priority.label }}</text>
</view>
</view>
</view>
<view class="form-group">
<text class="form-label">截止日期</text>
<picker mode="date" :value="newTask.dueDate" @change="onDateChange">
<view class="date-picker">
<text class="date-text">{{
newTask.dueDate || "选择日期"
}}</text>
<i class="fas fa-calendar-alt"></i>
</view>
</picker>
</view>
<view class="form-group">
<text class="form-label">标签</text>
<input
v-model="tagInput"
placeholder="输入标签后按回车添加"
class="form-input"
@confirm="addTag"
/>
<view class="tags-display" v-if="newTask.tags.length > 0">
<view
v-for="(tag, index) in newTask.tags"
:key="index"
class="tag-item"
>
<text class="tag-text">{{ tag }}</text>
<view class="tag-remove" @click="removeTag(index)">
<i class="fas fa-times"></i>
</view>
</view>
</view>
</view>
</view>
<view class="modal-footer">
<button class="btn-cancel" @click="closeModal">取消</button>
<button
class="btn-confirm"
@click="saveTask"
:disabled="!newTask.title"
>
保存
</button>
</view>
</view>
</view>
</scroll-view>
</view>
</template>
<script>
import { ref, computed, onMounted, nextTick } from "vue";
import { useTaskStore } from "../../src/store/taskStore.js";
export default {
name: "TasksPage",
setup() {
const taskStore = useTaskStore();
// 响应式数据
const showSearch = ref(false);
const showAddTask = ref(false);
const showSortOptions = ref(false);
const searchKeyword = ref("");
const tagInput = ref("");
const newTask = ref({
title: "",
description: "",
priority: "medium",
dueDate: "",
tags: [],
});
// 筛选选项
const filterOptions = [
{ label: "全部", value: "all" },
{ label: "待完成", value: "pending" },
{ label: "已完成", value: "completed" },
{ label: "已逾期", value: "overdue" },
];
// 排序选项
const sortOptions = [
{ label: "按截止日期", value: "dueDate" },
{ label: "按优先级", value: "priority" },
{ label: "按创建时间", value: "created" },
];
// 优先级选项
const priorityOptions = [
{ label: "低", value: "low" },
{ label: "中", value: "medium" },
{ label: "高", value: "high" },
];
// 计算属性
const filteredTasks = computed(() => taskStore.filteredTasks);
const taskStats = computed(() => taskStore.taskStats);
const currentFilter = computed(() => taskStore.currentFilter);
const currentSort = computed(() => taskStore.currentSort);
// 方法
const handleSearch = () => {
taskStore.setSearchKeyword(searchKeyword.value);
};
const clearSearch = () => {
searchKeyword.value = "";
taskStore.setSearchKeyword("");
};
const setFilter = (filter) => {
taskStore.setFilter(filter);
// 滚动到选中的标签
nextTick(() => {
const activeTab = document.querySelector('.filter-tab.active');
if (activeTab) {
activeTab.scrollIntoView({
behavior: 'smooth',
block: 'nearest',
inline: 'center'
});
}
});
};
const setSort = (sort) => {
taskStore.setSort(sort);
showSortOptions.value = false;
};
const getCurrentSortLabel = () => {
const option = sortOptions.find((opt) => opt.value === currentSort.value);
return option ? option.label : "按截止日期";
};
const getFilterCount = (filter) => {
switch (filter) {
case "all":
return taskStore.tasks.length;
case "pending":
return taskStore.tasks.filter((task) => !task.completed).length;
case "completed":
return taskStore.tasks.filter((task) => task.completed).length;
case "overdue":
return taskStore.getOverdueTasks().length;
default:
return 0;
}
};
const toggleTask = (id) => {
taskStore.toggleTask(id);
};
const editTask = (task) => {
newTask.value = { ...task };
showAddTask.value = true;
};
const deleteTask = (id) => {
uni.showModal({
title: "确认删除",
content: "确定要删除这个任务吗?",
success: (res) => {
if (res.confirm) {
taskStore.deleteTask(id);
uni.showToast({
title: "删除成功",
icon: "success",
});
}
},
});
};
const isOverdue = (task) => {
if (task.completed || !task.dueDate) return false;
return new Date(task.dueDate) < new Date();
};
const formatDate = (dateString) => {
const date = new Date(dateString);
const now = new Date();
const diffTime = date - now;
const diffDays = Math.ceil(diffTime / (1000 * 60 * 60 * 24));
if (diffDays === 0) return "今天";
if (diffDays === 1) return "明天";
if (diffDays === -1) return "昨天";
if (diffDays < 0) return `${Math.abs(diffDays)}天前`;
if (diffDays <= 7) return `${diffDays}天后`;
return date.toLocaleDateString();
};
const onDateChange = (e) => {
newTask.value.dueDate = e.detail.value;
};
const addTag = () => {
const tag = tagInput.value.trim();
if (tag && !newTask.value.tags.includes(tag)) {
newTask.value.tags.push(tag);
tagInput.value = "";
}
};
const removeTag = (index) => {
newTask.value.tags.splice(index, 1);
};
const saveTask = () => {
if (!newTask.value.title.trim()) {
uni.showToast({
title: "请输入任务标题",
icon: "none",
});
return;
}
if (newTask.value.id) {
// 编辑任务
taskStore.updateTask(newTask.value.id, newTask.value);
uni.showToast({
title: "任务更新成功",
icon: "success",
});
} else {
// 添加任务
taskStore.addTask(newTask.value);
uni.showToast({
title: "任务添加成功",
icon: "success",
});
}
closeModal();
};
const closeModal = () => {
showAddTask.value = false;
newTask.value = {
title: "",
description: "",
priority: "medium",
dueDate: "",
tags: [],
};
tagInput.value = "";
};
// 生命周期
onMounted(() => {
taskStore.loadFromStorage();
taskStore.initSampleData();
});
return {
showSearch,
showAddTask,
showSortOptions,
searchKeyword,
tagInput,
newTask,
filterOptions,
sortOptions,
priorityOptions,
filteredTasks,
taskStats,
currentFilter,
currentSort,
handleSearch,
clearSearch,
setFilter,
setSort,
getCurrentSortLabel,
getFilterCount,
toggleTask,
editTask,
deleteTask,
isOverdue,
formatDate,
onDateChange,
addTag,
removeTag,
saveTask,
closeModal,
};
},
};
</script>
<style scoped>
/* 主容器 */
.tasks-page {
min-height: 100vh;
background: var(--gradient-surface);
position: relative;
padding-top: calc(var(--status-bar-height) + 88rpx);
}
/* 支持安全区域的设备 */
@supports (padding: max(0px)) {
.tasks-page {
padding-top: calc(var(--status-bar-height) + 88rpx + env(safe-area-inset-top));
}
}
/* 页面头部样式已移至统一导航 */
/* 搜索容器 */
.search-container {
padding: 24rpx 30rpx;
background: var(--white);
border-bottom: 1rpx solid var(--border-light);
box-shadow: var(--shadow);
}
.search-input-container {
position: relative;
display: flex;
align-items: center;
background: var(--gray-lighter);
border-radius: 20rpx;
padding: 0 24rpx;
border: 2rpx solid var(--border-light);
transition: all 0.3s cubic-bezier(0.4, 0, 0.2, 1);
box-shadow: var(--shadow);
}
.search-input-container:focus-within {
border-color: var(--primary-color);
box-shadow: 0 0 0 4rpx var(--info-light);
background: var(--white);
}
.search-icon {
color: var(--text-secondary);
font-size: 28rpx;
margin-right: 16rpx;
}
.search-input {
flex: 1;
padding: 28rpx 0;
border: none;
background: transparent;
font-size: 30rpx;
color: var(--text-color);
font-weight: 500;
letter-spacing: 0.2rpx;
}
.search-input::placeholder {
color: var(--text-muted);
font-weight: 400;
}
.clear-button {
color: var(--text-muted);
font-size: 26rpx;
padding: 12rpx;
transition: all 0.3s ease;
border-radius: 12rpx;
}
.clear-button:active {
color: var(--primary-color);
background: var(--info-light);
}
/* 统计概览 */
.stats-overview {
padding: 30rpx;
}
.stats-grid {
display: grid;
grid-template-columns: repeat(2, 1fr);
gap: 20rpx;
}
.stat-card {
background: var(--white);
border-radius: 24rpx;
padding: 32rpx 24rpx;
display: flex;
align-items: center;
gap: 20rpx;
box-shadow: var(--shadow-md);
border: 2rpx solid var(--border-light);
transition: all 0.3s cubic-bezier(0.4, 0, 0.2, 1);
position: relative;
overflow: hidden;
}
.stat-card::before {
content: '';
position: absolute;
top: 0;
left: 0;
right: 0;
bottom: 0;
background: var(--gradient-surface);
opacity: 0.2;
z-index: 0;
}
.stat-card::after {
content: '';
position: absolute;
top: 0;
left: 0;
right: 0;
height: 4rpx;
background: var(--gradient-primary);
opacity: 0;
transition: opacity 0.3s ease;
z-index: 1;
}
.stat-card:active {
transform: translateY(-4rpx);
box-shadow: var(--shadow-lg);
border-color: var(--primary-color);
}
.stat-card:active::after {
opacity: 1;
}
.stat-icon {
width: 72rpx;
height: 72rpx;
border-radius: 20rpx;
display: flex;
align-items: center;
justify-content: center;
font-size: 28rpx;
color: var(--white);
box-shadow: var(--shadow);
position: relative;
z-index: 1;
}
.stat-card.total .stat-icon {
background: var(--gradient-primary);
}
.stat-card.pending .stat-icon {
background: var(--gradient-warning);
}
.stat-card.completed .stat-icon {
background: var(--gradient-success);
}
.stat-card.overdue .stat-icon {
background: var(--gradient-error);
}
.stat-content {
flex: 1;
position: relative;
z-index: 1;
}
.stat-number {
font-size: 42rpx;
font-weight: 800;
color: var(--text-color);
margin-bottom: 6rpx;
display: block;
letter-spacing: -0.5rpx;
}
.stat-label {
font-size: 26rpx;
color: var(--text-secondary);
font-weight: 600;
letter-spacing: 0.2rpx;
}
/* 筛选容器 */
.filter-container {
padding: 0 30rpx 20rpx;
display: flex;
flex-direction: row;
align-items: center;
justify-content: space-between;
gap: 20rpx;
}
.filter-tabs-wrapper {
flex: 1;
display: flex;
justify-content: flex-start;
overflow: hidden;
position: relative;
}
.filter-tabs-wrapper::after {
content: '';
position: absolute;
right: 0;
top: 0;
bottom: 0;
width: 20rpx;
background: linear-gradient(to left, rgba(255, 255, 255, 0.8), transparent);
pointer-events: none;
z-index: 1;
}
.filter-tabs {
display: flex;
gap: 8rpx;
overflow-x: auto;
overflow-y: hidden;
padding: 8rpx 0;
justify-content: flex-start;
flex-wrap: nowrap;
min-width: 0;
width: 100%;
-webkit-overflow-scrolling: touch;
scrollbar-width: none;
-ms-overflow-style: none;
}
.filter-tabs::-webkit-scrollbar {
display: none;
}
.filter-tab {
padding: 16rpx 24rpx;
border-radius: 20rpx;
background: var(--white);
display: flex;
align-items: center;
gap: 8rpx;
box-shadow: var(--shadow);
border: 2rpx solid var(--border-light);
transition: all 0.3s cubic-bezier(0.4, 0, 0.2, 1);
white-space: nowrap;
flex-shrink: 0;
flex-grow: 0;
min-width: fit-content;
cursor: pointer;
position: relative;
overflow: hidden;
}
.filter-tab::before {
content: '';
position: absolute;
top: 0;
left: 0;
right: 0;
bottom: 0;
background: var(--gradient-primary);
opacity: 0;
transition: opacity 0.3s ease;
}
.filter-tab.active {
background: var(--gradient-primary);
color: var(--white);
border-color: transparent;
box-shadow: var(--shadow-md);
}
.filter-tab.active::before {
opacity: 0;
}
.filter-text {
font-size: 28rpx;
font-weight: 600;
color: var(--text-color);
position: relative;
z-index: 1;
letter-spacing: 0.2rpx;
}
.filter-tab.active .filter-text {
color: var(--white);
}
.filter-badge {
background: var(--gray-dark);
color: var(--white);
font-size: 22rpx;
padding: 6rpx 12rpx;
border-radius: 12rpx;
min-width: 36rpx;
text-align: center;
font-weight: 700;
position: relative;
z-index: 1;
box-shadow: 0 2rpx 4rpx rgba(0, 0, 0, 0.1);
}
.sort-container {
display: flex;
align-items: center;
gap: 8rpx;
padding: 10rpx 16rpx;
background: var(--white);
border-radius: 16rpx;
box-shadow: var(--shadow);
border: 1rpx solid var(--border);
transition: all 0.3s ease;
flex-shrink: 0;
}
.sort-container:active {
transform: scale(0.98);
}
.sort-text {
font-size: 26rpx;
color: var(--text-color);
font-weight: 500;
}
.sort-icon {
font-size: 20rpx;
color: var(--text-secondary);
transition: transform 0.3s ease;
}
.sort-icon.rotated {
transform: rotate(180deg);
}
/* 排序下拉 */
.sort-dropdown {
position: absolute;
top: 160rpx;
right: 30rpx;
background: var(--white);
border-radius: 16rpx;
box-shadow: 0 8rpx 24rpx rgba(0, 0, 0, 0.15);
z-index: 200;
min-width: 200rpx;
border: 1rpx solid var(--border);
overflow: hidden;
}
.sort-option {
padding: 20rpx 24rpx;
display: flex;
justify-content: space-between;
align-items: center;
border-bottom: 1rpx solid var(--border);
transition: background-color 0.3s ease;
}
.sort-option:last-child {
border-bottom: none;
}
.sort-option.active {
background: var(--background);
}
.sort-option:active {
background: var(--gray-light);
}
.sort-option-text {
font-size: 26rpx;
color: var(--text-color);
font-weight: 500;
}
.sort-option i {
color: var(--primary-color);
font-size: 20rpx;
}
/* 任务列表容器 */
.task-list-container {
flex: 1;
padding: 0 30rpx;
min-height: 0;
}
.task-list {
height: calc(100vh - 500rpx);
padding-bottom: 20rpx;
}
/* 任务卡片 */
.task-card {
background: var(--white);
border-radius: 24rpx;
margin-bottom: 20rpx;
box-shadow: var(--shadow-md);
border: 2rpx solid var(--border-light);
transition: all 0.3s cubic-bezier(0.4, 0, 0.2, 1);
overflow: hidden;
position: relative;
}
.task-card::before {
content: '';
position: absolute;
top: 0;
left: 0;
right: 0;
bottom: 0;
background: var(--gradient-surface);
opacity: 0.3;
z-index: 0;
}
.task-card:active {
transform: translateY(-4rpx);
box-shadow: var(--shadow-lg);
border-color: var(--primary-color);
}
.task-card.completed {
opacity: 0.7;
background: var(--gray-lighter);
border-color: var(--gray);
}
.task-card.completed::before {
opacity: 0.1;
}
.task-card.overdue {
border-left: 8rpx solid var(--red);
background: linear-gradient(90deg, var(--error-light) 0%, var(--white) 100%);
border-color: var(--red-light);
}
.task-card.overdue::before {
background: linear-gradient(90deg, var(--error-light) 0%, var(--white) 100%);
opacity: 0.5;
}
.task-main {
display: flex;
align-items: flex-start;
gap: 20rpx;
padding: 32rpx 28rpx;
position: relative;
z-index: 1;
}
.task-checkbox-container {
margin-top: 4rpx;
}
.custom-checkbox {
width: 36rpx;
height: 36rpx;
border: 2rpx solid #cbd5e0;
border-radius: 10rpx;
display: flex;
align-items: center;
justify-content: center;
transition: all 0.3s ease;
background: #ffffff;
}
.custom-checkbox.checked {
background: linear-gradient(135deg, #10b981, #34d399);
border-color: #10b981;
color: #ffffff;
}
.custom-checkbox i {
font-size: 20rpx;
}
.task-content {
flex: 1;
}
.task-header {
display: flex;
justify-content: space-between;
align-items: center;
margin-bottom: 12rpx;
}
.task-title {
font-size: 32rpx;
font-weight: 600;
color: #1a202c;
flex: 1;
line-height: 1.4;
}
.task-priority-indicator {
width: 16rpx;
height: 16rpx;
border-radius: 50%;
margin-left: 12rpx;
}
.task-priority-indicator.high {
color: #dc2626;
}
.task-priority-indicator.medium {
color: #f59e0b;
}
.task-priority-indicator.low {
color: #059669;
}
.task-description {
font-size: 26rpx;
color: #718096;
line-height: 1.5;
margin-bottom: 16rpx;
display: block;
}
.task-footer {
display: flex;
justify-content: space-between;
align-items: center;
flex-wrap: wrap;
gap: 12rpx;
}
.task-tags {
display: flex;
gap: 8rpx;
flex-wrap: wrap;
}
.task-tag {
background: linear-gradient(135deg, #dbeafe, #bfdbfe);
color: #1e40af;
font-size: 22rpx;
padding: 6rpx 12rpx;
border-radius: 12rpx;
font-weight: 500;
}
.task-due-date {
display: flex;
align-items: center;
gap: 6rpx;
color: #718096;
font-size: 24rpx;
}
.due-date-text {
font-weight: 500;
}
.task-actions {
display: flex;
gap: 12rpx;
padding: 0 28rpx 32rpx;
position: relative;
z-index: 1;
}
.action-button {
width: 56rpx;
height: 56rpx;
border-radius: 16rpx;
background: var(--gray-lighter);
color: var(--text-secondary);
display: flex;
align-items: center;
justify-content: center;
font-size: 22rpx;
transition: all 0.3s cubic-bezier(0.4, 0, 0.2, 1);
border: 2rpx solid var(--border-light);
box-shadow: var(--shadow);
}
.action-button.edit {
color: var(--primary-color);
background: var(--info-light);
border-color: var(--primary-light);
}
.action-button.edit:active {
background: var(--primary-color);
color: var(--white);
transform: scale(0.95);
}
.action-button.delete {
color: var(--red);
background: var(--error-light);
border-color: var(--red-light);
}
.action-button.delete:active {
background: var(--red);
color: var(--white);
transform: scale(0.95);
}
/* 空状态 */
.empty-state {
text-align: center;
padding: 80rpx 30rpx;
color: #718096;
}
.empty-icon-container {
margin-bottom: 30rpx;
}
.empty-icon {
font-size: 120rpx;
color: #cbd5e0;
}
.empty-title {
font-size: 32rpx;
font-weight: 600;
color: #4a5568;
margin-bottom: 16rpx;
display: block;
}
.empty-description {
font-size: 26rpx;
color: #718096;
margin-bottom: 40rpx;
display: block;
}
.empty-action {
display: inline-flex;
align-items: center;
gap: 12rpx;
padding: 16rpx 32rpx;
background: linear-gradient(135deg, #4f46e5, #7c3aed);
color: #ffffff;
border-radius: 20rpx;
font-size: 28rpx;
font-weight: 600;
box-shadow: 0 4rpx 12rpx rgba(79, 70, 229, 0.3);
transition: all 0.3s ease;
}
.empty-action:active {
transform: scale(0.95);
box-shadow: 0 2rpx 8rpx rgba(79, 70, 229, 0.4);
}
/* 弹窗样式 */
.modal-overlay {
position: fixed;
top: 0;
left: 0;
right: 0;
bottom: 0;
background: rgba(0, 0, 0, 0.6);
display: flex;
align-items: center;
justify-content: center;
z-index: 1000;
padding: 40rpx;
backdrop-filter: blur(4rpx);
}
.modal-content {
background: #ffffff;
border-radius: 24rpx;
width: 100%;
max-width: 600rpx;
max-height: 80vh;
overflow: hidden;
box-shadow: 0 20rpx 60rpx rgba(0, 0, 0, 0.3);
}
.modal-header {
padding: 30rpx;
border-bottom: 1rpx solid #e2e8f0;
display: flex;
justify-content: space-between;
align-items: center;
background: linear-gradient(135deg, #f7fafc, #edf2f7);
}
.modal-title {
font-size: 32rpx;
font-weight: 700;
color: #1a202c;
}
.close-btn {
width: 48rpx;
height: 48rpx;
border-radius: 12rpx;
background: #f7fafc;
color: #718096;
display: flex;
align-items: center;
justify-content: center;
font-size: 24rpx;
transition: all 0.3s ease;
border: 1rpx solid #e2e8f0;
}
.close-btn:active {
background: #edf2f7;
color: #4a5568;
}
.modal-body {
padding: 30rpx;
max-height: 60vh;
overflow-y: auto;
}
.form-group {
margin-bottom: 30rpx;
}
.form-label {
display: block;
font-size: 26rpx;
font-weight: 600;
color: #2d3748;
margin-bottom: 12rpx;
}
.form-input,
.form-textarea {
width: 100%;
padding: 20rpx;
border: 2rpx solid #e2e8f0;
border-radius: 12rpx;
font-size: 28rpx;
color: #2d3748;
background: #ffffff;
transition: all 0.3s ease;
}
.form-input:focus,
.form-textarea:focus {
border-color: #4f46e5;
box-shadow: 0 0 0 4rpx rgba(79, 70, 229, 0.1);
outline: none;
}
.form-textarea {
height: 120rpx;
resize: none;
}
.priority-options {
display: flex;
gap: 16rpx;
}
.priority-option {
flex: 1;
padding: 16rpx;
border: 2rpx solid #e2e8f0;
border-radius: 12rpx;
display: flex;
align-items: center;
gap: 12rpx;
transition: all 0.3s ease;
background: #ffffff;
}
.priority-option.active {
border-color: #4f46e5;
background: #e0e7ff;
box-shadow: 0 0 0 4rpx rgba(79, 70, 229, 0.1);
}
.priority-dot {
width: 16rpx;
height: 16rpx;
border-radius: 50%;
}
.priority-dot.high {
background: #dc2626;
}
.priority-dot.medium {
background: #f59e0b;
}
.priority-dot.low {
background: #059669;
}
.priority-text {
font-size: 26rpx;
color: #2d3748;
font-weight: 500;
}
.date-picker {
padding: 20rpx;
border: 2rpx solid #e2e8f0;
border-radius: 12rpx;
display: flex;
justify-content: space-between;
align-items: center;
background: #ffffff;
transition: all 0.3s ease;
}
.date-picker:active {
border-color: #4f46e5;
box-shadow: 0 0 0 4rpx rgba(79, 70, 229, 0.1);
}
.date-text {
font-size: 28rpx;
color: #2d3748;
font-weight: 500;
}
.tags-display {
display: flex;
flex-wrap: wrap;
gap: 8rpx;
margin-top: 12rpx;
}
.tag-item {
background: linear-gradient(135deg, #dbeafe, #bfdbfe);
color: #1e40af;
padding: 8rpx 16rpx;
border-radius: 16rpx;
display: flex;
align-items: center;
gap: 8rpx;
font-size: 24rpx;
font-weight: 500;
}
.tag-remove {
color: #1e40af;
font-size: 20rpx;
transition: color 0.3s ease;
}
.tag-remove:active {
color: #1e3a8a;
}
.modal-footer {
padding: 30rpx;
border-top: 1rpx solid #e2e8f0;
display: flex;
gap: 20rpx;
background: #f7fafc;
}
.btn-cancel,
.btn-confirm {
flex: 1;
height: 80rpx;
border-radius: 16rpx;
font-size: 28rpx;
font-weight: 600;
border: none;
transition: all 0.3s ease;
}
.btn-cancel {
background: #ffffff;
color: #718096;
border: 2rpx solid #e2e8f0;
}
.btn-cancel:active {
background: #f7fafc;
color: #4a5568;
}
.btn-confirm {
background: linear-gradient(135deg, #4f46e5, #7c3aed);
color: #ffffff;
box-shadow: 0 4rpx 12rpx rgba(79, 70, 229, 0.3);
}
.btn-confirm:active {
transform: translateY(1rpx);
box-shadow: 0 2rpx 8rpx rgba(79, 70, 229, 0.4);
}
.btn-confirm:disabled {
opacity: 0.6;
transform: none;
box-shadow: none;
}
/* 响应式设计 */
@media screen and (max-width: 750rpx) {
.stats-grid {
grid-template-columns: 1fr;
gap: 16rpx;
}
.filter-container {
flex-direction: column;
gap: 16rpx;
align-items: stretch;
}
.filter-tabs-wrapper {
justify-content: flex-start;
}
.filter-tabs {
justify-content: flex-start;
overflow-x: auto;
overflow-y: hidden;
-webkit-overflow-scrolling: touch;
scrollbar-width: none;
-ms-overflow-style: none;
}
.filter-tabs::-webkit-scrollbar {
display: none;
}
.filter-tab {
flex-shrink: 0;
flex-grow: 0;
min-width: fit-content;
}
.task-list {
height: calc(100vh - 600rpx);
}
}
</style>