Files
document-server/frontend/todos.html

675 lines
31 KiB
HTML

<!DOCTYPE html>
<html lang="ko">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>할일관리 - Document Server</title>
<script src="https://cdn.tailwindcss.com"></script>
<script src="https://unpkg.com/alpinejs@3.x.x/dist/cdn.min.js" defer></script>
<link rel="stylesheet" href="https://cdnjs.cloudflare.com/ajax/libs/font-awesome/6.0.0/css/all.min.css">
<style>
[x-cloak] { display: none !important; }
/* 모바일 최적화 */
@media (max-width: 768px) {
.container { padding-left: 1rem; padding-right: 1rem; }
.todo-input-inner { padding: 16px; }
.todo-textarea { font-size: 16px; } /* iOS 줌 방지 */
.todo-card { margin-bottom: 12px; }
.modal-content { margin: 1rem; max-height: 90vh; }
}
/* memos/트위터 스타일 입력창 */
.todo-input-container {
background: linear-gradient(135deg, #6366f1 0%, #8b5cf6 100%);
border-radius: 16px;
padding: 2px;
box-shadow: 0 4px 20px rgba(99, 102, 241, 0.3);
}
.todo-input-inner {
background: white;
border-radius: 14px;
padding: 20px;
}
.todo-textarea {
resize: none;
border: none;
outline: none;
font-size: 16px;
line-height: 1.6;
min-height: 80px;
font-family: -apple-system, BlinkMacSystemFont, 'Segoe UI', Roboto, sans-serif;
}
.todo-textarea::placeholder {
color: #9ca3af;
font-weight: 400;
}
/* 카드 스타일 */
.todo-card {
transition: all 0.2s ease;
border-radius: 12px;
border: 1px solid #e5e7eb;
background: white;
position: relative;
overflow: hidden;
}
.todo-card::before {
content: '';
position: absolute;
left: 0;
top: 0;
bottom: 0;
width: 4px;
transition: all 0.2s ease;
}
.todo-card.draft::before { background: #9ca3af; }
.todo-card.scheduled::before { background: #3b82f6; }
.todo-card.active::before { background: #f59e0b; }
.todo-card.completed::before { background: #10b981; }
.todo-card.delayed::before { background: #ef4444; }
.todo-card:active {
transform: scale(0.98);
box-shadow: 0 2px 8px rgba(0,0,0,0.1);
}
/* 상태 배지 */
.status-badge {
font-size: 11px;
padding: 4px 10px;
border-radius: 20px;
font-weight: 600;
text-transform: uppercase;
letter-spacing: 0.5px;
}
.status-draft { background: #f3f4f6; color: #6b7280; }
.status-scheduled { background: #dbeafe; color: #1d4ed8; }
.status-active { background: #fef3c7; color: #d97706; }
.status-completed { background: #d1fae5; color: #065f46; }
.status-delayed { background: #fee2e2; color: #dc2626; }
/* 시간 배지 */
.time-badge {
background: #f0f9ff;
color: #0369a1;
font-size: 10px;
padding: 3px 8px;
border-radius: 12px;
font-weight: 500;
}
/* 탭 스타일 */
.tab-container {
background: white;
border-radius: 12px;
padding: 4px;
box-shadow: 0 2px 8px rgba(0,0,0,0.1);
}
.tab-button {
border-radius: 8px;
transition: all 0.2s ease;
font-weight: 500;
font-size: 14px;
}
.tab-button.active {
background: linear-gradient(135deg, #6366f1 0%, #8b5cf6 100%);
color: white;
box-shadow: 0 2px 8px rgba(99, 102, 241, 0.3);
}
/* 버튼 스타일 */
.btn-primary {
background: linear-gradient(135deg, #6366f1 0%, #8b5cf6 100%);
border: none;
border-radius: 12px;
font-weight: 600;
transition: all 0.2s ease;
box-shadow: 0 2px 8px rgba(99, 102, 241, 0.3);
}
.btn-primary:hover {
transform: translateY(-1px);
box-shadow: 0 4px 12px rgba(99, 102, 241, 0.4);
}
.btn-primary:active {
transform: translateY(0);
}
/* 액션 버튼들 */
.action-btn {
border-radius: 8px;
font-size: 12px;
font-weight: 500;
padding: 6px 12px;
transition: all 0.2s ease;
border: none;
}
.action-btn:active {
transform: scale(0.95);
}
/* 댓글 버블 */
.comment-bubble {
background: #f8fafc;
border-radius: 12px;
padding: 12px;
margin-top: 8px;
border-left: 3px solid #e2e8f0;
}
/* 모달 스타일 */
.modal-content {
border-radius: 16px;
box-shadow: 0 20px 40px rgba(0,0,0,0.15);
}
/* 빈 상태 */
.empty-state {
padding: 3rem 1rem;
text-align: center;
color: #6b7280;
}
.empty-state i {
margin-bottom: 1rem;
opacity: 0.5;
}
/* 스와이프 힌트 */
.swipe-hint {
font-size: 12px;
color: #9ca3af;
text-align: center;
padding: 8px;
background: #f9fafb;
border-radius: 8px;
margin-bottom: 16px;
}
/* 풀 투 리프레시 스타일 */
.pull-refresh {
text-align: center;
padding: 20px;
color: #6366f1;
font-weight: 500;
}
/* 햅틱 피드백 애니메이션 */
@keyframes haptic-pulse {
0% { transform: scale(1); }
50% { transform: scale(1.05); }
100% { transform: scale(1); }
}
.haptic-feedback {
animation: haptic-pulse 0.1s ease-in-out;
}
</style>
</head>
<body class="bg-gray-50 min-h-screen" x-data="todosApp()" x-init="init()" x-cloak>
<!-- 헤더 -->
<div id="header-container"></div>
<!-- 메인 컨텐츠 -->
<main class="container mx-auto px-4 pt-20 pb-8">
<!-- 페이지 헤더 -->
<div class="text-center mb-8">
<h1 class="text-3xl md:text-4xl font-bold text-gray-900 mb-3">
<i class="fas fa-tasks text-indigo-600 mr-3"></i>
할일관리
</h1>
<p class="text-lg md:text-xl text-gray-600 mb-4">효율적인 일정 관리와 생산성 향상</p>
<!-- 간단한 통계 -->
<div class="flex justify-center space-x-6 text-sm text-gray-500">
<div class="flex items-center">
<i class="fas fa-inbox w-4 h-4 mr-1 text-gray-400"></i>
<span>검토필요 <strong x-text="stats.draft_count || 0"></strong></span>
</div>
<div class="flex items-center">
<i class="fas fa-tasks w-4 h-4 mr-1 text-blue-500"></i>
<span>진행중 <strong x-text="stats.todo_count || 0"></strong></span>
</div>
<div class="flex items-center">
<i class="fas fa-check w-4 h-4 mr-1 text-green-500"></i>
<span>완료 <strong x-text="stats.completed_count || 0"></strong></span>
</div>
</div>
</div>
<!-- 할일 입력 (memos/트위터 스타일) -->
<div class="max-w-2xl mx-auto mb-8">
<div class="todo-input-container">
<div class="todo-input-inner">
<textarea
x-model="newTodoContent"
@keydown.ctrl.enter="createTodo()"
placeholder="새로운 할일을 입력하세요... (Ctrl+Enter로 저장)"
class="todo-textarea w-full"
rows="3"
></textarea>
<div class="flex items-center justify-between mt-4">
<div class="text-sm text-gray-500">
<i class="fas fa-lightbulb mr-1"></i>
Ctrl+Enter로 빠르게 저장하세요
</div>
<button
@click="createTodo(); hapticFeedback($event.target)"
:disabled="!newTodoContent.trim() || loading"
class="btn-primary px-6 py-3 text-white rounded-full disabled:opacity-50 disabled:cursor-not-allowed"
>
<i class="fas fa-plus mr-2"></i>
<span x-show="!loading">추가</span>
<span x-show="loading">
<i class="fas fa-spinner fa-spin mr-2"></i>저장 중...
</span>
</button>
</div>
</div>
</div>
</div>
<!-- 탭 네비게이션 -->
<div class="max-w-4xl mx-auto mb-6">
<!-- 모바일용 스와이프 힌트 -->
<div class="swipe-hint md:hidden">
<i class="fas fa-hand-pointer mr-1"></i>
탭을 눌러서 전환하세요
</div>
<div class="tab-container">
<div class="grid grid-cols-3 gap-1">
<button
@click="activeTab = 'draft'; hapticFeedback($event.target)"
:class="activeTab === 'draft' ? 'tab-button active' : 'tab-button text-gray-600 hover:text-gray-900'"
class="px-4 py-3 text-center"
>
<div class="flex flex-col items-center">
<i class="fas fa-inbox text-lg mb-1"></i>
<span class="text-sm font-medium">검토필요</span>
<span class="text-xs opacity-75">(<span x-text="stats.draft_count || 0"></span>)</span>
</div>
</button>
<button
@click="activeTab = 'todo'; hapticFeedback($event.target)"
:class="activeTab === 'todo' ? 'tab-button active' : 'tab-button text-gray-600 hover:text-gray-900'"
class="px-4 py-3 text-center"
>
<div class="flex flex-col items-center">
<i class="fas fa-tasks text-lg mb-1"></i>
<span class="text-sm font-medium">TODO</span>
<span class="text-xs opacity-75">(<span x-text="stats.todo_count || 0"></span>)</span>
</div>
</button>
<button
@click="activeTab = 'completed'; hapticFeedback($event.target)"
:class="activeTab === 'completed' ? 'tab-button active' : 'tab-button text-gray-600 hover:text-gray-900'"
class="px-4 py-3 text-center"
>
<div class="flex flex-col items-center">
<i class="fas fa-check text-lg mb-1"></i>
<span class="text-sm font-medium">완료된일</span>
<span class="text-xs opacity-75">(<span x-text="stats.completed_count || 0"></span>)</span>
</div>
</button>
</div>
</div>
</div>
<!-- 할일 목록 -->
<div class="max-w-6xl mx-auto">
<!-- 검토필요 탭 -->
<div x-show="activeTab === 'draft'" class="space-y-4">
<template x-for="todo in draftTodos" :key="todo.id">
<div class="todo-card draft bg-white rounded-lg shadow-sm p-6">
<div class="flex items-start justify-between">
<div class="flex-1">
<p class="text-gray-900 mb-3" x-text="todo.content"></p>
<div class="flex items-center space-x-2">
<span class="status-badge status-draft">검토필요</span>
<span class="text-xs text-gray-500" x-text="formatDate(todo.created_at)"></span>
</div>
</div>
<div class="flex items-center space-x-2 ml-4">
<button
@click="openScheduleModal(todo)"
class="px-3 py-1 bg-blue-600 text-white text-sm rounded-md hover:bg-blue-700"
>
<i class="fas fa-calendar-plus mr-1"></i>일정설정
</button>
<button
@click="openSplitModal(todo)"
class="px-3 py-1 bg-green-600 text-white text-sm rounded-md hover:bg-green-700"
>
<i class="fas fa-cut mr-1"></i>분할
</button>
</div>
</div>
</div>
</template>
<div x-show="draftTodos.length === 0" class="text-center py-12">
<i class="fas fa-inbox text-6xl text-gray-300 mb-4"></i>
<p class="text-gray-500">검토가 필요한 할일이 없습니다</p>
</div>
</div>
<!-- TODO 탭 -->
<div x-show="activeTab === 'todo'" class="space-y-4">
<template x-for="todo in activeTodos" :key="todo.id">
<div class="todo-card active bg-white rounded-lg shadow-sm p-6" x-data="{ newMemo: '', addingMemo: false }">
<div class="flex items-start justify-between">
<div class="flex-1">
<p class="text-gray-900 mb-3" x-text="todo.content"></p>
<div class="flex items-center space-x-2 mb-3">
<span class="status-badge status-active">진행중</span>
<span class="time-badge" x-text="`${todo.estimated_minutes}분`"></span>
<span class="text-xs text-gray-500" x-text="formatDate(todo.start_date)"></span>
</div>
</div>
<div class="flex items-center space-x-2 ml-4">
<button
@click="completeTodo(todo.id); hapticFeedback($event.target)"
class="action-btn bg-green-600 text-white hover:bg-green-700"
>
<i class="fas fa-check mr-1"></i>완료
</button>
<button
@click="openScheduleModal(todo); hapticFeedback($event.target)"
class="action-btn bg-orange-600 text-white hover:bg-orange-700"
>
<i class="fas fa-clock mr-1"></i>지연
</button>
<button
@click="toggleMemo(todo.id); hapticFeedback($event.target)"
class="action-btn bg-indigo-600 text-white hover:bg-indigo-700"
>
<i class="fas fa-sticky-note mr-1"></i>메모
<span x-show="getTodoMemoCount(todo.id) > 0" class="ml-1 bg-white text-indigo-600 rounded-full px-1 text-xs" x-text="getTodoMemoCount(todo.id)"></span>
</button>
</div>
</div>
<!-- 인라인 메모 섹션 -->
<div x-show="showMemoForTodo[todo.id]" x-transition class="mt-4 border-t pt-4">
<!-- 기존 메모들 -->
<div x-show="getTodoMemos(todo.id).length > 0" class="space-y-2 mb-3">
<template x-for="memo in getTodoMemos(todo.id)" :key="memo.id">
<div class="comment-bubble">
<p class="text-sm text-gray-700" x-text="memo.content"></p>
<div class="text-xs text-gray-500 mt-1" x-text="formatRelativeTime(memo.created_at)"></div>
</div>
</template>
</div>
<!-- 새 메모 입력 -->
<div class="flex space-x-2">
<input
type="text"
x-model="newMemo"
placeholder="메모를 입력하세요..."
class="flex-1 px-3 py-2 border border-gray-300 rounded-lg text-sm focus:outline-none focus:ring-2 focus:ring-indigo-500"
@keydown.enter="if(newMemo.trim()) { addingMemo = true; addTodoMemo(todo.id, newMemo).then(() => { newMemo = ''; addingMemo = false; }); }"
>
<button
@click="if(newMemo.trim()) { addingMemo = true; addTodoMemo(todo.id, newMemo).then(() => { newMemo = ''; addingMemo = false; }); hapticFeedback($event.target); }"
:disabled="!newMemo.trim() || addingMemo"
class="action-btn bg-indigo-600 text-white hover:bg-indigo-700 disabled:opacity-50"
>
<i class="fas fa-plus" x-show="!addingMemo"></i>
<i class="fas fa-spinner fa-spin" x-show="addingMemo"></i>
</button>
</div>
</div>
</div>
</template>
<div x-show="activeTodos.length === 0" class="empty-state">
<i class="fas fa-tasks text-4xl"></i>
<p class="text-lg font-medium">할 일이 없습니다</p>
<p class="text-sm">검토필요에서 일정을 설정해보세요</p>
</div>
</div>
<!-- 예정된일 탭 -->
<div x-show="activeTab === 'scheduled'" class="space-y-4">
<template x-for="todo in scheduledTodos" :key="todo.id">
<div class="todo-card scheduled bg-white rounded-lg shadow-sm p-6">
<div class="flex items-start justify-between">
<div class="flex-1">
<p class="text-gray-900 mb-3" x-text="todo.content"></p>
<div class="flex items-center space-x-2">
<span class="status-badge status-scheduled">예정됨</span>
<span class="time-badge" x-text="`${todo.estimated_minutes}분`"></span>
<span class="text-xs text-gray-500" x-text="formatDate(todo.start_date)"></span>
</div>
</div>
</div>
</div>
</template>
<div x-show="scheduledTodos.length === 0" class="text-center py-12">
<i class="fas fa-calendar text-6xl text-gray-300 mb-4"></i>
<p class="text-gray-500">예정된 할일이 없습니다</p>
</div>
</div>
<!-- 완료된일 탭 -->
<div x-show="activeTab === 'completed'" class="space-y-4">
<template x-for="todo in completedTodos" :key="todo.id">
<div class="todo-card completed bg-white rounded-lg shadow-sm p-6">
<div class="flex items-start justify-between">
<div class="flex-1">
<p class="text-gray-900 mb-3" x-text="todo.content"></p>
<div class="flex items-center space-x-2">
<span class="status-badge status-completed">완료</span>
<span class="time-badge" x-text="`${todo.estimated_minutes}분`"></span>
<span class="text-xs text-gray-500" x-text="formatDate(todo.completed_at)"></span>
</div>
</div>
</div>
</div>
</template>
<div x-show="completedTodos.length === 0" class="text-center py-12">
<i class="fas fa-check text-6xl text-gray-300 mb-4"></i>
<p class="text-gray-500">완료된 할일이 없습니다</p>
</div>
</div>
</div>
</main>
<!-- 일정 설정 모달 -->
<div x-show="showScheduleModal" x-transition class="fixed inset-0 bg-black bg-opacity-50 z-50 flex items-center justify-center p-4">
<div class="bg-white rounded-2xl shadow-2xl max-w-md w-full">
<div class="p-6 border-b">
<h3 class="text-xl font-bold text-gray-900" x-text="activeTab === 'todo' ? '일정 지연' : '일정 설정'"></h3>
<p class="text-sm text-gray-600 mt-1" x-text="activeTab === 'todo' ? '새로운 날짜와 시간을 설정하세요' : '날짜와 예상 소요시간을 설정하세요'"></p>
</div>
<div class="p-6">
<div class="mb-4">
<label class="block text-sm font-medium text-gray-700 mb-2">시작 날짜</label>
<input
type="date"
x-model="scheduleForm.start_date"
class="w-full px-3 py-2 border border-gray-300 rounded-lg focus:ring-2 focus:ring-indigo-500"
>
</div>
<div class="mb-6">
<label class="block text-sm font-medium text-gray-700 mb-2">예상 소요시간</label>
<select
x-model="scheduleForm.estimated_minutes"
class="w-full px-3 py-2 border border-gray-300 rounded-lg focus:ring-2 focus:ring-indigo-500"
>
<option value="15">15분</option>
<option value="30">30분</option>
<option value="60">1시간</option>
<option value="90">1시간 30분</option>
<option value="120">2시간</option>
</select>
<p class="text-xs text-gray-500 mt-1">
<i class="fas fa-info-circle mr-1"></i>
하루 8시간 초과 시 경고가 표시됩니다
</p>
</div>
<div class="flex space-x-3">
<button
@click="scheduleTodo(); hapticFeedback($event.target)"
:disabled="!scheduleForm.start_date || !scheduleForm.estimated_minutes"
class="flex-1 btn-primary px-4 py-2 text-white rounded-lg disabled:opacity-50"
x-text="activeTab === 'todo' ? '지연 설정' : '일정 설정'"
>
</button>
<button
@click="closeScheduleModal(); hapticFeedback($event.target)"
class="px-4 py-2 bg-gray-300 text-gray-700 rounded-lg hover:bg-gray-400"
>
취소
</button>
</div>
</div>
</div>
</div>
<!-- 분할 모달 -->
<div x-show="showSplitModal" x-transition class="fixed inset-0 bg-black bg-opacity-50 z-50 flex items-center justify-center p-4">
<div class="bg-white rounded-2xl shadow-2xl max-w-lg w-full">
<div class="p-6 border-b">
<h3 class="text-xl font-bold text-gray-900">할일 분할</h3>
<p class="text-sm text-gray-600 mt-1">2시간 이상의 작업을 작은 단위로 나누어 관리하세요</p>
</div>
<div class="p-6">
<div class="space-y-4 mb-6">
<template x-for="(subtask, index) in splitForm.subtasks" :key="index">
<div class="flex items-center space-x-3">
<div class="flex-1">
<input
type="text"
x-model="splitForm.subtasks[index]"
:placeholder="`하위 작업 ${index + 1}`"
class="w-full px-3 py-2 border border-gray-300 rounded-lg focus:ring-2 focus:ring-purple-500"
>
</div>
<select
x-model="splitForm.estimated_minutes_per_task[index]"
class="px-3 py-2 border border-gray-300 rounded-lg focus:ring-2 focus:ring-purple-500"
>
<option value="1">1분</option>
<option value="30">30분</option>
<option value="60">1시간</option>
</select>
<button
x-show="splitForm.subtasks.length > 2"
@click="splitForm.subtasks.splice(index, 1); splitForm.estimated_minutes_per_task.splice(index, 1)"
class="px-2 py-2 text-red-600 hover:bg-red-50 rounded"
>
<i class="fas fa-trash"></i>
</button>
</div>
</template>
</div>
<div class="flex items-center justify-between mb-6">
<button
@click="splitForm.subtasks.push(''); splitForm.estimated_minutes_per_task.push(30)"
class="px-3 py-1 bg-gray-200 text-gray-700 text-sm rounded-md hover:bg-gray-300"
>
<i class="fas fa-plus mr-1"></i>하위 작업 추가
</button>
<span class="text-xs text-gray-500">최대 10개까지 가능</span>
</div>
<div class="flex space-x-3">
<button
@click="splitTodo()"
class="flex-1 px-4 py-2 bg-purple-600 text-white rounded-lg hover:bg-purple-700"
>
분할하기
</button>
<button
@click="closeSplitModal()"
class="px-4 py-2 bg-gray-300 text-gray-700 rounded-lg hover:bg-gray-400"
>
취소
</button>
</div>
</div>
</div>
</div>
<!-- 댓글 모달 -->
<div x-show="showCommentModal" x-transition class="fixed inset-0 bg-black bg-opacity-50 z-50 flex items-center justify-center p-4">
<div class="bg-white rounded-2xl shadow-2xl max-w-lg w-full max-h-[80vh] overflow-y-auto">
<div class="p-6 border-b">
<h3 class="text-xl font-bold text-gray-900">메모 작성</h3>
</div>
<div class="p-6">
<!-- 기존 댓글들 -->
<div x-show="currentTodoComments.length > 0" class="mb-4 space-y-3">
<template x-for="comment in currentTodoComments" :key="comment.id">
<div class="comment-bubble">
<p class="text-gray-900 text-sm" x-text="comment.content"></p>
<p class="text-xs text-gray-500 mt-2" x-text="formatDate(comment.created_at)"></p>
</div>
</template>
</div>
<!-- 새 댓글 입력 -->
<div class="mb-4">
<textarea
x-model="commentForm.content"
placeholder="메모를 입력하세요..."
class="w-full px-3 py-2 border border-gray-300 rounded-lg focus:ring-2 focus:ring-purple-500 resize-none"
rows="3"
></textarea>
</div>
<div class="flex space-x-3">
<button
@click="addComment()"
:disabled="!commentForm.content.trim()"
class="flex-1 px-4 py-2 bg-purple-600 text-white rounded-lg hover:bg-purple-700 disabled:opacity-50"
>
메모 추가
</button>
<button
@click="closeCommentModal()"
class="px-4 py-2 bg-gray-300 text-gray-700 rounded-lg hover:bg-gray-400"
>
닫기
</button>
</div>
</div>
</div>
</div>
<!-- 헤더 로더 -->
<script src="static/js/header-loader.js"></script>
<!-- API 및 앱 스크립트 -->
<script src="static/js/api.js?v=2025012627"></script>
<script src="static/js/todos.js?v=2025012627"></script>
</body>
</html>